#include <bits/stdc++.h>
using namespace std;
#define endl '\n'
#define int long long
const int N = 2e5, oo = 2e18, MOD = 1e9+7;
int modpow(int b, int p) {
int res = 1;
while (p) {
if (p & 1) res = (b * res) % MOD;
b = (b * b) % MOD;
p >>= 1;
}
return res;
}
int inv(int a) {
return modpow(a, MOD - 2);
}
vector<int> spf;
void sieve(int n) { // O (n . log(n))
spf.assign(n + 1,0 );
vector<bool> is_prime(n+1, true);
is_prime[0] = is_prime[1] = false;
for (int i = 2; i <= n; i++) {
if (is_prime[i]) {
spf[i] = i;
for (int j = 2 * i; j <= n; j+=i) {
is_prime[j] = false;
if (spf[j] == 0) {
spf[j] = i;
}
}
}
}
}
struct Node2 {
int sum, lazy = 0;
bool isLazy = false;
Node2(){
sum=oo;
}
Node2(int x) : sum(x) {}
void updateNode(int x, int lx, int rx) {
sum=min(sum,x);
lazy=x;
isLazy=true;
}
};
struct SagaraWithLazy2 {
#define leftChild (node * 2 + 1)
#define rightChild (node * 2 + 2)
#define mid ((lx + rx >> 1))
int treeSize;
vector<Node2> segData;
void build(int node, int lx, int rx, vector<int>& arr) {
if (rx - lx == 1) {
if (lx < (int)arr.size())
segData[node] = Node2(arr[lx]);
return;
}
build(leftChild, lx, mid, arr);
build(rightChild, mid, rx, arr);
segData[node] = merge(segData[leftChild], segData[rightChild]);
}
Node2 merge(Node2& L, Node2& R) {
Node2 res = Node2();
res.sum=min(L.sum,R.sum);
return res;
}
void propagate(int node, int lx, int rx) {
if (rx - lx == 1 or !segData[node].isLazy) return;
segData[leftChild].updateNode(segData[node].lazy, lx, mid);
segData[rightChild].updateNode(segData[node].lazy, mid, rx);
segData[node].isLazy = segData[node].lazy = 0;
}
void update(int l, int r, int x, int node, int lx, int rx) {
propagate(node, lx, rx);
if (l >= rx or lx >= r) return;
if (l <= lx and rx <= r) {
segData[node].updateNode(x, lx, rx);
return;
}
update(l, r, x, leftChild, lx, mid);
update(l, r, x, rightChild, mid, rx);
segData[node] = merge(segData[leftChild], segData[rightChild]);
}
Node2 query(int l, int r, int node, int lx, int rx) {
propagate(node, lx, rx);
if (l >= rx or lx >= r) return Node2();
if (l <= lx and rx <= r) {
return segData[node];
}
Node2 left = query(l, r, leftChild, lx, mid);
Node2 right = query(l, r, rightChild, mid, rx);
return merge(left, right);
}
public:
SagaraWithLazy2(int n, vector<int>& arr) {
treeSize = 1;
while (treeSize < n) treeSize <<= 1;
segData = vector<Node2>(treeSize << 1);
build(0, 0, treeSize, arr);
}
void update(int l, int r, int x) {
update(l, r, x, 0, 0, treeSize);
}
int query(int l, int r) {
Node2 a = query(l, r, 0, 0, treeSize);
return (a.sum);
}
#undef LeftChild
#undef RightChild
#undef mid
};
struct Node {
int prod;
int sum, lazy = 0;
bool isLazy = false;
Node() :prod(1) { }
Node(int x) : prod(x) {}
void updateNode(int x, int lx, int rx) {
prod *= modpow(x,rx-lx);
prod %= MOD;
if(lazy==0)lazy=1;
lazy *= x;
lazy%=MOD;
isLazy = true;
}
};
struct SagaraWithLazy {
#define leftChild (node * 2 + 1)
#define rightChild (node * 2 + 2)
#define mid ((lx + rx >> 1))
int treeSize;
vector<Node> segData;
void build(int node, int lx, int rx, vector<int>& arr) {
if (rx - lx == 1) {
if (lx < (int)arr.size())
segData[node] = Node(arr[lx]);
return;
}
build(leftChild, lx, mid, arr);
build(rightChild, mid, rx, arr);
segData[node] = merge(segData[leftChild], segData[rightChild]);
}
Node merge(Node& L, Node& R) {
Node res = Node();
res.prod = (L.prod * R.prod) % MOD;
return res;
}
void propagate(int node, int lx, int rx) {
if (rx - lx == 1 or !segData[node].isLazy) return;
segData[leftChild].updateNode(segData[node].lazy, lx, mid);
segData[rightChild].updateNode(segData[node].lazy, mid, rx);
segData[node].isLazy = segData[node].lazy = 0;
}
void update(int l, int r, int x, int node, int lx, int rx) {
propagate(node, lx, rx);
if (l >= rx or lx >= r) return;
if (l <= lx and rx <= r) {
segData[node].updateNode(x, lx, rx);
return;
}
update(l, r, x, leftChild, lx, mid);
update(l, r, x, rightChild, mid, rx);
segData[node] = merge(segData[leftChild], segData[rightChild]);
}
Node query(int l, int r, int node, int lx, int rx) {
propagate(node, lx, rx);
if (l >= rx or lx >= r) return Node();
if (l <= lx and rx <= r) {
return segData[node];
}
Node left = query(l, r, leftChild, lx, mid);
Node right = query(l, r, rightChild, mid, rx);
return merge(left, right);
}
public:
SagaraWithLazy(int n, vector<int>& arr) {
treeSize = 1;
while (treeSize < n) treeSize <<= 1;
segData = vector<Node>(treeSize << 1);
build(0, 0, treeSize, arr);
}
void update(int l, int r, int x) {
update(l, r, x, 0, 0, treeSize);
}
int query(int l, int r) {
Node a = query(l, r, 0, 0, treeSize);
return (a.prod ) % MOD;
}
#undef LeftChild
#undef RightChild
#undef mid
};
void solve() {
int n, q; cin >> n >> q;
vector<int> a(n),b(n);
for (int i = 0; i < n; i++) cin >> a[i],b[i]=spf[a[i]];
SagaraWithLazy st(n, a);
SagaraWithLazy2 st2(n,b);
while (q--) {
int t; cin >> t;
if (t == 1) {
int l, r; cin >> l >> r;
l--, r--;
int prod = st.query(l, r + 1);
int sp = st2.query(l, r + 1);
cout << (prod * inv(sp)) % MOD << endl;
} else {
int l, r, x; cin >> l >> r >> x;
st.update(l-1, r, x);
st2.update(l-1, r, spf[x]);
}
}
}
signed main() {
ios_base::sync_with_stdio(false);
cin.tie(NULL); cout.tie(NULL);
// #ifndef ONLINE_JUDGE
// freopen("input.txt", "r", stdin);
// freopen("output.txt", "w", stdout);
// #endif
int t; t = 1;
sieve(1e7);
cin >> t;
while (t--) solve();
return 0;
}
#include <bits/stdc++.h>
using namespace std;

#define endl     '\n'
#define int      long long

const int N = 2e5, oo = 2e18, MOD = 1e9+7;


int modpow(int b, int p) {
    int res = 1;
    while (p) {
        if (p & 1) res = (b * res) % MOD;
        b = (b * b)  % MOD;
        p >>= 1;
    }
    return res;
}

int inv(int a) {
    return modpow(a, MOD - 2);
}

vector<int> spf;
void sieve(int n) { // O (n . log(n))
    spf.assign(n + 1,0 );
   vector<bool> is_prime(n+1, true);
   is_prime[0] = is_prime[1] = false;
   for (int i = 2; i <= n; i++) {
      if (is_prime[i]) {
         spf[i] = i;
         for (int j = 2 * i; j <= n; j+=i) {
            is_prime[j] = false;
            if (spf[j] == 0) {
                spf[j] = i;
            }
         }
      }
   }
}



struct Node2 {
    int sum, lazy = 0;
    bool isLazy = false;
    Node2(){
        sum=oo;
    }
    Node2(int x) : sum(x) {}
    void updateNode(int x, int lx, int rx) {
        sum=min(sum,x);
        lazy=x;
        isLazy=true;
    }
};


struct SagaraWithLazy2 {
#define leftChild  (node * 2 + 1)
#define rightChild (node * 2 + 2)
#define mid        ((lx + rx >> 1))

    int treeSize;
    vector<Node2> segData;

    void build(int node, int lx, int rx, vector<int>& arr) {
        if (rx - lx == 1) {
            if (lx < (int)arr.size())
                segData[node] = Node2(arr[lx]);
            return;
        }

        build(leftChild, lx, mid, arr);
        build(rightChild, mid, rx, arr);

        segData[node] = merge(segData[leftChild], segData[rightChild]);
    }

    Node2 merge(Node2& L, Node2& R) {
        Node2 res = Node2();
        res.sum=min(L.sum,R.sum);
        return res;
    }


    void propagate(int node, int lx, int rx) {
        if (rx - lx == 1 or !segData[node].isLazy) return;

        segData[leftChild].updateNode(segData[node].lazy, lx, mid);
        segData[rightChild].updateNode(segData[node].lazy, mid, rx);

        segData[node].isLazy = segData[node].lazy = 0;
    }

    void update(int l, int r, int x, int node, int lx, int rx) {
        propagate(node, lx, rx);

        if (l >= rx or lx >= r) return;
        if (l <= lx and rx <= r) {
            segData[node].updateNode(x, lx, rx);
            return;
        }

        update(l, r, x, leftChild, lx, mid);
        update(l, r, x, rightChild, mid, rx);

        segData[node] = merge(segData[leftChild], segData[rightChild]);
    }


    Node2 query(int l, int r, int node, int lx, int rx) {
        propagate(node, lx, rx);

        if (l >= rx or lx >= r) return Node2();
        if (l <= lx and rx <= r) {
            return segData[node];
        }

        Node2 left = query(l, r, leftChild, lx, mid);
        Node2 right = query(l, r, rightChild, mid, rx);

        return merge(left, right);
    }

public:
    SagaraWithLazy2(int n, vector<int>& arr) {
        treeSize = 1;
        while (treeSize < n) treeSize <<= 1;
        segData = vector<Node2>(treeSize << 1);
        build(0, 0, treeSize, arr);
    }

    void update(int l, int r, int x) {
        update(l, r, x, 0, 0, treeSize);
    }

    int query(int l, int r) {
        Node2 a = query(l, r, 0, 0, treeSize);
        return (a.sum);
    }
#undef LeftChild
#undef RightChild
#undef mid
};

struct Node {
    int prod;
    int sum, lazy = 0;
    bool isLazy = false;

    Node() :prod(1) { }
    Node(int x) :  prod(x) {}
    void updateNode(int x, int lx, int rx) {
        prod *= modpow(x,rx-lx);
        prod %= MOD;
        if(lazy==0)lazy=1;
        lazy *= x;
        lazy%=MOD;
        isLazy = true;
    }
};


struct SagaraWithLazy {
#define leftChild  (node * 2 + 1)
#define rightChild (node * 2 + 2)
#define mid        ((lx + rx >> 1))

    int treeSize;
    vector<Node> segData;

    void build(int node, int lx, int rx, vector<int>& arr) {
        if (rx - lx == 1) {
            if (lx < (int)arr.size())
                segData[node] = Node(arr[lx]);
            return;
        }

        build(leftChild, lx, mid, arr);
        build(rightChild, mid, rx, arr);

        segData[node] = merge(segData[leftChild], segData[rightChild]);
    }

    Node merge(Node& L, Node& R) {
        Node res = Node();
        res.prod = (L.prod * R.prod) % MOD;
        return res;
    }


    void propagate(int node, int lx, int rx) {
        if (rx - lx == 1 or !segData[node].isLazy) return;

        segData[leftChild].updateNode(segData[node].lazy, lx, mid);
        segData[rightChild].updateNode(segData[node].lazy, mid, rx);

        segData[node].isLazy = segData[node].lazy = 0;
    }

    void update(int l, int r, int x, int node, int lx, int rx) {
        propagate(node, lx, rx);

        if (l >= rx or lx >= r) return;
        if (l <= lx and rx <= r) {
            segData[node].updateNode(x, lx, rx);
            return;
        }

        update(l, r, x, leftChild, lx, mid);
        update(l, r, x, rightChild, mid, rx);

        segData[node] = merge(segData[leftChild], segData[rightChild]);
    }


    Node query(int l, int r, int node, int lx, int rx) {
        propagate(node, lx, rx);

        if (l >= rx or lx >= r) return Node();
        if (l <= lx and rx <= r) {
            return segData[node];
        }

        Node left = query(l, r, leftChild, lx, mid);
        Node right = query(l, r, rightChild, mid, rx);

        return merge(left, right);
    }

public:
    SagaraWithLazy(int n, vector<int>& arr) {
        treeSize = 1;
        while (treeSize < n) treeSize <<= 1;
        segData = vector<Node>(treeSize << 1);
        build(0, 0, treeSize, arr);
    }

    void update(int l, int r, int x) {
        update(l, r, x, 0, 0, treeSize);
    }

    int query(int l, int r) {
        Node a = query(l, r, 0, 0, treeSize);
        return (a.prod ) % MOD;
    }
#undef LeftChild
#undef RightChild
#undef mid
};



void solve() {
    int n, q; cin >> n >> q;
    vector<int> a(n),b(n);
    for (int i = 0; i < n; i++) cin >> a[i],b[i]=spf[a[i]];

    SagaraWithLazy st(n, a);

    SagaraWithLazy2 st2(n,b);
    while (q--) {
        int t; cin >> t;
        if (t == 1) {
            int l, r; cin >> l >> r;
            l--, r--;
            int prod = st.query(l, r + 1);
            int sp = st2.query(l, r + 1);
            cout << (prod * inv(sp)) % MOD << endl;

        } else {
            int l, r, x; cin >> l >> r >> x;
            st.update(l-1, r, x);
            st2.update(l-1, r, spf[x]);

        }
    }
}


signed main() {
    ios_base::sync_with_stdio(false); 
    cin.tie(NULL); cout.tie(NULL);
    // #ifndef ONLINE_JUDGE 
    //    freopen("input.txt", "r", stdin); 
    //    freopen("output.txt", "w", stdout); 
    // #endif 
    int t; t = 1;
    sieve(1e7);
    cin >> t;
    while (t--) solve();
    return 0;
}