Greasy Fork

Greasy Fork is available in English.

PT签到+首页喊魔力(安全优化版)

安全优化的PT签到脚本,修复XSS风险,增强稳定性

// ==UserScript==
// @name         PT签到+首页喊魔力(安全优化版)
// @namespace    http://tampermonkey.net/
// @version      0.1
// @license      MIT
// @author       40
// @description  安全优化的PT签到脚本,修复XSS风险,增强稳定性
// @grant        GM_setValue
// @grant        GM_getValue
// @icon         
// @include      https://*/*
// ==/UserScript==

(function () {
    'use strict';

    // 安全工具函数
    const utils = {
        escapeXPath: (str) => str.replace(/["'\\]/g, '\\$&'),
        log: (msg) => console.log(`[PT签到插件] ${new Date().toISOString()} ${msg}`),
        error: (msg) => console.error(`[PT签到插件] ERROR: ${msg}`)
    };

    // 配置数据
    const CONFIG = {
        attendance: {
            texts: [
                "签到得魔力", "簽到得魔力", "签到领魔力", "簽到领魔力",
                "签到赚魔力", "簽到赚魔力", "签到得猫粮", "簽到得猫粮",
                "签到得杏仁", "簽到得杏仁", "签到得冰晶", "簽到得冰晶",
                "签到得爆米花", "簽到得爆米花", "签 到", "簽 到",
                "签到", "簽到", "每日签到", "每日簽到", "每日打卡",
                "签到得G值", "签到得音浪", "签到金元宝", "签到得金魂币",
                "签到得星焱", "簽到得G值", "簽到得音浪", "簽到金元宝",
                "簽到得金魂币", "簽到得星焱", "签到得电力", "簽到得电力",
                "签到得金币", "簽到得金币", "签到得鲸币", "簽到得鲸币"
            ],
            delay: 12000
        },
        moli: {
            sites: [
                {
                    host: "ptvicomo.net",
                    inputId: "shbox_text",
                    submitId: "hbsubmit",
                    messages: ["小象求象草"]
                },
                {
                    host: "cyanbug.net",
                    inputId: "shbox_text",
                    submitId: "hbsubmit",
                    messages: ["青虫娘 求魔力", "青虫娘 求上传", "青虫娘 求VIP", "青虫娘 求彩虹ID"]
                },
            ],
            delay: 500,
            interval: 3000
        }
    };

    // 核心功能
    class PTAutoSign {
        constructor() {
            this.host = window.location.host;
            this.href = window.location.href;
            this.today = this.formatDate(new Date());
            this.attendanceKey = `${this.host}_ATTENDANCE_DAY`;
            this.moliKey = `${this.host}_MOLI_DAY`;
        }

        formatDate(date) {
            return date.toISOString().split('T')[0]; // YYYY-MM-DD
        }

        safeXPath(query) {
            return document.evaluate(
                query,
                document,
                null,
                XPathResult.UNORDERED_NODE_SNAPSHOT_TYPE,
                null
            );
        }

        async handleAttendance() {
            const lastDay = GM_getValue(this.attendanceKey, "");
            if (lastDay === this.today) {
                utils.log(`[${this.host}] 今日已签到`);
                return;
            }

            for (const text of CONFIG.attendance.texts) {
                try {
                    const query = `//*[contains(text(), "${utils.escapeXPath(text)}")]`;
                    const elements = this.safeXPath(query);

                    for (let i = 0; i < elements.snapshotLength; i++) {
                        const element = elements.snapshotItem(i);
                        if (!element) continue;

                        const elementText = element.textContent || element.innerText;
                        if (elementText.includes("已") || elementText.includes("详情")) {
                            GM_setValue(this.attendanceKey, this.today);
                            utils.log(`[${this.host}] 已签到标记更新`);
                            return;
                        }

                        if (elementText.includes(text)) {
                            if (this.host.includes("ourbits") && !this.href.includes("attendance.php")) {
                                setTimeout(() => {
                                    element.click();
                                    GM_setValue(this.attendanceKey, this.today);
                                    utils.log(`[${this.host}] 延迟签到成功`);
                                }, 5000);
                            } else {
                                element.click();
                                GM_setValue(this.attendanceKey, this.today);
                                utils.log(`[${this.host}] 立即签到成功`);
                            }
                            return;
                        }
                    }
                } catch (error) {
                    utils.error(`签到处理失败: ${error}`);
                }
            }
        }

        async handleMoliRequest() {
            const lastDay = GM_getValue(this.moliKey, "");
            if (lastDay === this.today) {
                utils.log(`[${this.host}] 今日已请求魔力`);
                return;
            }

            const siteConfig = CONFIG.moli.sites.find(s => this.host.includes(s.host));
            if (!siteConfig) return;

            try {
                const input = document.getElementById(siteConfig.inputId);
                const submit = document.getElementById(siteConfig.submitId);
                if (!input || !submit) return;

                siteConfig.messages.forEach((msg, index) => {
                    setTimeout(() => {
                        try {
                            input.value = msg;
                            submit.click();
                            utils.log(`[${this.host}] 魔力请求: ${msg}`);
                            if (index === siteConfig.messages.length - 1) {
                                GM_setValue(this.moliKey, this.today);
                            }
                        } catch (error) {
                            utils.error(`魔力请求失败: ${error}`);
                        }
                    }, CONFIG.moli.interval * index);
                });
            } catch (error) {
                utils.error(`魔力处理失败: ${error}`);
            }
        }

        init() {
            setTimeout(() => this.handleAttendance(), CONFIG.attendance.delay);
            setTimeout(() => this.handleMoliRequest(), CONFIG.moli.delay);
        }
    }

    new PTAutoSign().init();
})();