/**
* Problem : QTREE - Query on a tree
* Problem URL : https://w...content-available-to-author-only...j.com/problems/QTREE/
*/
#include <bits/stdc++.h>
using namespace std;
using ll = long long;
using pii = pair < int, int >;
// This section is for Policy Based Data Structure, more precisely Ordered Set.
// All Functions of set works here, in addition we have order_of_key() and find_by_order() function. find_by_order() returns iterator
#include<ext/pb_ds/assoc_container.hpp>
#include<ext/pb_ds/tree_policy.hpp>
#include<functional>
using namespace __gnu_pbds;
// 1. Custom Set
template < class T, class COMP >
using custom_set = tree < T, null_type, COMP, rb_tree_tag, tree_order_statistics_node_update >;
// 2. Ordered Set
template < class T >
using ordered_set = tree < T, null_type, less < T >, rb_tree_tag, tree_order_statistics_node_update >;
// 3. Ordered Multi Set
// For storing multiple occurance of same value we need to use pair. In pair first value is our key and second is useless count variable.
// order_of_key(make_pair(key, 0)) returns first occurance of a value, order_of_key(make_pair(key, INT_MAX)) returns last occurance.
template < class T >
using ordered_multiset = tree < pair < T, int >, null_type, less < pair < T, int > >, rb_tree_tag, tree_order_statistics_node_update >;
// 4. Ordered Map
template < class T, class U >
using ordered_map = tree < T, U, less < T >, rb_tree_tag, tree_order_statistics_node_update >;
// Policy End
const int mx = 10005;
const int BITS = 14;
vector < pii > adj[mx];
pii edges[mx];
int SA[mx], ST[6 * mx];
int nodes[mx][5];
int edgeCount, currentChain, n, chainHeads[mx];
int tin[mx], tout[mx], up[mx][BITS], timer;
void addEdge (int id, int u, int v, int w) {
adj[u].emplace_back (v, id);
adj[v].emplace_back (u, id);
edges[id].first = w;
}
void dfs (int u, int p, int d) {
nodes[u][0] = p, nodes[u][1] = d, nodes[u][2] = 1;
tin[u] = timer++;
if (p == -1)
up[u][0] = u;
else
up[u][0] = p;
for (int i = 1; i < BITS; i++) {
up[u][i] = up[up[u][i - 1]][i - 1];
}
for (pii & pi : adj[u]) {
if (pi.first != p) {
dfs (pi.first, u, d + 1);
nodes[u][2]++;
edges[pi.second].second = pi.first;
}
}
tout[u] = timer;
}
void HLD (int u, int id) {
if (chainHeads[currentChain] == -1) {
chainHeads[currentChain] = u;
}
nodes[u][4] = currentChain;
nodes[u][3] = edgeCount;
SA[edgeCount++] = edges[id].first;
int sc = -1, si;
for (pii & pi : adj[u]) {
if (pi.first != nodes[u][0]) {
if (sc == -1 || nodes[sc][2] < nodes[pi.first][2]) {
sc = pi.first; si = pi.second;
}
}
}
if (sc != -1) {
HLD (sc, si);
}
for (pii & pi : adj[u]) {
if (pi.first != nodes[u][0] && pi.first != sc) {
currentChain++;
HLD (pi.first, pi.second);
}
}
}
bool is_ancestor (int u, int v) {
return tin[u] <= tin[v] && tout[u] >= tout[v];
}
int LCA (int u, int v) {
if (is_ancestor (u, v)) return u;
if (is_ancestor (v, u)) return v;
for (int i = BITS - 1; i >= 0; i--) {
if (!is_ancestor (up[u][i], v))
u = up[u][i];
}
return up[u][0];
}
void build (int low, int high, int pos) {
if (low == high) {
ST[pos] = SA[low];
return;
}
int mid = low + high;
mid >>= 1;
int pos1 = (pos << 1) + 1;
int pos2 = pos1 + 1;
build (low, mid, pos1);
build (mid + 1, high, pos2);
ST[pos] = ST[pos1] > ST[pos2] ? ST[pos1] : ST[pos2];
}
void updateUtil (int low, int high, int index, int value, int pos) {
if (low > index || high < index) return;
if (low == high) {
ST[pos] = value;
return;
}
int mid = low + high;
mid >>= 1;
int pos1 = (pos << 1) + 1;
int pos2 = pos1 + 1;
updateUtil (low, mid, index, value, pos1);
updateUtil (mid + 1, high, index, value, pos2);
ST[pos] = ST[pos1] > ST[pos2] ? ST[pos1] : ST[pos2];
}
void update (int id, int value) {
updateUtil (0, n - 1, nodes[edges[id].second][3], value, 0);
}
int queryUtil (int low, int high, int left, int right, int pos) {
if (left <= low && right >= high) return ST[pos];
if (left > high || low > right) return -1;
int mid = low + high;
mid >>= 1;
int pos1 = (pos << 1) + 1;
int pos2 = pos1 + 1;
int a = queryUtil (low, mid, left, right, pos1);
int b = queryUtil (mid + 1, high, left, right, pos2);
return a > b ? a : b;
}
int query (int left, int right) {
return queryUtil (0, n - 1, left, right, 0);
}
int crawlTree (int u, int v) {
int cu, cv = nodes[v][4], ans = 0;
while (true) {
cu = nodes[u][4];
if (cu == cv) {
if (u != v) {
int a = query (nodes[v][3] + 1, nodes[u][3]);
if (ans < a) ans = a;
}
break;
} else {
int a = query (nodes[chainHeads[cu]][3], nodes[u][3]);
if (ans < a) ans = a;
u = nodes[chainHeads[cu]][0];
}
}
return ans;
}
int maxEdge (int u, int v) {
int lca = LCA (u, v);
// cerr << "LCA (" << u + 1 << ", " << v + 1 << ") = " << lca + 1 << endl;
int a = crawlTree (u, lca);
int b = crawlTree (v, lca);
return a > b ? a : b;
}
void solve () {
scanf ("%d", &n);
/**
vector < pii > adj[mx];
pii edges[mx];
SegmentTree S;
TreeNode node[mx];
int edgeCount, currentChain, n, chainHeads[mx];
int tin[mx], tout[mx], up[mx][BITS];
*/
for (int i = 0; i < n; i++) {
adj[i].clear();
}
memset (chainHeads, -1, sizeof (chainHeads));
edgeCount = currentChain = timer = 0;
int u, v, w;
for (int i = 0; i < n - 1; i++) {
// cin >> u >> v >> w;
scanf ("%d %d %d", &u, &v, &w);
u--, v--;
addEdge (i, u, v, w);
}
int root = 0, p = -1, d = 0;
dfs (root, p, d);
edges[n - 1] = {-1, -1};
HLD (root, n - 1);
build (0, edgeCount - 1, 0);
// for (int i = 0; i < n; i++) {
// for (int j = 0; j < n; j++) {
// cerr << i << " -- " << j << " => " << maxEdge (i, j) << endl;
// }
// }
// for (int i = 0; i < n - 1; i++) {
// update (i, 1);
// }
// string s;
while (true) {
char s[10];
scanf ("%s", s);
// cin >> s;
if (s[0] == 'D') {
break;
}
if (s[0] == 'C') {
// cin >> u >> v;
scanf ("%d %d", &u, &v);
u--;
update (u, v);
} else {
// cin >> u >> v;
scanf ("%d %d", &u, &v);
u--, v--;
cout << maxEdge (u, v) << endl;
}
}
}
int main() {
// ios_base::sync_with_stdio (false);
// cin.tie (nullptr); cout.tie(nullptr);
int t = 1;
scanf ("%d", &t);
while (t--) {
solve();
}
return 0;
}
/**
 * Problem : QTREE - Query on a tree
 * Problem URL : https://w...content-available-to-author-only...j.com/problems/QTREE/
 */

#include <bits/stdc++.h>
using namespace std;
using ll = long long;
using pii = pair < int, int >;

// This section is for Policy Based Data Structure, more precisely Ordered Set.
// All Functions of set works here, in addition we have order_of_key() and find_by_order() function. find_by_order() returns iterator
#include<ext/pb_ds/assoc_container.hpp>
#include<ext/pb_ds/tree_policy.hpp>
#include<functional>
using namespace __gnu_pbds;

// 1. Custom Set
template < class T, class COMP >
using custom_set = tree < T, null_type, COMP, rb_tree_tag, tree_order_statistics_node_update >;

// 2. Ordered Set
template < class T >
using ordered_set = tree < T, null_type, less < T >, rb_tree_tag, tree_order_statistics_node_update >;

// 3. Ordered Multi Set
// For storing multiple occurance of same value we need to use pair. In pair first value is our key and second is useless count variable.
// order_of_key(make_pair(key, 0)) returns first occurance of a value, order_of_key(make_pair(key, INT_MAX)) returns last occurance.
template < class T >
using ordered_multiset = tree < pair < T, int >, null_type, less < pair < T, int > >, rb_tree_tag, tree_order_statistics_node_update >;

// 4. Ordered Map
template < class T, class U >
using ordered_map = tree < T, U, less < T >, rb_tree_tag, tree_order_statistics_node_update >;
// Policy End

const int mx = 10005;
const int BITS = 14;

vector < pii > adj[mx];
pii edges[mx];
int SA[mx], ST[6 * mx];
int nodes[mx][5];
int edgeCount, currentChain, n, chainHeads[mx];
int tin[mx], tout[mx], up[mx][BITS], timer;

void addEdge (int id, int u, int v, int w) {
	adj[u].emplace_back (v, id);
	adj[v].emplace_back (u, id);
	edges[id].first = w;
}

void dfs (int u, int p, int d) {
	nodes[u][0] = p, nodes[u][1] = d, nodes[u][2] = 1;
	tin[u] = timer++;
	if (p == -1)
		up[u][0] = u;
	else
		up[u][0] = p;
	for (int i = 1; i < BITS; i++) {
		up[u][i] = up[up[u][i - 1]][i - 1];
	}
	for (pii & pi : adj[u]) {
		if (pi.first != p) {
			dfs (pi.first, u, d + 1);
			nodes[u][2]++;
			edges[pi.second].second = pi.first;
		}
	}
	tout[u] = timer;
}

void HLD (int u, int id) {
	if (chainHeads[currentChain] == -1) {
		chainHeads[currentChain] = u;
	}
	nodes[u][4] = currentChain;
	nodes[u][3] = edgeCount;
	SA[edgeCount++] = edges[id].first;
	int sc = -1, si;
	for (pii & pi : adj[u]) {
		if (pi.first != nodes[u][0]) {
			if (sc == -1 || nodes[sc][2] < nodes[pi.first][2]) {
				sc = pi.first; si = pi.second;
			}
		}
	}
	if (sc != -1) {
		HLD (sc, si);
	}
	
	for (pii & pi : adj[u]) {
		if (pi.first != nodes[u][0] && pi.first != sc) {
			currentChain++;
			HLD (pi.first, pi.second);
		}
	}
}

bool is_ancestor (int u, int v) {
	return tin[u] <= tin[v] && tout[u] >= tout[v];
}

int LCA (int u, int v) {
	if (is_ancestor (u, v)) return u;
	if (is_ancestor (v, u)) return v;
	for (int i = BITS - 1; i >= 0; i--) {
		if (!is_ancestor (up[u][i], v))
			u = up[u][i];
	}
	return up[u][0];
}

void build (int low, int high, int pos) {
	if (low == high) {
		ST[pos] = SA[low];
		return;
	}
	int mid = low + high;
	mid >>= 1;
	int pos1 = (pos << 1) + 1;
	int pos2 = pos1 + 1;
	build (low, mid, pos1);
	build (mid + 1, high, pos2);
	ST[pos] = ST[pos1] > ST[pos2] ? ST[pos1] : ST[pos2];
}

void updateUtil (int low, int high, int index, int value, int pos) {
	if (low > index || high < index) return;	
	if (low == high) {
		ST[pos] = value;
		return;
	}
	int mid = low + high;
	mid >>= 1;
	int pos1 = (pos << 1)  + 1;
	int pos2 = pos1 + 1;
	updateUtil (low, mid, index, value, pos1);
	updateUtil (mid + 1, high, index, value, pos2);
	ST[pos] = ST[pos1] > ST[pos2] ? ST[pos1] : ST[pos2];
}

void update (int id, int value) {
	updateUtil (0, n - 1, nodes[edges[id].second][3], value, 0);
}

int queryUtil (int low, int high, int left, int right, int pos) {
	if (left <= low && right >= high) return ST[pos];
	if (left > high || low > right) return -1;
	int mid = low + high;
	mid >>= 1;
	int pos1 = (pos << 1)  + 1;
	int pos2 = pos1 + 1;
	int a = queryUtil (low, mid, left, right, pos1);
	int b = queryUtil (mid + 1, high, left, right, pos2);
	return a > b ? a : b;
}

int query (int left, int right) {
	return queryUtil (0, n - 1, left, right, 0);
}

int crawlTree (int u, int v) {
	int cu, cv = nodes[v][4], ans = 0;
	while (true) {
		cu = nodes[u][4];
		if (cu == cv) {
			if (u != v) {
				int a = query (nodes[v][3] + 1, nodes[u][3]);
				if (ans < a) ans = a;
			}
			break;
		} else {
			int a = query (nodes[chainHeads[cu]][3], nodes[u][3]);
			if (ans < a) ans = a;
			u = nodes[chainHeads[cu]][0];
		}
	}
	return ans;
}

int maxEdge (int u, int v) {
	int lca = LCA (u, v);
	// cerr << "LCA (" << u + 1 << ", " << v + 1 << ") = " << lca + 1 << endl;
	int a = crawlTree (u, lca);
	int b = crawlTree (v, lca);
	return a > b ? a : b;
}

void solve () {
	scanf ("%d", &n);
	/**
		vector < pii > adj[mx];
		pii edges[mx];
		SegmentTree S;
		TreeNode node[mx];
		int edgeCount, currentChain, n, chainHeads[mx];
		int tin[mx], tout[mx], up[mx][BITS];
	*/
	for (int i = 0; i < n; i++) {
		adj[i].clear();
	}
	memset (chainHeads, -1, sizeof (chainHeads));
	edgeCount = currentChain = timer = 0;
	int u, v, w;
	for (int i = 0; i < n - 1; i++) {
		// cin >> u >> v >> w;
		scanf ("%d %d %d", &u, &v, &w);
		u--, v--;
		addEdge (i, u, v, w);
	}
	int root = 0, p = -1, d = 0;
	dfs (root, p, d);
	edges[n - 1] = {-1, -1};
	HLD (root, n - 1);
	build (0, edgeCount - 1, 0);
	// for (int i = 0; i < n; i++) {
		// for (int j = 0; j < n; j++) {
			// cerr << i << " -- " << j << " => " << maxEdge (i, j) << endl;
		// }
	// }
	// for (int i = 0; i < n - 1; i++) {
		// update (i, 1);
	// }
	// string s;
	while (true) {
		char s[10];
		scanf ("%s", s);
		// cin >> s;
		if (s[0] == 'D') {
			break;
		}
		if (s[0] == 'C') {
			// cin >> u >> v;
			scanf ("%d %d", &u, &v);
			u--;
			update (u, v);
		} else {
			// cin >> u >> v;
			scanf ("%d %d", &u, &v);
			u--, v--;
			cout << maxEdge (u, v) << endl;
		}
	}
}

int main() {
	// ios_base::sync_with_stdio (false);
	// cin.tie (nullptr); cout.tie(nullptr);
	int t = 1;
	scanf ("%d", &t);
	while (t--) {
		solve();
	}
	return 0;
}

