/*
树的秘密.(secret.cpp/c/pas)
Prime最近在研究树这种奇怪的数据结构,他把树的意思向周边各位组的大神介绍了一遍:树是由根结点和若干颗子树构成的。树是由一个集合以及在该集合上定义的一种关系构成的。集合中的元素称为树的结点,所定义的关系称为父子关系。父子关系在树的结点之间建立了一个层次结构。在这种层次结构中有一个结点具有特殊的地位,这个结点称为该树的根结点,或称为树根。
大神们表示不能理解这种奇怪的东西 ,然后Prime又耐心的解释了一遍的树的意思,Prime讲数据结构就好比张三丰教张无忌太极一样,每一遍都不一样!他这次是这样解释的:单个结点是一棵树,树根就是该结点本身。
设T1,T2,..,Tk是树,它们的根结点分别为n1,n2,..,nk。用一个新结点n作为n1,n2,..,nk的父亲,则得到一棵新树,结点n就是新树的根。我们称n1,n2,..,nk为一组兄弟结点,它们都是结点n的子结点。我们还称T1,T2,..,Tk为结点n的子树。
空集合也是树,称为空树。空树中没有结点。
好吧,大神们纷纷的懂了树的意思,于是好奇的他们每人给prime出了一个问题。
蕾蕾想知道树上任意两点之间的路径权值和是多少,他认为他期末考试的分数是这个和的1000倍!
陶爷,是一个文艺沉静的大神,他觉得树应有一种静美,所以他希望知道任意两点之间的路径中权值最大的一条边是什么,以便他写文章来讽刺这条边!
ZCJ是一个爱闹事的XAG逗比,他会不时的改变某一条的边的权值,但是他技术太弱,不会改变边所连是哪对节点。
树是在是一种神秘的数据结构,prime表示他参透不了其中的秘密!
这时Prime周围的tty,owaski,lyx,zxo以及众大神开始嘲讽prime太弱,肯定解决不了,是的Prime他好弱啥都不会,于是他把问题交给了大神您,谢谢你!
*/
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <cmath>
#include <ctime>
#include <queue>
#include <set>
#include <map>
#define REP(I,A,B) for(int I=A,END=B;I<=END;I++)
#define REPD(I,A,B) for(int I=A,END=B;I>=END;I--)
#define RI(X) scanf("%d",&X)
#define RS(X) scanf("%s",&X)
#define GCH getchar()
#define PCH(X) putchar(X)
#define MAX(A,B) (((A)>(B))?(A):(B))
#define MIN(A,B) (((A)<(B))?(A):(B))
#define MS(X,Y) memset(X,Y,sizeof(X))
#define MC(X,Y,var,len) memcpy((X),(Y),sizeof(var)*(len))
#define debug(...) fprintf(stderr,__VA_ARGS__)
#define MAXN 100005
struct line
{
int y;
int value;
int next;
}bian[MAXN*2]={0};
int head[MAXN]={0};
int fa[MAXN]={0};
struct node
{
int deep;
int fa;
int son;
int w;
int size;
int top;
}tree[MAXN]={0};
struct sg_tree
{
int lazy;
int max;
int sum;
}sg[MAXN*4]={0};
int n,m,tot=0;
using namespace std;
void open()
{
freopen("secret.in","r",stdin);
freopen("secret.out","w",stdout);
}
void close()
{
fclose(stdin);
fclose(stdout);
}
void add(int x,int y,int value)
{
tot++;
bian[tot].y=y;
bian[tot].value=value;
bian[tot].next=head[x];
head[x]=tot;
}
void init()
{
RI(n);
RI(m);
int x,y,value;
REP(i,1,n-1)
{
RI(x);
RI(y);
RI(value);
add(x,y,value);
add(y,x,value);
}
}
void dfs(int p)//求 deep fa size son
{
int i;
tree[p].size++;
for (i=head[p];i!=0;i=bian[i].next)
if (tree[bian[i].y].deep==0)
{
tree[bian[i].y].deep=tree[p].deep+1;
tree[bian[i].y].fa=p;
dfs(bian[i].y);
if (tree[tree[p].son].size<tree[bian[i].y].size)
{
tree[p].son=bian[i].y;
}
tree[p].size+=tree[bian[i].y].size;
}
}
void decomposition(int p)
{
int i;
if (tree[p].son!=0)
{
tree[tree[p].son].top=tree[p].top;
tot++;
tree[tree[p].son].w=tot;
decomposition(tree[p].son);
}
for (i=head[p];i!=0;i=bian[i].next)
if (tree[p].fa!=bian[i].y && tree[p].son!=bian[i].y)
{
tree[bian[i].y].top=bian[i].y;
tot++;
tree[bian[i].y].w=tot;
fa[tot]=bian[i].value;
decomposition(bian[i].y);
}
else if (tree[p].son==bian[i].y)
{
fa[tree[tree[p].son].w]=bian[i].value;
}
}
#define lc(p) (p*2)
#define rc(p) (p*2+1)
void down(int p,int l,int r)
{
int mid=(l+r)/2;
if (sg[p].lazy!=0)
{
sg[rc(p)].max=sg[lc(p)].max=sg[p].lazy;
sg[lc(p)].sum=(mid-l+1)*sg[p].lazy;
sg[rc(p)].sum=(r-mid)*sg[p].lazy;
sg[rc(p)].lazy=sg[lc(p)].lazy=sg[p].lazy;
sg[p].lazy=0;
}
}
void up(int p)
{
sg[p].max=max(sg[lc(p)].max,sg[rc(p)].max);
sg[p].sum=sg[lc(p)].sum+sg[rc(p)].sum;
}
void insert(int p,int l,int r,int ll,int rr,int value)
{
if (l==ll && r==rr)
{
sg[p].lazy=value;
sg[p].max=value;
sg[p].sum=(r-l+1)*value;
return ;
}
int mid=(l+r)/2;
down(p,l,r);
if (rr<=mid)
insert(lc(p),l,mid,ll,rr,value);
else if (mid<ll)
insert(rc(p),mid+1,r,ll,rr,value);
else
insert(lc(p),l,mid,ll,mid,value),insert(rc(p),mid+1,r,mid+1,rr,value);
up(p);
}
int squery(int p,int l,int r,int ll,int rr)
{
if (l==ll && r==rr)
{
return sg[p].sum;
}
int mid=(l+r)/2;
down(p,l,r);
if (rr<=mid)
return squery(lc(p),l,mid,ll,rr);
else if (mid<ll)
return squery(rc(p),mid+1,r,ll,rr);
else
return squery(lc(p),l,mid,ll,mid)+squery(rc(p),mid+1,r,mid+1,rr);
}
int mquery(int p,int l,int r,int ll,int rr)
{
if (l==ll && r==rr)
{
return sg[p].max;
}
int mid=(l+r)/2;
down(p,l,r);
if (rr<=mid)
return mquery(lc(p),l,mid,ll,rr);
else if (mid<ll)
return mquery(rc(p),mid+1,r,ll,rr);
else
return max(mquery(lc(p),l,mid,ll,mid),mquery(rc(p),mid+1,r,mid+1,rr));
}
void maketree()
{
REP(i,1,n)
insert(1,1,n,tree[i].w,tree[i].w,fa[tree[i].w]);
}
int sfind(int u,int v)
{
int fu=tree[u].top,fv=tree[v].top;
int sum=0;
while (fu!=fv)
{
if (tree[fu].deep<tree[fv].deep)
{
swap(u,v);
swap(fu,fv);
}
sum+=squery(1,1,n,tree[fu].w,tree[u].w);
u=tree[fu].fa;
fu=tree[u].top;
}
if (u==v) //已然是LCA
{
return sum;
}
else if (tree[u].deep>tree[v].deep)
{
swap(u,v);
}
return sum+squery(1,1,n,tree[tree[u].son].w,tree[v].w);
}
int mfind(int u,int v)
{
int fu=tree[u].top,fv=tree[v].top;
int tmp=0;
while (fu!=fv)
{
if (tree[fu].deep<tree[fv].deep)
{
swap(u,v);
swap(fu,fv);
}
tmp=max(tmp,mquery(1,1,n,tree[fu].w,tree[u].w));
u=tree[fu].fa;
fu=tree[u].top;
}
if (u==v)
{
return tmp;
}
else if (tree[u].deep>tree[v].deep)
{
swap(u,v);
}
return max(tmp,mquery(1,1,n,tree[tree[u].son].w,tree[v].w));
}
void change(int u,int v,int value)
{
int fu=tree[u].top,fv=tree[v].top;
while (fu!=fv)
{
if (tree[fu].deep<tree[fv].deep)
{
swap(u,v);
swap(fu,fv);
}
insert(1,1,n,tree[fu].w,tree[u].w,value);
u=tree[fu].fa;
fu=tree[u].top;
}
if (u==v)
{
return ;
}
else if (tree[u].deep>tree[v].deep)
{
swap(u,v);
}
insert(1,1,n,tree[tree[u].son].w,tree[v].w,value);
}
void work()
{
scanf("\n");
char way;
int x,y,value;
REP(_,1,m)
{
way=GCH;
GCH;
GCH;
RI(x);
RI(y);
if (way=='C')
{
RI(value);
change(x,y,value);
}
else if (way=='S')
{
printf("%d\n",sfind(x,y));
}
else
{
printf("%d\n",mfind(x,y));
}
GCH;
}
}
int main()
{
open();
init();
tree[1].deep=1;
dfs(1);
tot=0;
tot++;
tree[1].w=tot;
tree[1].top=1;
fa[1]=0;
decomposition(1);
maketree();
work();
close();
return 0;
}
/*
树的秘密.(secret.cpp/c/pas)
Prime最近在研究树这种奇怪的数据结构，他把树的意思向周边各位组的大神介绍了一遍：树是由根结点和若干颗子树构成的。树是由一个集合以及在该集合上定义的一种关系构成的。集合中的元素称为树的结点，所定义的关系称为父子关系。父子关系在树的结点之间建立了一个层次结构。在这种层次结构中有一个结点具有特殊的地位，这个结点称为该树的根结点，或称为树根。
大神们表示不能理解这种奇怪的东西 ，然后Prime又耐心的解释了一遍的树的意思，Prime讲数据结构就好比张三丰教张无忌太极一样，每一遍都不一样!他这次是这样解释的：单个结点是一棵树，树根就是该结点本身。
设T1,T2,..,Tk是树，它们的根结点分别为n1,n2,..,nk。用一个新结点n作为n1,n2,..,nk的父亲，则得到一棵新树，结点n就是新树的根。我们称n1,n2,..,nk为一组兄弟结点，它们都是结点n的子结点。我们还称T1,T2,..,Tk为结点n的子树。
空集合也是树，称为空树。空树中没有结点。
好吧，大神们纷纷的懂了树的意思，于是好奇的他们每人给prime出了一个问题。
蕾蕾想知道树上任意两点之间的路径权值和是多少，他认为他期末考试的分数是这个和的1000倍！
陶爷，是一个文艺沉静的大神，他觉得树应有一种静美，所以他希望知道任意两点之间的路径中权值最大的一条边是什么，以便他写文章来讽刺这条边！
ZCJ是一个爱闹事的XAG逗比,他会不时的改变某一条的边的权值，但是他技术太弱，不会改变边所连是哪对节点。
树是在是一种神秘的数据结构，prime表示他参透不了其中的秘密！
这时Prime周围的tty,owaski,lyx,zxo以及众大神开始嘲讽prime太弱，肯定解决不了，是的Prime他好弱啥都不会，于是他把问题交给了大神您，谢谢你！

*/
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <cmath>
#include <ctime>
#include <queue>
#include <set>
#include <map>
#define REP(I,A,B) for(int I=A,END=B;I<=END;I++)
#define REPD(I,A,B) for(int I=A,END=B;I>=END;I--)
#define RI(X) scanf("%d",&X)
#define RS(X) scanf("%s",&X)
#define GCH getchar()
#define PCH(X) putchar(X)
#define MAX(A,B) (((A)>(B))?(A):(B))
#define MIN(A,B) (((A)<(B))?(A):(B))
#define MS(X,Y) memset(X,Y,sizeof(X))
#define MC(X,Y,var,len) memcpy((X),(Y),sizeof(var)*(len))
#define debug(...) fprintf(stderr,__VA_ARGS__)
#define MAXN 100005
struct line
{
  int y;
  int value;
  int next;
}bian[MAXN*2]={0};
int head[MAXN]={0};
int fa[MAXN]={0};
struct node
{
  int deep;
  int fa;
  int son;
  int w;
  int size;
  int top;
}tree[MAXN]={0};
struct sg_tree
{
  int lazy;
  int max;
  int sum;
}sg[MAXN*4]={0};
int n,m,tot=0;
using namespace std;
void open()
{
  freopen("secret.in","r",stdin);
  freopen("secret.out","w",stdout);
}
void close()
{
  fclose(stdin);
  fclose(stdout);
}
void add(int x,int y,int value)
{
  tot++;
  bian[tot].y=y;
  bian[tot].value=value;
  bian[tot].next=head[x];
  head[x]=tot;
}
void init()
{
  RI(n);
  RI(m);
  int x,y,value;
  REP(i,1,n-1)
  {
    RI(x);
    RI(y);
    RI(value);
    add(x,y,value);
    add(y,x,value);
  }
}
void dfs(int p)//求 deep fa size son 
{
  int i;
  tree[p].size++;
  for (i=head[p];i!=0;i=bian[i].next)
    if (tree[bian[i].y].deep==0) 
    {
      tree[bian[i].y].deep=tree[p].deep+1;
      tree[bian[i].y].fa=p;
      dfs(bian[i].y);
      if (tree[tree[p].son].size<tree[bian[i].y].size)
      {
	    tree[p].son=bian[i].y;
	  }
	  tree[p].size+=tree[bian[i].y].size;
    }
}
void decomposition(int p)
{
  int i;
  if (tree[p].son!=0)
  {
    tree[tree[p].son].top=tree[p].top;
    tot++;
    tree[tree[p].son].w=tot;
    decomposition(tree[p].son);
  }
  for (i=head[p];i!=0;i=bian[i].next)
    if (tree[p].fa!=bian[i].y && tree[p].son!=bian[i].y)
    {
	  tree[bian[i].y].top=bian[i].y;
	  tot++;
	  tree[bian[i].y].w=tot;
	  fa[tot]=bian[i].value;
	  decomposition(bian[i].y);
	}
	else if (tree[p].son==bian[i].y)
	{
	  fa[tree[tree[p].son].w]=bian[i].value;
	}
}
#define lc(p) (p*2)
#define rc(p) (p*2+1)
void down(int p,int l,int r)
{
  int mid=(l+r)/2;
   if (sg[p].lazy!=0)
   {
     sg[rc(p)].max=sg[lc(p)].max=sg[p].lazy;
     sg[lc(p)].sum=(mid-l+1)*sg[p].lazy;
     sg[rc(p)].sum=(r-mid)*sg[p].lazy;
     sg[rc(p)].lazy=sg[lc(p)].lazy=sg[p].lazy;
     sg[p].lazy=0;
   }
}
void up(int p)
{
  sg[p].max=max(sg[lc(p)].max,sg[rc(p)].max);
  sg[p].sum=sg[lc(p)].sum+sg[rc(p)].sum;
}
void insert(int p,int l,int r,int ll,int rr,int value)
{
  if (l==ll && r==rr)
  {
    sg[p].lazy=value;
    sg[p].max=value;
    sg[p].sum=(r-l+1)*value;
    return ;
  }
  int mid=(l+r)/2;
  down(p,l,r);
  if (rr<=mid)
    insert(lc(p),l,mid,ll,rr,value);
  else if (mid<ll)
    insert(rc(p),mid+1,r,ll,rr,value);
  else 
    insert(lc(p),l,mid,ll,mid,value),insert(rc(p),mid+1,r,mid+1,rr,value);
  up(p);
}
int squery(int p,int l,int r,int ll,int rr)
{
  if (l==ll && r==rr)
  {
    return sg[p].sum;
  }
  int mid=(l+r)/2;
  down(p,l,r);
  if (rr<=mid)
    return squery(lc(p),l,mid,ll,rr);
  else if (mid<ll)
    return squery(rc(p),mid+1,r,ll,rr);
  else
    return squery(lc(p),l,mid,ll,mid)+squery(rc(p),mid+1,r,mid+1,rr);
}
int mquery(int p,int l,int r,int ll,int rr)
{
  if (l==ll && r==rr)
  {
    return sg[p].max;
  }
  int mid=(l+r)/2;
  down(p,l,r);
  if (rr<=mid)
    return mquery(lc(p),l,mid,ll,rr);
  else if (mid<ll)
    return mquery(rc(p),mid+1,r,ll,rr);
  else
    return max(mquery(lc(p),l,mid,ll,mid),mquery(rc(p),mid+1,r,mid+1,rr));
}
void maketree()
{
  REP(i,1,n)
    insert(1,1,n,tree[i].w,tree[i].w,fa[tree[i].w]);
}
int sfind(int u,int v)
{
  int fu=tree[u].top,fv=tree[v].top;
  int sum=0;
  while (fu!=fv)
  {
    if (tree[fu].deep<tree[fv].deep)
    {
	  swap(u,v);
	  swap(fu,fv);
	}
	sum+=squery(1,1,n,tree[fu].w,tree[u].w);
	u=tree[fu].fa;
	fu=tree[u].top;
  }
  if (u==v) //已然是LCA
  {
    return sum;
  } 
  else if (tree[u].deep>tree[v].deep)
  {
    swap(u,v);
  }
  return sum+squery(1,1,n,tree[tree[u].son].w,tree[v].w);
}
int mfind(int u,int v)
{
  int fu=tree[u].top,fv=tree[v].top;
  int tmp=0;
  while (fu!=fv)
  {
    if (tree[fu].deep<tree[fv].deep)
    {
	  swap(u,v);
	  swap(fu,fv);
	}
	tmp=max(tmp,mquery(1,1,n,tree[fu].w,tree[u].w));
	u=tree[fu].fa;
	fu=tree[u].top;
  }
  if (u==v)
  {
    return tmp;
  }
  else if (tree[u].deep>tree[v].deep)
  {
    swap(u,v);
  }
  return max(tmp,mquery(1,1,n,tree[tree[u].son].w,tree[v].w));
}
void change(int u,int v,int value)
{
  int fu=tree[u].top,fv=tree[v].top;
  while (fu!=fv)
  {
    if (tree[fu].deep<tree[fv].deep)
    {
	  swap(u,v);
	  swap(fu,fv);
	}
	insert(1,1,n,tree[fu].w,tree[u].w,value);
	u=tree[fu].fa;
	fu=tree[u].top;
  }
  if (u==v)
  {
    return ;
  }
  else if (tree[u].deep>tree[v].deep)
  {
    swap(u,v);
  }
  insert(1,1,n,tree[tree[u].son].w,tree[v].w,value);
}
void work()
{
  scanf("\n");
  char way;
  int x,y,value;
  REP(_,1,m)
  {
    way=GCH;
    GCH;
    GCH;
    RI(x);
    RI(y);
    if (way=='C')
    {
	  RI(value);
	  change(x,y,value);
	}
    else if (way=='S')
    {
	  printf("%d\n",sfind(x,y));
	}
	else
	{
	  printf("%d\n",mfind(x,y));
	}
    GCH;
  }
}
int main()
{
  open();
  init();
  tree[1].deep=1;
  dfs(1);
  tot=0;
  tot++;
  tree[1].w=tot;
  tree[1].top=1;
  fa[1]=0;
  decomposition(1);
  maketree();
  work();
  close();
  return 0;
}