/***********Template Starts Here***********/
#include <bits/stdc++.h>
#define pb push_back
#define nl puts ("")
#define sp printf ( " " )
#define phl printf ( "hello\n" )
#define ff first
#define ss second
#define POPCOUNT __builtin_popcountll
#define RIGHTMOST __builtin_ctzll
#define LEFTMOST(x) (63-__builtin_clzll((x)))
#define MP make_pair
#define FOR(i,x,y) for(vlong i = (x) ; i <= (y) ; ++i)
#define ROF(i,x,y) for(vlong i = (y) ; i >= (x) ; --i)
#define CLR(x,y) memset(x,y,sizeof(x))
#define UNIQUE(V) (V).erase(unique((V).begin(),(V).end()),(V).end())
#define MIN(a,b) ((a)<(b)?(a):(b))
#define MAX(a,b) ((a)>(b)?(a):(b))
#define NUMDIGIT(x,y) (((vlong)(log10((x))/log10((y))))+1)
#define SQ(x) ((x)*(x))
#define ABS(x) ((x)<0?-(x):(x))
#define FABS(x) ((x)+eps<0?-(x):(x))
#define ALL(x) (x).begin(),(x).end()
#define LCM(x,y) (((x)/gcd((x),(y)))*(y))
#define SZ(x) ((vlong)(x).size())
#define NORM(x) if(x>=mod)x-=mod;
using namespace std;
typedef long long vlong;
typedef unsigned long long uvlong;
typedef pair < int, int > pii;
typedef pair < vlong, vlong > pll;
typedef vector<pii> vii;
typedef vector<int> vi;
const vlong inf = 2147383647;
/***********Template Ends Here***********/
#define LCANODE 100010
#define LCADEPTH 20
vi adj[LCANODE];
struct LCA{
private:
int tab[LCANODE][LCADEPTH], lev[LCANODE], vis[LCANODE], q[LCANODE], cc, N, M, root;
public:
int par[LCANODE];
private:
void bfs ( int s ) {
vis[s] = cc;
lev[s] = 0;
par[s] = -1; ///Set par[root] = -1
int qh = 0, qt = 0;
q[qt++] = s;
int t;
while ( qh < qt ) {
s = q[qh++];
FOR(i,0,SZ(adj[s])-1){
t = adj[s][i];
if ( vis[t] != cc ) {
par[t] = s;
vis[t] = cc;
lev[t] = lev[s] + 1;
q[qt++] = t;
}
}
}
}
void calculate (){
int i,j,p;
for (i=0;i<=N;i++){
tab[i][0]=par[i];
}
for(i=1;i<=M;i++){
for(j=0;j<=N;j++){
p=tab[j][i-1];
if(p==-1) tab[j][i]=-1;
else tab[j][i]=tab[p][i-1];
}
}
}
public:
void clear () {
cc++;
FOR(i,0,N) adj[i].clear();
}
LCA() {
cc = 1;
}
void setVar ( int n, int r ) {
N = n;
M = 0;
int temp = n+1;
while ( temp ) {
M++;
temp /= 2;
}
root = r;
clear();
}
void precal () {
bfs ( root );
calculate();
}
int findLCA(int s,int t){
int dif,pos,i;
if(lev[s]!=lev[t] ) {
if(lev[s]>lev[t]) swap( s, t );
dif=lev[t]-lev[s];
while(dif){
pos=RIGHTMOST(dif);
t=tab[t][pos];
dif-=1<<pos;
}
}
if (s==t)return s;
for(i=M;i>=0;i--){
if(tab[s][i]!=tab[t][i]){
s=tab[s][i];
t=tab[t][i];
}
}
return tab[s][0];
}
}lca;
int child[100010];
int pretrav[100010], pre = 0;
int preStart[100010];
void dfs ( int s, int p ) {
child[s] = 1;
preStart[s] = pre;
pretrav[pre++] = s;
FOR(i,0,SZ(adj[s])-1){
int t = adj[s][i];
if ( t == p ) continue;
dfs ( t, s );
child[s] += child[t];
}
}
int color[100010], arr[11][100010];
void dfs2 ( int s, int p, int d, int c ) {
arr[c][s] = d;
FOR(i,0,SZ(adj[s])-1){
int t = adj[s][i];
if ( t == p ) continue;
int cost = 0;
if ( color[t] == c ) cost = 1;
dfs2 ( t, s, d + cost, c );
}
}
struct node {
int val;
int lazy;
}tn[11][4*100000];
void build ( int col, int t, int b, int e ) {
int m = ( b + e ) / 2;
int u = t * 2;
int v = u + 1;
tn[col][t].lazy = 0; ///lazyClear();
if ( b == e ) {
tn[col][t].val = arr[col][pretrav[b]];
return;
}
build( col, u, b, m );
build( col, v, m + 1, e );
}
int qb, qe, qv;
void lazyPropagate ( node &p, node &u, node &v ) {
u.lazy += p.lazy;
v.lazy += p.lazy;
}
void update ( int col, int t, int b, int e ) {
int m = ( b + e ) / 2;
int u = t * 2;
int v = u + 1;
if ( qb <= b && e <= qe ) {
tn[col][t].lazy += qv;///lazyUpdate();
tn[col][t].val += tn[col][t].lazy;///calculate();
if ( b != e ) lazyPropagate ( tn[col][t], tn[col][u], tn[col][v] );
tn[col][t].lazy = 0; ///LazyClear
return;
}
if ( qe < b || e < qb ) {
tn[col][t].val += tn[col][t].lazy;///calculate();
if ( b != e ) lazyPropagate ( tn[col][t], tn[col][u], tn[col][v] );
tn[col][t].lazy = 0; ///LazyClear
return;
}
lazyPropagate ( tn[col][t], tn[col][u], tn[col][v] );
tn[col][t].lazy = 0; ///LazyClear
update( col, u, b, m );
update( col, v, m + 1, e );
}
int qvv[11];
void query ( int t, int b, int e ) {
int m = ( b + e ) / 2;
int u = t * 2;
int v = u + 1;
if ( qb <= b && e <= qe ) {
FOR(i,1,10) tn[i][t].val += tn[i][t].lazy;///calculate();
if ( b != e ) {
FOR(i,1,10) lazyPropagate ( tn[i][t], tn[i][u], tn[i][v] );
}
FOR(i,1,10) tn[i][t].lazy = 0; ///LazyClear
FOR(i,1,10) qvv[i] = tn[i][t].val;
return;
}
if ( qe < b || e < qb ) {
FOR(i,1,10) tn[i][t].val += tn[i][t].lazy;///calculate();
if ( b != e ) {
FOR(i,1,10) lazyPropagate ( tn[i][t], tn[i][u], tn[i][v] );
}
FOR(i,1,10) tn[i][t].lazy = 0; ///LazyClear
return;
}
FOR(i,1,10) lazyPropagate ( tn[i][t], tn[i][u], tn[i][v] );
FOR(i,1,10) tn[i][t].lazy = 0; ///LazyClear
query( u, b, m );
query( v, m + 1, e );
}
int main () {
int kase;
scanf ( "%d", &kase );
while ( kase-- ) {
int n, m;
scanf ( "%d %d", &n, &m );
lca.setVar( n, 0 );
CLR(color,0);
FOR(i,1,n){
int c;
scanf ( "%d", &c );
color[i] = c; ///Set color
}
FOR(i,0,n-2){
int a, b;
scanf ( "%d %d", &a, &b );
adj[a].pb ( b );
adj[b].pb ( a );
}
adj[0].pb ( 1 ); ///Dummy node
lca.precal();
pre = 0;
dfs ( 0, -1 ); ///Calculate child, starting position in pre-traversal
///Calculate distance in each color tree
FOR(i,1,10){
dfs2 ( 0, -1, 0, i );
}
///Preparations complete. Start segment tree
FOR(i,1,10) {
build ( i, 1, 0, n );
}
while ( m-- ) {
int t;
scanf ( "%d", &t );
if ( t == 0 ) {
int u, c;
scanf ( "%d %d", &u, &c );
///Change color from color[u] to c
int p = color[u];
///Update the subtree of u of color p with -1
qb = preStart[u]; qe = qb + child[u] - 1; qv = -1;
update ( p, 1, 0, n );
///Update the subtree of u of color c with +1
qv = 1;
update ( c, 1, 0, n );
color[u] = c;
}
else {
int u, v;
scanf ( "%d %d", &u, &v );
///Find distance in each color tree
int res = 0;
int tres[11];
CLR(tres,0);
qb = preStart[u]; qe = qb;
query( 1, 0, n );
FOR(i,1,10) tres[i] = qvv[i];
qb = preStart[v]; qe = qb;
query( 1, 0, n );
FOR(i,1,10) tres[i] += qvv[i];
int par = lca.findLCA( u, v );
qb = preStart[par]; qe = qb;
query( 1, 0, n );
FOR(i,1,10) tres[i] -= 2 * qvv[i];
FOR(i,1,10){
int col = color[par];
if ( col == i ) tres[i]++;
}
FOR(i,1,10) res = MAX(res,tres[i]);
printf ( "%d\n", res );
}
}
}
return 0;
}
/***********Template Starts Here***********/
#include <bits/stdc++.h>

#define pb push_back
#define nl puts ("")
#define sp printf ( " " )
#define phl printf ( "hello\n" )
#define ff first
#define ss second
#define POPCOUNT __builtin_popcountll
#define RIGHTMOST __builtin_ctzll
#define LEFTMOST(x) (63-__builtin_clzll((x)))
#define MP make_pair
#define FOR(i,x,y) for(vlong i = (x) ; i <= (y) ; ++i)
#define ROF(i,x,y) for(vlong i = (y) ; i >= (x) ; --i)
#define CLR(x,y) memset(x,y,sizeof(x))
#define UNIQUE(V) (V).erase(unique((V).begin(),(V).end()),(V).end())
#define MIN(a,b) ((a)<(b)?(a):(b))
#define MAX(a,b) ((a)>(b)?(a):(b))
#define NUMDIGIT(x,y) (((vlong)(log10((x))/log10((y))))+1)
#define SQ(x) ((x)*(x))
#define ABS(x) ((x)<0?-(x):(x))
#define FABS(x) ((x)+eps<0?-(x):(x))
#define ALL(x) (x).begin(),(x).end()
#define LCM(x,y) (((x)/gcd((x),(y)))*(y))
#define SZ(x) ((vlong)(x).size())
#define NORM(x) if(x>=mod)x-=mod;

using namespace std;

typedef long long vlong;
typedef unsigned long long uvlong;
typedef pair < int, int > pii;
typedef pair < vlong, vlong > pll;
typedef vector<pii> vii;
typedef vector<int> vi;

const vlong inf = 2147383647;

/***********Template Ends Here***********/

#define LCANODE 100010
#define LCADEPTH 20
vi adj[LCANODE];
struct LCA{
    private:
    int tab[LCANODE][LCADEPTH], lev[LCANODE], vis[LCANODE], q[LCANODE], cc, N, M, root;

    public:
    int par[LCANODE];

    private:
    void bfs ( int s ) {
        vis[s] = cc;
        lev[s] = 0;
        par[s] = -1; ///Set par[root] = -1

        int qh = 0, qt = 0;
        q[qt++] = s;
        int t;
        while ( qh < qt ) {
            s = q[qh++];
            FOR(i,0,SZ(adj[s])-1){
                t = adj[s][i];
                if ( vis[t] != cc ) {
                    par[t] = s;
                    vis[t] = cc;
                    lev[t] = lev[s] + 1;
                    q[qt++] = t;
                }
            }
        }
    }

    void calculate (){
        int i,j,p;
        for (i=0;i<=N;i++){
           tab[i][0]=par[i];
        }

        for(i=1;i<=M;i++){
            for(j=0;j<=N;j++){
                p=tab[j][i-1];
                if(p==-1) tab[j][i]=-1;
                else tab[j][i]=tab[p][i-1];
            }
        }
    }

    public:
    void clear () {
        cc++;
        FOR(i,0,N) adj[i].clear();
    }

    LCA() {
        cc = 1;
    }

    void setVar ( int n, int r ) {
        N = n;

        M = 0;
        int temp = n+1;
        while ( temp ) {
            M++;
            temp /= 2;
        }
        root = r;
        clear();
    }
    void precal () {
        bfs ( root );
        calculate();
    }
    int findLCA(int s,int t){
        int dif,pos,i;
        if(lev[s]!=lev[t] ) {
            if(lev[s]>lev[t]) swap( s, t );
            dif=lev[t]-lev[s];
            while(dif){
                pos=RIGHTMOST(dif);
                t=tab[t][pos];
                dif-=1<<pos;
            }
        }
        if (s==t)return s;
        for(i=M;i>=0;i--){
            if(tab[s][i]!=tab[t][i]){
                s=tab[s][i];
                t=tab[t][i];
            }
        }
        return tab[s][0];
    }
}lca;

int child[100010];
int pretrav[100010], pre = 0;
int preStart[100010];
void dfs ( int s, int p ) {
    child[s] = 1;
    preStart[s] = pre;
    pretrav[pre++] = s;

    FOR(i,0,SZ(adj[s])-1){
        int t = adj[s][i];
        if ( t == p ) continue;
        dfs ( t, s );
        child[s] += child[t];
    }
}

int color[100010], arr[11][100010];

void dfs2 ( int s, int p, int d, int c ) {
    arr[c][s] = d;

    FOR(i,0,SZ(adj[s])-1){
        int t = adj[s][i];
        if ( t == p ) continue;
        int cost = 0;
        if ( color[t] == c ) cost = 1;

        dfs2 ( t, s, d + cost, c );
    }
}

struct node {
    int val;
    int lazy;
}tn[11][4*100000];

void build ( int col, int t, int b, int e ) {
    int m = ( b + e ) / 2;
    int u = t * 2;
    int v = u + 1;

    tn[col][t].lazy = 0; ///lazyClear();

    if ( b == e ) {
        tn[col][t].val = arr[col][pretrav[b]];
        return;
    }

    build( col, u, b, m );
    build( col, v, m + 1, e );
}

int qb, qe, qv;
void lazyPropagate ( node &p, node &u, node &v ) {
    u.lazy += p.lazy;
    v.lazy += p.lazy;
}

void update ( int col, int t, int b, int e ) {
    int m = ( b + e ) / 2;
    int u = t * 2;
    int v = u + 1;

    if ( qb <= b && e <= qe ) {
        tn[col][t].lazy += qv;///lazyUpdate();
        tn[col][t].val += tn[col][t].lazy;///calculate();
        if ( b != e ) lazyPropagate ( tn[col][t], tn[col][u], tn[col][v] );
        tn[col][t].lazy = 0; ///LazyClear
        return;
    }
    if ( qe < b || e < qb ) {
        tn[col][t].val += tn[col][t].lazy;///calculate();
        if ( b != e ) lazyPropagate ( tn[col][t], tn[col][u], tn[col][v] );
        tn[col][t].lazy = 0; ///LazyClear
        return;
    }

    lazyPropagate ( tn[col][t], tn[col][u], tn[col][v] );
    tn[col][t].lazy = 0; ///LazyClear

    update( col, u, b, m );
    update( col, v, m + 1, e );

}

int qvv[11];

void query ( int t, int b, int e ) {
    int m = ( b + e ) / 2;
    int u = t * 2;
    int v = u + 1;

    if ( qb <= b && e <= qe ) {
        FOR(i,1,10) tn[i][t].val += tn[i][t].lazy;///calculate();
        if ( b != e ) {
            FOR(i,1,10) lazyPropagate ( tn[i][t], tn[i][u], tn[i][v] );
        }
        FOR(i,1,10) tn[i][t].lazy = 0; ///LazyClear
        FOR(i,1,10) qvv[i] = tn[i][t].val;
        return;
    }
    if ( qe < b || e < qb ) {
        FOR(i,1,10) tn[i][t].val += tn[i][t].lazy;///calculate();
        if ( b != e ) {
            FOR(i,1,10) lazyPropagate ( tn[i][t], tn[i][u], tn[i][v] );
        }
        FOR(i,1,10) tn[i][t].lazy = 0; ///LazyClear
        return;
    }

    FOR(i,1,10) lazyPropagate ( tn[i][t], tn[i][u], tn[i][v] );
    FOR(i,1,10) tn[i][t].lazy = 0; ///LazyClear

    query( u, b, m );
    query( v, m + 1, e );

}

int main () {

    int kase;
    scanf ( "%d", &kase );

    while ( kase-- ) {
        int n, m;
        scanf ( "%d %d", &n, &m );

        lca.setVar( n, 0 );

        CLR(color,0);
        FOR(i,1,n){
            int c;
            scanf ( "%d", &c );
            color[i] = c; ///Set color
        }

        FOR(i,0,n-2){
            int a, b;
            scanf ( "%d %d", &a, &b );
            adj[a].pb ( b );
            adj[b].pb ( a );
        }
        adj[0].pb ( 1 ); ///Dummy node

        lca.precal();

        pre = 0;
        dfs ( 0, -1 ); ///Calculate child, starting position in pre-traversal

        ///Calculate distance in each color tree
        FOR(i,1,10){
            dfs2 ( 0, -1, 0, i );
        }


        ///Preparations complete. Start segment tree
        FOR(i,1,10) {
            build ( i, 1, 0, n );
        }

        while ( m-- ) {
            int t;
            scanf ( "%d", &t );

            if ( t == 0 ) {
                int u, c;
                scanf ( "%d %d", &u, &c );

                ///Change color from color[u] to c
                int p = color[u];
                ///Update the subtree of u of color p with -1
                qb = preStart[u]; qe = qb + child[u] - 1; qv = -1;
                update ( p, 1, 0, n );

                ///Update the subtree of u of color c with +1
                qv = 1;
                update ( c, 1, 0, n );
                color[u] = c;
            }
            else {
                int u, v;
                scanf ( "%d %d", &u, &v );

                ///Find distance in each color tree
                int res = 0;

                int tres[11];
                CLR(tres,0);

                qb = preStart[u]; qe = qb;
                query( 1, 0, n );

                FOR(i,1,10) tres[i] = qvv[i];


                qb = preStart[v]; qe = qb;
                query( 1, 0, n );

                FOR(i,1,10) tres[i] += qvv[i];

                int par = lca.findLCA( u, v );

                qb = preStart[par]; qe = qb;
                query( 1, 0, n );
                FOR(i,1,10) tres[i] -= 2 * qvv[i];

                FOR(i,1,10){
                    int col = color[par];
                    if ( col == i ) tres[i]++;
                }

                FOR(i,1,10) res = MAX(res,tres[i]);

                printf ( "%d\n", res );
            }
        }
    }

    return 0;
}
