The101Bypass

Globe The 101 developer and openwrt bypass

// ==UserScript==
// @name         The101Bypass
// @namespace    http://tampermonkey.net/
// @version      2025-10-07
// @description  Globe The 101 developer and openwrt bypass
// @author       Rouel
// @match        *://192.168.0.1/*
// @icon         
// @grant        none
// @run-at       document-start
// ==/UserScript==


(function () {
    'use strict';

    const editedKey = 'edited'
    const devResponse = { "data": { "develop_mode": 1 }, "code": 0 }
    const encodedDevUrl = new URL("data:application/json," + encodeURIComponent(JSON.stringify(devResponse)))
    const devPath = '/cgi-bin/luci/admin/jt_system/get_develop_mode'

    function isCgi(u) {
        return u.pathname.startsWith("/cgi-bin/")
    }

    function updateSearchParams(u) {
        const iscgi = isCgi(u)
        if (!iscgi) return
        u.searchParams.set("flag", "get_develop_mode")
        u.searchParams.set("t", new Date().getTime())
    }

    function addOWRTLink() {
        console.log("add openwrt link")
        const element = document.evaluate("/html/body/div/div/section/div/div/div[3]/div[1]/div[2]/div/div[2]/div", document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue

        if (!element) {
            setTimeout(addOWRTLink, 1000)
            return
        }

        const link = document.createElement("a")
        link.href = "/cgi-bin/luci/admin/status/overview?flag=get_develop_mode&t="
        link.innerHTML = "OWRT"
        link.classList.add("el-col", "el-col-6")
        link.style = 'padding-left: 15px; padding-right: 30px; font-size: 16px; color: "rgb(0, 119, 200)"'
        element.appendChild(link)

        // dumb style
        document.evaluate("/html/body/div/div/section/div/div/div[3]/div[1]/div[2]/div/div[2]/div", document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue.style.marginLeft = '-50px'
    }

    function rewriteUrls() {
        // update urls
        console.log("update links")

        document.querySelectorAll(`a:not([${editedKey}])`).forEach(a => {
            a.setAttribute(editedKey, '')
            if (!a.href) return
            const u = new URL(a.href)
            if (!isCgi(u)) return
            updateSearchParams(u)
            a.href = u.toString()
        })

        document.querySelectorAll(`[onclick]:not([${editedKey}])`).forEach((a) => {
            a.setAttribute(editedKey, '')
            const f = a.onclick.toString();
            if (typeof f !== "string") return;
            const m = f.match(/'(\/cgi-bin\/.*)'/)
            console.log({ m })
            if (!m) return

            const prevClick = a.onclick

            a.onclick = () => {
                prevClick()
                const newUrl = new URL(m[1], window.location.protocol + "//" + window.location.hostname)
                updateSearchParams(newUrl)
                location.href = newUrl.toString()
            }
        });
    }


    function updateLuci() {
        rewriteUrls()
    }

    if (!isCgi(new URL(window.location))) {
        addOWRTLink()
    } else {
        updateLuci()
        setInterval(() => updateLuci(), 1000)
    }


    //////////////////////////////////////////////////////////////////////////////////
    //////////////////////////////////////////////////////////////////////////////////
    // fetch interceptor
    window.nativeFetch = window.fetch;

    window.customFetch = async (request, headers) => {
        console.log('fetch')
        var req;
        var response;

        if (typeof request == 'string') {
            const u = new URL(request)
            updateSearchParams(u)
            req = new Request(u.toString(), headers);
            response = await window.nativeFetch(req);
            response.requestInputObject = req;
        } else {
            const u = new URL(request.url)
            updateSearchParams(u)
            const newRequest = new Request(u.toString(), request)
            response = await window.nativeFetch(request, headers);
        }

        if (typeof request == 'object') {
            response.requestInputObject = request;
        } else {
            response.requestInputURL = request;
            response.requestInputObject = req;

        }

        if (headers) { response.requestInputHeaders = headers; }

        return response;
    }

    window.fetch = window.customFetch

    //////////////////////////////////////////////////////////////////////////////////
    //////////////////////////////////////////////////////////////////////////////////
    // xml interceptor


    function xmlCustomRequest(method, url, _async, data) {
        console.log("xml")
        let u = new URL(url, window.location.protocol + "//" + window.location.hostname)

        if (url.startsWith(devPath)) { u = encodedDevUrl; }

        updateSearchParams(u)
        console.log({ method, url: u.toString(), _async, data })
        this.method = method;
        this.requestURL = u.toString();
        this.asynch = _async;
        this.requestHeaders = new Map();
        return this.nativeOpen(method, u.toString(), _async);
    }

    XMLHttpRequest.prototype.nativeOpen = XMLHttpRequest.prototype.open;
    XMLHttpRequest.prototype.customOpen = xmlCustomRequest
    XMLHttpRequest.prototype.open = XMLHttpRequest.prototype.customOpen;

    //////////////////////////////////////////////////////////////////////////////////

    XMLHttpRequest.prototype.nativeSend = XMLHttpRequest.prototype.send;
    XMLHttpRequest.prototype.customSend = function (data) {
        console.log({ data, 'send': 1, url: this.url, type: typeof data })
        return XMLHttpRequest.prototype.nativeSend.apply(this, [data]);
    }
    XMLHttpRequest.prototype.send = XMLHttpRequest.prototype.customSend
})();