import java.io.*;
import java.util.*;
class CLSHDN{
static long INF
= Long.
MAX_VALUE>>2;
static class Edge{
int fr, to, wt;
Edge(int a, int b, int c){
fr = a; to = b; wt = c;
}
@Override
public int hashCode(){
int hc = 31*fr;
hc = hc*31 + to;
return hc;
}
@Override
public boolean equals
(Object o
){ if(o==null || !(o instanceof Edge))
return false;
Edge e = (Edge)o;
return fr==e.fr && to==e.to;
}
}
static class Vertex{
ArrayList<Edge> edges, sptEdges; int ilnd;
Vertex(){
edges = new ArrayList<>(); sptEdges = new ArrayList<>(); ilnd = -1;
}
}
static class KempeTree{
long t[];
int N;
KempeTree(int n){
t = new long[N+n];
}
void rangeUpdate(int i, int j, long v){
for(i+=N, j+=N; i<=j; i=(i+1)/2, j=(j-1)/2){
if(i%2==1)
t
[i
] = Math.
min(t
[i
], v
); if(j%2==0)
t
[j
] = Math.
min(t
[j
], v
); }
}
long query(int i){
long v = INF;
for(int j=i+N; j>0; j/=2)
return v;
}
}
static int n, m, s, t, q;
static Vertex g[];
static Edge el[];
static long ds[], dt[];
static KempeTree kt;
static HashSet<Edge> bridges;
int T = re.nextInt();
while(T-->0){
n = re.nextInt();
m = re.nextInt();
g = new Vertex[n];
el = new Edge[m*2];
for(int i=0; i<n; i++)
g[i] = new Vertex();
for(int i=0; i<m; i++){
int u = re.nextInt();
int v = re.nextInt();
int w = re.nextInt();
el[2*i] = new Edge(u, v, w);
el[2*i+1] = new Edge(v, u, w);
g[u].edges.add(el[2*i]);
g[v].edges.add(el[2*i+1]);
}
s = re.nextInt();
t = re.nextInt();
work();
q = re.nextInt();
while(q-->0){
int u = re.nextInt();
int v = re.nextInt();
if(ds[v]<ds[u]){
int t = v; v = u; u = t;
}
if(bridges.contains(new Edge(u, v, 0))){
long out = kt.query(g[u].ilnd);
System.
out.
println(out
>=INF
?"Infinity":out
); }
else
}
}
}
static void work(){
// COMPUTE DS AND DT
Edge prev[] = new Edge[n];
dt = dijkstra(g, t, s, prev);
ds = dijkstra(g, s, t, prev);
// FIND ALL BRIDGES
ArrayList<Edge> opt = new ArrayList<>();
bridges = new HashSet<>();
int nb = findBridges(opt);
//IDENTIFY ISLANDS
identifyIslands(prev);
//GET BYPASS VALUES
kt = new KempeTree(nb);
fillBypass();
}
static int findBridges(ArrayList<Edge> opt){
for(int i=0; i<el.length; i++){
Edge e = el[i];
if(ds[e.fr]+e.wt+dt[e.to]==ds[t]){
opt.add(e);
el[i] = null;
}
}
public int compare(Edge e1, Edge e2){
if(ds[e1.fr]==ds[e2.fr])
return ds[e1.to]==ds[e2.to] ? 0 : ds[e1.to]<ds[e2.to]? -1 : 1;
else
return ds[e1.fr]<ds[e2.fr] ? -1 : 1;
}
});
int nb = 0;
if(opt.size()==1){
bridges.add(opt.get(0)); nb++;
}
else{
if(ds[opt.get(0).to]<=ds[opt.get(1).fr]){
bridges.add(opt.get(0)); nb++;
}
Edge e = opt.get(opt.size()-1);
if(ds[e.fr]>=ds[opt.get(opt.size()-2).to] && !bridges.contains(e)){
bridges.add(e); nb++;
}
for(int i=1; i<opt.size()-1; i++){
e = opt.get(i);
if(ds[e.fr]>=ds[opt.get(i-1).to] && ds[e.to]<=ds[opt.get(i+1).fr]){
bridges.add(e); nb++;
}
}
}
return nb;
}
static void identifyIslands(Edge prev[]){
for(Edge e : prev)
if(e!=null)
g[e.fr].sptEdges.add(e);
class State{
int v, currIlnd;
State(int v1, int ci){
v = v1; currIlnd = ci;
}
}
Deque<State> stack = new LinkedList<>();
stack.push(new State(s, 0));
while(!stack.isEmpty()){
State st = stack.pop();
g[st.v].ilnd = st.currIlnd;
for(Edge e : g[st.v].sptEdges){
if(bridges.contains(e))
stack.push(new State(e.to, st.currIlnd+1));
else
stack.push(new State(e.to, st.currIlnd));
}
}
}
static void fillBypass(){
for(Edge e : el)
if(e!=null)
kt.rangeUpdate(g[e.fr].ilnd, g[e.to].ilnd-1, ds[e.fr]+e.wt+dt[e.to]);
}
static long[] dijkstra(Vertex g[], int s, int t, Edge prev[]){
long dist[] = new long[g.length];
dist[s] = 0;
prev[s] = null;
PriorityQueue<Integer> pq = new PriorityQueue<>(new Comparator<Integer>(){
return dist[u]==dist[v] ? 0 : dist[u]<dist[v] ? -1 : 1;
}
});
pq.add(s);
while(!pq.isEmpty()){
int curr = pq.poll();
if(curr==t)
continue;
if(dist[curr]>=INF)
break;
Vertex v = g[curr];
for(Edge e : v.edges){
if(dist[curr]+e.wt < dist[e.to]){
pq.remove(e.to);
dist[e.to] = dist[curr]+e.wt;
prev[e.to] = e;
pq.add(e.to);
}
}
}
return dist;
}
}
}
while(!st.hasMoreTokens())
return st.nextToken();
}
}
return Double.
parseDouble(next
()); }
}
import java.io.*;
import java.util.*;
class CLSHDN{
    static long INF = Long.MAX_VALUE>>2;
    
    static class Edge{
        int fr, to, wt;
        Edge(int a, int b, int c){
            fr = a; to = b; wt = c;
        }

        @Override
        public int hashCode(){
            int hc = 31*fr;
            hc = hc*31 + to;
            return hc;
        }

        @Override
        public boolean equals(Object o){
            if(o==null || !(o instanceof Edge))
                return false;
            Edge e = (Edge)o;
            return fr==e.fr && to==e.to;
        }
        
    }

    static class Vertex{
        ArrayList<Edge> edges, sptEdges; int ilnd;
        Vertex(){
            edges = new ArrayList<>(); sptEdges = new ArrayList<>(); ilnd = -1;
        }
    }
    
    static class KempeTree{
        long t[];
        int N;
        KempeTree(int n){
            N = (int)Math.pow(2, Math.ceil(Math.log(n)/Math.log(2)));
            t = new long[N+n];
            Arrays.fill(t, INF);
        }
        
        void rangeUpdate(int i, int j, long v){
            for(i+=N, j+=N; i<=j; i=(i+1)/2, j=(j-1)/2){
                if(i%2==1)
                    t[i] = Math.min(t[i], v);
                if(j%2==0)
                    t[j] = Math.min(t[j], v);
            }
        }
        
        long query(int i){
            long v = INF;
            for(int j=i+N; j>0; j/=2)
                v = Math.min(t[j], v);
            return v;
        }
    }

    static int n, m, s, t, q;
    static Vertex g[];
    static Edge el[];
    static long ds[], dt[];
    static KempeTree kt;
    static HashSet<Edge> bridges;
    public static void main(String args[])throws Exception{
        Reader re = new Reader(System.in);
        int T = re.nextInt();
        while(T-->0){
            n = re.nextInt();
            m = re.nextInt();
            g = new Vertex[n];
            el = new Edge[m*2];
            for(int i=0; i<n; i++)
                g[i] = new Vertex();
            for(int i=0; i<m; i++){
                int u = re.nextInt();
                int v = re.nextInt();
                int w = re.nextInt();
                el[2*i] = new Edge(u, v, w);
                el[2*i+1] = new Edge(v, u, w);
                g[u].edges.add(el[2*i]);
                g[v].edges.add(el[2*i+1]);
            }
            s = re.nextInt();
            t = re.nextInt();
            work();
            q = re.nextInt();
            while(q-->0){
                int u = re.nextInt();
                int v = re.nextInt();
                if(ds[v]<ds[u]){
                    int t = v; v = u; u = t;
                }
                if(bridges.contains(new Edge(u, v, 0))){
                    long out = kt.query(g[u].ilnd);
                    System.out.println(out>=INF?"Infinity":out);
                }
                else
                    System.out.println(ds[t]);
            }
        }
    }

    static void work(){
        // COMPUTE DS AND DT
        Edge prev[] = new Edge[n];
        dt = dijkstra(g, t, s, prev);
        ds = dijkstra(g, s, t, prev);

        // FIND ALL BRIDGES
        ArrayList<Edge> opt = new ArrayList<>();
        bridges = new HashSet<>();
        int nb = findBridges(opt);

        //IDENTIFY ISLANDS
        identifyIslands(prev);

        //GET BYPASS VALUES
        kt = new KempeTree(nb);
        fillBypass();
    }

    static int findBridges(ArrayList<Edge> opt){
        for(int i=0; i<el.length; i++){
            Edge e = el[i];
            if(ds[e.fr]+e.wt+dt[e.to]==ds[t]){
                opt.add(e);
                el[i] = null;
            }
        }
        Collections.sort(opt, new Comparator<Edge>(){
                public int compare(Edge e1, Edge e2){
                    if(ds[e1.fr]==ds[e2.fr])
                        return ds[e1.to]==ds[e2.to] ? 0 : ds[e1.to]<ds[e2.to]? -1 : 1;
                    else
                        return ds[e1.fr]<ds[e2.fr] ? -1 : 1;
                }
            });
        int nb = 0;
        if(opt.size()==1){
            bridges.add(opt.get(0)); nb++;
        }
        else{
            if(ds[opt.get(0).to]<=ds[opt.get(1).fr]){
                bridges.add(opt.get(0)); nb++;
            }
            Edge e = opt.get(opt.size()-1);
            if(ds[e.fr]>=ds[opt.get(opt.size()-2).to] && !bridges.contains(e)){
                bridges.add(e); nb++;
            }
            for(int i=1; i<opt.size()-1; i++){
                e = opt.get(i);
                if(ds[e.fr]>=ds[opt.get(i-1).to] && ds[e.to]<=ds[opt.get(i+1).fr]){
                    bridges.add(e); nb++;
                }
            }
        }
        return nb;
    }

    static void identifyIslands(Edge prev[]){
        for(Edge e : prev)
            if(e!=null)
                g[e.fr].sptEdges.add(e);
        class State{
            int v, currIlnd;
            State(int v1, int ci){
                v = v1; currIlnd = ci;
            }
        }
        Deque<State> stack = new LinkedList<>();
        stack.push(new State(s, 0));
        while(!stack.isEmpty()){
            State st = stack.pop();
            g[st.v].ilnd = st.currIlnd;
            for(Edge e : g[st.v].sptEdges){
                if(bridges.contains(e))
                    stack.push(new State(e.to, st.currIlnd+1));
                else
                    stack.push(new State(e.to, st.currIlnd));
            }
        }
    }

    static void fillBypass(){
        for(Edge e : el)
            if(e!=null)
                kt.rangeUpdate(g[e.fr].ilnd, g[e.to].ilnd-1, ds[e.fr]+e.wt+dt[e.to]);
    }

    static long[] dijkstra(Vertex g[], int s, int t, Edge prev[]){
        long dist[] = new long[g.length];
        Arrays.fill(dist, INF);
        dist[s] = 0;
        prev[s] = null;
        PriorityQueue<Integer> pq = new PriorityQueue<>(new Comparator<Integer>(){
                    public int compare(Integer u, Integer v){
                        return dist[u]==dist[v] ? 0 : dist[u]<dist[v] ? -1 : 1;
                    }
                });
        pq.add(s);

        while(!pq.isEmpty()){
            int curr = pq.poll();
            if(curr==t)
                continue;
            if(dist[curr]>=INF)
                break;
            Vertex v = g[curr];
            for(Edge e : v.edges){
                if(dist[curr]+e.wt < dist[e.to]){
                    pq.remove(e.to);
                    dist[e.to] = dist[curr]+e.wt;
                    prev[e.to] = e;
                    pq.add(e.to);
                }
            }
        }
        return dist;
    }
}

class Reader{
    BufferedReader br;
    StringTokenizer st;
    Reader(InputStream in) throws Exception{
        br = new BufferedReader(new InputStreamReader(in));
        st = new StringTokenizer("");
    }

    String next() throws Exception{
        while(!st.hasMoreTokens())
            st = new StringTokenizer(br.readLine());
        return st.nextToken();
    }

    int nextInt() throws Exception{
        return Integer.parseInt(next());
    }

    double nextDouble() throws Exception{
        return Double.parseDouble(next());
    }
}