#include<bits/stdc++.h>
using namespace std;using ll=long long;const int maxN=1e7+5;
using ll = long long;
const int MOD = 1e9 + 7;
template<ll M>
struct modint {
static ll _pow(ll n, ll k) {
ll r = 1;
for (; k > 0; k >>= 1, n = (n*n)%M)
if (k&1) r = (r*n)%M;
return r;
}
ll v; modint(ll n = 0) : v(n%M) { v += (M&(0-(v<0))); }
friend string to_string(const modint n) { return to_string(n.v); }
friend istream& operator>>(istream& i, modint& n) { return i >> n.v; }
friend ostream& operator<<(ostream& o, const modint n) { return o << n.v; }
template<typename T> explicit operator T() { return T(v); }
friend bool operator==(const modint n, const modint m) { return n.v == m.v; }
friend bool operator!=(const modint n, const modint m) { return n.v != m.v; }
friend bool operator<(const modint n, const modint m) { return n.v < m.v; }
friend bool operator<=(const modint n, const modint m) { return n.v <= m.v; }
friend bool operator>(const modint n, const modint m) { return n.v > m.v; }
friend bool operator>=(const modint n, const modint m) { return n.v >= m.v; }
modint& operator+=(const modint n) { v += n.v; v -= (M&(0-(v>=M))); return *this; }
modint& operator-=(const modint n) { v -= n.v; v += (M&(0-(v<0))); return *this; }
modint& operator*=(const modint n) { v = (v*n.v)%M; return *this; }
modint& operator/=(const modint n) { v = (v*_pow(n.v, M-2))%M; return *this; }
friend modint operator+(const modint n, const modint m) { return modint(n) += m; }
friend modint operator-(const modint n, const modint m) { return modint(n) -= m; }
friend modint operator*(const modint n, const modint m) { return modint(n) *= m; }
friend modint operator/(const modint n, const modint m) { return modint(n) /= m; }
modint& operator++() { return *this += 1; }
modint& operator--() { return *this -= 1; }
modint operator++(int) { modint t = *this; return *this += 1, t; }
modint operator--(int) { modint t = *this; return *this -= 1, t; }
modint operator+() { return *this; }
modint operator-() { return modint(0) -= *this; }
modint pow(const ll k) const {
return k < 0 ? _pow(v, M-1-(-k%(M-1))) : _pow(v, k);
}
modint inv() const { return _pow(v, M-2); }
};
using mint = modint<int(MOD)>;
// check add
class segtree {
public:
struct node {
mint mul = 1;
mint add = 1;
void apply(int l, int r, mint x) {
mul = mul*(x.pow(r-l+1));
add = add*x;
}
};
node unite(const node &a, const node &b) const {
node res;
res.mul = a.mul*b.mul;
return res;
}
inline void push(int x, int l, int r) {
int y = (l + r) >> 1;
int z = x + ((y - l + 1) << 1);
if (tree[x].add != 1) { // it's wrong because of me
tree[x + 1].apply(l, y, tree[x].add);
tree[z].apply(y + 1, r, tree[x].add);
tree[x].add = 1;
}
}
inline void pull(int x, int z) {
tree[x] = unite(tree[x + 1], tree[z]);
}
int n;
vector<node> tree;
template <typename M>
void build(int x, int l, int r, const vector<M> &v) {
if (l == r) {
tree[x].apply(l, r, v[l]);
return;
}
int y = (l + r) >> 1;
int z = x + ((y - l + 1) << 1);
build(x + 1, l, y, v);
build(z, y + 1, r, v);
pull(x, z);
}
node query(int x, int l, int r, int LL, int rr) {
if (LL <= l && r <= rr) {
return tree[x];
}
int y = (l + r) >> 1;
int z = x + ((y - l + 1) << 1);
push(x, l, r);
node res{};
if (rr <= y) {
res = query(x + 1, l, y, LL, rr);
}
else{
if (LL > y) {
res = query(z, y + 1, r, LL, rr);
}
else{
res = unite(query(x + 1, l, y, LL, rr), query(z, y + 1, r, LL, rr));
}
}
pull(x, z);
return res;
}
template <typename... M>
void update(int x, int l, int r, int LL, int rr, const M&... v) {
if (LL <= l && r <= rr) {
tree[x].apply(l, r, v...);
return;
}
int y = (l + r) >> 1;
int z = x + ((y - l + 1) << 1);
push(x, l, r);
if (LL <= y) {
update(x + 1, l, y, LL, rr, v...);
}
if (rr > y) {
update(z, y + 1, r, LL, rr, v...);
}
pull(x, z);
}
template <typename M>
segtree(const vector<M> &v) {
n = v.size();
assert(n > 0);
tree.resize(2 * n - 1);
build(0, 0, n - 1, v);
}
node query(int LL, int rr) {
if(LL > rr)
return {};
assert(0 <= LL && LL <= rr && rr <= n - 1);
return query(0, 0, n - 1, LL, rr);
}
template <typename... M>
void update(int LL, int rr, const M&... v) {
assert(0 <= LL && LL <= rr && rr <= n - 1);
update(0, 0, n - 1, LL, rr, v...);
}
};
class segtree2 {
public:
struct node {
int mn = 1e9;
int add = 1e9;
void apply(int l, int r, int x) {
mn = min(mn,x);
add = min(add,x);
}
};
node unite(const node &a, const node &b) const {
node res;
res.mn=min(a.mn,b.mn);
return res;
}
inline void push(int x, int l, int r) {
int y = (l + r) >> 1;
int z = x + ((y - l + 1) << 1);
if (tree[x].add != 1e9) { // it's wrong because of me
tree[x + 1].apply(l, y, tree[x].add);
tree[z].apply(y + 1, r, tree[x].add);
tree[x].add = 1e9;
}
}
inline void pull(int x, int z) {
tree[x] = unite(tree[x + 1], tree[z]);
}
int n;
vector<node> tree;
template <typename M>
void build(int x, int l, int r, const vector<M> &v) {
if (l == r) {
tree[x].apply(l, r, v[l]);
return;
}
int y = (l + r) >> 1;
int z = x + ((y - l + 1) << 1);
build(x + 1, l, y, v);
build(z, y + 1, r, v);
pull(x, z);
}
node query(int x, int l, int r, int LL, int rr) {
if (LL <= l && r <= rr) {
return tree[x];
}
int y = (l + r) >> 1;
int z = x + ((y - l + 1) << 1);
push(x, l, r);
node res{};
if (rr <= y) {
res = query(x + 1, l, y, LL, rr);
}
else{
if (LL > y) {
res = query(z, y + 1, r, LL, rr);
}
else{
res = unite(query(x + 1, l, y, LL, rr), query(z, y + 1, r, LL, rr));
}
}
pull(x, z);
return res;
}
template <typename... M>
void update(int x, int l, int r, int LL, int rr, int val) {
// if(tree[x].mn<=val){
// }
if (LL <= l && r <= rr) {
tree[x].apply(l, r, val);
return;
}
int y = (l + r) >> 1;
int z = x + ((y - l + 1) << 1);
push(x, l, r);
if (LL <= y) {
update(x + 1, l, y, LL, rr, val);
}
if (rr > y) {
update(z, y + 1, r, LL, rr, val);
}
pull(x, z);
}
template <typename M>
segtree2(const vector<M> &v) {
n = v.size();
assert(n > 0);
tree.resize(2 * n - 1);
build(0, 0, n - 1, v);
}
node query(int LL, int rr) {
if(LL > rr)
return {};
assert(0 <= LL && LL <= rr && rr <= n - 1);
return query(0, 0, n - 1, LL, rr);
}
template <typename... M>
void update(int LL, int rr, int x) {
assert(0 <= LL && LL <= rr && rr <= n - 1);
update(0, 0, n - 1, LL, rr, x);
}
};
int spf[maxN],lst[maxN];
vector<int> prms;
void pre(){
for(int i = 2;i<maxN;++i){
if(spf[i]==0){
spf[i] = i;
prms.push_back(i);
}
for(auto&p:prms){
if(p*i>=maxN||p>spf[i]) break;
spf[p*i]=p;
}
}
for(int i=1;i<maxN;++i){
if(spf[i]==i) lst[i] = i;
else lst[i] = lst[i-1];
}
}
void solve(){
int n;cin>>n;
int q;cin>>q;
vector<int> a(n+1), b(n+1);
for(int i=1;i<=n;++i){
cin>>a[i];
b[i]=spf[a[i]];
}
segtree st1(a);
segtree2 st2(b);
while(q--){
ll type, l,r;cin>>type>>l>>r;
if(type==1){
mint mul = st1.query(l,r).mul;
mint mn = st2.query(l,r).mn;
mul/=mn;
// cout << mul << " " << mn << '\n';
cout<<mul<<'\n';
}
else{
int x;cin>>x;
st1.update(l,r,x);
st2.update(l,r,spf[x]);
}
}
}
int32_t main(){ios_base::sync_with_stdio(0); cin.tie(0);
pre();
int tt=1;cin>>tt;while(tt--)
solve();return 0;}
#include<bits/stdc++.h>
using namespace std;using ll=long long;const int maxN=1e7+5;
using ll = long long;
const int MOD = 1e9 + 7;
 
template<ll M>
struct modint {
 
    static ll _pow(ll n, ll k) {
        ll r = 1;
        for (; k > 0; k >>= 1, n = (n*n)%M)
            if (k&1) r = (r*n)%M;
        return r;
    }
 
    ll v; modint(ll n = 0) : v(n%M) { v += (M&(0-(v<0))); }
    
    friend string to_string(const modint n) { return to_string(n.v); }
    friend istream& operator>>(istream& i, modint& n) { return i >> n.v; }
    friend ostream& operator<<(ostream& o, const modint n) { return o << n.v; }
    template<typename T> explicit operator T() { return T(v); }
 
    friend bool operator==(const modint n, const modint m) { return n.v == m.v; }
    friend bool operator!=(const modint n, const modint m) { return n.v != m.v; }
    friend bool operator<(const modint n, const modint m) { return n.v < m.v; }
    friend bool operator<=(const modint n, const modint m) { return n.v <= m.v; }
    friend bool operator>(const modint n, const modint m) { return n.v > m.v; }
    friend bool operator>=(const modint n, const modint m) { return n.v >= m.v; }
 
    modint& operator+=(const modint n) { v += n.v; v -= (M&(0-(v>=M))); return *this; }
    modint& operator-=(const modint n) { v -= n.v; v += (M&(0-(v<0))); return *this; }
    modint& operator*=(const modint n) { v = (v*n.v)%M; return *this; }
    modint& operator/=(const modint n) { v = (v*_pow(n.v, M-2))%M; return *this; }
    friend modint operator+(const modint n, const modint m) { return modint(n) += m; }
    friend modint operator-(const modint n, const modint m) { return modint(n) -= m; }
    friend modint operator*(const modint n, const modint m) { return modint(n) *= m; }
    friend modint operator/(const modint n, const modint m) { return modint(n) /= m; }
    modint& operator++() { return *this += 1; }
    modint& operator--() { return *this -= 1; }
    modint operator++(int) { modint t = *this; return *this += 1, t; }
    modint operator--(int) { modint t = *this; return *this -= 1, t; }
    modint operator+() { return *this; }
    modint operator-() { return modint(0) -= *this; }
 
    modint pow(const ll k) const {
        return k < 0 ? _pow(v, M-1-(-k%(M-1))) : _pow(v, k);
    }
    modint inv() const { return _pow(v, M-2); }
};
 
using mint = modint<int(MOD)>;
// check add
class segtree {
    public:
    struct node {
        mint mul = 1;
        mint add = 1;

        void apply(int l, int r, mint x) {
            mul = mul*(x.pow(r-l+1));
            add = add*x;
        }
    };
 
    node unite(const node &a, const node &b) const {
        node res;
        res.mul = a.mul*b.mul;
        return res;
    }
 
    inline void push(int x, int l, int r) {
        int y = (l + r) >> 1;
        int z = x + ((y - l + 1) << 1);
        if (tree[x].add != 1) { // it's wrong because of me
            tree[x + 1].apply(l, y, tree[x].add);
            tree[z].apply(y + 1, r, tree[x].add);
            tree[x].add = 1;
        }
    }
 
    inline void pull(int x, int z) {
        tree[x] = unite(tree[x + 1], tree[z]);
    }
 
    int n;
    vector<node> tree;
 
    template <typename M>
    void build(int x, int l, int r, const vector<M> &v) {
        if (l == r) {
            tree[x].apply(l, r, v[l]);
            return;
        }
        int y = (l + r) >> 1;
        int z = x + ((y - l + 1) << 1);
        build(x + 1, l, y, v);
        build(z, y + 1, r, v);
        pull(x, z);
    }
 
    node query(int x, int l, int r, int LL, int rr) {
        if (LL <= l && r <= rr) {
            return tree[x];
        }
        int y = (l + r) >> 1;
        int z = x + ((y - l + 1) << 1);
        push(x, l, r);
        node res{};
        if (rr <= y) {
            res = query(x + 1, l, y, LL, rr);
        } 
        else{
            if (LL > y) {
                res = query(z, y + 1, r, LL, rr);
            } 
            else{
                res = unite(query(x + 1, l, y, LL, rr), query(z, y + 1, r, LL, rr));
            }
        }
        pull(x, z);
        return res;
    }
 
    template <typename... M>
    void update(int x, int l, int r, int LL, int rr, const M&... v) {
        if (LL <= l && r <= rr) {
            tree[x].apply(l, r, v...);
            return;
        }
        int y = (l + r) >> 1;
        int z = x + ((y - l + 1) << 1);
        push(x, l, r);
        if (LL <= y) {
            update(x + 1, l, y, LL, rr, v...);
        }
        if (rr > y) {
            update(z, y + 1, r, LL, rr, v...);
        }
        pull(x, z);
    }
    
    template <typename M>
    segtree(const vector<M> &v) {
        n = v.size();
        assert(n > 0);
        tree.resize(2 * n - 1);
        build(0, 0, n - 1, v);
    }
 
    node query(int LL, int rr) {
        if(LL > rr)
            return {};
        assert(0 <= LL && LL <= rr && rr <= n - 1);
        return query(0, 0, n - 1, LL, rr);
    }
 
    template <typename... M>
    void update(int LL, int rr, const M&... v) {
        assert(0 <= LL && LL <= rr && rr <= n - 1);
        update(0, 0, n - 1, LL, rr, v...);
    }
};
class segtree2 {
    public:
    struct node {
        int mn = 1e9;
        int add = 1e9;
        void apply(int l, int r, int x) {
            mn = min(mn,x);
            add = min(add,x);
        }
    };
 
    node unite(const node &a, const node &b) const {
        node res;
        res.mn=min(a.mn,b.mn);
        return res;
    }
 
    inline void push(int x, int l, int r) {
        int y = (l + r) >> 1;
        int z = x + ((y - l + 1) << 1);
        if (tree[x].add != 1e9) { // it's wrong because of me
            tree[x + 1].apply(l, y, tree[x].add);
            tree[z].apply(y + 1, r, tree[x].add);
            tree[x].add = 1e9;
        }
    }
 
    inline void pull(int x, int z) {
        tree[x] = unite(tree[x + 1], tree[z]);
    }
 
    int n;
    vector<node> tree;
 
    template <typename M>
    void build(int x, int l, int r, const vector<M> &v) {
        if (l == r) {
            tree[x].apply(l, r, v[l]);
            return;
        }
        int y = (l + r) >> 1;
        int z = x + ((y - l + 1) << 1);
        build(x + 1, l, y, v);
        build(z, y + 1, r, v);
        pull(x, z);
    }
 
    node query(int x, int l, int r, int LL, int rr) {
        if (LL <= l && r <= rr) {
            return tree[x];
        }
        int y = (l + r) >> 1;
        int z = x + ((y - l + 1) << 1);
        push(x, l, r);
        node res{};
        if (rr <= y) {
            res = query(x + 1, l, y, LL, rr);
        } 
        else{
            if (LL > y) {
                res = query(z, y + 1, r, LL, rr);
            } 
            else{
                res = unite(query(x + 1, l, y, LL, rr), query(z, y + 1, r, LL, rr));
            }
        }
        pull(x, z);
        return res;
    }
 
    template <typename... M>
    void update(int x, int l, int r, int LL, int rr, int val) {
        // if(tree[x].mn<=val){
        // }
        if (LL <= l && r <= rr) {
            tree[x].apply(l, r, val);
            return;
        }
        int y = (l + r) >> 1;
        int z = x + ((y - l + 1) << 1);
        push(x, l, r);
        if (LL <= y) {
            update(x + 1, l, y, LL, rr, val);
        }
        if (rr > y) {
            update(z, y + 1, r, LL, rr, val);
        }
        pull(x, z);
    }
    
    template <typename M>
    segtree2(const vector<M> &v) {
        n = v.size();
        assert(n > 0);
        tree.resize(2 * n - 1);
        build(0, 0, n - 1, v);
    }
 
    node query(int LL, int rr) {
        if(LL > rr)
            return {};
        assert(0 <= LL && LL <= rr && rr <= n - 1);
        return query(0, 0, n - 1, LL, rr);
    }
 
    template <typename... M>
    void update(int LL, int rr, int x) {
        assert(0 <= LL && LL <= rr && rr <= n - 1);
        update(0, 0, n - 1, LL, rr, x);
    }
};
int spf[maxN],lst[maxN];
vector<int> prms;
void pre(){
    for(int i = 2;i<maxN;++i){
        if(spf[i]==0){
            spf[i] = i;
            prms.push_back(i);
        }
        for(auto&p:prms){
            if(p*i>=maxN||p>spf[i]) break;
            spf[p*i]=p;
        }
    }
    for(int i=1;i<maxN;++i){
        if(spf[i]==i) lst[i] = i;
        else lst[i] = lst[i-1];
    }
}
void solve(){
    int n;cin>>n;
    int q;cin>>q;
    vector<int> a(n+1), b(n+1);
    for(int i=1;i<=n;++i){
        cin>>a[i];
        b[i]=spf[a[i]];
    }
    segtree st1(a);
    segtree2 st2(b);

    while(q--){
        ll type, l,r;cin>>type>>l>>r;
        if(type==1){
            mint mul = st1.query(l,r).mul;
            mint mn = st2.query(l,r).mn;
            mul/=mn;
            // cout << mul << " " << mn << '\n';
            cout<<mul<<'\n';
        }
        else{
            int x;cin>>x;
            st1.update(l,r,x);
            st2.update(l,r,spf[x]);
        }
    }
}
int32_t main(){ios_base::sync_with_stdio(0); cin.tie(0);
pre();
int tt=1;cin>>tt;while(tt--)
solve();return 0;}