#include <cstdio>
#include <algorithm>
#include <vector>
#include <cstring>
#include <climits>
using namespace std;
struct edge
{
int u, v, w, c, ci;
edge(int u, int v, int w, int c, int ci)
: u(u), v(v), w(w), c(c), ci(ci) {}
};
struct vertex
{
int p, d, se, c, ci;
vector<int> a;
vector<int> lca;
vertex():
p(-1),d(0),se(-1),c(-1),ci(-1),a(),lca() {}
};
struct chain
{
int je, h;
vector<int> e, st;
chain(int je, int h): je(je), h(h) {}
};
struct graph
{
vector<edge> e;
vector<vertex> v;
vector<chain> c;
};
void hld(int u, int c, int ci, graph& g);
int dfs(int u, int p, graph& g);
int log2(int n);
void comp_lca(graph &g);
int lca(int u, int v, graph &g);
void create_sts(graph &g);
int build_st(int ci, int idx, int l, int r, graph &g);
void update(int ci, int idx, int l, int r, int i, int nv, graph& g);
int path_max(int u, int v, graph& g);
int main()
{
int t, n, u, v, w;
graph g;
char q[10];
scanf("%d", &t);
while(t--)
{
scanf("%d", &n);
g.e.clear();
g.v.clear();
g.c.clear();
g.v.resize(n, vertex());
for(int i = 0; i < n-1; i++)
{
scanf("%d %d %d", &u, &v, &w);
u--; v--;
g.e.push_back(edge(u, v, w, -1, -1));
g.v[u].a.push_back(i);
g.v[v].a.push_back(i);
}
dfs(0, -1, g);
g.c.push_back(chain(-1, 0));
hld(0, 0, 0, g);
comp_lca(g);
create_sts(g);
while(true)
{
scanf("%s", q);
if(strcmp(q, "DONE") == 0)
break;
scanf("%d %d", &u, &v);
if(strcmp(q, "QUERY") == 0)
{
u--; v--;
w = lca(u, v, g);
if(u == v)
printf("0\n");
else
printf("%d\n", max(path_max(u, w, g), path_max(v, w, g)));
}
if(strcmp(q, "CHANGE") == 0)
{
u--;
g.e[u].w = v;
if(g.e[u].c != -1)
update(g.e[u].c,
0, 0, g.c[g.e[u].c].e.size()-1,
g.e[u].ci, v, g);
}
}
}
return 0;
}
int dfs(int u, int p, graph& g)
{
int count, m_count, t_count, v, se;
vector<int>::iterator it;
count = 1;
m_count = 0;
se = -1;
g.v[u].p = p;
g.v[u].d = p == -1 ? 1 : g.v[p].d + 1;
for(it = g.v[u].a.begin(); it != g.v[u].a.end(); it++)
{
v = u ^ g.e[*it].u ^ g.e[*it].v;
if(v == p)
continue;
t_count = dfs(v, u, g);
count += t_count;
if(t_count > m_count)
{
m_count = t_count;
se = *it;
}
}
g.v[u].se = se;
return count;
}
void hld(int u, int c, int ci, graph& g)
{
int v, se;
vector<int>::iterator it;
g.v[u].c = c;
g.v[u].ci = ci;
se = g.v[u].se;
if(se == -1)
return;
v = u ^ g.e[se].u ^ g.e[se].v;
g.e[se].c = c;
g.c[c].e.push_back(se);
g.e[se].ci = g.c[c].e.size()-1;
hld(v, c, ci+1, g);
for(it = g.v[u].a.begin(); it != g.v[u].a.end(); it++)
{
v = u ^ g.e[*it].u ^ g.e[*it].v;
if(v == g.v[u].p || *it == se)
continue;
g.c.push_back(chain(*it, v));
hld(v, g.c.size()-1, 0, g);
}
}
int log2(int n)
{
int e = 0, te = 1;
while(te < n)
{
te *= 2;
e++;
}
return e;
}
void comp_lca(graph &g)
{
int maxj = log2(g.v.size());
for(int i = 0; i < g.v.size(); i++)
{
g.v[i].lca.resize(maxj+1, -1);
g.v[i].lca[0] = g.v[i].p;
}
for(int j = 1; j <= maxj; j++)
{
for(int i = 0; i < g.v.size(); i++)
{
if(g.v[i].lca[j-1] != -1)
g.v[i].lca[j] = g.v[g.v[i].lca[j-1]].lca[j-1];
else
g.v[i].lca[j] = -1;
}
}
}
int lca(int u, int v, graph &g)
{
int diff = g.v[v].d - g.v[u].d, j;
if(diff < 0)
return lca(v, u, g);
j = 0;
while(diff > 0)
{
if(diff % 2)
v = g.v[v].lca[j];
diff /= 2;
j++;
}
if(u == v)
return u;
while(g.v[u].p != g.v[v].p)
{
j = 0;
while(g.v[u].lca[j] != g.v[v].lca[j])
j++;
u = g.v[u].lca[j-1];
v = g.v[v].lca[j-1];
}
return g.v[u].p;
}
void create_sts(graph& g)
{
for(int i = 0; i < g.c.size(); i++)
{
if(g.c[i].e.size() == 0)
continue;
g.c[i].st.resize(4*g.c[i].e.size(), 0);
build_st(i, 0, 0, g.c[i].e.size()-1, g);
}
}
int build_st(int ci, int idx, int l, int r, graph &g)
{
int ridx, lidx, mid;
lidx = (idx<<1)+1;
ridx = lidx+1;
mid = (l+r)>>1;
if(l==r)
{
g.c[ci].st[idx] = g.e[g.c[ci].e[l]].w;
return g.c[ci].st[idx];
}
g.c[ci].st[idx] = max(
build_st(ci, lidx, l, mid, g),
build_st(ci, ridx, mid+1, r, g));
}
int query(int ci, int idx, int l, int r, int a, int b, graph& g)
{
int lidx, ridx, mid, mx = INT_MIN;
lidx = (idx<<1)+1;
ridx = lidx+1;
mid = (l+r)>>1;
if(l==a && r==b)
return g.c[ci].st[idx];
if(a <= mid)
mx = max(mx, query(ci, lidx, l, mid, a, min(mid,b), g));
if(b > mid)
mx = max(mx, query(ci, ridx, mid+1, r, max(mid+1, a), b, g));
return mx;
}
void update(int ci, int idx, int l, int r, int i, int nv, graph& g)
{
int lidx, ridx, mid;
lidx = (idx<<1)+1;
ridx = lidx + 1;
mid = (l+r)>>1;
if(l==r)
{
g.c[ci].st[idx] = nv;
return;
}
if(i <= mid)
update(ci, lidx, l, mid, i, nv, g);
else
update(ci, ridx, mid+1, r, i, nv, g);
g.c[ci].st[idx] = max(g.c[ci].st[lidx], g.c[ci].st[ridx]);
}
int path_max(int u, int v, graph &g)
{
int mx = INT_MIN, i, j;
while(true)
{
if(u == v)
return mx;
i = g.v[u].c == g.v[v].c ? g.v[v].ci : 0;
j = g.v[u].ci - 1;
if(j >= 0)
mx = max(mx, query(g.v[u].c,
0, 0, g.c[g.v[u].c].e.size()-1,
i, j, g));
if(g.v[u].c == g.v[v].c)
return mx;
mx = max(mx, g.e[g.c[g.v[u].c].je].w);
u = g.v[g.c[g.v[u].c].h].p;
}
}
#include <cstdio>
#include <algorithm>
#include <vector>
#include <cstring>
#include <climits>

using namespace std;

struct edge
{
    int u, v, w, c, ci;
    edge(int u, int v, int w, int c, int ci)
        : u(u), v(v), w(w), c(c), ci(ci) {}
};

struct vertex
{
    int p, d, se, c, ci;
    vector<int> a;
    vector<int> lca;
    vertex():
        p(-1),d(0),se(-1),c(-1),ci(-1),a(),lca() {}
};

struct chain
{
    int je, h;
    vector<int> e, st;
    chain(int je, int h): je(je), h(h) {}
};

struct graph
{
    vector<edge> e;
    vector<vertex> v;
    vector<chain> c;
};

void hld(int u, int c, int ci, graph& g);
int dfs(int u, int p, graph& g);
int log2(int n);
void comp_lca(graph &g);
int lca(int u, int v, graph &g);
void create_sts(graph &g);
int build_st(int ci, int idx, int l, int r, graph &g);
void update(int ci, int idx, int l, int r, int i, int nv, graph& g);
int path_max(int u, int v, graph& g);

int main()
{
    int t, n, u, v, w;
    graph g;
    char q[10];

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

        g.e.clear();
        g.v.clear();
        g.c.clear();
        g.v.resize(n, vertex());
        
        for(int i = 0; i < n-1; i++)
        {
            scanf("%d %d %d", &u, &v, &w);
            u--; v--;
            g.e.push_back(edge(u, v, w, -1, -1));
            g.v[u].a.push_back(i);
            g.v[v].a.push_back(i);
        }

        dfs(0, -1, g);        
        g.c.push_back(chain(-1, 0));
        hld(0, 0, 0, g);
        comp_lca(g);
        create_sts(g);

        while(true)
        {
            scanf("%s", q);

            if(strcmp(q, "DONE") == 0)
                break;

            scanf("%d %d", &u, &v);

            if(strcmp(q, "QUERY") == 0)
            {
                u--; v--;
                w = lca(u, v, g);
                if(u == v)
                    printf("0\n");
                else
                    printf("%d\n", max(path_max(u, w, g), path_max(v, w, g)));
            }

            if(strcmp(q, "CHANGE") == 0)
            {
                u--;
                g.e[u].w = v;
                
                if(g.e[u].c != -1)
                    update(g.e[u].c,
                           0, 0, g.c[g.e[u].c].e.size()-1,
                           g.e[u].ci, v, g);
            }
        }        
    }

    return 0;
}

int dfs(int u, int p, graph& g)
{
    int count, m_count, t_count, v, se;
    vector<int>::iterator it;

    count = 1;
    m_count = 0;
    se = -1;
    g.v[u].p = p;
    g.v[u].d = p == -1 ? 1 : g.v[p].d + 1;

    for(it = g.v[u].a.begin(); it != g.v[u].a.end(); it++)
    {
        v = u ^ g.e[*it].u ^ g.e[*it].v;

        if(v == p)
            continue;

        t_count = dfs(v, u, g);
        count += t_count;

        if(t_count > m_count)
        {
            m_count = t_count;
            se = *it;
        }
    }

    g.v[u].se = se;
    return count;
}

void hld(int u, int c, int ci, graph& g)
{
    int v, se;
    vector<int>::iterator it;

    g.v[u].c = c;
    g.v[u].ci = ci;
    se = g.v[u].se;
    
    if(se == -1)
        return;
        
    v = u ^ g.e[se].u ^ g.e[se].v;
    g.e[se].c = c;
    g.c[c].e.push_back(se);
    g.e[se].ci = g.c[c].e.size()-1;

    
    hld(v, c, ci+1, g);

    for(it = g.v[u].a.begin(); it != g.v[u].a.end(); it++)
    {
        v = u ^ g.e[*it].u ^ g.e[*it].v;
        
        if(v == g.v[u].p || *it == se)
            continue;
        
        g.c.push_back(chain(*it, v));
        hld(v, g.c.size()-1, 0, g);
    }
}

int log2(int n)
{
    int e = 0, te = 1;
    while(te < n)
    {
        te *= 2;
        e++;
    }
    return e;
}

void comp_lca(graph &g)
{
    int maxj = log2(g.v.size());

    for(int i = 0; i < g.v.size(); i++)
    {
        g.v[i].lca.resize(maxj+1, -1);
        g.v[i].lca[0] = g.v[i].p;
    }
            
    for(int j = 1; j <= maxj; j++)
    {
        for(int i = 0; i < g.v.size(); i++)
        {            
            if(g.v[i].lca[j-1] != -1)
                g.v[i].lca[j] = g.v[g.v[i].lca[j-1]].lca[j-1];
            else
                g.v[i].lca[j] = -1;
        }
    }
}

int lca(int u, int v, graph &g)
{
    int diff = g.v[v].d - g.v[u].d, j;

    if(diff < 0)
        return lca(v, u, g);

    j = 0;
    while(diff > 0)
    {
        if(diff % 2)
            v = g.v[v].lca[j];
        diff /= 2;
        j++;
    }

    if(u == v)
        return u;
    
    while(g.v[u].p != g.v[v].p)
    {
        j = 0;
        while(g.v[u].lca[j] != g.v[v].lca[j])
            j++;
        u = g.v[u].lca[j-1];
        v = g.v[v].lca[j-1];
    }

    return g.v[u].p;
}

void create_sts(graph& g)
{
    for(int i = 0; i < g.c.size(); i++)
    {
        if(g.c[i].e.size() == 0)
            continue;
        
        g.c[i].st.resize(4*g.c[i].e.size(), 0);
        build_st(i, 0, 0, g.c[i].e.size()-1, g);
    }    
}

int build_st(int ci, int idx, int l, int r, graph &g)
{
    int ridx, lidx, mid;

    lidx = (idx<<1)+1;
    ridx = lidx+1;
    mid = (l+r)>>1;
    
    if(l==r)
    {
        g.c[ci].st[idx] = g.e[g.c[ci].e[l]].w;
        return g.c[ci].st[idx];
    }

    g.c[ci].st[idx] = max(
        build_st(ci, lidx, l, mid, g),
        build_st(ci, ridx, mid+1, r, g));
}

int query(int ci, int idx, int l, int r, int a, int b, graph& g)
{
    int lidx, ridx, mid, mx = INT_MIN;

    lidx = (idx<<1)+1;
    ridx = lidx+1;
    mid = (l+r)>>1;

    if(l==a && r==b)
        return g.c[ci].st[idx];
    if(a <= mid)
        mx = max(mx, query(ci, lidx, l, mid, a, min(mid,b), g));
    if(b > mid)
        mx = max(mx, query(ci, ridx, mid+1, r, max(mid+1, a), b, g));
    return mx;
}

void update(int ci, int idx, int l, int r, int i, int nv, graph& g)
{
    int lidx, ridx, mid;

    lidx = (idx<<1)+1;
    ridx = lidx + 1;
    mid = (l+r)>>1;
    
    if(l==r)
    {
        g.c[ci].st[idx] = nv;
        return;
    }

    if(i <= mid)
        update(ci, lidx, l, mid, i, nv, g);
    else        
        update(ci, ridx, mid+1, r, i, nv, g);

    g.c[ci].st[idx] = max(g.c[ci].st[lidx], g.c[ci].st[ridx]);    
}

int path_max(int u, int v, graph &g)
{
    int mx = INT_MIN, i, j;
    
    while(true)
    {
        if(u == v)
            return mx;

        i = g.v[u].c == g.v[v].c ? g.v[v].ci : 0;
        j = g.v[u].ci - 1;
        if(j >= 0)
            mx = max(mx, query(g.v[u].c,
                               0, 0, g.c[g.v[u].c].e.size()-1,
                               i, j, g));

        if(g.v[u].c == g.v[v].c)
            return mx;
        
        mx = max(mx, g.e[g.c[g.v[u].c].je].w);
        u = g.v[g.c[g.v[u].c].h].p;
    }
}
