#include <bits/stdc++.h>
using namespace std;
#define pb push_back
#define mp make_pair
#define ins insert
#define fi first
#define se second
#define INF 1000000000
#define NINF -1000000000
#define LOGN 22
typedef int ll;
const int N = int(1e5)+10;
vector<pair<int,int>> *graph;
bool isBlack[N],isSpecial[N];
int subset_sz[N],chain_group[N];
int data[N],chain_pos[N];
typedef struct {
int a;int b;int wt;
} edge;
edge *Edge;
typedef struct {
vector<int> a;int head;int parent=-1;
int len=0;} chain;
chain *Chain;
map<pair<int,int>,int> e2i;
////////////LEAST COMMON ANCESTOR O(LOG(N)) PER QUERY/////////////////////
int level[N];
int DP[LOGN][N];
void dfs0(int u){
subset_sz[u]=1;
int sidx=-1,swt=-1,sit=-1;
for(int it=0;it<graph[u].size();++it)
{int v=graph[u][it].fi;
if(v!=DP[0][u])
{
DP[0][v]=u;
level[v]=level[u]+1;
dfs0(v);
if(subset_sz[v]>swt){sidx=v;swt=subset_sz[v];sit=it;}
subset_sz[u]+=subset_sz[v];
}
}
if(sidx>-1){
isSpecial[sidx]=true;
data[u]=graph[u][sit].se;}
}
void preprocess(int n){
level[0]=0;
DP[0][0]=0;
dfs0(0);
for(int i=1;i<LOGN;i++)
for(int j=0;j<n;j++)
DP[i][j] = DP[i-1][DP[i-1][j]];
}
int lca(int a,int b){
if(level[a]>level[b])swap(a,b);
int d = level[b]-level[a];
for(int i=0;i<LOGN;i++)
if(d&(1<<i))
b=DP[i][b];
if(a==b)return a;
for(int i=LOGN-1;i>=0;i--)
if(DP[i][a]!=DP[i][b])
a=DP[i][a],b=DP[i][b];
return DP[0][a];
}
//////////////////////////////////////////////////////////////////////////
/////////////////HEAVY-LIGHT DECOMPOSITION////////////////////////////////
int max_chain_no=-1;
void hld(int u,int chain_no,int from){
Chain[chain_no].a.pb(u); ++Chain[chain_no].len;
if(Chain[chain_no].len==1) {
Chain[chain_no].parent=from;
Chain[chain_no].head=u; }
chain_group[u]=chain_no; chain_pos[u]=Chain[chain_no].len-1;
int sz=graph[u].size();
for(int i=0;i<graph[u].size();i++){
int v=graph[u][i].fi;
if(v!=from) {
if(isSpecial[v]){hld(v,chain_no,u);}
else {hld(v,++max_chain_no,u);}
}
}
}
////////////////////CHARACTERIZING OUR DECOMPOSED CHAINS//////////////////////
vector<vector<int>> seg_tree;
int join(int a,int b){return a>b?a:b;}
void build(){
for(int it=0;it<=max_chain_no;it++){
int n= Chain[it].len;
vector<int> tree(2*n);
for (int i=0; i<n; i++)
tree[n+i] = data[Chain[it].a[i]];
for (int i = n - 1; i > 0; --i)
tree[i] = join(tree[i<<1] ,tree[i<<1 | 1]);
seg_tree.pb(tree);
}
}
void updateTree(int idx, int val,int chain_no){
int n= Chain[chain_no].len;
seg_tree[chain_no][idx+n] = val; idx= idx+n;
for (int i=idx; i > 1; i >>= 1)
seg_tree[chain_no][i>>1] = join(seg_tree[chain_no][i],seg_tree[chain_no][i^1]);
}
int queryTree(int l, int r,int chain_no) {
int res = NINF;r++;
int n= Chain[chain_no].len;
for (l += n, r += n; l < r; l >>= 1, r >>= 1){
if (l&1) res = join(res,seg_tree[chain_no][l++]);
if (r&1) res = join(res,seg_tree[chain_no][--r]);
}
return res;
}
void update(int i,int wt){
int a= Edge[i].a, b= Edge[i].b;
if(DP[0][a]==b){swap(a,b);}
if(chain_group[a]==chain_group[b]){updateTree(a,wt,chain_group[a]);}
Edge[i].wt=wt;
}
int query(int a,int b){
int answer=NINF,answer2=NINF;
int c= lca(a,b);
int x=a,j,k;
//printf("[[%d]]",c);
while(x!=c){
if(chain_group[x]==chain_group[c]){
answer= max(answer,queryTree(chain_pos[c],chain_pos[DP[0][x]],chain_group[x])); break; }
else{
if(x!=Chain[chain_group[x]].head)answer= max(answer,queryTree(chain_pos[Chain[chain_group[x]].head],chain_pos[DP[0][x]],chain_group[x]));
}
k=Chain[chain_group[x]].head; x= Chain[chain_group[x]].parent; j=x;
if(j>k){swap(j,k);}
// printf("---%d,%d->%d---",j,k,Edge[e2i[mp(j,k)]].wt);
answer = max(answer,Edge[e2i[mp(j,k)]].wt);
}
x=b;
while(x!=c){
if(chain_group[x]==chain_group[c]){
answer2= max(answer2,queryTree(chain_pos[c],chain_pos[DP[0][x]],chain_group[x])); break; }
else{
if(x!=Chain[chain_group[x]].head)answer2= max(answer2,queryTree(chain_pos[Chain[chain_group[x]].head],chain_pos[DP[0][x]],chain_group[x]));
}
k=Chain[chain_group[x]].head; x= Chain[chain_group[x]].parent; j=x;
if(j>k){swap(j,k);}
//printf("+++%d,%d->%d+++",j,k,Edge[e2i[mp(j,k)]].wt);
answer2 = max(answer2,Edge[e2i[mp(j,k)]].wt);
}
//printf("[%d %d][%d %d]",a,answer,b,answer2);
return max(answer,answer2);
}
//////////////////////////////////////////////////////////////////////////////
int main() {
int t;cin>>t;
string pt;
while(getline(cin,pt) and t--){
int n;cin>>n; e2i.clear();
graph= new vector<pair<int,int>> [n];
Edge = new edge[n];
for(int i=0;i<n-1;i++)
{int u,v,w;cin>>u>>v>>w;u--;v--;
graph[u].pb(mp(v,w));graph[v].pb(mp(u,w));
Edge[i].a=u;Edge[i].b=v; Edge[i].wt= w;
if(u>v)swap(u,v); e2i[mp(u,v)]=i;}
for(int i=0;i<n;i++){isBlack[i]=false;isSpecial[i]=false;data[i]=NINF;}
max_chain_no=-1;
preprocess(n);
Chain = new chain[N];
hld(0,++max_chain_no,-1);
seg_tree.clear();
build();
string type;
while(true)
{
cin>>type;if(type=="DONE"){break;}
int q1,q2;cin>>q1>>q2;//scanf("%d%d",&q1,&q2);
if(type=="CHANGE"){update(q1-1,q2);}
else if(type=="QUERY"){printf("%d\n",query(q1-1,q2-1));}
else {break;}
}
/* printf("[[[%d]]]",max_chain_no);
puts("");
for(int i=0;i<=max_chain_no;i++)
{
printf("[%d]",Chain[i].len);for(int j=0;j<Chain[i].a.size();j++){
printf("{%d %d %d}",Chain[i].a[j]+1,data[Chain[i].a[j]],chain_group[Chain[i].a[j]]);
}puts("");
// printf("%d ",Chain[i].parent);
}
//printf("--[%d]--",queryTree(1,3,0));
*/
}
return 0;
}
/*
1
3
1 2 1
2 3 2
QUERY 1 2
CHANGE 1 3
QUERY 1 2
DONE
*/
/*
1
23
1 2 3
1 3 16
1 4 5
1 5 20
2 6 2
2 8 4
2 9 5
2 10 11
3 11 18
4 12 23
19 23 15
6 7 1
9 13 6
9 15 8
10 18 12
10 19 14
13 14 7
15 16 10
15 17 9
18 20 13
19 21 16
21 22 17
QUERY 22 3
QUERY 3 22
QUERY 22 11
QUERY 11 22
QUERY 22 4
QUERY 4 22
QUERY 22 12
QUERY 12 22
QUERY 14 23
QUERY 23 14
DONE
*/
#include <bits/stdc++.h>
using namespace std;
#define pb push_back
#define mp make_pair 
#define ins insert
#define fi first
#define se second
#define INF   1000000000
#define NINF -1000000000
#define LOGN 22
typedef int ll;
const int N = int(1e5)+10;
vector<pair<int,int>> *graph;

bool isBlack[N],isSpecial[N];
int subset_sz[N],chain_group[N];
int data[N],chain_pos[N];

typedef struct {
 int a;int b;int wt;
} edge;

edge *Edge;

typedef struct {
 vector<int> a;int head;int parent=-1;
  int len=0;} chain;

chain *Chain;

map<pair<int,int>,int> e2i;
////////////LEAST COMMON ANCESTOR O(LOG(N)) PER QUERY/////////////////////
int level[N];
int DP[LOGN][N];
void dfs0(int u){
    subset_sz[u]=1; 
    int sidx=-1,swt=-1,sit=-1;
	for(int it=0;it<graph[u].size();++it)
    {int v=graph[u][it].fi;
		if(v!=DP[0][u])
		{
			DP[0][v]=u;
			level[v]=level[u]+1;
			dfs0(v);
            if(subset_sz[v]>swt){sidx=v;swt=subset_sz[v];sit=it;}
            subset_sz[u]+=subset_sz[v];
		}
    }
    if(sidx>-1){
      isSpecial[sidx]=true;
        data[u]=graph[u][sit].se;}
}
void preprocess(int n){
    level[0]=0;
	DP[0][0]=0;
	dfs0(0);
	for(int i=1;i<LOGN;i++)
		for(int j=0;j<n;j++)
			DP[i][j] = DP[i-1][DP[i-1][j]];
}
int lca(int a,int b){
	if(level[a]>level[b])swap(a,b);
	int d = level[b]-level[a];
	for(int i=0;i<LOGN;i++)
		if(d&(1<<i))
			b=DP[i][b];
	if(a==b)return a;
	for(int i=LOGN-1;i>=0;i--)
		if(DP[i][a]!=DP[i][b])
			a=DP[i][a],b=DP[i][b];
	return DP[0][a];
}
//////////////////////////////////////////////////////////////////////////

/////////////////HEAVY-LIGHT DECOMPOSITION////////////////////////////////
int max_chain_no=-1;
void hld(int u,int chain_no,int from){   
    Chain[chain_no].a.pb(u); ++Chain[chain_no].len;
    if(Chain[chain_no].len==1) {
     Chain[chain_no].parent=from;
     Chain[chain_no].head=u;   }
     chain_group[u]=chain_no; chain_pos[u]=Chain[chain_no].len-1;
    int sz=graph[u].size();
    for(int i=0;i<graph[u].size();i++){
        int v=graph[u][i].fi;
        if(v!=from) {
          if(isSpecial[v]){hld(v,chain_no,u);}
            else {hld(v,++max_chain_no,u);}
           }
    }
}
////////////////////CHARACTERIZING OUR DECOMPOSED CHAINS//////////////////////

vector<vector<int>> seg_tree;

int join(int a,int b){return a>b?a:b;}
void build(){
    for(int it=0;it<=max_chain_no;it++){
     int n= Chain[it].len;   
     vector<int> tree(2*n);
        
     for (int i=0; i<n; i++)    
        tree[n+i] = data[Chain[it].a[i]];
     for (int i = n - 1; i > 0; --i)     
        tree[i] = join(tree[i<<1] ,tree[i<<1 | 1]);
        
     seg_tree.pb(tree);
    }
}
void updateTree(int idx, int val,int chain_no){
    int n= Chain[chain_no].len;
    seg_tree[chain_no][idx+n] = val;  idx= idx+n;
     
    for (int i=idx; i > 1; i >>= 1)
    seg_tree[chain_no][i>>1] = join(seg_tree[chain_no][i],seg_tree[chain_no][i^1]);
}
int queryTree(int l, int r,int chain_no) { 
    int res = NINF;r++;
    int n= Chain[chain_no].len;
    for (l += n, r += n; l < r; l >>= 1, r >>= 1){
        if (l&1) res = join(res,seg_tree[chain_no][l++]);
        if (r&1) res = join(res,seg_tree[chain_no][--r]);
    }
    return res; 
}

void update(int i,int wt){
 int a= Edge[i].a, b= Edge[i].b;
 if(DP[0][a]==b){swap(a,b);}
 if(chain_group[a]==chain_group[b]){updateTree(a,wt,chain_group[a]);}
 Edge[i].wt=wt;
}
int query(int a,int b){
    
 int answer=NINF,answer2=NINF;
 int c= lca(a,b);
 int x=a,j,k;
 //printf("[[%d]]",c);
 while(x!=c){
     
  if(chain_group[x]==chain_group[c]){
     answer= max(answer,queryTree(chain_pos[c],chain_pos[DP[0][x]],chain_group[x]));  break; }   
  else{
     if(x!=Chain[chain_group[x]].head)answer= max(answer,queryTree(chain_pos[Chain[chain_group[x]].head],chain_pos[DP[0][x]],chain_group[x]));   
  }
   k=Chain[chain_group[x]].head; x= Chain[chain_group[x]].parent; j=x;
  if(j>k){swap(j,k);}
 // printf("---%d,%d->%d---",j,k,Edge[e2i[mp(j,k)]].wt);
  answer = max(answer,Edge[e2i[mp(j,k)]].wt);
 }
    
 x=b;
 while(x!=c){
     
  if(chain_group[x]==chain_group[c]){
     answer2= max(answer2,queryTree(chain_pos[c],chain_pos[DP[0][x]],chain_group[x]));  break; }   
  else{
     if(x!=Chain[chain_group[x]].head)answer2= max(answer2,queryTree(chain_pos[Chain[chain_group[x]].head],chain_pos[DP[0][x]],chain_group[x]));   
  }
  k=Chain[chain_group[x]].head; x= Chain[chain_group[x]].parent; j=x;
  if(j>k){swap(j,k);}
  //printf("+++%d,%d->%d+++",j,k,Edge[e2i[mp(j,k)]].wt);
 answer2 = max(answer2,Edge[e2i[mp(j,k)]].wt);
 }
 //printf("[%d %d][%d %d]",a,answer,b,answer2);
 return max(answer,answer2);
}

//////////////////////////////////////////////////////////////////////////////

int main() {
    int t;cin>>t;
    string pt;
    while(getline(cin,pt) and t--){
        
    int n;cin>>n; e2i.clear();
    graph= new vector<pair<int,int>> [n];
    Edge = new edge[n];
        
    for(int i=0;i<n-1;i++)
    {int u,v,w;cin>>u>>v>>w;u--;v--;
     graph[u].pb(mp(v,w));graph[v].pb(mp(u,w)); 
     Edge[i].a=u;Edge[i].b=v; Edge[i].wt= w;    
     if(u>v)swap(u,v); e2i[mp(u,v)]=i;} 
        
    for(int i=0;i<n;i++){isBlack[i]=false;isSpecial[i]=false;data[i]=NINF;}
    max_chain_no=-1;    
    preprocess(n); 
    Chain = new chain[N];
    hld(0,++max_chain_no,-1);
    seg_tree.clear();
    build();
        
    string type;
    
    while(true)
    {
        cin>>type;if(type=="DONE"){break;}
        int q1,q2;cin>>q1>>q2;//scanf("%d%d",&q1,&q2);
        if(type=="CHANGE"){update(q1-1,q2);}
        else if(type=="QUERY"){printf("%d\n",query(q1-1,q2-1));}
        else {break;}
    }
        
    /*    printf("[[[%d]]]",max_chain_no);
        puts("");
     for(int i=0;i<=max_chain_no;i++)
     {
      printf("[%d]",Chain[i].len);for(int j=0;j<Chain[i].a.size();j++){
          printf("{%d %d %d}",Chain[i].a[j]+1,data[Chain[i].a[j]],chain_group[Chain[i].a[j]]);
      }puts("");   
        // printf("%d ",Chain[i].parent);
     } 
      //printf("--[%d]--",queryTree(1,3,0));  
      */
    }
    return 0;
}
/*
1
3
1 2 1
2 3 2
QUERY 1 2
CHANGE 1 3
QUERY 1 2
DONE

*/
/*
1
23
1 2 3
1 3 16 
1 4 5 
1 5 20
2 6 2
2 8 4
2 9 5 
2 10 11
3 11 18
4 12 23
19 23 15
6 7 1 
9 13 6
9 15 8
10 18 12
10 19 14
13 14 7 
15 16 10
15 17 9
18 20 13
19 21 16
21 22 17
QUERY 22 3
QUERY 3 22 
QUERY 22 11
QUERY 11 22
QUERY 22 4
QUERY 4 22
QUERY 22 12
QUERY 12 22
QUERY 14 23
QUERY 23 14
DONE
*/
