跳转到内容

MediaWiki:Gadget-abusefilter33test.js:修订间差异

来自萌娘共享
代码变动:103d1a56 - feat: rename (#594) by U:AnnAngela, co-authored-by: GH:github-actions[bot]
标签由机器人或全自动脚本执行的操作
代码变动:103d1a56 - feat: rename (#594) by U:AnnAngela, co-authored-by: GH:github-actions[bot]
标签由机器人或全自动脚本执行的操作
第8行: 第8行:
/* <pre> */
/* <pre> */


// <pre>
"use strict";
"use strict";
function _array_like_to_array(arr, len) {
$(() => (async () => {
    if (len == null || len > arr.length) len = arr.length;
     if (!mw.config.get("wgUserGroups").includes("sysop") && !mw.config.get("wgUserGroups").includes("staff")) {
    for(var i = 0, arr2 = new Array(len); i < len; i++)arr2[i] = arr[i];
         return;
    return arr2;
    }
}
     if (mw.config.get("wgCanonicalSpecialPageName") !== "AbuseLog") {
function _array_without_holes(arr) {
     if (Array.isArray(arr)) return _array_like_to_array(arr);
}
function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) {
    try {
         var info = gen[key](arg);
        var value = info.value;
     } catch (error) {
        reject(error);
         return;
         return;
     }
     }
     if (info.done) {
     const getLogVar = (varName) => $(`.mw-abuselog-details-${varName} .mw-abuselog-var-value .mw-abuselog-var-value`).text();
        resolve(value);
    const timestampVar = getLogVar("timestamp");
     } else {
    const userName = getLogVar("user_name");
        Promise.resolve(value).then(_next, _throw);
    const articlePrefixedtext = getLogVar("article_prefixedtext");
     const filter = [...document.querySelectorAll('a[href*="/Special:%E6%BB%A5%E7%94%A8%E8%BF%87%E6%BB%A4%E5%99%A8/"]')].filter(({ href }) => /%E6%BB%A5%E7%94%A8%E8%BF%87%E6%BB%A4%E5%99%A8\/[1-9]\d*$/.test(href))?.[0]?.href?.match?.(/[1-9]\d*$/)?.[0];
    if (timestampVar.length === 0 || userName.length === 0 || articlePrefixedtext.length === 0 || typeof filter !== "string") {
        return;
     }
     }
}
    const symbolEnter = (str) => typeof str === "string" ? str.replace(/\n/g, "↵") : str;
function _async_to_generator(fn) {
    const api = new mw.Api();
    return function() {
    const pTable = $("<table/>");
         var self = this, args = arguments;
    $("#mw-content-text > fieldset > p:first").after(pTable.css("width", "100%").addClass("wikitable"));
         return new Promise(function(resolve, reject) {
    pTable.empty().append("<tbody><tr><td style=\"text-align: center;\">加载中(<span id=\"abusefiltertest-progress\">0</span>/2)……</td></tr></tbody>");
             var gen = fn.apply(self, args);
    const progress = $("#abusefiltertest-progress");
             function _next(value) {
    try {
                 asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value);
        const __details = (await api.post({
            action: "query",
            assertuser: mw.config.get("wgUserName"),
            list: "abuselog",
            afluser: userName,
            afltitle: articlePrefixedtext,
            aflprop: "details",
            aflfilter: +filter,
        })).query.abuselog;
        if (__details.length === 0) {
            throw "无法找到对应用户名和页面标题的滥用过滤器日志";
        }
        progress.text(1);
        const _details = Array.from(new Map(__details.filter(({ details: { timestamp } }) => timestampVar === timestamp).map((n) => [n.new_wikitext, n])).values());
        if (_details.length === 0) {
            console.info(__details);
            throw "无法找到对应时间戳的滥用过滤器日志";
         }
        progress.text(2);
        const _rules = (await api.post({
            action: "query",
            assertuser: mw.config.get("wgUserName"),
            list: "abusefilters",
            abfstartid: +filter,
            abfendid: +filter,
            abfprop: "pattern",
        })).query.abusefilters[0].pattern.replace(/\r/g, "");
        const _stringRules = Array.from(_rules.match(/\/\* string rule start \*\/[\s\S]+?(?=\/\* string rule end \*\/)/g) || []).map((r) => r.replace(/\/\* regex rule start \*\//g, "").split("\n")).flat().map((r) => r.replace(/^ *"?/, "").replace(/"?,? *$/, "")).filter((r) => r !== "");
         const stringRules = new Set(_stringRules);
        const _regexRules = Array.from(_rules.match(/\/\* regex rule start \*\/[\s\S]+?(?=\/\* regex rule end \*\/)/g) || []).map((r) => r.replace(/\/\* regex rule start \*\//g, "").split("\n") || []).flat().map((r) => r.replace(/^ *"?/, "").replace(/"?,? *$/, "")).filter((r) => r !== "");
        const regexRules = new Set(_regexRules);
        if (stringRules.size + regexRules.size === 0) {
            throw "无法找到符合格式的规则";
        }
        for (const { details } of _details) {
             const addedLines = details.added_lines || details.new_wikitext;
            const result = [];
            const table = $("<table/>").css("width", "100%").addClass("wikitable abusefiltertest").hide();
             pTable.after(table);
            for (const sr of stringRules.values()) {
                 if (addedLines.includes(sr)) {
                    const splits = addedLines.split(sr);
                    splits.forEach((str, i) => {
                        if (i === splits.length - 1) {
                            return;
                        }
                        const next = splits[i + 1];
                        const start = Math.max(0, str.length - 20);
                        const end = Math.min(next.length, 20);
                        result.push({
                            isRegexp: false,
                            rule: sr,
                            before: str.substring(start, start + 20),
                            string: sr,
                            after: next.substring(0, end),
                            start,
                        });
                    });
                }
             }
             }
             function _throw(err) {
             for (const rr of regexRules.values()) {
                 asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err);
                 const regex = RegExp(`.{0,20}${rr}.{0,20}`, "ig");
                const findstring = RegExp(rr, "ig");
                if (regex.test(addedLines)) {
                    const splits = addedLines.match(regex);
                    let lastFromIndex = 0;
                    splits.forEach((str) => {
                        const string = str.match(findstring)[0];
                        const split = str.split(string);
                        const before = split[0];
                        const after = split.slice(1).join(string);
                        const start = addedLines.indexOf(string, lastFromIndex);
                        lastFromIndex = start + string.length;
                        result.push({
                            isRegex: true,
                            rule: rr,
                            before,
                            string,
                            after,
                            start,
                        });
                    });
                }
             }
             }
             _next(undefined);
             table.empty();
        });
            if (result.length > 0) {
    };
                table.append(`<caption style="font-weight: 700;">命中防滥用过滤器${filter}的内容</caption><tr><th style="text-align: right;">规则(请勿泄漏!!!)</th><th style="text-align: left;">命中字符串上下文</th></tr>`);
}
                result.sort(({ start: a }, { start: b }) => a - b);
function _iterable_to_array(iter) {
                const results = [];
    if (typeof Symbol !== "undefined" && iter[Symbol.iterator] != null || iter["@@iterator"] != null) return Array.from(iter);
                result.forEach((r) => {
}
                    if (results.filter(({ isRegex, start }) => isRegex === r.isRegex && start === r.start).length === 0) {
function _non_iterable_spread() {
                         results.push(r);
    throw new TypeError("Invalid attempt to spread non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
}
function _to_consumable_array(arr) {
    return _array_without_holes(arr) || _iterable_to_array(arr) || _unsupported_iterable_to_array(arr) || _non_iterable_spread();
}
function _unsupported_iterable_to_array(o, minLen) {
    if (!o) return;
    if (typeof o === "string") return _array_like_to_array(o, minLen);
    var n = Object.prototype.toString.call(o).slice(8, -1);
    if (n === "Object" && o.constructor) n = o.constructor.name;
    if (n === "Map" || n === "Set") return Array.from(n);
    if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _array_like_to_array(o, minLen);
}
function _ts_generator(thisArg, body) {
    var f, y, t, _ = {
        label: 0,
        sent: function() {
            if (t[0] & 1) throw t[1];
            return t[1];
        },
        trys: [],
        ops: []
    }, g = Object.create((typeof Iterator === "function" ? Iterator : Object).prototype);
    return g.next = verb(0), g["throw"] = verb(1), g["return"] = verb(2), typeof Symbol === "function" && (g[Symbol.iterator] = function() {
        return this;
    }), g;
    function verb(n) {
        return function(v) {
            return step([
                n,
                v
            ]);
        };
    }
    function step(op) {
        if (f) throw new TypeError("Generator is already executing.");
        while(g && (g = 0, op[0] && (_ = 0)), _)try {
            if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
            if (y = 0, t) op = [
                op[0] & 2,
                t.value
            ];
            switch(op[0]){
                case 0:
                case 1:
                    t = op;
                    break;
                case 4:
                    _.label++;
                    return {
                        value: op[1],
                        done: false
                    };
                case 5:
                    _.label++;
                    y = op[1];
                    op = [
                        0
                    ];
                    continue;
                case 7:
                    op = _.ops.pop();
                    _.trys.pop();
                    continue;
                default:
                    if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) {
                         _ = 0;
                        continue;
                     }
                     }
                     if (op[0] === 3 && (!t || op[1] > t[0] && op[1] < t[3])) {
                });
                         _.label = op[1];
                console.info(results);
                         break;
                results.forEach(({ isRegex, rule, before, string, after, start }) => {
                    const tr = $("<tr/>");
                    const label = $("<td/>");
                    const ruleCode = $("<code/>");
                    ruleCode.text(rule);
                    label.addClass("mw-label");
                    label.text(isRegex ? "正则表达式" : "字符串");
                    label.append(symbolEnter(ruleCode)).append(":");
                     tr.append(label);
                    const input = $("<td/>");
                    input.addClass("mw-input");
                    const beforeSpan = $("<span/>");
                    beforeSpan.text(symbolEnter(before)).css({
                        padding: "0 .25em",
                    });
                    const strSpan = $("<span/>");
                    strSpan.text(symbolEnter(string)).css({
                        "font-weight": "700",
                        "text-decoration": "underline",
                        padding: "0 .25em",
                    });
                    const afterSpan = $("<span/>");
                    afterSpan.text(symbolEnter(after)).css({
                         padding: "0 .25em",
                    });
                    if (before.length >= 20) {
                        const beforeDot = $("<span/>");
                        beforeDot.text(`[${start}] ……`).css({
                            "padding-right": ".25em",
                        });
                         input.append(beforeDot);
                     }
                     }
                     if (op[0] === 6 && _.label < t[1]) {
                    input.append(beforeSpan).append(strSpan).append(afterSpan);
                        _.label = t[1];
                     if (after.length >= 20) {
                         t = op;
                        const afterDot = $("<span/>");
                         break;
                        afterDot.text("……").css({
                            "padding-left": ".25em",
                         });
                         input.append(afterDot);
                     }
                     }
                     if (t && _.label < t[2]) {
                     tr.append(input);
                        _.label = t[2];
                    table.append(tr);
                        _.ops.push(op);
                });
                        break;
            }
                    }
            else {
                    if (t[2]) _.ops.pop();
                table.empty().append("<tbody><tr><td style=\"text-align: center;\">现有规则没有匹配项……</td></tr></tbody>");
                    _.trys.pop();
                    continue;
             }
             }
            op = body.call(thisArg, _);
        } catch (e) {
            op = [
                6,
                e
            ];
            y = 0;
        } finally{
            f = t = 0;
         }
         }
         if (op[0] & 5) throw op[1];
         pTable.hide();
         return {
        $(".abusefiltertest").show();
             value: op[0] ? op[1] : void 0,
    }
            done: true
    catch (e) {
         };
        console.info("abusefiltertest", e);
        $(".abusefiltertest").remove();
        pTable.empty();
         if (filter === 33) {
             pTable.append(`<tbody><tr><td style="text-align: ${typeof e === "string" ? "center" : "left"};">发生错误:${typeof e === "string" ? e : `${e} ${e.stack.split("\n")[1].trim()}`}</td></tr></tbody>`);
         }
     }
     }
}
})());  
$(function() {
    return function() {
        return _async_to_generator(function() {
            var _filter__href_match, _filter__href_match1, _filter__href, _filter_, _filter, getLogVar, timestampVar, userName, articlePrefixedtext, filter, symbolEnter, api, pTable, progress, __details, _details, _rules, _stringRules, stringRules, _regexRules, regexRules, _iteratorNormalCompletion, _didIteratorError, _iteratorError, _loop, _iterator, _step, e;
            return _ts_generator(this, function(_state) {
                switch(_state.label){
                    case 0:
                        if (!mw.config.get("wgUserGroups").includes("sysop") && !mw.config.get("wgUserGroups").includes("staff")) {
                            return [
                                2
                            ];
                        }
                        if (mw.config.get("wgCanonicalSpecialPageName") !== "AbuseLog") {
                            return [
                                2
                            ];
                        }
                        getLogVar = function(varName) {
                            return $(".mw-abuselog-details-".concat(varName, " .mw-abuselog-var-value .mw-abuselog-var-value")).text();
                        };
                        timestampVar = getLogVar("timestamp");
                        userName = getLogVar("user_name");
                        articlePrefixedtext = getLogVar("article_prefixedtext");
                        filter = (_filter = _to_consumable_array(document.querySelectorAll('a[href*="/Special:%E6%BB%A5%E7%94%A8%E8%BF%87%E6%BB%A4%E5%99%A8/"]')).filter(function(param) {
                            var href = param.href;
                            return /%E6%BB%A5%E7%94%A8%E8%BF%87%E6%BB%A4%E5%99%A8\/[1-9]\d*$/.test(href);
                        })) === null || _filter === void 0 ? void 0 : (_filter_ = _filter[0]) === null || _filter_ === void 0 ? void 0 : (_filter__href = _filter_.href) === null || _filter__href === void 0 ? void 0 : (_filter__href_match1 = _filter__href.match) === null || _filter__href_match1 === void 0 ? void 0 : (_filter__href_match = _filter__href_match1.call(_filter__href, /[1-9]\d*$/)) === null || _filter__href_match === void 0 ? void 0 : _filter__href_match[0];
                        if (timestampVar.length === 0 || userName.length === 0 || articlePrefixedtext.length === 0 || typeof filter !== "string") {
                            return [
                                2
                            ];
                        }
                        symbolEnter = function(str) {
                            return typeof str === "string" ? str.replace(/\n/g, "↵") : str;
                        };
                        api = new mw.Api();
                        pTable = $("<table/>");
                        $("#mw-content-text > fieldset > p:first").after(pTable.css("width", "100%").addClass("wikitable"));
                        pTable.empty().append("<tbody><tr><td style=\"text-align: center;\">加载中(<span id=\"abusefiltertest-progress\">0</span>/2)……</td></tr></tbody>");
                        progress = $("#abusefiltertest-progress");
                        _state.label = 1;
                    case 1:
                        _state.trys.push([
                            1,
                            4,
                            ,
                            5
                        ]);
                        return [
                            4,
                            api.post({
                                action: "query",
                                assertuser: mw.config.get("wgUserName"),
                                list: "abuselog",
                                afluser: userName,
                                afltitle: articlePrefixedtext,
                                aflprop: "details",
                                aflfilter: +filter
                            })
                        ];
                    case 2:
                        __details = _state.sent().query.abuselog;
                        if (__details.length === 0) {
                            throw "无法找到对应用户名和页面标题的滥用过滤器日志";
                        }
                        progress.text(1);
                        _details = Array.from(new Map(__details.filter(function(param) {
                            var timestamp = param.details.timestamp;
                            return timestampVar === timestamp;
                        }).map(function(n) {
                            return [
                                n.new_wikitext,
                                n
                            ];
                        })).values());
                        if (_details.length === 0) {
                            console.info(__details);
                            throw "无法找到对应时间戳的滥用过滤器日志";
                        }
                        progress.text(2);
                        return [
                            4,
                            api.post({
                                action: "query",
                                assertuser: mw.config.get("wgUserName"),
                                list: "abusefilters",
                                abfstartid: +filter,
                                abfendid: +filter,
                                abfprop: "pattern"
                            })
                        ];
                    case 3:
                        _rules = _state.sent().query.abusefilters[0].pattern.replace(/\r/g, "");
                        _stringRules = Array.from(_rules.match(/\/\* string rule start \*\/[\s\S]+?(?=\/\* string rule end \*\/)/g) || []).map(function(r) {
                            return r.replace(/\/\* regex rule start \*\//g, "").split("\n");
                        }).flat().map(function(r) {
                            return r.replace(/^ *"?/, "").replace(/"?,? *$/, "");
                        }).filter(function(r) {
                            return r !== "";
                        });
                        stringRules = new Set(_stringRules);
                        _regexRules = Array.from(_rules.match(/\/\* regex rule start \*\/[\s\S]+?(?=\/\* regex rule end \*\/)/g) || []).map(function(r) {
                            return r.replace(/\/\* regex rule start \*\//g, "").split("\n") || [];
                        }).flat().map(function(r) {
                            return r.replace(/^ *"?/, "").replace(/"?,? *$/, "");
                        }).filter(function(r) {
                            return r !== "";
                        });
                        regexRules = new Set(_regexRules);
                        if (stringRules.size + regexRules.size === 0) {
                            throw "无法找到符合格式的规则";
                        }
                        _iteratorNormalCompletion = true, _didIteratorError = false, _iteratorError = undefined;
                        try {
                            _loop = function() {
                                var details = _step.value.details;
                                var addedLines = details.added_lines || details.new_wikitext;
                                var result = [];
                                var table = $("<table/>").css("width", "100%").addClass("wikitable abusefiltertest").hide();
                                pTable.after(table);
                                var _iteratorNormalCompletion = true, _didIteratorError = false, _iteratorError = undefined;
                                try {
                                    var _loop = function() {
                                        var sr = _step1.value;
                                        if (addedLines.includes(sr)) {
                                            var splits = addedLines.split(sr);
                                            splits.forEach(function(str, i) {
                                                if (i === splits.length - 1) {
                                                    return;
                                                }
                                                var next = splits[i + 1];
                                                var start = Math.max(0, str.length - 20);
                                                var end = Math.min(next.length, 20);
                                                result.push({
                                                    isRegexp: false,
                                                    rule: sr,
                                                    before: str.substring(start, start + 20),
                                                    string: sr,
                                                    after: next.substring(0, end),
                                                    start: start
                                                });
                                            });
                                        }
                                    };
                                    for(var _iterator = stringRules.values()[Symbol.iterator](), _step1; !(_iteratorNormalCompletion = (_step1 = _iterator.next()).done); _iteratorNormalCompletion = true)_loop();
                                } catch (err) {
                                    _didIteratorError = true;
                                    _iteratorError = err;
                                } finally{
                                    try {
                                        if (!_iteratorNormalCompletion && _iterator.return != null) {
                                            _iterator.return();
                                        }
                                    } finally{
                                        if (_didIteratorError) {
                                            throw _iteratorError;
                                        }
                                    }
                                }
                                var _iteratorNormalCompletion1 = true, _didIteratorError1 = false, _iteratorError1 = undefined;
                                try {
                                    var _loop1 = function() {
                                        var rr = _step2.value;
                                        var regex = RegExp(".{0,20}".concat(rr, ".{0,20}"), "ig");
                                        var findstring = RegExp(rr, "ig");
                                        if (regex.test(addedLines)) {
                                            var splits = addedLines.match(regex);
                                            var lastFromIndex = 0;
                                            splits.forEach(function(str) {
                                                var string = str.match(findstring)[0];
                                                var split = str.split(string);
                                                var before = split[0];
                                                var after = split.slice(1).join(string);
                                                var start = addedLines.indexOf(string, lastFromIndex);
                                                lastFromIndex = start + string.length;
                                                result.push({
                                                    isRegex: true,
                                                    rule: rr,
                                                    before: before,
                                                    string: string,
                                                    after: after,
                                                    start: start
                                                });
                                            });
                                        }
                                    };
                                    for(var _iterator1 = regexRules.values()[Symbol.iterator](), _step2; !(_iteratorNormalCompletion1 = (_step2 = _iterator1.next()).done); _iteratorNormalCompletion1 = true)_loop1();
                                } catch (err) {
                                    _didIteratorError1 = true;
                                    _iteratorError1 = err;
                                } finally{
                                    try {
                                        if (!_iteratorNormalCompletion1 && _iterator1.return != null) {
                                            _iterator1.return();
                                        }
                                    } finally{
                                        if (_didIteratorError1) {
                                            throw _iteratorError1;
                                        }
                                    }
                                }
                                table.empty();
                                if (result.length > 0) {
                                    table.append('<caption style="font-weight: 700;">命中防滥用过滤器'.concat(filter, '的内容</caption><tr><th style="text-align: right;">规则(请勿泄漏!!!)</th><th style="text-align: left;">命中字符串上下文</th></tr>'));
                                    result.sort(function(param, param1) {
                                        var a = param.start, b = param1.start;
                                        return a - b;
                                    });
                                    var results = [];
                                    result.forEach(function(r) {
                                        if (results.filter(function(param) {
                                            var isRegex = param.isRegex, start = param.start;
                                            return isRegex === r.isRegex && start === r.start;
                                        }).length === 0) {
                                            results.push(r);
                                        }
                                    });
                                    console.info(results);
                                    results.forEach(function(param) {
                                        var isRegex = param.isRegex, rule = param.rule, before = param.before, string = param.string, after = param.after, start = param.start;
                                        var tr = $("<tr/>");
                                        var label = $("<td/>");
                                        var ruleCode = $("<code/>");
                                        ruleCode.text(rule);
                                        label.addClass("mw-label");
                                        label.text(isRegex ? "正则表达式" : "字符串");
                                        label.append(symbolEnter(ruleCode)).append(":");
                                        tr.append(label);
                                        var input = $("<td/>");
                                        input.addClass("mw-input");
                                        var beforeSpan = $("<span/>");
                                        beforeSpan.text(symbolEnter(before)).css({
                                            padding: "0 .25em"
                                        });
                                        var strSpan = $("<span/>");
                                        strSpan.text(symbolEnter(string)).css({
                                            "font-weight": "700",
                                            "text-decoration": "underline",
                                            // "border-left": "1px solid #a2a9b1",
                                            padding: "0 .25em"
                                        });
                                        var afterSpan = $("<span/>");
                                        afterSpan.text(symbolEnter(after)).css({
                                            // "border-left": "1px solid #a2a9b1",
                                            padding: "0 .25em"
                                        });
                                        if (before.length >= 20) {
                                            var beforeDot = $("<span/>");
                                            beforeDot.text("[".concat(start, "] ……")).css({
                                                // "border-right": "1px solid #a2a9b1",
                                                "padding-right": ".25em"
                                            });
                                            input.append(beforeDot);
                                        }
                                        input.append(beforeSpan).append(strSpan).append(afterSpan);
                                        if (after.length >= 20) {
                                            var afterDot = $("<span/>");
                                            afterDot.text("……").css({
                                                // "border-left": "1px solid #a2a9b1",
                                                "padding-left": ".25em"
                                            });
                                            input.append(afterDot);
                                        }
                                        tr.append(input);
                                        table.append(tr);
                                    });
                                } else {
                                    table.empty().append("<tbody><tr><td style=\"text-align: center;\">现有规则没有匹配项……</td></tr></tbody>");
                                }
                            };
                            for(_iterator = _details[Symbol.iterator](); !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true)_loop();
                        } catch (err) {
                            _didIteratorError = true;
                            _iteratorError = err;
                        } finally{
                            try {
                                if (!_iteratorNormalCompletion && _iterator.return != null) {
                                    _iterator.return();
                                }
                            } finally{
                                if (_didIteratorError) {
                                    throw _iteratorError;
                                }
                            }
                        }
                        pTable.hide();
                        $(".abusefiltertest").show();
                        return [
                            3,
                            5
                        ];
                    case 4:
                        e = _state.sent();
                        console.info("abusefiltertest", e);
                        $(".abusefiltertest").remove();
                        pTable.empty();
                        if (filter === 33) {
                            pTable.append('<tbody><tr><td style="text-align: '.concat(typeof e === "string" ? "center" : "left", ';">发生错误:').concat(typeof e === "string" ? e : "".concat(e, " ").concat(e.stack.split("\n")[1].trim()), "</td></tr></tbody>"));
                        }
                        return [
                            3,
                            5
                        ];
                    case 5:
                        return [
                            2
                        ];
                }
            });
        })();
    }();
}); // </pre>


/* </pre> */
/* </pre> */

2025年7月24日 (四) 20:47的版本

/**
 * -------------------------------------------------------------------------
 * !!! DON'T MODIFY THIS PAGE MANUALLY, YOUR CHANGES WILL BE OVERWRITTEN !!!
 * -------------------------------------------------------------------------
 */
var _addText = '{{GHIACode|page=GHIA:MoegirlPediaInterfaceCodes/blob/master/src/gadgets/abusefilter33test/Gadget-abusefilter33test.js|user=[[U:AnnAngela]]|co-authors=GH:github-actions[bot]|longId=103d1a563ea4ccc8ff29fb55c9bcd88329a56eb5|shortId=103d1a56|summary=feat: rename (#594)|body=<nowiki>Co-authored-by: github-actions[bot] <41898282+github-actions[bot]📧users.noreply.github.com></nowiki>}}'; 

/* <pre> */

"use strict";
$(() => (async () => {
    if (!mw.config.get("wgUserGroups").includes("sysop") && !mw.config.get("wgUserGroups").includes("staff")) {
        return;
    }
    if (mw.config.get("wgCanonicalSpecialPageName") !== "AbuseLog") {
        return;
    }
    const getLogVar = (varName) => $(`.mw-abuselog-details-${varName} .mw-abuselog-var-value .mw-abuselog-var-value`).text();
    const timestampVar = getLogVar("timestamp");
    const userName = getLogVar("user_name");
    const articlePrefixedtext = getLogVar("article_prefixedtext");
    const filter = [...document.querySelectorAll('a[href*="/Special:%E6%BB%A5%E7%94%A8%E8%BF%87%E6%BB%A4%E5%99%A8/"]')].filter(({ href }) => /%E6%BB%A5%E7%94%A8%E8%BF%87%E6%BB%A4%E5%99%A8\/[1-9]\d*$/.test(href))?.[0]?.href?.match?.(/[1-9]\d*$/)?.[0];
    if (timestampVar.length === 0 || userName.length === 0 || articlePrefixedtext.length === 0 || typeof filter !== "string") {
        return;
    }
    const symbolEnter = (str) => typeof str === "string" ? str.replace(/\n/g, "↵") : str;
    const api = new mw.Api();
    const pTable = $("<table/>");
    $("#mw-content-text > fieldset > p:first").after(pTable.css("width", "100%").addClass("wikitable"));
    pTable.empty().append("<tbody><tr><td style=\"text-align: center;\">加载中(<span id=\"abusefiltertest-progress\">0</span>/2)……</td></tr></tbody>");
    const progress = $("#abusefiltertest-progress");
    try {
        const __details = (await api.post({
            action: "query",
            assertuser: mw.config.get("wgUserName"),
            list: "abuselog",
            afluser: userName,
            afltitle: articlePrefixedtext,
            aflprop: "details",
            aflfilter: +filter,
        })).query.abuselog;
        if (__details.length === 0) {
            throw "无法找到对应用户名和页面标题的滥用过滤器日志";
        }
        progress.text(1);
        const _details = Array.from(new Map(__details.filter(({ details: { timestamp } }) => timestampVar === timestamp).map((n) => [n.new_wikitext, n])).values());
        if (_details.length === 0) {
            console.info(__details);
            throw "无法找到对应时间戳的滥用过滤器日志";
        }
        progress.text(2);
        const _rules = (await api.post({
            action: "query",
            assertuser: mw.config.get("wgUserName"),
            list: "abusefilters",
            abfstartid: +filter,
            abfendid: +filter,
            abfprop: "pattern",
        })).query.abusefilters[0].pattern.replace(/\r/g, "");
        const _stringRules = Array.from(_rules.match(/\/\* string rule start \*\/[\s\S]+?(?=\/\* string rule end \*\/)/g) || []).map((r) => r.replace(/\/\* regex rule start \*\//g, "").split("\n")).flat().map((r) => r.replace(/^ *"?/, "").replace(/"?,? *$/, "")).filter((r) => r !== "");
        const stringRules = new Set(_stringRules);
        const _regexRules = Array.from(_rules.match(/\/\* regex rule start \*\/[\s\S]+?(?=\/\* regex rule end \*\/)/g) || []).map((r) => r.replace(/\/\* regex rule start \*\//g, "").split("\n") || []).flat().map((r) => r.replace(/^ *"?/, "").replace(/"?,? *$/, "")).filter((r) => r !== "");
        const regexRules = new Set(_regexRules);
        if (stringRules.size + regexRules.size === 0) {
            throw "无法找到符合格式的规则";
        }
        for (const { details } of _details) {
            const addedLines = details.added_lines || details.new_wikitext;
            const result = [];
            const table = $("<table/>").css("width", "100%").addClass("wikitable abusefiltertest").hide();
            pTable.after(table);
            for (const sr of stringRules.values()) {
                if (addedLines.includes(sr)) {
                    const splits = addedLines.split(sr);
                    splits.forEach((str, i) => {
                        if (i === splits.length - 1) {
                            return;
                        }
                        const next = splits[i + 1];
                        const start = Math.max(0, str.length - 20);
                        const end = Math.min(next.length, 20);
                        result.push({
                            isRegexp: false,
                            rule: sr,
                            before: str.substring(start, start + 20),
                            string: sr,
                            after: next.substring(0, end),
                            start,
                        });
                    });
                }
            }
            for (const rr of regexRules.values()) {
                const regex = RegExp(`.{0,20}${rr}.{0,20}`, "ig");
                const findstring = RegExp(rr, "ig");
                if (regex.test(addedLines)) {
                    const splits = addedLines.match(regex);
                    let lastFromIndex = 0;
                    splits.forEach((str) => {
                        const string = str.match(findstring)[0];
                        const split = str.split(string);
                        const before = split[0];
                        const after = split.slice(1).join(string);
                        const start = addedLines.indexOf(string, lastFromIndex);
                        lastFromIndex = start + string.length;
                        result.push({
                            isRegex: true,
                            rule: rr,
                            before,
                            string,
                            after,
                            start,
                        });
                    });
                }
            }
            table.empty();
            if (result.length > 0) {
                table.append(`<caption style="font-weight: 700;">命中防滥用过滤器${filter}的内容</caption><tr><th style="text-align: right;">规则(请勿泄漏!!!)</th><th style="text-align: left;">命中字符串上下文</th></tr>`);
                result.sort(({ start: a }, { start: b }) => a - b);
                const results = [];
                result.forEach((r) => {
                    if (results.filter(({ isRegex, start }) => isRegex === r.isRegex && start === r.start).length === 0) {
                        results.push(r);
                    }
                });
                console.info(results);
                results.forEach(({ isRegex, rule, before, string, after, start }) => {
                    const tr = $("<tr/>");
                    const label = $("<td/>");
                    const ruleCode = $("<code/>");
                    ruleCode.text(rule);
                    label.addClass("mw-label");
                    label.text(isRegex ? "正则表达式" : "字符串");
                    label.append(symbolEnter(ruleCode)).append(":");
                    tr.append(label);
                    const input = $("<td/>");
                    input.addClass("mw-input");
                    const beforeSpan = $("<span/>");
                    beforeSpan.text(symbolEnter(before)).css({
                        padding: "0 .25em",
                    });
                    const strSpan = $("<span/>");
                    strSpan.text(symbolEnter(string)).css({
                        "font-weight": "700",
                        "text-decoration": "underline",
                        padding: "0 .25em",
                    });
                    const afterSpan = $("<span/>");
                    afterSpan.text(symbolEnter(after)).css({
                        padding: "0 .25em",
                    });
                    if (before.length >= 20) {
                        const beforeDot = $("<span/>");
                        beforeDot.text(`[${start}] ……`).css({
                            "padding-right": ".25em",
                        });
                        input.append(beforeDot);
                    }
                    input.append(beforeSpan).append(strSpan).append(afterSpan);
                    if (after.length >= 20) {
                        const afterDot = $("<span/>");
                        afterDot.text("……").css({
                            "padding-left": ".25em",
                        });
                        input.append(afterDot);
                    }
                    tr.append(input);
                    table.append(tr);
                });
            }
            else {
                table.empty().append("<tbody><tr><td style=\"text-align: center;\">现有规则没有匹配项……</td></tr></tbody>");
            }
        }
        pTable.hide();
        $(".abusefiltertest").show();
    }
    catch (e) {
        console.info("abusefiltertest", e);
        $(".abusefiltertest").remove();
        pTable.empty();
        if (filter === 33) {
            pTable.append(`<tbody><tr><td style="text-align: ${typeof e === "string" ? "center" : "left"};">发生错误:${typeof e === "string" ? e : `${e} ${e.stack.split("\n")[1].trim()}`}</td></tr></tbody>`);
        }
    }
})()); 

/* </pre> */