// ROOT : DRAGON3012009
#include <bits/stdc++.h>
#define ll long long
#define el "\n"
#define fast ios_base::sync_with_stdio(0); cin.tie(0); cout.tie(0);
#define __ROOT__ int main()
#define fi first
#define se second
#define M 1000000007
#define MAXN 100001
#define GIOIHAN 1000001
#define BLOCK_SIZE 425
#define MAX_NODE 1001001
#define LOG 19
#define ALPHA_SIZE 26
#define BASE 311
#define NAME "check"
#define compare(v) sort((v).begin(), (v).end()); (v).erase(unique((v).begin(), (v).end()), (v).end()); // dùng để nén sort mảng compare
using namespace std;
const ll MOD[] = {(ll)1e9 + 2277, (ll)1e9 + 5277, (ll)1e9 + 8277, (ll)1e9 + 9277, (ll) 1e9 + 7 };
const ll NMOD = 1;
const int dx[] = {-1, 0, 1,0};
const int dy[] = {0, 1, 0, -1};
//**Variable**//
ll n ;
ll tour[MAXN] ;
ll par[MAXN] ;
ll fin[MAXN] ;
ll st[MAXN] ;
ll high[MAXN] ;
ll sz[MAXN] ;
ll chainHead[MAXN] ;
ll chainID[MAXN] ;
ll NodeVal[MAXN] ;
ll curchain = 1 ;
ll timedfs =0 ;
vector<ll>adj[MAXN] ;
//**Struct**//
struct E {
ll u, v, w;
};
vector<E> edge ;
struct Seg {
struct Data {
ll mi, ma ;
};
Data val[MAXN << 2 ] ;
ll lazy[MAXN<<2 ] ;
void resett() {
for(ll i = 0 ; i <(MAXN <<2 ) ; i ++ ) {
val[i].ma = INT_MIN;val[i]. mi = INT_MAX ;
}
memset(lazy, 0, sizeof lazy ) ;
}
Data Merge(Data a, Data b ) {
Data res ;
res.ma = max(a.ma, b.ma ) ;
res.mi = min(a.mi, b.mi) ;
return res ;
}
void build(ll id, ll l,ll r ) {
if(l == r ) {
val[id].mi = val[id].ma = NodeVal[tour[l]];
} else {
ll m = (l + r) >> 1 ;
build(id<<1, l, m ) ;
build(id<<1|1, m + 1, r ) ;
val[id ] =Merge(val[id<<1], val[id<<1|1]) ;
}
}
void fix(ll id, ll l, ll r ) {
if(lazy[id]) {
// ll tmp_ma = val[id].ma ;
// ll tmp_mi = val[id].mi ;
swap(val[id].ma, val[id].mi);
val[id].ma = -val[id].ma ;
val[id].mi = -val[id].mi;
if(l!=r ) {
lazy[id<<1] ^= lazy[id] ;
lazy[id<<1|1] ^=lazy[id] ;
}
lazy[id] = 0 ;
}
}
void update(ll id, ll l, ll r, ll u, ll v ) {
fix(id, l, r) ;
if(u > r || v < l ) return ;
if(u <= l && v >= r ) {
lazy[id] ^= 1 ;
fix(id, l, r ) ;
return ;
}
ll m = (l + r) >> 1 ;
update(id<< 1, l, m, u, v );
update(id<<1|1, m + 1, r, u, v ) ;
val[id] = Merge(val[id<<1], val[id<<1|1]) ;
}
ll get(ll id, ll l, ll r,ll u, ll v ) {
fix(id, l, r ) ;
if(u > r || v < l ) return INT_MIN ;
if(u <= l && v >= r ) {
return val[id].ma ;
}
ll m = (l + r) >> 1 ;
return max(get(id<<1, l, m, u, v ), get(id<< 1 |1, m + 1, r, u, v )) ;
}
void updatepos(ll id , ll l ,ll r , ll u , ll v , ll value) {
fix(id , l , r ) ;
if(u > r || v < l ) return ;
if(u <= l && v >= r ) {
val[id].ma = val[id].mi = value ;
return ;
}
ll m = l + r >> 1;
updatepos(id<<1 , l , m , u , v ,value) ;
updatepos(id<<1|1 , m + 1 , r ,u , v , value ) ;
val[id] =Merge(val[id<<1] , val[id<<1|1]) ;
}
} seg;
//**Function**//
void resetall() {
memset(tour, 0, sizeof tour) ;
memset(par, 0, sizeof par) ;
memset(st, 0, sizeof st) ;
memset(fin, 0, sizeof fin) ;
memset(high, 0, sizeof high) ;
memset(sz, 0, sizeof sz) ;
memset(chainHead, 0, sizeof chainHead) ;
memset(chainID, 0, sizeof chainID) ;
memset(NodeVal, 0, sizeof NodeVal ) ;
seg.resett();
edge.resize(0) ;
curchain = 1 ;
timedfs =0 ;
for(ll i = 0 ; i <= n ; i ++ ) adj[i].resize(0);
}
void dfs(ll u, ll p ) {
sz[u] = 1 ;
for(ll v : adj[u]) {
if(p != v ) {
high[v] = high[u] + 1 ;
par[v]= u ;
dfs(v, u) ;
sz[u] += sz[v];
}
}
}
void hld(ll u, ll p ) {
if(!chainHead[curchain]) {
chainHead[curchain ] = u ;
}
st[u] = ++ timedfs ;
tour[timedfs] = u;
chainID[u]= curchain ;
ll nxt = 0 ;
for(ll v : adj[u]) {
if(v == p ) continue ;
if(nxt == 0 || sz[v] > sz[nxt]) nxt = v;
}
if(nxt) hld(nxt, u) ;
for(ll v : adj[u]) {
if(v != p && v != nxt ) {
curchain ++ ;
hld(v, u ) ;
}
}
fin[u] = timedfs ;
}
ll LCA(ll u, ll v) {
while (chainID[u] != chainID[v]) {
if (high[chainHead[chainID[u]]] > high[chainHead[chainID[v]]])
u = par[chainHead[chainID[u]]];
else
v = par[chainHead[chainID[v]]];
}
return high[u] < high[v] ? u : v;
}
ll query(ll u, ll v ) {
if (u == v) {
return 0 ;
}
ll lca = LCA(u, v), ma = INT_MIN ;
while(chainID[u] != chainID[lca]) {
ma = max(ma, seg.get(1, 1, n,st[chainHead[chainID[u]]] , st[u] ));
u = par[chainHead[chainID[u]]];
}
while(chainID[v] != chainID[lca]) {
ma = max(ma, seg.get(1, 1, n, st[chainHead[chainID[v]]] , st[v] ));
v = par[chainHead[chainID[v]]] ;
}
if(u == v ) return ma ;
if(high[u] > high[v]) swap(u, v ) ;
return max(ma, seg.get(1, 1, n, st[u] + 1, st[v])) ;
}
void daodau(ll u, ll v ) {
ll lca = LCA(u, v) ;
while(chainID[u] != chainID[lca]) {
seg.update(1, 1, n, st[chainHead[chainID[u]]] , st[u] );
u = par[chainHead[chainID[u]]] ;
}
while(chainID[v] != chainID[lca]) {
seg.update(1, 1, n,st[chainHead[chainID[v]]] , st[v] );
v = par[chainHead[chainID[v]]] ;
}
if(u == v ) return ;
if(high[u] > high[v]) swap(u, v ) ;
seg.update(1, 1, n, st[u] + 1, st[v]) ;
}
void init() {
cin>>n ;
for(ll i = 1 ; i <= n - 1 ; i ++ ) {
ll x, y, w ;
cin>>x>>y>>w ;
edge.push_back({x, y, w }) ;
adj[x].push_back(y ) ;
adj[y].push_back(x) ;
}
dfs(1, 1 ) ;
for(ll i = 0 ; i < edge.size() ; i ++ ) {
ll u = edge[i].u;
ll v = edge[i].v ;
if(u == par[v]) swap(u, v ) ;// u : child
NodeVal[u] = edge[i].w;
}
hld(1, 1 ) ;
seg.build(1, 1, n );
}
void solve() {
// cout<<LCA(2 , 5) << " dưa" ;
//cout<<st[9] << " ew" << el ;
string str;
do {
cin>>str;
// cout<<str<<el ;
if (str == "DONE") return ;
// cout<<str<<el ;
if (str == "QUERY") {
ll x, y ;
cin >> x >> y;
cout << query(x, y) << el;
} else if (str == "NEGATE") {
ll x, y ;
cin >> x >> y;
daodau(x, y);
} else if(str == "CHANGE" ) {
ll x, y ;
cin >> x >> y;
ll u = edge[x - 1].u;
ll v = edge[x - 1].v;
edge[x-1 ].w = y ;
if (u == par[v]) swap(u, v); // u : child
seg.updatepos(1, 1, n, st[u] , st[u], y);
}
} while(str != "DONE") ;
}
__ROOT__ {
// freopen(NAME".inp" , "r" , stdin);
// freopen(NAME".out" , "w", stdout) ;
fast;
ll t;
cin >> t;
for(ll i = 1 ; i <= t ; i ++ ) {
init();
solve();
resetall() ;
}
}
// ROOT : DRAGON3012009
#include <bits/stdc++.h>
#define ll long long
#define el "\n"
#define fast ios_base::sync_with_stdio(0); cin.tie(0); cout.tie(0);
#define __ROOT__ int main()
#define fi first
#define se second
#define M 1000000007
#define MAXN 100001
#define GIOIHAN 1000001
#define BLOCK_SIZE 425
#define MAX_NODE 1001001
#define LOG 19
#define ALPHA_SIZE 26
#define BASE 311
#define NAME "check"
#define compare(v) sort((v).begin(), (v).end()); (v).erase(unique((v).begin(), (v).end()), (v).end()); // dùng để nén sort mảng compare
using namespace std;
const ll MOD[] = {(ll)1e9 + 2277, (ll)1e9 + 5277, (ll)1e9 + 8277, (ll)1e9 + 9277, (ll) 1e9 + 7 };
const ll NMOD = 1;
const int dx[] = {-1, 0, 1,0};
const int dy[] = {0, 1, 0, -1};
//**Variable**//
ll n ;
ll tour[MAXN] ;
ll par[MAXN] ;
ll fin[MAXN] ;
ll st[MAXN] ;
ll high[MAXN] ;
ll sz[MAXN] ;
ll chainHead[MAXN] ;
ll chainID[MAXN] ;
ll NodeVal[MAXN] ;
ll curchain = 1  ;
ll timedfs =0 ;
vector<ll>adj[MAXN] ;
//**Struct**//
struct E {
    ll u, v, w;

};
vector<E> edge ;
struct Seg {
    struct Data {
        ll mi, ma  ;
    };
    Data val[MAXN << 2 ] ;
    ll lazy[MAXN<<2 ] ;
    void resett() {
        for(ll i = 0  ; i <(MAXN <<2  ) ; i ++ ) {
            val[i].ma = INT_MIN;val[i]. mi = INT_MAX ;
        }
        memset(lazy, 0, sizeof lazy  ) ;
    }
    Data Merge(Data a, Data b ) {
        Data res  ;
        res.ma = max(a.ma, b.ma ) ;
        res.mi = min(a.mi, b.mi)  ;
        return res ;
    }
    void build(ll id, ll l,ll r ) {
        if(l == r ) {
            val[id].mi = val[id].ma = NodeVal[tour[l]];
        } else {
            ll m = (l + r) >> 1 ;
            build(id<<1, l, m ) ;
            build(id<<1|1, m + 1, r  ) ;
            val[id ] =Merge(val[id<<1], val[id<<1|1]) ;
        }
    }
    void fix(ll id,  ll l, ll r ) {
        if(lazy[id]) {
//            ll tmp_ma = val[id].ma ;
//            ll tmp_mi = val[id].mi ;
            swap(val[id].ma, val[id].mi);
            val[id].ma = -val[id].ma ;
            val[id].mi = -val[id].mi;
            if(l!=r ) {
                lazy[id<<1] ^= lazy[id] ;
                lazy[id<<1|1] ^=lazy[id] ;
            }
            lazy[id]  = 0 ;
        }
    }
    void update(ll id, ll l, ll r, ll u, ll v ) {
        fix(id, l, r) ;
        if(u > r  || v < l   ) return  ;
        if(u <= l && v >= r ) {
            lazy[id] ^= 1 ;
            fix(id, l, r ) ;
            return ;
        }
        ll m = (l + r) >> 1 ;
        update(id<< 1, l, m, u,  v );
        update(id<<1|1, m + 1, r, u, v  ) ;
        val[id] = Merge(val[id<<1], val[id<<1|1]) ;
    }

    ll get(ll id, ll l, ll r,ll u, ll v ) {
        fix(id, l, r  )  ;
        if(u > r || v < l ) return INT_MIN  ;
        if(u <= l && v >= r ) {
            return val[id].ma ;
        }
        ll m = (l + r) >> 1 ;
        return max(get(id<<1,  l, m, u, v ), get(id<< 1 |1, m + 1, r, u, v ))  ;
    }

    void updatepos(ll id , ll l ,ll r , ll u , ll v  , ll value) {
    fix(id , l , r  ) ;
    if(u > r || v < l ) return ;
    if(u <= l && v >= r  ) {
        val[id].ma = val[id].mi = value  ;
        return ;
    }
    ll m = l  + r >> 1;
    updatepos(id<<1 , l , m  , u , v ,value) ;
    updatepos(id<<1|1 , m + 1 , r ,u , v , value ) ;
    val[id] =Merge(val[id<<1] , val[id<<1|1]) ;
    }
} seg;
//**Function**//
void resetall() {
    memset(tour, 0, sizeof tour) ;
    memset(par, 0, sizeof par) ;
    memset(st, 0, sizeof st) ;
    memset(fin, 0, sizeof fin) ;
    memset(high, 0, sizeof high) ;
    memset(sz, 0, sizeof sz) ;
    memset(chainHead, 0, sizeof chainHead) ;
    memset(chainID, 0, sizeof chainID) ;
    memset(NodeVal, 0, sizeof NodeVal ) ;
    seg.resett();
    edge.resize(0) ;
    curchain = 1  ;
    timedfs =0 ;
    for(ll i = 0 ; i <= n ; i ++ ) adj[i].resize(0);
}
void dfs(ll u, ll p ) {
    sz[u] = 1 ;
    for(ll v : adj[u]) {
        if(p != v ) {
            high[v] = high[u] + 1 ;
            par[v]= u ;
            dfs(v, u) ;
            sz[u] += sz[v];
        }
    }
}

void hld(ll u, ll p ) {
    if(!chainHead[curchain]) {
        chainHead[curchain ] = u ;
    }
    st[u] = ++ timedfs ;
    tour[timedfs] = u;
    chainID[u]= curchain ;
    ll nxt = 0 ;
    for(ll v : adj[u]) {
        if(v == p ) continue ;
        if(nxt == 0 || sz[v] > sz[nxt]) nxt = v;
    }
    if(nxt) hld(nxt, u) ;
    for(ll v : adj[u]) {
        if(v != p && v != nxt ) {
            curchain ++ ;
            hld(v, u ) ;
        }
    }
    fin[u] = timedfs ;
}
ll LCA(ll u, ll v) {
    while (chainID[u] != chainID[v]) {

        if (high[chainHead[chainID[u]]] > high[chainHead[chainID[v]]])
            u = par[chainHead[chainID[u]]];
        else
            v = par[chainHead[chainID[v]]];
    }
    return high[u] < high[v] ? u : v;
}

ll query(ll u, ll v ) {
      if (u == v) {
               return 0 ;
            }
    ll lca = LCA(u, v), ma = INT_MIN ;
    while(chainID[u] != chainID[lca]) {
        ma = max(ma, seg.get(1, 1, n,st[chainHead[chainID[u]]]   , st[u] ));
        u = par[chainHead[chainID[u]]];
    }
    while(chainID[v] != chainID[lca]) {
        ma = max(ma, seg.get(1, 1, n, st[chainHead[chainID[v]]]  , st[v] ));
        v = par[chainHead[chainID[v]]] ;
    }
    if(u == v ) return ma ;
    if(high[u] > high[v]) swap(u, v ) ;
    return max(ma, seg.get(1, 1, n, st[u] + 1, st[v])) ;
}
void daodau(ll u, ll v ) {
    ll lca = LCA(u, v) ;
    while(chainID[u] != chainID[lca]) {
        seg.update(1, 1, n, st[chainHead[chainID[u]]]    , st[u] );
        u = par[chainHead[chainID[u]]] ;
    }
    while(chainID[v] != chainID[lca]) {
        seg.update(1, 1, n,st[chainHead[chainID[v]]] , st[v] );
        v = par[chainHead[chainID[v]]] ;
    }
    if(u == v ) return ;
    if(high[u] > high[v]) swap(u, v ) ;
    seg.update(1, 1, n, st[u] + 1, st[v]) ;
}
void init() {
    cin>>n ;
    for(ll i = 1 ; i <= n - 1 ; i ++ ) {
        ll x, y, w ;
        cin>>x>>y>>w ;
        edge.push_back({x, y, w }) ;
        adj[x].push_back(y ) ;
        adj[y].push_back(x)  ;
    }
    dfs(1, 1   ) ;
    for(ll i = 0 ; i < edge.size() ; i ++ )  {
        ll u = edge[i].u;
        ll v = edge[i].v ;
        if(u == par[v]) swap(u, v ) ;// u : child
        NodeVal[u] = edge[i].w;
    }
    hld(1, 1 ) ;
    seg.build(1, 1, n );
}

void solve() {
//    cout<<LCA(2 , 5) << " dưa" ;
//cout<<st[9] << " ew" << el ;
    string str;
    do {
        cin>>str;
//          cout<<str<<el ;
        if (str == "DONE") return ;
//            cout<<str<<el ;
        if (str == "QUERY") {
            ll x, y ;
            cin >> x >> y;
            cout << query(x, y) << el;
        } else if (str == "NEGATE") {
            ll x, y ;
            cin >> x >> y;
            daodau(x, y);
        } else if(str == "CHANGE" ) {
            ll x, y ;
            cin >> x >> y;
            ll u = edge[x - 1].u;
            ll v = edge[x - 1].v;
            edge[x-1 ].w = y ;
            if (u == par[v]) swap(u, v); // u : child
            seg.updatepos(1, 1, n, st[u] , st[u], y);
        }
    } while(str != "DONE") ;
}


__ROOT__ {
//    freopen(NAME".inp" , "r" , stdin);
//              freopen(NAME".out" , "w", stdout) ;
    fast;
    ll t;
    cin >> t;
    for(ll i = 1 ; i <= t ; i ++ ) {
        init();
        solve();
        resetall() ;
    }
}

