/**********************************************
Author : smiley007
***********************************************/
//Data Structure Includes
#include <vector>
#include <queue>
#include <deque>
#include <bitset>
#include <stack>
#include <list>
#include <set>
#include <map>
#include <unordered_set>
//Other Includes
#include <cstdio>
#include <string>
#include <cstring>
#include <cstdlib>
#include <iostream>
#include <cmath>
#include <cctype>
#include <cassert>
#include <numeric>
#include <algorithm>
#include <ctime>
#include <sstream>
#include <climits>
#include <iomanip>
using namespace std ;
typedef long long ll;
typedef pair<int,int> pii;
typedef pair<ll,ll> pll;
typedef vector<int> vi;
typedef vector<ll> vll;
typedef vector<vll> vvll;
typedef vector<vvll> vvvll;
typedef vector<pii> vpii;
typedef vector<vpii> vvpii;
typedef vector<vpii> vvpii;
typedef vector<pll> vpll;
typedef vector<vpll> vvpll;
typedef vector<vvpll> vvvpll;
typedef vector<vi> vvi;
typedef vector<string> vs;
typedef vector<vs> vvs;
#ifdef LocalHost
#define eprintf(...) fprintf(stderr,__VA_ARGS__)
#endif
//Code Begins Here ------ >>>
class HLD {
private:
const static int N = 1e4 + 44;
const static int LN = 14;
int parent[N], depth[N], subsize[N];
int chainNum[N], valuePos[N], chainPos[N], chainHead[N] = {-1};
int tree[2 * N];
vpii g[N];
int a[N], b[N];
int p[N][LN]; //[#node][height from node]
int currChain = 0; // # of available chains
int n;
int ptr = 0;
public:
void reset() {
ptr = 0;
currChain = 0;
for (int i = 1; i <= n; i++) {
chainHead[i] = -1;
g[i].clear();
for (int j = 0; j < LN; j++) p[i][j] = -1;
}
}
void read() {
scanf("%d", &n);
reset();
for (int i = 1; i <= n - 1; i++) {
int x, y, z;
scanf("%d %d %d", &x, &y, &z);
a[i] = x; b[i] = y;
g[x].push_back(make_pair(y, z));
g[y].push_back(make_pair(x, z));
}
}
void solve() {
dfs(1, 0, -1);
for (int i = 1; i <= n; i++) {
p[i][0] = parent[i];
}
for (int i = 1; i < LN; i++) {
for (int j = 1; j <= n; j++) if (p[j][i - 1] != -1){
p[j][i] = p[p[j][i - 1]][i - 1];
}
}
make_hld(1, -1, 0);
build(1, ptr, 1);
}
/**
(u -> current node, l -> current level, p -> parent node)
**/
void dfs(int u, int l, int p) {
parent[u] = p;
depth[u] = l;
int res = 1;
for (auto it : g[u]) {
int v = it.first;
if (v != p) {
dfs(v, l + 1, u);
res += subsize[v];
}
}
subsize[u] = res;
return;
}
int getLCA(int a, int b) {
if (depth[a] < depth[b]) swap(a, b);
int diff = depth[a] - depth[b];
for (int i = LN - 1; i >= 0; i--) if ((diff >> i) & 1) {
a = p[a][i];
}
if (a == b) return a;
for (int i = LN - 1; i >= 0; i--) {
if (p[a][i] != p[b][i]) {
a = p[a][i];
b = p[b][i];
}
}
return p[a][0];
}
void make_hld(int curr, int p, int cst) {
if (chainHead[currChain] == -1) {
chainHead[currChain] = curr;
}
chainNum[curr] = currChain;
chainPos[curr] = ++ptr;
valuePos[ptr] = cst;
int idx = -1, mx = -1;
for (int i = 0; i < (int)g[curr].size(); i++) {
int v = g[curr][i].first;
if (v != p) {
if (mx < subsize[v]) {
mx = subsize[v];
idx = i;
}
}
}
if (idx != -1) make_hld(g[curr][idx].first, curr, g[curr][idx].second);
for (int i = 0; i < (int)g[curr].size(); i++) {
int v = g[curr][i].first;
if (v != p && i != idx) {
currChain++;
make_hld(g[curr][i].first, curr, g[curr][i].second);
}
}
}
void build(int l, int r, int idx) {
if (l == r) {
tree[idx] = valuePos[l];
return;
}
int mid = (l + r) >> 1;
build(l, mid, idx + idx);
build(mid + 1, r, idx + idx + 1);
tree[idx] = max(tree[idx + idx], tree[idx + idx + 1]);
}
void update_tree(int l, int r, int idx, int pos, int val) {
if (l == r) {
tree[idx] = val;
return;
}
int mid = (l + r) >> 1;
if (pos <= mid) update_tree(l, mid, idx + idx, pos, val);
else update_tree(mid + 1, r, idx + idx + 1, pos, val);
tree[idx] = max(tree[idx + idx], tree[idx + idx + 1]);
}
void update(int numEdge, int cst) {
int u = a[numEdge];
int v = b[numEdge];
int pu = chainPos[u];
int pv = chainPos[v];
update_tree(1, ptr, 1, max(pu, pv), cst);
}
int query_tree(int l, int r, int idx, int a, int b) {
if (l > b || r < a) return 0;
if (a <= l && r <= b) return tree[idx];
int mid = (l + r) >> 1;
int ans = query_tree(l, mid, idx + idx, a, b);
ans = max(ans, query_tree(mid + 1, r, idx + idx + 1, a, b));
return ans;
}
int query(int u, int v) {
int ans = 0;
while (u != v) {
int cu = chainNum[u];
int cv = chainNum[v];
int pu = chainPos[u];
int pv = chainPos[v];
int h = chainHead[cu];
int ph = chainPos[h];
if (cu == cv) {
ans = max(ans, query_tree(1, ptr, 1, pv + 1, pu));
break;
} else {
ans = max(ans, query_tree(1, ptr, 1, ph, pu));
u = parent[h];
}
}
return ans;
}
// void out() {
// cout << "depth : \n";
// for (int i = 1; i <= n; i++)
// cout << i << " : " << depth[i] << "\n";
// cout << "\n\nparent : \n";
// for (int i = 1; i <= n; i++)
// cout << i << " : " << parent[i] << "\n";
// cout << "\n\nsubtree size : \n";
// for (int i = 1; i <= n; i++)
// cout << i << " : " << subsize[i] << "\n";
// cout << "\n";
// cout << "currChain : " << currChain << "\n";
// for (int i = 0; i <= currChain; i++) {
// for (int j = 0; j < chainSize[i]; j++) {
// cout << chains[i][j] << " ";
// }
// cout << "\n\n";
// }
// }
};
int main(){
#ifdef LocalHost
freopen("input.txt","r",stdin);
#endif
int t;
cin >> t;
HLD obj = HLD();
while (t--) {
obj.read();
obj.solve();
while (true) {
string s;
cin >> s;
if (s[0] == 'D') break;
int a, b;
cin >> a >> b;
if (s[0] == 'Q') {
int lca = obj.getLCA(a, b);
cout << max(obj.query(a, lca), obj.query(b, lca)) << "\n";
} else {
obj.update(a, b);
}
}
}
#ifdef LocalHost
cout<<endl<<"Execution time = "<<(float)clock()/CLOCKS_PER_SEC <<" s"<<endl;
#endif
return 0;
}
/**********************************************
            Author : smiley007  
***********************************************/

//Data Structure Includes
#include <vector>
#include <queue>
#include <deque>
#include <bitset>
#include <stack>
#include <list>
#include <set>          
#include <map>
#include <unordered_set>

//Other Includes
#include <cstdio>
#include <string>
#include <cstring>
#include <cstdlib>
#include <iostream>
#include <cmath>
#include <cctype>
#include <cassert>
#include <numeric>
#include <algorithm>
#include <ctime>
#include <sstream>
#include <climits>
#include <iomanip>

using namespace std ;
typedef long long ll;
typedef pair<int,int> pii;
typedef pair<ll,ll> pll;
typedef vector<int> vi;
typedef vector<ll> vll;
typedef vector<vll> vvll;
typedef vector<vvll> vvvll;
typedef vector<pii> vpii;
typedef vector<vpii> vvpii;
typedef vector<vpii> vvpii;
typedef vector<pll> vpll;
typedef vector<vpll> vvpll;
typedef vector<vvpll> vvvpll;
typedef vector<vi> vvi;
typedef vector<string> vs;
typedef vector<vs> vvs;

#ifdef LocalHost
    #define eprintf(...) fprintf(stderr,__VA_ARGS__)
#endif

//Code Begins Here ------ >>>


class HLD {

private:
    const static int N = 1e4 + 44;
    const static int LN = 14;
    int parent[N], depth[N], subsize[N];
    int chainNum[N], valuePos[N], chainPos[N], chainHead[N] = {-1};
    int tree[2 * N];
    vpii g[N];
    int a[N], b[N];
    int p[N][LN]; //[#node][height from node]
    int currChain = 0; // # of available chains
    int n;
    int ptr = 0;

public:

    void reset() {
        ptr = 0;
        currChain = 0;

        for (int i = 1; i <= n; i++) {
            chainHead[i] = -1;
            g[i].clear();
            for (int j = 0; j < LN; j++)    p[i][j] = -1;
        }

    }

     void read() {
        scanf("%d", &n);
        reset();
        for (int i = 1; i <= n - 1; i++) {
            int x, y, z;
            scanf("%d %d %d", &x, &y, &z);
            a[i] = x;   b[i] = y;
            g[x].push_back(make_pair(y, z));
            g[y].push_back(make_pair(x, z));
        }

    }

    void solve() {
        
        dfs(1, 0, -1); 
        
        for (int i = 1; i <= n; i++) {
            p[i][0] = parent[i];
        }
        for (int i = 1; i < LN; i++) {
            for (int j = 1; j <= n; j++) if (p[j][i - 1] != -1){
                p[j][i] = p[p[j][i - 1]][i - 1];
            }
        }
        make_hld(1, -1, 0);
        build(1, ptr, 1);
        
    }

   

    /**
    (u -> current node, l -> current level, p -> parent node)
    **/
    void dfs(int u, int l, int p) {
        parent[u] = p;
        depth[u] = l;
        int res = 1;
        for (auto it : g[u]) {
            int v = it.first;
            if (v != p) {
                dfs(v, l + 1, u);
                res += subsize[v];
            }
        }
        subsize[u] = res;
        return;
    }

    int getLCA(int a, int b) {
        if (depth[a] < depth[b])    swap(a, b);
        int diff = depth[a] - depth[b];
        for (int i = LN - 1; i >= 0; i--)   if ((diff >> i) & 1) {
            a = p[a][i];
        }
        if (a == b) return a;
        for (int i = LN - 1; i >= 0; i--) {
            if (p[a][i] != p[b][i]) {
                a = p[a][i];
                b = p[b][i];
            }
        }
        return p[a][0];
    }


    void make_hld(int curr, int p, int cst) {
        if (chainHead[currChain] == -1) {
            chainHead[currChain] = curr;
        }  
        chainNum[curr] = currChain;
        chainPos[curr] = ++ptr;
        valuePos[ptr] = cst;
        int idx = -1, mx = -1;
        for (int i = 0; i < (int)g[curr].size(); i++) {
            int v = g[curr][i].first;
            if (v != p) {
                if (mx < subsize[v]) {
                    mx = subsize[v];
                    idx = i;
                }
            }
        }
        if (idx != -1)  make_hld(g[curr][idx].first, curr, g[curr][idx].second);
        for (int i = 0; i < (int)g[curr].size(); i++) {
            int v = g[curr][i].first;
            if (v != p && i != idx) {
                currChain++;
                make_hld(g[curr][i].first, curr, g[curr][i].second);
            }
        }
    }

    void build(int l, int r, int idx) {
        if (l == r) {
            tree[idx] = valuePos[l];
            return;
        } 
        int mid = (l + r) >> 1;
        build(l, mid, idx + idx);
        build(mid + 1, r, idx + idx + 1);
        tree[idx] = max(tree[idx + idx], tree[idx + idx + 1]);
    }

    void update_tree(int l, int r, int idx, int pos, int val) {
        if (l == r) {
            tree[idx] = val;
            return;
        }
        int mid = (l + r) >> 1;
        if (pos <= mid) update_tree(l, mid, idx + idx, pos, val);
        else    update_tree(mid + 1, r, idx + idx + 1, pos, val);
        tree[idx] = max(tree[idx + idx], tree[idx + idx + 1]);
    }

    void update(int numEdge, int cst) {
        int u = a[numEdge];
        int v = b[numEdge];

        int pu = chainPos[u];
        int pv = chainPos[v];

        update_tree(1, ptr, 1, max(pu, pv), cst);
        
    }

    int query_tree(int l, int r, int idx, int a, int b) {
        if (l > b || r < a) return 0;
        if (a <= l && r <= b)   return tree[idx];
        int mid = (l + r) >> 1;
        int ans = query_tree(l, mid, idx + idx, a, b);
        ans = max(ans, query_tree(mid + 1, r, idx + idx + 1, a, b));
        return ans;
    }

    int query(int u, int v) {
        int ans = 0;
        while (u != v) {
            int cu = chainNum[u];
            int cv = chainNum[v];
            int pu = chainPos[u];
            int pv = chainPos[v];
            int h = chainHead[cu];
            int ph = chainPos[h];
            if (cu == cv) {   
                ans = max(ans, query_tree(1, ptr, 1, pv + 1, pu));
                break;
            } else {
                ans = max(ans, query_tree(1, ptr, 1, ph, pu));
                u = parent[h];
            }
        }
        return ans;
    }   


    // void out() {
    //     cout << "depth : \n";
    //     for (int i = 1; i <= n; i++)
    //         cout << i << " :  " << depth[i] << "\n";
    //     cout << "\n\nparent : \n";
    //     for (int i = 1; i <= n; i++)
    //         cout << i << " :  " << parent[i] << "\n";
    //     cout << "\n\nsubtree size : \n";
    //     for (int i = 1; i <= n; i++)
    //         cout << i << " :  " << subsize[i] << "\n";
    //     cout << "\n";
    //     cout << "currChain : " << currChain << "\n";
    //     for (int i = 0; i <= currChain; i++) {
    //         for (int j = 0; j < chainSize[i]; j++) {
    //             cout << chains[i][j] << " ";
    //         }
    //         cout << "\n\n";
    //     }
    // }


};



int main(){
    #ifdef LocalHost
        freopen("input.txt","r",stdin);
    #endif

    int t;
    cin >> t;
    HLD obj = HLD();
    while (t--) {
        obj.read();
        obj.solve();
        while (true) {
            string s;
            cin >> s;
            if (s[0] == 'D')    break;
            int a, b;
            cin >> a >> b;
            if (s[0] == 'Q') {
                int lca = obj.getLCA(a, b);
                cout << max(obj.query(a, lca), obj.query(b, lca)) << "\n";
            } else {
                obj.update(a, b);
            }
        }
    }











    #ifdef LocalHost
        cout<<endl<<"Execution time = "<<(float)clock()/CLOCKS_PER_SEC <<" s"<<endl;
    #endif

    

    return 0;
}