// ==UserScript==
// @name auto-task
// @namespace auto-task
// @version 5.0.4
// @description 自动完成 Freeanywhere,Giveawaysu,GiveeClub,Givekey,Gleam,Indiedb,keyhub,OpiumPulses,Opquests,SweepWidget 等网站的任务。
// @description:en Automatically complete the tasks of FreeAnyWhere, GiveawaySu, GiveeClub, Givekey, Gleam, Indiedb, keyhub, OpiumPulses, Opquests, SweepWidget websites.
// @author HCLonely
// @license MIT
// @run-at document-start
// @homepage https://auto-task-doc.js.org/
// @supportURL https://github.com/HCLonely/auto-task/issues
// @icon https://auto-task.hclonely.com/favicon.ico
// @tag games
// @include *://freeanywhere.net/*
// @include *://giveaway.su/giveaway/view/*
// @include *://givee.club/*/event/*
// @include *://givekey.ru/giveaway/*
// @include *://www.indiedb.com/giveaways*
// @include *://key-hub.eu/giveaway/*
// @include *://keylol.com/*
// @include *://www.opiumpulses.com/giveaways
// @include *://prys.revadike.com/giveaway/?id=*
// @include *://opquests.com/quests/*
// @include *://gleam.io/*
// @include *://sweepwidget.com/view/*
// @include *://giveawayhopper.com/c/*
// @include *://discord.com/*
// @include *://www.twitch.tv/*
// @include *://www.youtube.com/*
// @include *://m.youtube.com/*
// @include *://*.reddit.com/*
// @include *://twitter.com/settings/account?k*
// @include *://x.com/settings/account*
// @include *://steamcommunity.com/*
// @include *://store.steampowered.com/*
// @include *://give.gamesforfarm.local/*
// @include *://gamesforfarm-testing.ru/*
// @include *://mee6.xyz/*
// @include *://gamesforfarm.com/*
// @include https://auto-task.hclonely.com/setting.html
// @include https://auto-task.hclonely.com/history.html
// @include https://auto-task-doc.js.org/setting.html
// @include https://auto-task-doc.js.org/history.html
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_listValues
// @grant GM_deleteValue
// @grant GM_addStyle
// @grant GM_xmlhttpRequest
// @grant GM_registerMenuCommand
// @grant GM_info
// @grant GM_openInTab
// @grant GM_setClipboard
// @grant GM_getResourceText
// @grant GM_cookie
// @grant GM_addValueChangeListener
// @grant GM_removeValueChangeListener
// @grant unsafeWindow
// @grant window.close
// @grant window.localStorage
// @grant window.sessionStorage
// @grant window.focus
// @connect auto-task.hclonely.com
// @connect auto-task-doc.js.org
// @connect cdn.jsdelivr.net
// @connect store.steampowered.com
// @connect steamcommunity.com
// @connect login.steampowered.com
// @connect twitter.com
// @connect x.com
// @connect abs.twimg.com
// @connect api.twitter.com
// @connect youtube.com
// @connect www.youtube.com
// @connect facebook.com
// @connect instagram.com
// @connect vk.com
// @connect twitch.tv
// @connect www.twitch.tv
// @connect gql.twitch.tv
// @connect github.com
// @connect discordapp.com
// @connect discord.gg
// @connect discord.com
// @connect www.reddit.com
// @connect oauth.reddit.com
// @connect raw.githubusercontent.com
// @connect t.me
// @connect bit.ly
// @connect giveaway.su
// @connect google.com
// @connect www.vloot.io
// @connect givee.club
// @connect gleam.io
// @connect www.indiedb.com
// @connect key-hub.eu
// @connect opquests.com
// @connect itch.io
// @connect auto-task.hclonely.com
// @connect giveawayhopper.com
// @connect freeanywhere.net
// @connect *
// @require https://cdn.jsdelivr.net/npm/[email protected]/dist/jquery.min.js
// @require https://cdn.jsdelivr.net/npm/[email protected]/dist/js.cookie.min.js
// @require https://cdn.jsdelivr.net/npm/[email protected]/src/sha1.min.js
// @require https://cdn.jsdelivr.net/npm/sweetalert2@11/dist/sweetalert2.min.js
// @resource style https://cdn.jsdelivr.net/npm/[email protected]/dist/sweetalert2.min.css
// @require https://cdn.jsdelivr.net/npm/[email protected]/dist/keyboard.min.js
// @require https://cdn.jsdelivr.net/npm/[email protected]/dayjs.min.js
// @require https://cdn.jsdelivr.net/gh/tinygo-org/tinygo@3e60eeb368f25f237a512e7553fd6d70f36dc74c/targets/wasm_exec.min.js
// @require https://cdn.jsdelivr.net/npm/[email protected]/dist/inspect.min.js
// @require https://cdn.jsdelivr.net/npm/[email protected]/dist/browser.min.js
// @noframes
// ==/UserScript==
console.log('%c%s', 'color:blue', 'Auto-Task[Load]: 脚本开始加载');
/*
* @Author : HCLonely
* @Date : 2025-06-15 14:59:17
* @LastEditTime : 2025-08-18 19:05:01
* @LastEditors : HCLonely
* @FilePath : /auto-task/src/scripts/checkDependence.js
* @Description :
*/
const neededDependencies = ['jQuery', 'Cookies', 'sha1', 'Swal', 'keyboardJS', 'dayjs', 'Go', 'util', 'browser'];
const missingDependencies = neededDependencies.filter(dependency => typeof window[dependency] === 'undefined');
if (missingDependencies.length > 0) {
console.log('%c%s', 'color:red', `[Auto-Task] 脚本加载失败,缺少的依赖:${missingDependencies.join(', ')}`);
if (confirm(`[Auto-Task] 脚本依赖加载失败,请刷新重试或安装全依赖版本,是否前往安装全依赖版本?\n缺少的依赖:${missingDependencies.join(', ')}`)) {
GM_openInTab('https://github.com/HCLonely/auto-task/raw/main/dist/auto-task.all.user.js', { active: true });
}
}
(function(Swal, Cookies, browser, util, dayjs, keyboardJS) {
'use strict';
const tokenKeyPattern = /token|auth|session|jwt|key|secret|api[-_]?key|bearer|authorization|access[-_]?token|refresh[-_]?token|sid/i;
const tokenStringPatterns = [ /([A-Za-z0-9-_]{10,})\.([A-Za-z0-9-_]{10,})\.([A-Za-z0-9-_]{10,})/g, /(Bearer|Basic)\s+([A-Za-z0-9\-._~+/]+=*)/gi, /\b([a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12})\b/gi, /\b(eyJ[A-Za-z0-9\-_]+)\b/g ];
const maskToken = str => {
if (typeof str !== 'string' || str.length < 8) {
return str;
}
return str.replace(/^([A-Za-z0-9\-_+/=]{4})[A-Za-z0-9\-_+/=]+([A-Za-z0-9\-_+/=]{4})$/, '$1***$2');
};
const maskObject = obj => {
if (Array.isArray(obj)) {
return obj.map(maskObject);
} else if (obj && typeof obj === 'object') {
const newObj = {};
for (const key in obj) {
if (tokenKeyPattern.test(key) && typeof obj[key] === 'string') {
newObj[key] = maskToken(obj[key]);
} else {
newObj[key] = maskObject(obj[key]);
}
}
return newObj;
}
if (typeof obj === 'string' && obj.length > 8) {
return maskString(obj);
}
return obj;
};
const maskString = str => {
let masked = str;
for (const pattern of tokenStringPatterns) {
masked = masked.replace(pattern, ((match, ...groups) => {
if (groups.length >= 3 && match.includes('.')) {
return groups.map((seg => seg.length > 8 ? `${seg.slice(0, 4)}***${seg.slice(-4)}` : seg)).join('.');
}
if (match.length > 8) {
return `${match.slice(0, 4)}***${match.slice(-4)}`;
}
return match;
}));
}
return masked;
};
const maskArgs = args => args.map((arg => {
if (typeof arg === 'string') {
return maskString(arg);
} else if (typeof arg === 'object' && arg !== null) {
return maskObject(arg);
}
return arg;
}));
const consoleLogHook = () => {
const originalLog = console.log;
window.__allLogs = window.__allLogs || [];
console.log = function(...args) {
const maskedArgs = maskArgs(args);
window.__allLogs.push(maskedArgs);
originalLog.apply(console, maskedArgs);
};
};
const defaultGlobalOptions = {
doTask: {
discord: {
servers: true
},
twitch: {
channels: true
},
twitter: {
users: true,
retweets: true
},
vk: {
names: true
},
youtube: {
channels: true,
likes: true
},
reddit: {
reddits: true
},
steam: {
groups: true,
officialGroups: true,
wishlists: true,
follows: true,
forums: true,
workshops: true,
curators: true,
workshopVotes: true,
announcements: true,
licenses: true,
playtests: true,
playTime: true
}
},
undoTask: {
discord: {
servers: true
},
twitch: {
channels: true
},
twitter: {
users: true,
retweets: true
},
vk: {
names: true
},
youtube: {
channels: true,
likes: true
},
reddit: {
reddits: true
},
steam: {
groups: true,
officialGroups: true,
wishlists: true,
follows: true,
forums: true,
workshops: true,
curators: true,
playTime: true
}
},
ASF: {
AsfEnabled: false,
AsfIpcUrl: '',
AsfIpcPassword: '',
AsfBotname: 'asf',
steamWeb: false,
preferASF: false,
steamWebApiKey: ''
},
position: {
buttonSideX: 'right',
buttonSideY: 'top',
buttonDistance: '15,30',
showButtonSideX: 'right',
showButtonSideY: 'top',
showButtonDistance: '15,30',
logSideX: 'right',
logSideY: 'bottom',
logDistance: '10,10'
},
hotKey: {
doTaskKey: 'alt + d',
undoTaskKey: 'alt + u',
toggleLogKey: 'alt + l'
},
other: {
twitterVerifyId: '783214',
youtubeVerifyChannel: 'UCrXUsMBcfTVqwAS7DKg9C0Q',
autoUpdateSource: 'jsdelivr',
language: 'zh',
checkLogin: true,
checkLeftKey: true,
defaultShowButton: true,
defaultShowLog: true,
debug: false,
receivePreview: true
}
};
const userDefinedGlobalOptions = GM_getValue('globalOptions') || {};
const deepMerge = (target, source) => {
try {
const result = {
...target
};
for (const [key, value] of Object.entries(source)) {
const targetValue = target[key];
if (isObject(value) && isObject(targetValue)) {
result[key] = deepMerge(targetValue, value);
} else if (value !== undefined) {
result[key] = value;
}
}
return result;
} catch (error) {
console.log('%c%s', 'color:white;background:red', `Auto-Task[Error]: deepMerge\n${error.stack}`);
return target;
}
};
const isObject = value => value !== null && typeof value === 'object' && !Array.isArray(value);
const globalOptions = deepMerge(defaultGlobalOptions, userDefinedGlobalOptions);
var style = '.colorful-button,#auto-task-buttons a.auto-task-website-btn,.show-button-div a.auto-task-website-btn,body.auto-task-options .auto-task-form table button{position:relative !important;padding:14px 28px !important;text-align:center !important;color:#fff !important;text-decoration:none !important;background:#2196f3 !important;border-radius:30px !important;text-transform:capitalize !important;font-weight:600 !important;letter-spacing:.5px !important;border:none !important;transition:all .2s ease !important;display:inline-block !important;line-height:1.5 !important;margin:8px !important;margin-bottom:12px !important;box-sizing:border-box !important;min-height:50px !important;min-width:140px !important;outline:none !important;vertical-align:middle !important;white-space:nowrap !important;font-size:18px !important}.colorful-button:hover,#auto-task-buttons a.auto-task-website-btn:hover,.show-button-div a.auto-task-website-btn:hover,body.auto-task-options .auto-task-form table button:hover{background:#1976d2 !important;box-shadow:0 4px 8px rgba(0,0,0,.1) !important;cursor:pointer !important;color:#fff !important;text-decoration:none !important}.colorful-button:active,#auto-task-buttons a.auto-task-website-btn:active,.show-button-div a.auto-task-website-btn:active,body.auto-task-options .auto-task-form table button:active{transform:translateY(1px) !important;color:#fff !important;text-decoration:none !important}.colorful-button:focus,#auto-task-buttons a.auto-task-website-btn:focus,.show-button-div a.auto-task-website-btn:focus,body.auto-task-options .auto-task-form table button:focus{color:#fff !important;text-decoration:none !important;outline:none !important}#auto-task-info{position:fixed;bottom:10px;right:10px;width:60%;max-width:500px;max-height:60%;overflow-y:auto;color:#000;background-color:#fff;padding-left:5px;z-index:999999999 !important;border:solid 2px #add8e6;border-radius:10px;font-size:14px !important}#auto-task-info li{text-align:left;display:block !important;align-items:baseline !important}#auto-task-info li .before-icon{display:inline-block !important;width:14px !important;height:14px !important;position:relative !important;top:2px !important;margin-right:5px !important;background-size:14px !important;background-repeat:no-repeat !important;flex-shrink:0 !important}#auto-task-info li font.before{color:#57bae8;margin-right:5px}#auto-task-info li a.high-light{color:#00aeff;font-weight:bold}#auto-task-info .success{color:green}#auto-task-info .error{color:red}#auto-task-info .warning{color:blue}#auto-task-info .info{color:#ff0}#auto-task-info .update-text{color:green;border:solid 2px #8dcb69;margin:5px 10px 5px 20px;border-radius:10px;padding:5px 20px}.auto-task-keylol{display:inline-block;text-transform:capitalize;margin-left:10px;text-decoration:none !important;border:solid 1px;border-radius:5px;padding:0 2px}.auto-task-keylol[selected=selected]{background-color:blue !important;color:#fff !important}.auto-task-form table{font-family:verdana,arial,sans-serif;font-size:11px;color:#333;border-width:1px;border-color:#999;border-collapse:collapse;width:100%}.auto-task-form table thead td{border-width:1px;padding:8px;border-style:solid;border-color:#a9c6c9;font-weight:bold;background-color:#fff}.auto-task-form table tbody tr{background-color:#d4e3e5}.auto-task-form table tbody tr:hover{background-color:#ff6 !important}.auto-task-form table tbody tr th{background-color:#c3dde0;border-width:1px;padding:8px;border-style:solid;border-color:#a9c6c9;text-transform:capitalize}.auto-task-form table tbody tr td{border-width:1px;padding:8px;border-style:solid;border-color:#a9c6c9}.swal2-modal{width:70% !important;max-width:1000px !important}.swal2-modal #swal2-title{text-align:center !important}body.auto-task-options{padding-top:10px;text-align:center}body.auto-task-options .auto-task-form{width:80%;max-width:1000px;margin:0 auto;padding-bottom:20px}body.auto-task-options .auto-task-form table input.editOption{width:80%}body.auto-task-options .auto-task-form table #getTwitterUserId,body.auto-task-options .auto-task-form table #getYoutubeChannelId{margin-top:5px}body.auto-task-options .auto-task-form table button{z-index:1;position:relative !important;padding:5px 20px !important;text-align:center !important;color:#fff !important;text-decoration:none !important;background:#2196f3 !important;border-radius:30px !important;text-transform:capitalize !important;font-weight:600 !important;letter-spacing:.5px !important;border:none !important;transition:all .2s ease !important;display:inline-block !important;line-height:1 !important;margin:8px !important;box-sizing:border-box !important;min-height:30px !important;min-width:140px !important;outline:none !important;vertical-align:middle !important;white-space:nowrap !important;font-size:15px !important}body.auto-task-options .auto-task-form table input[type=text]{outline-style:none;border:1px solid #ccc;border-radius:3px;padding:5px 10px;font-size:14px}body.auto-task-options .auto-task-form table input[type=text]:focus{border-color:#66afe9;outline:0;box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 8px rgba(102,175,233,.6)}body.auto-task-options .auto-task-form table label{position:relative;width:160px;height:80px;cursor:pointer;transform:scale(0.25);margin:-25% 0;top:-30px;display:inline-block}body.auto-task-options .auto-task-form table label input{position:relative;z-index:1;appearance:none}body.auto-task-options .auto-task-form table label input:checked~span{background:#05be05;box-shadow:0 15px 25px rgba(5,190,5,.4)}body.auto-task-options .auto-task-form table label input:checked~span i{left:84px}body.auto-task-options .auto-task-form table label input:checked~span i::before{background:#05be05;box-shadow:35px 0 0 #05be05}body.auto-task-options .auto-task-form table label input:checked~span i::after{bottom:12px;height:15px;border-bottom-left-radius:15px;border-bottom-right-radius:15px;background:#05be05}body.auto-task-options .auto-task-form table label span{position:absolute;top:0;left:0;width:100%;height:100%;background:#fe0000;border-radius:80px;transition:.5s;box-shadow:0 15px 25px rgba(254,0,0,.4)}body.auto-task-options .auto-task-form table label span i{position:absolute;top:4px;left:4px;width:72px;height:72px;background:#fff;border-radius:50%}body.auto-task-options .auto-task-form table label span i::before{content:"";position:absolute;top:22px;left:12px;width:12px;height:12px;border-radius:50%;background:#fe0000;box-shadow:35px 0 0 #fe0000;transition:.5s}body.auto-task-options .auto-task-form table label span i::after{content:"";position:absolute;bottom:15px;left:calc(50% - 15px);width:30px;height:6px;border-radius:6px;background:#fe0000;transition:.5s}body.auto-task-history{font-size:15px;font-weight:400;line-height:1.5}body.auto-task-history .container a{color:#007bff;text-decoration:none;background-color:rgba(0,0,0,0)}body.auto-task-history .container .card{width:80%;max-width:800px;border-radius:10px;background:rgba(118,118,118,.1019607843);border-top:1px solid hsla(0,0%,100%,.5019607843);-webkit-backdrop-filter:blur(20px);backdrop-filter:blur(20px);box-shadow:0 15px 25px rgba(0,0,0,.1019607843);margin:20px auto;position:relative;display:flex;flex-direction:column;word-wrap:break-word;-webkit-background-clip:border-box;background-clip:border-box;border:1px solid rgba(0,0,0,.125);border-radius:.25rem}body.auto-task-history .container .card .title{text-align:center;font-size:30px;font-weight:bold;margin:5px 0}body.auto-task-history .container .card .title a:hover{text-decoration:none;background:#93e1ff;border-radius:10px;padding:3px}body.auto-task-history .container .card ul{margin-bottom:25px}body.auto-task-history .container .card ul li{margin-bottom:5px;line-height:20px}body.auto-task-history .container .card ul a:hover{text-decoration:underline}body.auto-task-history .container .card .delete-task{right:10px;width:38px;height:35px;position:absolute;font-size:24px;cursor:pointer;border-radius:10px}body.auto-task-history .container .card .delete-task:hover{background:#fff}body.auto-task-history .container .card .time{right:5px;position:absolute;bottom:0;color:#e83e8c;font-family:\'SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace\';font-size:15px}#auto-task-buttons,.show-button-div{position:fixed !important;top:30px;right:15px;width:150px !important;z-index:999999999 !important;padding:8px !important;border-radius:12px !important}#auto-task-buttons p,.show-button-div p{line-height:normal !important;height:auto !important;text-align:center !important;margin:8px 0 !important;padding:0 !important;font-size:16px !important;color:#333 !important}#auto-task-buttons a.auto-task-website-btn,.show-button-div a.auto-task-website-btn{width:140px !important;min-height:30px !important;line-height:1.5 !important;font-size:16px !important;display:block !important;margin:0 auto !important;padding:8px 16px !important}.show-button-div{width:40px !important;cursor:pointer !important;padding:4px !important}.show-button-div a.auto-task-website-btn{right:-15px !important}.show-button-div a.auto-task-website-btn::after{content:"✓" !important;position:absolute !important;left:12px !important;top:50% !important;transform:translateY(-50%) !important;font-size:18px !important;font-weight:bold !important;color:#fff !important}.auto-task-capitalize{text-transform:capitalize !important}.swal2-file:focus,.swal2-input:focus,.swal2-textarea:focus{box-shadow:inset 0px 0px 4px 1px rgba(100,150,200,.5) !important}.swal2-checkbox-custom{align-items:center;justify-content:center;background:#fff;color:inherit;margin:1em auto}.swal2-checkbox-custom input{flex-shrink:0;margin:0 .4em}.giveaway-actions #getKey{display:none !important}.auto-task-giveaway-status{color:#fff;border-radius:10px;padding:0 5px;margin-left:5px}.auto-task-giveaway-status.active{background-color:#5cb85c}.auto-task-giveaway-status.not-active{background-color:#d9534f}';
const data$1 = {
website: '网站',
type: '类型',
edit: '编辑',
whiteList: '白名单',
skipTask: '跳过撤销任务',
whiteListOptions: '白名单设置',
changeWhiteListOption: '设置白名单(%0)',
whiteListNotFound: '找不到此项白名单: %0',
changeWhiteListSuccess: '白名单修改成功,刷新生效!',
changeWebsiteOptions: '网站设置',
changeGlobalOptions: '全局设置',
ok: '是',
save: '保存',
close: '关闭',
return: '返回',
option: '选项',
value: '值',
websiteOptions: '当前网站设置',
changeWebsiteOptionsSuccess: '更改当前网站设置成功,刷新生效!',
changeGlobalOptionsSuccess: '更改全局设置成功,刷新生效!',
needLogin: '请先登录!',
getTasksInfo: '正在获取并处理任务信息...',
gettingKey: '正在获取Key...',
verifyingTask: '正在验证任务',
notice: '自动任务脚本提醒',
noKeysLeft: '此页面已经没有剩余key了,是否关闭?',
giveawayEnded: '此活动已结束,是否关闭?',
giveawayNotWork: '此活动因某些原因(已结束/暂停/未开始...)不可用(如果是脚本误判请及时反馈),是否关闭?',
confirm: '确定',
cancel: '取消',
unKnown: '未知',
unKnownTaskType: '未识别的任务',
doing: '正在做任务',
allTasksComplete: '所有任务已完成!',
getTaskIdFailed: '获取任务Id失败!',
initSuccess: '%0 初始化成功!',
initFailed: '%0 初始化失败!',
errorLink: '链接错误: %0',
needInit: '请先初始化',
verifyingAuth: '正在验证%0凭证...',
updatingAuth: '正在更新%0凭证...',
refreshingToken: '正在刷新%0凭证...',
settingToken: '正在设置%0凭证...',
steamStoreTab: 'Steam商店(弹窗)',
steamCommunityTab: 'Steam社区(弹窗)',
initing: '正在初始化...',
getFailed: '获取%0失败!',
checkLoginFailed: '检测登录状态失败!',
checkLeftKeyFailed: '检测剩余Key失败!',
userId: '用户Id',
joiningGiveaway: '正在加入赠Key',
needJoinGiveaway: '需要先加入赠Key',
cannotUndo: '此网站不支持取消任务',
verifyAuth: '正在验证 %0 凭证...',
closePageNotice: '如果此页面没有自动关闭,请自行关闭本页面。',
errorReport: '检测到脚本报错,是否前往反馈BUG?',
visitingLink: '正在访问链接...',
doTask: '做任务',
undoTask: '撤销任务',
verifyTask: '验证任务',
getKey: '获取Key',
selectAll: '全选',
selectNone: '全不选',
invertSelect: '反选',
doFreeTask: '加入免费赠品',
doPointTask: '加入点数赠品',
skipTaskOption: '设置中已配置跳过任务',
other: '其他',
globalOptions: '全局设置',
checkLogin: '登录检测</br>需要登录的网站自动登录,部分网站支持',
checkLeftKey: '剩余Key检测</br>赠Key活动结束提示是否关闭,部分网站支持',
twitterVerifyId: '通过尝试关注该账号验证Twitter凭证</br>默认为Twitter官方帐号 783214</br>不想关注官方账号可以改为自己的帐号',
youtubeVerifyChannel: '通过尝试订阅该频道验证YouTube凭证</br>默认为YouTube官方频道 UCrXUsMBcfTVqwAS7DKg9C0Q</br>不想关注官方频道可以改为自己的频道',
autoUpdateSource: '更新源</br>github: 需代理,实时更新</br>jsdelivr: 可不用代理,更新有延迟</br>standby: 备用</br>auto: 依次使用github, jsdelivr, standby源进行尝试更新',
saveGlobalOptions: '保存设置',
settingPage: '设置页面',
name: '名称',
version: '版本',
scriptManager: '脚本管理器',
script: '脚本',
environment: '环境',
os: '系统',
browser: '浏览器',
getId: '获取 %0 id',
getTwitterUserId: '获取Twitter用户id(获取id功能仅在设置页面可用)',
getYoutubeChannelId: '获取Youtube频道id(获取id功能仅在设置页面可用)',
showButton: '显示按钮',
hideButton: '隐藏按钮',
showLog: '显示日志',
hideLog: '隐藏日志',
defaultShowButton: '默认显示按钮',
defaultShowLog: '默认显示日志',
debug: '输出调试日志,不要开启此选项!',
receivePreview: '接收预览版更新',
position: '组件位置',
buttonSideX: '按钮区域水平方向定位(实时预览功能仅在设置页面可用)</br>left: 靠左 | right: 靠右',
buttonSideY: '按钮区域垂直方向定位(实时预览功能仅在设置页面可用)</br>top: 靠上 | bottom: 靠下',
buttonDistance: '按钮区域距边缘的距离(实时预览功能仅在设置页面可用)</br>格式: X距离,Y距离',
showButtonSideX: '显示按钮水平方向定位(实时预览功能仅在设置页面可用)</br>left: 靠左 | right: 靠右',
showButtonSideY: '显示按钮垂直方向定位(实时预览功能仅在设置页面可用)</br>top: 靠上 | bottom: 靠下',
showButtonDistance: '显示按钮距边缘的距离(实时预览功能仅在设置页面可用)</br>格式: X距离,Y距离',
logSideX: '日志区域水平方向定位(实时预览功能仅在设置页面可用)</br>left: 靠左 | right: 靠右',
logSideY: '日志区域垂直方向定位(实时预览功能仅在设置页面可用)</br>top: 靠上 | bottom: 靠下',
logDistance: '日志区域距边缘的距离(实时预览功能仅在设置页面可用)</br>格式: X距离,Y距离',
hotKey: '快捷键',
doTaskKey: '做任务快捷键</br>(实时预览功能仅在设置页面可用)',
undoTaskKey: '撤销任务快捷键</br>(实时预览功能仅在设置页面可用)',
toggleLogKey: '显示/隐藏日志快捷键</br>(实时预览功能仅在设置页面可用)',
tasksHistory: '任务历史',
clearHistory: '清空历史',
clearHistoryFinished: '已清空任务历史!',
deleteTask: '删除任务',
lastChangeTime: '最后一次修改时间',
clearTaskFinished: '删除以下任务完成!',
clearTaskFailed: '删除任务失败,没有找到任务名!',
syncData: '数据同步',
settingData: '正在上传数据...',
gettingData: '正在获取数据...',
help: '帮助',
fileName: '文件名',
upload2gist: '同步到Gist',
downloadFromGist: '从Gist同步',
saveAndTest: '保存配置并测试',
testSuccess: '测试成功!',
testFailed: '测试失败!',
saveAndTestNotice: '请先保存配置并测试!',
processingData: '正在处理数据...',
updatingData: '正在上传数据...',
syncDataSuccess: '同步数据成功!',
syncDataFailed: '同步数据失败,请在控制台查看错误信息!',
downloadingData: '正在下载数据...',
checkedNoData: '没有检测到远程数据,请确认配置是否正确!',
savingData: '正在保存数据...',
syncHistory: '同步任务历史',
checkUpdateFailed: '检测更新失败',
newVersionNotice: '检测到新版本V%0, <a class="high-light" href="%1" target="_blank">点此更新</a>',
language: '语言</br>目前仅支持zh: 中文, en: 英文',
gistOptions: 'Gist 设置',
swalNotice: '检测到您第一次安装V4版本脚本,请前往阅读用前必读内容!',
echoNotice: '检测到您第一次安装V4版本脚本,请<a class="high-light" href="%0" target="_blank">点此前往</a>阅读用前必读内容!',
noticeLink: 'https://auto-task-doc.js.org/guide/#用前必读',
toGithub: '前往Github反馈',
toKeylol: '前往其乐论坛反馈',
copySuccess: '错误信息已复制到剪切板,是否前往其乐论坛反馈?',
copyFailed: '请复制下方错误信息后前往Keylol论坛反馈!',
updateText: '%0 版本更新内容:',
Active: '进行中',
Ended: '已结束',
Banned: '已封禁',
Paused: '已暂停',
notStart: '未开始',
noRemoteData: '检测到远程无数据',
errorRemoteDataFormat: '远程数据格式错误',
updateHistory: '历史更新记录<a class="high-light" href="https://auto-task-doc.js.org/logs/" target="_blank">点此查看</a>',
AsfEnabled: '使用ASF做Steam相关任务(需<a href="https://github.com/chr233/ASFEnhance" target="_blank">ASFEnhance</a>插件)',
steamWeb: '同时使用Steam Web API做任务',
preferASF: '优先使用ASF',
AsfIpcUrl: 'ASF IPC 地址',
AsfIpcPassword: 'ASF IPC 密码',
versionNotMatched: '脚本管理器版本过低,需TamperMonkey >= 5.2.0或TamperMonkey Beta >= 5.2.6196',
curatorLimitNotice: '失败:可能是鉴赏家关注数量限制,请<a class="high-light" href="https://store.steampowered.com/curators/home/" target="_blank">取关部分鉴赏家</a>后再试',
unknownScriptHandler: '未知脚本管理器,适用性未知',
debugModeNotice: '检测到 DEBUG 模式已开启,非必要请关闭!',
steamWebApiKey: 'Steam Web API 密钥<br>用于检测游戏挂机状态,<a href="https://steamcommunity.com/dev/apikey" target="_blank">点此申请</a>',
groups: '组',
officialGroups: '官方组',
wishlists: '愿望单',
follows: '游戏关注',
forums: '论坛',
workshops: '创意工坊收藏',
curators: '鉴赏家',
workshopVotes: '创意工坊点赞',
announcements: '社区通知',
steamCommunity: 'Steam社区',
steamStore: 'Steam商店',
licenses: '入库免费游戏',
playtests: '请求访问权限',
playTime: '挂时长',
needLoginSteamStore: '请先<a href="https://store.steampowered.com/login/" target="_blank">登录Steam商店</a>',
needLoginSteamCommunity: '请先<a href="https://steamcommunity.com/login/home/" target="_blank">登录Steam社区</a>',
joiningSteamGroup: '正在加入Steam组',
leavingSteamGroup: '正在退出Steam组',
gettingSteamGroupId: '正在获取Steam组Id',
joiningSteamOfficialGroup: '正在加入Steam官方组',
leavingSteamOfficialGroup: '正在退出Steam官方组',
gettingSteamOfficialGroupId: '正在获取Steam官方组Id',
subscribingForum: '正在订阅Steam论坛',
unsubscribingForum: '正在取消订阅Steam论坛',
gettingForumId: '正在获取Steam论坛Id',
followingCurator: '正在关注Steam鉴赏家',
unfollowingCurator: '正在取关Steam鉴赏家',
gettingCuratorId: '正在获取Steam鉴赏家Id',
addingToWishlist: '正在添加游戏到Steam愿望单',
removingFromWishlist: '正在从Steam愿望单移除游戏',
followingGame: '正在关注Steam游戏',
unfollowingGame: '正在取关Steam游戏',
favoritingWorkshop: '正在收藏Steam创意工坊物品',
unfavoritingWorkshop: '正在取消收藏Steam创意工坊物品',
gettingWorkshopAppId: '正在获取Steam创意工坊物品Id',
votingUpWorkshop: '正在点赞Steam创意工坊物品',
gettingAnnouncementParams: '正在获取Steam通知信息',
likingAnnouncement: '正在点赞Steam通知',
changingArea: '正在更换Steam地区: %0...',
notNeededChangeArea: '当前地区不需要更换',
noAnotherArea: '请检测是否开启正确开启代理',
gettingAreaInfo: '正在获取Steam地区信息...',
changeAreaNotice: '疑似锁区游戏,尝试换区执行',
steamFinishNotice: 'Steam任务完成,尝试将购物车地区换回',
gettingSubid: '正在获取游戏subid',
addingFreeLicense: '正在入库',
missParams: '缺少参数',
gettingLicenses: '正在获取Licenses...',
requestingPlayTestAccess: '正在请求访问权限',
gettingDemoAppid: '正在获取Steam游戏的Demo Appid',
tryChangeAreaNotice: '此功能无法检测游戏是否限区,因此会尝试换区后再入库,换区失败也不影响后续入库',
gettingUserInfo: '正在获取Steam用户社区凭证...',
retry: '重试',
owned: '已拥有',
redirect: '重定向',
noSubid: '无法获取,跳过',
noASFInstance: '未启用ASF,跳过挂时长任务',
initingASF: '正在初始化ASF...',
playingGames: '正在挂游戏时长[%0]...',
stoppingPlayGames: '正在停止挂游戏时长...',
stopPlayTimeTitle: 'Steam游戏挂机时长满足,是否结束挂机?',
stopPlayTimeText: '挂机已超时:%0 分钟',
ASFNotSupportted: '当前功能(%0)ASF无法实现,跳过',
checkingPlayGamesStatus: '正在检查挂游戏时长状态...',
gettingSteamId: '正在获取Steam ID...',
checkingPlayStatus: '正在检查挂机状态...',
noPlayStatus: '游戏未运行',
servers: '服务器',
joiningDiscordServer: '正在加入Discord服务器',
leavingDiscordServer: '正在退出Discord服务器',
gettingDiscordGuild: '正在获取Discord服务器Id',
getDiscordAuthFailed: '获取Discord凭证失败,请检测Discord帐号是否已登录',
discordImportantNotice: '重要提醒!!!',
discordImportantNoticeText: '由于Discord网站后台更新,目前使用此脚本加组后可能会导致Discord帐号被强制退出,且需要两步验证才能正常登录,请谨慎使用!!!',
continueDiscordTask: '本次执行Discord任务',
skipDiscordTask: '本次跳过Discord任务',
continueAndDontRemindAgain: '总是执行Discord任务且不再提醒',
gettingDiscordXContextProperties: '正在获取Discord加群参数',
captchaNeeded: '检测到人机验证,请手动完成!',
users: '用户',
loginIns: '请先<a href="https://www.instagram.com/accounts/login/" target="_blank">登录Instagram</a>',
insBanned: '您的Instagram账户已被封禁',
verifyingInsAuth: '正在验证Instagram凭证...',
gettingInsUserId: '正在获取Instagram用户Id',
followingIns: '正在关注Instagram用户',
unfollowingIns: '正在取关Instagram用户',
reddits: '社区/用户',
loginReddit: '请先<a href="https://www.reddit.com/login/" target="_blank">登录Reddit</a>',
changingRedditVersion: '正在切换Reddit为新版页面...',
joiningReddit: '正在加入Reddit社区',
leavingReddit: '正在退出Reddit社区',
followingRedditUser: '正在关注Reddit用户',
unfollowingRedditUser: '正在取关Reddit用户',
channels: '频道',
followingTwitchChannel: '正在关注Twitch频道',
unfollowingTwitchChannel: '正在取关Twitch频道',
gettingTwitchChannelId: '正在获取Twitch频道Id',
checkingTwitchIntegrity: '正在检查Twitch完整性...',
twitterUser: '推特用户',
retweets: '转推',
gettingTwitterUserId: '正在获取推特用户Id',
followingTwitterUser: '正在关注推特用户',
unfollowingTwitterUser: '正在取关推特用户',
retweetting: '正在转推',
unretweetting: '正在撤销转推',
names: '组/社区/动态',
loginVk: '请先<a href="https://vk.com/login/" target="_blank">登录Vk</a>',
gettingVkId: '正在获取Vk任务Id',
joiningVkGroup: '正在加入Vk组',
leavingVkGroup: '正在退出Vk组',
joiningVkPublic: '正在加入Vk社区',
leavingVkPublic: '正在退出Vk社区',
sendingVkWall: '正在转发Vk动态',
deletingVkWall: '正在撤销转发Vk动态',
youtubeChannel: 'YouTube频道',
likes: '点赞',
loginYtb: '请先<a href="https://accounts.google.com/ServiceLogin?service=youtube" target="_blank">登录YouTube</a>',
tryUpdateYtbAuth: '请尝试<a href="https://www.youtube.com/#auth" target="_blank">更新YouTube凭证</a>',
gettingYtbToken: '正在获取YouTube Token...',
followingYtbChannel: '正在订阅YouTube频道',
unfollowingYtbChannel: '正在退订YouTube频道',
likingYtbVideo: '正在点赞YouTube视频',
unlikingYtbVideo: '正在取消点赞YouTube视频',
giveKeyNoticeBefore: '每次验证间隔15s',
giveKeyNoticeAfter: '如果没有key, 请在<a href="https://givekey.ru/profile" target="_blank">https://givekey.ru/profile</a>查看',
noPoints: '点数不够,跳过抽奖',
getNeedPointsFailed: '获取所需点数失败,跳过抽奖',
joiningLottery: '正在加入抽奖',
doingGleamTask: '正在做Gleam任务...',
gettingGleamLink: '正在获取Gleam任务链接...',
gleamTaskNotice: '如果此页面长时间未关闭,请完成任一任务后自行关闭!',
verifiedGleamTasks: '已尝试验证所有任务,验证失败的任务请尝试手动验证或完成!',
campaign: '检测到人机验证,请手动完成!3秒后重新检测...',
gsNotice: '为避免得到"0000-0000-0000"key, 已自动屏蔽"Grab Key"按钮,获取key时请关闭脚本!',
giveeClubVerifyNotice: '正在验证任务...',
giveeClubVerifyFinished: '请等待验证完成后自行加入赠Key',
doingKeyhubTask: '正在做Keyhub任务...',
SweepWidgetNotice: '正在处理并验证任务,每次验证任务有1~3s间隔防止触发验证过快警告...',
tasksNotCompleted: '任务未完成',
notConnect: '社交平台未连接,跳过任务: %0',
tgTaskNotice: '检测到Telegram任务,需要手动完成',
updatingUserData: '正在更新用户数据...',
gettingUserGames: '正在获取用户游戏...',
confirmingTask: '正在跳过警告页面...',
unSupporttedTaskType: '不支持的任务类型: %0',
taskNotFinished: '有任务未完成,不获取密钥',
logCopied: '完整日志已复制到剪切板,请前往反馈!'
};
const data = {
website: 'Website',
type: 'Type',
edit: 'Edit',
whiteList: 'Whitelist',
skipTask: 'Skip undo task',
whiteListOptions: 'Whitelist options',
changeWhiteListOption: 'Whitelist option(%0)',
whiteListNotFound: 'Cannot find this whitelist: %0',
changeWhiteListSuccess: 'The whitelist is successfully modified, and the page refresh will take effect!',
changeWebsiteOptions: 'Website options',
changeGlobalOptions: 'Global options',
ok: 'OK',
save: 'Save',
close: 'Close',
return: 'Return',
option: 'Option',
value: 'Value',
websiteOptions: 'Current website settings',
changeWebsiteOptionsSuccess: 'The current website setting is changed successfully, and the page refresh will take effect!',
changeGlobalOptionsSuccess: 'The global setting is changed successfully, and the refresh will take effect!',
needLogin: 'Please log in first!',
getTasksInfo: 'Obtaining and processing task information...',
gettingKey: 'Getting Key...',
verifyingTask: 'Verifying task',
notice: 'Automatic task script notice',
noKeysLeft: 'There are no more keys left on this page. Do you want to close it?',
giveawayEnded: 'This event has ended, do you want to close it?',
giveawayNotWork: 'This activity is unavailable for some reasons (banned/ended/paused/not started...)' + ' (if it is a script misjudgment, please give us feedback in time), is it closed?',
confirm: 'Confirm',
cancel: 'Cancel',
unKnown: 'Unknown',
unKnownTaskType: 'Unrecognized task',
doing: 'Doing a task',
allTasksComplete: 'All tasks have been completed!',
getTaskIdFailed: 'Failed to obtain task Id!',
initSuccess: '%0 was initialized successfully!',
initFailed: '%0 initialization failed!',
errorLink: 'Link error: %0',
needInit: 'Please initialize first',
verifyingAuth: 'Verifying %0 token...',
updatingAuth: 'Update %0 token...',
refreshingToken: 'Refreshing %0 token...',
settingToken: 'Setting %0 token...',
steamStoreTab: 'Steam store (new tab)',
steamCommunityTab: 'Steam community(new tab)',
initing: 'Initializing...',
getFailed: 'Failed to get %0!',
checkLoginFailed: 'Failed to detect login status!',
checkLeftKeyFailed: 'Failed to detect the remaining keys!',
userId: 'User Id',
joiningGiveaway: 'Joining giveaway',
needJoinGiveaway: 'Need to join the giveaway first',
cannotUndo: 'This website does not support canceling tasks',
verifyAuth: 'Verifying %0 token...',
closePageNotice: 'f this page does not close automatically, please close this page yourself.',
errorReport: 'A script error is detected, do you want to report the BUG?',
visitingLink: 'Visiting link ...',
doTask: 'DoTask',
undoTask: 'UndoTask',
verifyTask: 'Verify',
getKey: 'GetKey',
selectAll: 'SelectAll',
selectNone: 'SelectNone',
invertSelect: 'InvertSelect',
doFreeTask: 'FreeTask',
doPointTask: 'PointTask',
skipTaskOption: 'Skip task has been configured in the settings',
other: 'Other',
globalOptions: 'Global Options',
checkLogin: 'Login detection</br>Need to log in to the website automatically log in, part of this website supports.',
checkLeftKey: 'Key remaining detection</br>The end of the giveaway event prompts whether to close or not, part of this website supports.',
twitterVerifyId: 'Verify Twitter token by trying to follow the account.</br>The default is the official Twitter account 783214.</br>' + 'If you don\'t want to follow the official account, you can change it to your own account.',
youtubeVerifyChannel: 'Verify YouTube token by trying to subscribe to the channel.</br>' + 'The default is the official YouTube channel UCrXUsMBcfTVqwAS7DKg9C0Q.</br>' + 'If you don\'t want to follow the official channel, you can change it to your own channel.',
autoUpdateSource: 'The source to update</br>github: Fast update.</br>jsdelivr: Update is delayed.</br>' + 'standby: Standby source.</br>auto: Try to update using github, jsdelivr, standby sources in turn.',
saveGlobalOptions: 'SaveSettings',
settingPage: 'Setting Page',
name: 'Name',
version: 'Version',
scriptManager: 'Script Manager',
script: 'Script',
environment: 'Environment',
os: 'OS',
browser: 'Browser',
getId: 'Get %0 id',
getTwitterUserId: 'Get Twitter user id (Get id function is only available on the settings page).',
getYoutubeChannelId: 'Get Youtube channel id (Get id function is only available on the settings page).',
showButton: 'ShowButton',
hideButton: 'HideButton',
showLog: 'ShowLog',
hideLog: 'HideLog',
defaultShowButton: 'Default display button',
defaultShowLog: 'Display log by default',
debug: 'Output debug log, do not enable this option!',
receivePreview: 'Receive preview updates',
position: 'Component position',
buttonSideX: 'Horizontal positioning of the button area (real-time preview function is only available on the setting page).' + '</br>left: left | right: right',
buttonSideY: 'The button area is positioned in the vertical direction (real-time preview function is only available on the settings page).' + '</br>top: top | bottom: bottom',
buttonDistance: 'The distance between the button area and the edge (the real-time preview function is only available on the setting page).' + '</br> Format: X distance, Y distance',
showButtonSideX: 'ShowButton horizontal positioning (real-time preview function is only available on the setting page).' + '</br>left: left | right: right',
showButtonSideY: 'ShowButton vertical positioning (real-time preview function is only available on the setting page).' + '</br>top: top | bottom: bottom',
showButtonDistance: 'The distance between the ShowButton and the edge (real-time preview function is only available on the setting page).' + '</br> Format: X distance, Y distance',
logSideX: 'Horizontal positioning of the log area (real-time preview function is only available on the setting page).' + '</br>left: left | right: right',
logSideY: 'Vertical positioning of the log area (real-time preview function is only available on the setting page).' + '</br>top: top | bottom: bottom',
logDistance: 'The distance between the log area and the edge (the real-time preview function is only available on the setting page).' + '</br> Format: X distance, Y distance',
hotKey: 'Shortcut key',
doTaskKey: 'DoTask shortcut keys</br> (real-time preview function is only available on the settings page).',
undoTaskKey: 'UndoTask shortcut keys</br> (real-time preview function is only available on the settings page).',
toggleLogKey: 'ShowLog/HideLog shortcut keys</br> (real-time preview function is only available on the settings page).',
tasksHistory: 'TasksHistory',
clearHistory: 'Clear history',
clearHistoryFinished: 'The mission history has been cleared!',
deleteTask: 'Delete task',
lastChangeTime: 'Last Change Time',
clearTaskFinished: 'Delete the following tasks completed!',
clearTaskFailed: 'Failed to delete the task, the task name was not found!',
syncData: 'DataSync',
settingData: 'Uploading data...',
gettingData: 'Getting data...',
help: 'Help',
fileName: 'Filename',
upload2gist: 'Sync to Gist',
downloadFromGist: 'Sync from Gist',
saveAndTest: 'Save configuration and test',
testSuccess: 'Test success!',
testFailed: 'Test failed!',
saveAndTestNotice: 'Please save the configuration and test first!',
processingData: 'Processing data...',
updatingData: 'Uploading data...',
syncDataSuccess: 'Synchronized data successfully!',
syncDataFailed: 'Failed to synchronize data, please check the error message on the console!',
downloadingData: 'Downloading data...',
checkedNoData: 'No remote data is detected, please confirm whether the configuration is correct!',
savingData: 'Saving data...',
syncHistory: 'Synchronize tasks history',
checkUpdateFailed: 'Check update failed',
newVersionNotice: 'Checked a new version V%0, <a class="high-light" href="%1" target="_blank">click to update</a>',
language: 'Language</br> Currently only supports zh: Chinese, en: English',
gistOptions: 'Gist Settings',
swalNotice: 'It is detected that you are installing the V4 version script for the first time' + ', please go to read the READ ME FIRST content before use!',
echoNotice: 'It is detected that you are installing the V4 version script for the first time' + ', please <a class="high-light" href="%0" target="_blank">click here</a> to read the READ ME FIRST content before use!',
noticeLink: 'https://auto-task-doc.js.org/en/guide/#read-me-first',
toGithub: 'Feedback(Github)',
toKeylol: 'Feedback(Keylol)',
copySuccess: 'The error message has been copied to the clipboard. Do you want to go to the Keylol forum to give feedback?',
copyFailed: 'Please copy the error information below and report back to the Keylol forum!',
updateText: 'Updates in version %0:',
Active: 'Active',
Ended: 'Ended',
Banned: 'Banned',
Paused: 'Paused',
notStart: 'notStart',
noRemoteData: 'No data remotely',
errorRemoteDataFormat: 'Remote data has wrong format',
updateHistory: '<a class="high-light" href="https://auto-task-doc.js.org/logs/" target="_blank">Click here</a>' + ' to view the historical update record.',
AsfEnabled: 'Use ASF to do Steam related tasks (requires <a href="https://github.com/chr233/ASFEnhance" target="_blank">ASFEnhance</a> plugin)',
steamWeb: 'Use Steam Web API to do Steam related tasks simultaneously',
preferASF: 'Prefer ASF to do Steam related tasks',
AsfIpcUrl: 'ASF IPC URL',
AsfIpcPassword: 'ASF IPC Password',
curatorLimitNotice: 'Failed: Maybe the curator follow limit is reached, please <a class="high-light" href="https://store.steampowered.com/curators/home/" target="_blank">unfollow some curators</a> and try again',
unknownScriptHandler: 'Unknown script handler, compatibility unknown',
debugModeNotice: 'Detected DEBUG mode enabled, please close it if it is not needed!',
steamWebApiKey: 'Steam Web API Key<br>Used to detect game idle status, <a href="https://steamcommunity.com/dev/apikey" target="_blank">click to apply</a>',
groups: 'Group',
officialGroups: 'Official Group',
wishlists: 'Wishlist',
follows: 'Follow Game',
forums: 'Forum',
workshops: 'Favorite Workshop',
curators: 'Curator',
workshopVotes: 'Voteup Workshop',
announcements: 'Announcement',
steamCommunity: 'Steam Community',
steamStore: 'Steam Store',
licenses: 'Add License',
playtests: 'Playtest Access',
needLoginSteamStore: 'Please <a href="https://store.steampowered.com/login/" target="_blank">log in to the Steam Store</a>',
needLoginSteamCommunity: 'Please <a href="https://steamcommunity.com/login/home/" target="_blank">log in to the Steam Community</a>',
joiningSteamGroup: 'Joining Steam Group',
leavingSteamGroup: 'Leaving Steam Group',
gettingSteamGroupId: 'Getting Steam Group Id',
joiningSteamOfficialGroup: 'Joining Steam Official Group',
leavingSteamOfficialGroup: 'Leaving Steam Official Group',
gettingSteamOfficialGroupId: 'Getting Steam Official Group Id',
subscribingForum: 'Subscribing the Steam Forum',
unsubscribingForum: 'Unsubscribing the Steam Forum',
gettingForumId: 'Getting Steam Forum Id',
followingCurator: 'Following Steam Curator',
unfollowingCurator: 'Unfollowing Steam Curator',
gettingCuratorId: 'Getting Steam Curator Id',
addingToWishlist: 'Adding the game to the Steam wishlist',
removingFromWishlist: 'Removing the game from the Steam wishlist',
followingGame: 'Following Steam games',
unfollowingGame: 'Unfollowing Steam games',
favoritingWorkshop: 'Favouring Steam Workshop Items',
unfavoritingWorkshop: 'Unfavoriting Steam Workshop Items',
gettingWorkshopAppId: 'Getting Steam Workshop Item Id',
votingUpWorkshop: 'Liking Steam workshop items',
gettingAnnouncementParams: 'Getting Steam announcement information',
likingAnnouncement: 'Liking Steam announcement',
changingArea: 'Changing Steam area: %0...',
notNeededChangeArea: 'The current area does not need to be changed',
noAnotherArea: 'Please check whether the proxy is turned on correctly',
gettingAreaInfo: 'Getting Steam area information...',
changeAreaNotice: 'Suspected of a locked zone game, try to change the zone to execute',
steamFinishNotice: 'Steam task completed, try to change the shopping cart area back to ',
gettingSubid: 'Getting subid',
addingFreeLicense: 'Adding free license',
missParams: 'Missing parameters',
gettingLicenses: 'Getting licenses...',
requestingPlayTestAccess: 'Requesting play test access',
gettingDemoAppid: 'Getting demo appid for steam game',
tryChangeAreaNotice: 'This function cannot detect whether the game is limited, so it will try to change the area before entering the library' + '. Failure to change the area will not affect the subsequent storage.',
versionNotMatched: 'The script manager version is too low, requiring TamperMonkey >= 5.2.0 or TamperMonkey Beta >= 5.2.6196',
gettingUserInfo: 'Getting steam user community credentials...',
retry: 'Retry',
owned: 'Owned',
redirect: 'Redirect',
noSubid: 'skip due to unrecognized',
noASFInstance: 'ASF is not enabled, skip idle time task',
initingASF: 'Initing ASF...',
playingGames: 'Playing games [%0]...',
stoppingPlayGames: 'Stopping play games...',
stopPlayTimeTitle: 'The Steam game idle time has finished. Do you want to end it?',
stopPlayTimeText: 'Time out: %0 minutes',
ASFNotSupportted: 'The current function (%0) cannot be implemented by ASF, skipping',
checkingPlayGamesStatus: 'Checking play games status...',
gettingSteamId: 'Getting Steam ID...',
checkingPlayStatus: 'Checking play status...',
noPlayStatus: 'Game not running',
servers: 'Server',
joiningDiscordServer: 'Joining Discord Server',
leavingDiscordServer: 'Leaving Discord Server',
gettingDiscordGuild: 'Getting Discord server Id',
getDiscordAuthFailed: 'Failed to get Discord token, please check whether the Discord account is logged in',
discordImportantNotice: 'Important Reminder! ! !',
discordImportantNoticeText: 'Due to the background update of the Discord website, currently using this script to join a group may cause the Discord account to be forcibly logged out, and two-step verification is required to log in normally, please use it with caution! ! !',
continueDiscordTask: 'Do Discord tasks this time.',
skipDiscordTask: 'Skip Discord tasks this time.',
continueAndDontRemindAgain: 'Always do Discord tasks and do not remind again.',
gettingDiscordXContextProperties: 'Getting Discord X context properties...',
captchaNeeded: 'Captcha detected, please complete it manually!',
users: 'User',
loginIns: 'Please <a href="https://www.instagram.com/accounts/login/" target="_blank">log in to Instagram</a>',
insBanned: 'Your Instagram account has been banned',
verifyingInsAuth: 'Verifying Instagram token...',
gettingInsUserId: 'Getting Instagram user Id',
followingIns: 'Following Instagram user',
unfollowingIns: 'Unfollowing Instagram user',
reddits: 'Reddit/User',
loginReddit: 'Please <a href="https://www.reddit.com/login/" target="_blank">log in to Reddit</a>',
changingRedditVersion: 'Switching Reddit to a new version page...',
joiningReddit: 'Joining the Reddit',
leavingReddit: 'Leaving the Reddit',
followingRedditUser: 'Following Reddit User',
unfollowingRedditUser: 'Unfollowing Reddit User',
channels: 'Channel',
followingTwitchChannel: 'Following Twitch Channel',
unfollowingTwitchChannel: 'Unfollowing Twitch Channel',
gettingTwitchChannelId: 'Getting Twitch Channel Id',
checkingTwitchIntegrity: 'Checking Twitch integrity...',
twitterUser: 'Twitter User',
retweets: 'Retweet',
gettingTwitterUserId: 'Getting Twitter User Id',
followingTwitterUser: 'Following Twitter User',
unfollowingTwitterUser: 'Unfollowing Twitter User',
retweetting: 'Retweetting',
unretweetting: 'Unretweetting',
names: 'Group/Public/Wall',
loginVk: 'Please <a href="https://vk.com/login/" target="_blank">log in to Vk</a>',
gettingVkId: 'Getting Vk task Id',
joiningVkGroup: 'Joining Vk Group',
leavingVkGroup: 'Leaving Vk Group',
joiningVkPublic: 'Joining Vk Public',
leavingVkPublic: 'Leaving Vk Public',
sendingVkWall: 'Sending Vk Wall',
deletingVkWall: 'Deleting Vk Wall',
youtubeChannel: 'YouTube Channel',
likes: 'Like',
loginYtb: 'Please <a href="https://accounts.google.com/ServiceLogin?service=youtube" target="_blank">log in to YouTube</a>',
tryUpdateYtbAuth: 'Please try to <a href="https://www.youtube.com/#auth" target="_blank">update YouTube token</a>',
gettingYtbToken: 'Getting YouTube Token...',
followingYtbChannel: 'Subscribing to YouTube channel',
unfollowingYtbChannel: 'Unsubscribing to YouTube channel',
likingYtbVideo: 'Liking YouTube video',
unlikingYtbVideo: 'Unliking YouTube video',
giveKeyNoticeBefore: 'Each verification interval is 15s',
giveKeyNoticeAfter: 'If there is no key, please check at <a href="https://givekey.ru/profile" target="_blank">https://givekey.ru/profile</a>',
noPoints: 'Not enough points, skip the lottery',
getNeedPointsFailed: 'ailed to obtain the required points, skip the lottery',
joiningLottery: 'Joining the lottery',
doingGleamTask: 'Doing Gleam Task...',
gettingGleamLink: 'Getting Gleam task link...',
gleamTaskNotice: 'If this page has not been closed for a long time, please close it yourself after completing any task!',
verifiedGleamTasks: 'Attempted to verify all tasks. If the verification fails, please try to verify manually or complete it!',
campaign: 'ReCAPTCHA detected, please complete it manually! 3 seconds later, re-verify...',
gsNotice: 'In order to avoid getting the "0000-0000-0000" key, the "Grab Key" button has been hidden,' + ' please close the script when obtaining the key!',
giveeClubVerifyNotice: 'Verifying task...',
giveeClubVerifyFinished: 'Wait for the verification to complete and join it by yourself',
doingKeyhubTask: 'Doing Keyhub Task...',
SweepWidgetNotice: 'The task is being processed and verified. ' + 'There is an interval of 1~3s for each verification task to prevent the triggering of too fast verification warning...',
tasksNotCompleted: 'Tasks Not Completed',
notConnect: 'Social platform is not connectted, skip task: %0',
tgTaskNotice: 'The telegram task is checked, need to do it yourself!',
updatingUserData: 'Updating user data...',
gettingUserGames: 'Getting user games...',
confirmingTask: 'Confirming task...',
unSupporttedTaskType: 'Unsupportted task type: %0',
taskNotFinished: 'There are tasks not completed, do not get the key',
logCopied: 'Full log has been copied to the clipboard, please go to feedback!'
};
const languages = {
zh: data$1,
en: data
};
const SUPPORTED_LANGUAGES = [ 'zh', 'en' ];
const getCurrentLanguage = () => {
const userLanguage = globalOptions.other.language;
return SUPPORTED_LANGUAGES.includes(userLanguage) ? userLanguage : 'en';
};
const replacePlaceholders = (text, args) => text.replace(/%([\d]+)/g, ((_, index) => args[parseInt(index, 10)] || ''));
const I18n = (key, ...args) => {
const currentLanguage = getCurrentLanguage();
const translation = languages[currentLanguage]?.[key];
if (!translation) {
console.warn(`Missing translation for key: ${key} in language: ${currentLanguage}`);
return key;
}
return replacePlaceholders(translation, args);
};
var DebugLevel;
(function(DebugLevel) {
DebugLevel['ERROR'] = 'error';
DebugLevel['WARN'] = 'warn';
DebugLevel['INFO'] = 'info';
DebugLevel['DEBUG'] = 'debug';
DebugLevel['TRACE'] = 'trace';
})(DebugLevel || (DebugLevel = {}));
const defaultConfig = {
enabled: false,
level: DebugLevel.INFO,
prefix: 'Auto-Task',
styles: {
error: 'color:#ff0000;font-weight:bold',
warn: 'color:#ffa500',
info: 'color:#a7a7a7',
debug: 'color:#808080',
trace: 'color:#87ceeb'
},
showTimestamp: true
};
class Debugger {
config;
levelPriority;
constructor(config = {}) {
this.config = {
...defaultConfig,
...config
};
this.levelPriority = {
error: 0,
warn: 1,
info: 2,
debug: 3,
trace: 4
};
}
getTimestamp() {
return (new Date).toLocaleString();
}
shouldLog(level) {
return this.config.enabled && this.levelPriority[level] <= this.levelPriority[this.config.level];
}
formatMessage(level, message) {
const parts = [ this.config.prefix ];
if (this.config.showTimestamp) {
parts.push(`[${this.getTimestamp()}]`);
}
parts.push(`[${level.toUpperCase()}]:`);
parts.push(message);
return parts.join(' ');
}
log(level, message, ...args) {
if (!this.shouldLog(level)) {
return;
}
const formattedMessage = this.formatMessage(level, message);
const style = this.config.styles[level];
if (args.length > 0) {
console.groupCollapsed('%c%s', style, formattedMessage);
args.forEach((arg => {
console.log(util.inspect(arg, {
showHidden: true,
depth: null,
colors: false
}));
}));
console.groupEnd();
} else {
console.log('%c%s', style, formattedMessage);
}
}
error(message, ...args) {
this.log(DebugLevel.ERROR, message, ...args);
}
warn(message, ...args) {
this.log(DebugLevel.WARN, message, ...args);
}
info(message, ...args) {
this.log(DebugLevel.INFO, message, ...args);
}
debug(message, ...args) {
this.log(DebugLevel.DEBUG, message, ...args);
}
trace(message, ...args) {
this.log(DebugLevel.TRACE, message, ...args);
if (this.shouldLog(DebugLevel.TRACE)) {
console.trace();
}
}
updateConfig(config) {
this.config = {
...this.config,
...config
};
}
enable() {
this.config.enabled = true;
}
disable() {
this.config.enabled = false;
}
setLevel(level) {
this.config.level = level;
}
}
let debugInstance;
const initDebug = () => {
if (!debugInstance) {
debugInstance = new Debugger({
enabled: window.DEBUG || false
});
if (window.DEBUG) {
debugInstance.setLevel(DebugLevel.DEBUG);
}
}
return debugInstance;
};
const debug = (...args) => {
const instance = initDebug();
return instance.debug(...args);
};
const getRunLogs = () => {
debug('开始获取运行日志');
const logElements = $('#auto-task-info>li');
const logs = logElements.length > 0 ? $.makeArray(logElements).map((element => element.innerText)).join('\n') : '';
debug('运行日志获取完成', {
logsLength: logs.length
});
return logs;
};
const getEnvironmentInfo = async () => {
debug('开始获取环境信息');
const envInfo = {
website: window.location.href,
browser: JSON.stringify(await browser.getInfo(), null, 2),
manager: `${GM_info.scriptHandler} ${GM_info.version}`,
userScript: GM_info.script.version,
logs: '',
runLogs: getRunLogs()
};
debug('环境信息获取完成', envInfo);
return envInfo;
};
const buildGithubIssueParams = async (name, errorStack, envInfo) => {
debug('开始构建GitHub Issue参数', {
name: name,
errorStackLength: errorStack.length
});
const params = {
title: `[BUG] 脚本报错: ${name}`,
labels: 'bug',
template: 'bug_report.yml',
website: envInfo.website,
browser: envInfo.browser,
manager: envInfo.manager,
'user-script': envInfo.userScript,
logs: errorStack || '',
'run-logs': ''
};
const runLogs = window.__allLogs.join('\n');
await GM_setClipboard(runLogs);
debug('GitHub Issue参数构建完成', params);
return params;
};
const generateGithubLink = async (name, errorStack, envInfo) => {
debug('开始生成GitHub Issue链接');
const params = new URLSearchParams(await buildGithubIssueParams(name, errorStack, envInfo));
const link = `https://github.com/HCLonely/auto-task/issues/new?${params.toString()}`;
debug('GitHub Issue链接生成完成', {
link: link
});
return link;
};
const logError = (name, errorStack) => {
debug('记录错误日志', {
name: name
});
console.log('%c%s', 'color:white;background:red', `Auto-Task[Error]: ${name}\n${errorStack}`);
};
const handleErrorReport = async (platform, name, errorStack, envInfo) => {
debug('开始处理错误报告', {
platform: platform,
name: name
});
{
const githubLink = await generateGithubLink(name, errorStack, envInfo);
debug('打开GitHub Issue链接', {
githubLink: githubLink
});
GM_openInTab(githubLink, {
active: true
});
}
};
async function throwError(error, name) {
debug('开始处理错误', {
name: name,
error: error
});
if (window.TRACE) {
debug('启用跟踪模式');
console.trace('%cAuto-Task[Trace]:', 'color:blue');
}
const errorStack = error.stack || '';
logError(name, errorStack);
debug('获取环境信息');
const envInfo = await getEnvironmentInfo();
envInfo.logs = errorStack;
debug('显示错误报告对话框');
const {isConfirmed: isConfirmed} = await Swal.fire({
title: I18n('errorReport'),
icon: 'error',
showCancelButton: true,
confirmButtonText: I18n('toGithub'),
cancelButtonText: I18n('close')
});
if (isConfirmed) {
debug('用户确认提交错误报告');
await handleErrorReport('github', name, errorStack, envInfo);
Swal.fire({
title: I18n('logCopied'),
icon: 'success',
showConfirmButton: false,
showCancelButton: true,
cancelButtonText: I18n('close')
});
} else {
debug('用户取消提交错误报告');
}
}
const parseHeaders = headerString => {
debug('开始解析HTTP头', {
headerString: headerString
});
const headers = {};
if (!headerString) {
debug('HTTP头为空,返回空对象');
return headers;
}
headerString.split('\n').forEach((header => {
const [name, ...values] = header.trim().split(':');
const value = values.join(':').trim();
if (!name || !value) {
return;
}
if (headers[name]) {
headers[name] = Array.isArray(headers[name]) ? [ ...headers[name], value ] : [ headers[name], value ];
} else {
headers[name] = value;
}
}));
if (headers['set-cookie'] && !Array.isArray(headers['set-cookie'])) {
headers['set-cookie'] = [ headers['set-cookie'] ];
}
debug('HTTP头解析完成', {
headers: headers
});
return headers;
};
const processResponse = (data, options) => {
debug('开始处理响应数据', {
responseType: options.responseType
});
const headers = parseHeaders(data.responseHeaders);
data.responseHeadersText = data.responseHeaders;
data.responseHeaders = headers;
data.finalUrl = headers.location || data.finalUrl;
debug('响应头处理完成', {
finalUrl: data.finalUrl
});
if (options.responseType === 'json' && data?.response && typeof data.response !== 'object') {
debug('尝试解析JSON响应');
try {
data.response = JSON.parse(data.responseText);
debug('JSON解析成功');
} catch {
debug('JSON解析失败,保持原始响应');
}
}
};
const httpRequest = async (options, times = 0) => {
debug('开始HTTP请求', {
url: options.url,
method: options.method,
retryTimes: times
});
if (window.TRACE) {
console.trace('%cAuto-Task[Trace]:', 'color:blue');
}
try {
const result = await new Promise((resolve => {
const requestObj = {
fetch: true,
timeout: 3e4,
ontimeout: data => {
debug('请求超时', {
url: options.url
});
resolve({
result: 'Error',
statusText: 'Timeout',
status: 601,
data: data,
options: options
});
},
onabort: () => {
debug('请求被中止', {
url: options.url
});
resolve({
result: 'Error',
statusText: 'Aborted',
status: 602,
data: undefined,
options: options
});
},
onerror: data => {
debug('请求发生错误', {
url: options.url,
error: data
});
resolve({
result: 'Error',
statusText: 'Error',
status: 603,
data: data,
options: options
});
},
onload: data => {
debug('请求加载完成', {
url: options.url,
status: data.status
});
processResponse(data, options);
resolve({
result: 'Success',
statusText: 'Load',
status: 600,
data: data,
options: options
});
},
...options,
responseType: options.dataType || options.responseType
};
debug('发送请求', {
requestObj: requestObj
});
GM_xmlhttpRequest(requestObj);
}));
if (window.DEBUG) {
console.log('%cAuto-Task[httpRequest]:', 'color:blue', result);
}
if (result.status !== 600 && times < 2) {
debug('请求失败,准备重试', {
status: result.status,
retryTimes: times + 1
});
return await httpRequest(options, times + 1);
}
debug('请求完成', {
status: result.status,
result: result.result
});
return result;
} catch (error) {
debug('请求发生JavaScript错误', {
error: error
});
console.log('%cAuto-Task[httpRequest]:', 'color:red', JSON.stringify({
errorMsg: error,
options: options
}));
throwError(error, 'httpRequest');
return {
result: 'JsError',
statusText: 'Error',
status: 604,
error: error,
options: options
};
}
};
var ASF = '<?xml version="1.0" encoding="UTF-8"?>\n<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="16" height="16" viewBox="0 0 16 16" version="1.1">\n<g id="surface1">\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(11.372549%,17.254902%,26.274511%);fill-opacity:1;" d="M 7.652344 0.175781 C 8.046875 0.351562 8.210938 0.59375 8.441406 0.953125 C 9.007812 1.746094 9.644531 2.539062 10.640625 2.761719 C 12.011719 2.949219 13.429688 2.882812 14.808594 2.796875 C 14.90625 2.792969 15.003906 2.785156 15.101562 2.78125 C 15.1875 2.773438 15.273438 2.769531 15.363281 2.761719 C 15.429688 2.769531 15.496094 2.777344 15.566406 2.78125 C 15.761719 3.082031 15.757812 3.191406 15.75 3.539062 C 15.75 3.625 15.746094 3.710938 15.746094 3.800781 C 15.742188 3.867188 15.742188 3.933594 15.738281 4 C 15.824219 4.027344 15.910156 4.058594 16 4.085938 C 15.988281 4.34375 15.976562 4.597656 15.960938 4.851562 C 15.957031 4.925781 15.957031 5 15.953125 5.074219 C 15.945312 5.175781 15.945312 5.175781 15.941406 5.285156 C 15.9375 5.378906 15.9375 5.378906 15.929688 5.480469 C 15.914062 5.652344 15.914062 5.652344 15.824219 5.914062 C 15.816406 6.058594 15.8125 6.203125 15.8125 6.34375 C 15.8125 6.433594 15.8125 6.519531 15.808594 6.613281 C 15.808594 6.707031 15.808594 6.800781 15.808594 6.898438 C 15.804688 7.097656 15.804688 7.300781 15.800781 7.5 C 15.796875 7.816406 15.792969 8.132812 15.792969 8.449219 C 15.789062 8.753906 15.785156 9.058594 15.78125 9.363281 C 15.78125 9.457031 15.78125 9.550781 15.78125 9.648438 C 15.777344 9.738281 15.777344 9.824219 15.773438 9.914062 C 15.773438 10.03125 15.773438 10.03125 15.773438 10.148438 C 15.738281 10.347656 15.738281 10.347656 15.625 10.503906 C 15.386719 10.671875 15.152344 10.625 14.871094 10.609375 C 14.867188 10.675781 14.859375 10.742188 14.855469 10.808594 C 14.816406 11.367188 14.746094 11.890625 14.609375 12.433594 C 14.3125 12.535156 14.105469 12.507812 13.804688 12.441406 C 13.6875 12.414062 13.6875 12.414062 13.570312 12.390625 C 13.480469 12.367188 13.480469 12.367188 13.390625 12.347656 C 13.359375 12.515625 13.359375 12.515625 13.324219 12.683594 C 13.226562 13.144531 13.054688 13.566406 12.871094 14 C 12.75 13.988281 12.628906 13.980469 12.511719 13.96875 C 12.410156 13.957031 12.410156 13.957031 12.308594 13.949219 C 12.050781 13.90625 11.8125 13.828125 11.566406 13.738281 C 11.492188 13.867188 11.492188 13.867188 11.417969 14 C 11.355469 14.113281 11.292969 14.222656 11.226562 14.335938 C 11.195312 14.394531 11.164062 14.449219 11.132812 14.507812 C 11 14.738281 10.882812 14.941406 10.695312 15.128906 C 10.257812 15.097656 9.957031 14.863281 9.601562 14.625 C 9.402344 14.503906 9.402344 14.503906 9.222656 14.542969 C 9.015625 14.617188 8.925781 14.703125 8.777344 14.863281 C 8.730469 14.914062 8.683594 14.964844 8.632812 15.019531 C 8.535156 15.125 8.433594 15.234375 8.335938 15.339844 C 7.972656 15.726562 7.972656 15.726562 7.675781 15.792969 C 7.4375 15.726562 7.367188 15.660156 7.21875 15.460938 C 7 15.1875 6.773438 14.941406 6.523438 14.695312 C 6.375 14.550781 6.230469 14.40625 6.085938 14.261719 C 6.023438 14.195312 5.957031 14.128906 5.890625 14.0625 C 2.175781 10.347656 2.175781 10.347656 1.945312 10.117188 C 1.738281 9.910156 1.527344 9.707031 1.316406 9.503906 C 1.160156 9.351562 1 9.199219 0.84375 9.042969 C 0.722656 8.925781 0.722656 8.925781 0.597656 8.808594 C 0.519531 8.734375 0.445312 8.660156 0.363281 8.582031 C 0.296875 8.515625 0.226562 8.445312 0.152344 8.375 C -0.03125 8.132812 -0.0351562 8.039062 0 7.738281 C 0.117188 7.570312 0.117188 7.570312 0.28125 7.402344 C 0.34375 7.339844 0.40625 7.277344 0.472656 7.210938 C 0.542969 7.144531 0.613281 7.074219 0.683594 7.003906 C 0.757812 6.933594 0.828125 6.859375 0.902344 6.785156 C 1.101562 6.585938 1.304688 6.386719 1.503906 6.1875 C 1.714844 5.980469 1.921875 5.773438 2.132812 5.5625 C 2.480469 5.214844 2.832031 4.863281 3.183594 4.515625 C 3.683594 4.023438 4.175781 3.53125 4.671875 3.039062 C 5.015625 2.695312 5.359375 2.355469 5.699219 2.015625 C 5.785156 1.929688 5.867188 1.847656 5.949219 1.765625 C 6.222656 1.492188 6.496094 1.222656 6.773438 0.949219 C 6.839844 0.882812 6.910156 0.8125 6.980469 0.742188 C 7.078125 0.648438 7.078125 0.648438 7.171875 0.550781 C 7.226562 0.496094 7.285156 0.441406 7.339844 0.386719 C 7.476562 0.261719 7.476562 0.261719 7.652344 0.175781 Z M 7.652344 0.175781 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(5.490196%,7.843138%,14.901961%);fill-opacity:1;" d="M 10.433594 2.78125 C 11.101562 2.777344 11.765625 2.773438 12.433594 2.769531 C 12.742188 2.769531 13.050781 2.765625 13.359375 2.761719 C 13.714844 2.757812 14.070312 2.757812 14.425781 2.757812 C 14.539062 2.753906 14.648438 2.753906 14.761719 2.753906 C 14.867188 2.753906 14.96875 2.753906 15.074219 2.753906 C 15.210938 2.753906 15.210938 2.753906 15.347656 2.75 C 15.421875 2.761719 15.492188 2.773438 15.566406 2.78125 C 15.761719 3.082031 15.757812 3.191406 15.75 3.539062 C 15.75 3.625 15.746094 3.710938 15.746094 3.800781 C 15.742188 3.867188 15.742188 3.933594 15.738281 4 C 15.824219 4.027344 15.910156 4.058594 16 4.085938 C 15.988281 4.332031 15.980469 4.578125 15.96875 4.824219 C 15.964844 4.929688 15.964844 4.929688 15.960938 5.035156 C 15.9375 5.492188 15.863281 5.90625 15.738281 6.347656 C 15.675781 6.3125 15.613281 6.273438 15.546875 6.234375 C 15.011719 5.921875 14.472656 5.609375 13.9375 5.296875 C 13.761719 5.195312 13.582031 5.089844 13.40625 4.988281 C 12.996094 4.746094 12.585938 4.507812 12.167969 4.273438 C 10.847656 3.527344 10.847656 3.527344 10.433594 2.957031 C 10.433594 2.898438 10.433594 2.839844 10.433594 2.78125 Z M 10.433594 2.78125 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(95.686275%,96.078432%,96.470588%);fill-opacity:1;" d="M 7.566406 5.304688 C 7.878906 5.410156 7.90625 5.480469 8.058594 5.761719 C 8.117188 5.859375 8.117188 5.859375 8.175781 5.964844 C 8.269531 6.195312 8.277344 6.363281 8.261719 6.609375 C 8.355469 6.550781 8.449219 6.492188 8.550781 6.433594 C 8.777344 6.296875 8.855469 6.261719 9.128906 6.261719 C 9.136719 6.425781 9.140625 6.589844 9.148438 6.753906 C 9.152344 6.894531 9.152344 6.894531 9.15625 7.035156 C 9.125 7.34375 9.058594 7.492188 8.871094 7.738281 C 8.703125 7.871094 8.703125 7.871094 8.527344 7.972656 C 8.46875 8.007812 8.410156 8.042969 8.351562 8.078125 C 8.292969 8.109375 8.234375 8.140625 8.175781 8.175781 C 8.113281 8.207031 8.054688 8.238281 7.996094 8.273438 C 7.710938 8.398438 7.511719 8.335938 7.21875 8.261719 C 6.832031 8.070312 6.476562 7.859375 6.261719 7.476562 C 6.167969 7.203125 6.164062 7.007812 6.167969 6.722656 C 6.167969 6.636719 6.171875 6.550781 6.171875 6.460938 C 6.171875 6.394531 6.171875 6.328125 6.175781 6.261719 C 6.492188 6.332031 6.753906 6.457031 7.042969 6.609375 C 7.050781 6.480469 7.050781 6.480469 7.058594 6.347656 C 7.113281 5.9375 7.332031 5.636719 7.566406 5.304688 Z M 7.566406 5.304688 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(5.882353%,8.235294%,15.294118%);fill-opacity:1;" d="M 15.652344 6.433594 C 15.679688 6.433594 15.710938 6.433594 15.738281 6.433594 C 15.746094 6.988281 15.75 7.539062 15.753906 8.09375 C 15.753906 8.28125 15.757812 8.46875 15.757812 8.65625 C 15.761719 8.925781 15.761719 9.195312 15.765625 9.464844 C 15.765625 9.550781 15.765625 9.636719 15.769531 9.722656 C 15.769531 9.800781 15.769531 9.878906 15.769531 9.960938 C 15.769531 10.027344 15.769531 10.097656 15.769531 10.167969 C 15.730469 10.398438 15.664062 10.472656 15.476562 10.609375 C 15.15625 10.625 15.15625 10.625 14.871094 10.609375 C 14.867188 10.675781 14.859375 10.742188 14.855469 10.808594 C 14.816406 11.367188 14.746094 11.890625 14.609375 12.433594 C 14.3125 12.535156 14.105469 12.507812 13.804688 12.441406 C 13.726562 12.421875 13.648438 12.40625 13.570312 12.390625 C 13.480469 12.367188 13.480469 12.367188 13.390625 12.347656 C 13.449219 12.027344 13.535156 11.730469 13.644531 11.425781 C 13.679688 11.332031 13.710938 11.238281 13.746094 11.144531 C 13.78125 11.046875 13.816406 10.953125 13.851562 10.851562 C 13.902344 10.710938 13.902344 10.710938 13.957031 10.566406 C 14.054688 10.289062 14.160156 10.015625 14.261719 9.738281 C 14.292969 9.65625 14.320312 9.574219 14.351562 9.488281 C 14.613281 8.792969 14.613281 8.792969 14.929688 8.523438 C 15.359375 8.140625 15.347656 7.753906 15.40625 7.203125 C 15.441406 6.886719 15.472656 6.707031 15.652344 6.433594 Z M 15.652344 6.433594 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(13.725491%,18.431373%,37.254903%);fill-opacity:1;" d="M 13.566406 9.824219 C 13.609375 9.914062 13.609375 9.914062 13.652344 10 C 13.539062 10.21875 13.417969 10.425781 13.289062 10.636719 C 13.207031 10.769531 13.125 10.90625 13.046875 11.039062 C 13.003906 11.109375 12.960938 11.183594 12.914062 11.253906 C 12.691406 11.632812 12.46875 12.015625 12.25 12.398438 C 12.207031 12.472656 12.160156 12.550781 12.117188 12.628906 C 11.8125 13.160156 11.53125 13.703125 11.269531 14.253906 C 11.113281 14.582031 10.933594 14.855469 10.695312 15.128906 C 10.3125 15.078125 10.042969 14.933594 9.710938 14.734375 C 9.574219 14.648438 9.574219 14.648438 9.429688 14.566406 C 9.359375 14.523438 9.289062 14.480469 9.21875 14.433594 C 9.253906 14.046875 9.523438 13.835938 9.789062 13.582031 C 9.832031 13.539062 9.875 13.496094 9.921875 13.449219 C 10.15625 13.21875 10.402344 12.996094 10.65625 12.78125 C 11.15625 12.34375 11.621094 11.871094 12.089844 11.398438 C 12.25 11.238281 12.414062 11.078125 12.574219 10.914062 C 12.679688 10.8125 12.78125 10.707031 12.886719 10.605469 C 12.933594 10.554688 12.980469 10.507812 13.03125 10.457031 C 13.234375 10.253906 13.40625 10.066406 13.566406 9.824219 Z M 13.566406 9.824219 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(95.294118%,95.686275%,96.078432%);fill-opacity:1;" d="M 6.175781 10.347656 C 6.503906 10.464844 6.769531 10.617188 7.054688 10.820312 C 7.128906 10.871094 7.203125 10.925781 7.277344 10.976562 C 7.445312 11.105469 7.59375 11.234375 7.738281 11.390625 C 7.769531 11.316406 7.769531 11.316406 7.800781 11.238281 C 7.941406 10.992188 8.105469 10.882812 8.335938 10.722656 C 8.410156 10.671875 8.488281 10.617188 8.566406 10.5625 C 8.761719 10.449219 8.90625 10.386719 9.128906 10.347656 C 9.21875 11.453125 9.21875 11.453125 8.914062 11.832031 C 8.617188 12.113281 8.285156 12.273438 7.914062 12.433594 C 7.882812 12.921875 7.855469 13.410156 7.824219 13.914062 C 7.683594 13.914062 7.539062 13.914062 7.390625 13.914062 C 7.390625 13.398438 7.390625 12.878906 7.390625 12.347656 C 7.246094 12.320312 7.105469 12.289062 6.957031 12.261719 C 6.5625 12.078125 6.359375 11.875 6.175781 11.476562 C 6.144531 11.097656 6.152344 10.726562 6.175781 10.347656 Z M 6.175781 10.347656 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(8.235294%,11.764706%,23.137255%);fill-opacity:1;" d="M 14.433594 9.042969 C 14.523438 9.21875 14.523438 9.21875 14.484375 9.378906 C 14.460938 9.445312 14.4375 9.507812 14.414062 9.574219 C 14.386719 9.648438 14.359375 9.71875 14.335938 9.792969 C 14.308594 9.871094 14.277344 9.949219 14.25 10.027344 C 14.222656 10.105469 14.195312 10.183594 14.164062 10.265625 C 14.007812 10.703125 13.84375 11.132812 13.675781 11.566406 C 13.503906 11.996094 13.386719 12.429688 13.285156 12.878906 C 13.179688 13.269531 13.03125 13.632812 12.871094 14 C 12.378906 13.96875 12 13.910156 11.566406 13.652344 C 11.703125 13.144531 11.960938 12.71875 12.222656 12.265625 C 12.269531 12.1875 12.316406 12.105469 12.363281 12.023438 C 12.460938 11.859375 12.554688 11.695312 12.648438 11.53125 C 12.765625 11.332031 12.882812 11.128906 12.996094 10.925781 C 13.402344 10.222656 13.859375 9.621094 14.433594 9.042969 Z M 14.433594 9.042969 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(0%,0%,0%);fill-opacity:1;" d="M 10.433594 2.78125 C 11.101562 2.777344 11.765625 2.773438 12.433594 2.769531 C 12.742188 2.769531 13.050781 2.765625 13.359375 2.761719 C 13.714844 2.757812 14.070312 2.757812 14.425781 2.757812 C 14.539062 2.753906 14.648438 2.753906 14.761719 2.753906 C 14.867188 2.753906 14.96875 2.753906 15.074219 2.753906 C 15.210938 2.753906 15.210938 2.753906 15.347656 2.75 C 15.421875 2.761719 15.492188 2.773438 15.566406 2.78125 C 15.761719 3.082031 15.757812 3.191406 15.75 3.539062 C 15.75 3.625 15.746094 3.710938 15.746094 3.800781 C 15.742188 3.867188 15.742188 3.933594 15.738281 4 C 14.917969 3.957031 14.128906 3.839844 13.324219 3.6875 C 13.066406 3.640625 12.808594 3.59375 12.546875 3.550781 C 12.363281 3.515625 12.175781 3.484375 11.992188 3.449219 C 11.78125 3.414062 11.566406 3.378906 11.355469 3.34375 C 11.226562 3.324219 11.226562 3.324219 11.097656 3.304688 C 11.027344 3.292969 10.953125 3.28125 10.878906 3.273438 C 10.664062 3.207031 10.570312 3.132812 10.433594 2.957031 C 10.433594 2.898438 10.433594 2.839844 10.433594 2.78125 Z M 10.433594 2.78125 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(95.686275%,96.078432%,96.470588%);fill-opacity:1;" d="M 9.128906 8.261719 C 9.183594 9.257812 9.183594 9.257812 9.011719 9.695312 C 8.605469 10.082031 8.152344 10.3125 7.597656 10.398438 C 7.09375 10.3125 6.6875 10.128906 6.347656 9.738281 C 6.074219 9.320312 6.136719 8.828125 6.175781 8.347656 C 6.539062 8.394531 6.765625 8.519531 7.066406 8.734375 C 7.140625 8.789062 7.214844 8.839844 7.292969 8.894531 C 7.476562 9.042969 7.476562 9.042969 7.566406 9.21875 C 7.621094 9.21875 7.679688 9.21875 7.738281 9.21875 C 7.835938 9.117188 7.933594 9.015625 8.03125 8.914062 C 8.738281 8.261719 8.738281 8.261719 9.128906 8.261719 Z M 9.128906 8.261719 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(7.843138%,11.372549%,22.745098%);fill-opacity:1;" d="M 11.042969 3.476562 C 11.648438 3.675781 12.246094 3.886719 12.84375 4.101562 C 12.984375 4.152344 12.984375 4.152344 13.125 4.207031 C 14.054688 4.542969 14.984375 4.878906 15.914062 5.21875 C 15.867188 5.597656 15.816406 5.972656 15.738281 6.347656 C 15.675781 6.3125 15.613281 6.273438 15.546875 6.234375 C 15.011719 5.921875 14.472656 5.609375 13.9375 5.296875 C 13.667969 5.140625 13.402344 4.984375 13.132812 4.828125 C 12.804688 4.636719 12.476562 4.445312 12.148438 4.253906 C 12.011719 4.179688 11.878906 4.101562 11.746094 4.023438 C 11.683594 3.988281 11.621094 3.949219 11.558594 3.914062 C 11.140625 3.671875 11.140625 3.671875 11.042969 3.476562 Z M 11.042969 3.476562 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(0%,0%,0.392157%);fill-opacity:1;" d="M 15.652344 6.433594 C 15.679688 6.433594 15.710938 6.433594 15.738281 6.433594 C 15.746094 6.988281 15.75 7.539062 15.753906 8.09375 C 15.753906 8.28125 15.757812 8.46875 15.757812 8.65625 C 15.761719 8.925781 15.761719 9.195312 15.765625 9.464844 C 15.765625 9.550781 15.765625 9.636719 15.769531 9.722656 C 15.769531 9.839844 15.769531 9.839844 15.769531 9.960938 C 15.769531 10.027344 15.769531 10.097656 15.769531 10.167969 C 15.730469 10.394531 15.664062 10.472656 15.476562 10.609375 C 15.199219 10.625 15.199219 10.625 14.957031 10.609375 C 14.875 10.285156 14.871094 10.046875 14.933594 9.71875 C 14.949219 9.636719 14.964844 9.550781 14.980469 9.464844 C 15 9.378906 15.015625 9.289062 15.03125 9.199219 C 15.050781 9.113281 15.066406 9.027344 15.082031 8.933594 C 15.203125 8.289062 15.203125 8.289062 15.304688 8.085938 C 15.34375 7.792969 15.375 7.5 15.40625 7.203125 C 15.441406 6.886719 15.472656 6.707031 15.652344 6.433594 Z M 15.652344 6.433594 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(86.274511%,87.450981%,89.019608%);fill-opacity:1;" d="M 8.609375 3.390625 C 8.78125 3.390625 8.953125 3.390625 9.128906 3.390625 C 9.128906 4.109375 9.128906 4.824219 9.128906 5.566406 C 8.988281 5.59375 8.84375 5.621094 8.695312 5.652344 C 8.609375 5.566406 8.609375 5.566406 8.597656 5.355469 C 8.597656 5.265625 8.601562 5.175781 8.601562 5.082031 C 8.601562 4.984375 8.601562 4.886719 8.601562 4.785156 C 8.601562 4.683594 8.601562 4.578125 8.601562 4.472656 C 8.605469 4.371094 8.605469 4.265625 8.605469 4.160156 C 8.605469 3.902344 8.605469 3.648438 8.609375 3.390625 Z M 8.609375 3.390625 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(93.725491%,94.117647%,94.509804%);fill-opacity:1;" d="M 6.175781 3.390625 C 6.316406 3.390625 6.460938 3.390625 6.609375 3.390625 C 6.609375 4.136719 6.609375 4.882812 6.609375 5.652344 C 6.464844 5.625 6.320312 5.59375 6.175781 5.566406 C 6.175781 4.847656 6.175781 4.128906 6.175781 3.390625 Z M 6.175781 3.390625 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(93.333334%,93.725491%,94.117647%);fill-opacity:1;" d="M 7.476562 2.433594 C 7.59375 2.433594 7.707031 2.433594 7.824219 2.433594 C 7.824219 3.179688 7.824219 3.925781 7.824219 4.695312 C 7.683594 4.695312 7.539062 4.695312 7.390625 4.695312 C 7.390625 4.335938 7.386719 3.976562 7.386719 3.613281 C 7.386719 3.511719 7.382812 3.410156 7.382812 3.300781 C 7.382812 3.203125 7.382812 3.105469 7.382812 3.003906 C 7.382812 2.914062 7.382812 2.824219 7.382812 2.730469 C 7.390625 2.523438 7.390625 2.523438 7.476562 2.433594 Z M 7.476562 2.433594 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(5.490196%,7.843138%,14.901961%);fill-opacity:1;" d="M 15.390625 6.871094 C 15.511719 7.203125 15.449219 7.484375 15.390625 7.824219 C 15.304688 7.769531 15.21875 7.710938 15.128906 7.652344 C 15.171875 7.328125 15.207031 7.148438 15.390625 6.871094 Z M 15.390625 6.871094 "/>\n</g>\n</svg>\n';
var Web = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 256 256" width="16" height="16">\n\t<title>favicon</title>\n\t<defs>\n\t\t<image width="256" height="256" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>\n';
var Discord$1 = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 256 256" width="16" height="16">\n\t<title>favicon</title>\n\t<defs>\n\t\t<image width="256" height="256" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>\n';
var Twitch$1 = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 32 32" width="16" height="16">\n\t<title>favicon-32-e29e246c157142c94346</title>\n\t<defs>\n\t\t<image width="28" height="32" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="2" y="0"/>\n</svg>\n';
var Instagram = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 76 76" width="16" height="16">\n\t<title>lam-fZmwmvn</title>\n\t<defs>\n\t\t<image width="76" height="76" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>';
var Twitter$1 = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024" width="16" height="16">\n\t<title>icon-ios</title>\n\t<defs>\n\t\t<image width="1024" height="1024" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>\n';
var Reddit$1 = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 192 192" width="16" height="16">\n\t<title>192x192</title>\n\t<defs>\n\t\t<image width="192" height="192" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>';
var Youtube$1 = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 48 48" width="16" height="16">\n\t<title>favicon_48x48</title>\n\t<defs>\n\t\t<image width="48" height="48" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>';
var Vk$1 = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 256 256" width="16" height="16">\n\t<title>fav_logo</title>\n\t<defs>\n\t\t<image width="256" height="256" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>';
var AutoTask = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 32 32" width="16" height="16">\n\t<title>favicon</title>\n\t<defs>\n\t\t<image width="32" height="32" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>';
const ICONS = {
'[ASF]': ASF,
'[Web]': Web,
'[Discord]': Discord$1,
'[Twitch]': Twitch$1,
'[Instagram]': Instagram,
'[Twitter]': Twitter$1,
'[Reddit]': Reddit$1,
'[Youtube]': Youtube$1,
'[Vk]': Vk$1,
'[AutoTask]': AutoTask
};
const generateLink = (url, text) => `<a href="${url}" target="_blank">${text}</a>`;
const createBaseElement = content => $(`<li>${content}<font class="log-status"></font></li>`).addClass('card-text');
const createPlatformElement = (type, text, id) => {
const urlGenerators = {
group: text => `https://steamcommunity.com/groups/${text}`,
officialGroup: text => `https://steamcommunity.com/games/${text}`,
forum: text => `https://steamcommunity.com/app/${text}/discussions/`,
curator: text => `https://store.steampowered.com/${text?.includes('/') ? text : `curator/${text}`}`,
app: text => `https://store.steampowered.com/app/${text}`,
sub: text => `https://steamdb.info/sub/${text}/`,
workshop: text => `https://steamcommunity.com/sharedfiles/filedetails/?id=${text}`,
announcement: (text, id) => `https://store.steampowered.com/news/app/${text}/view/${id}`,
discord: {
invite: text => `https://discord.com/invite/${text}`,
server: text => `https://discord.com/channels/@me/${text}`
},
twitch: text => `https://www.twitch.tv/${text}`,
instagram: text => `https://www.instagram.com/${text}/`,
twitter: text => `https://x.com/${text}`,
reddit: {
subreddit: text => `https://www.reddit.com/r/${text}/`,
user: text => `https://www.reddit.com/user/${text?.replace('u_', '')}`
},
youtube: {
channel: text => `https://www.youtube.com/channel/${text}`,
video: text => `https://www.youtube.com/watch?v=${text}`
},
vk: text => `https://vk.com/${text}/`
};
const typeMap = {
joiningSteamGroup: [ 'group' ],
leavingSteamGroup: [ 'group' ],
gettingSteamGroupId: [ 'group' ],
joiningSteamOfficialGroup: [ 'officialGroup' ],
leavingSteamOfficialGroup: [ 'officialGroup' ],
gettingSteamOfficialGroupId: [ 'officialGroup' ],
subscribingForum: [ 'forum' ],
unsubscribingForum: [ 'forum' ],
gettingForumId: [ 'forum' ],
followingCurator: [ 'curator' ],
unfollowingCurator: [ 'curator' ],
gettingCuratorId: [ 'curator' ],
addingToWishlist: [ 'app' ],
removingFromWishlist: [ 'app' ],
followingGame: [ 'app' ],
unfollowingGame: [ 'app' ],
gettingSubid: [ 'app' ],
addingFreeLicense: [ 'app', 'sub' ],
requestingPlayTestAccess: [ 'app' ],
gettingDemoAppid: [ 'app' ],
favoritingWorkshop: [ 'workshop' ],
unfavoritingWorkshop: [ 'workshop' ],
gettingWorkshopAppId: [ 'workshop' ],
votingUpWorkshop: [ 'workshop' ],
gettingAnnouncementParams: [ 'announcement' ],
likingAnnouncement: [ 'announcement' ],
joiningDiscordServer: [ 'discord', 'invite' ],
gettingDiscordGuild: [ 'discord', 'invite' ],
gettingDiscordXContextProperties: [ 'discord', 'invite' ],
leavingDiscordServer: [ 'discord', 'server' ],
followingTwitchChannel: [ 'twitch' ],
unfollowingTwitchChannel: [ 'twitch' ],
gettingTwitchChannelId: [ 'twitch' ],
gettingInsUserId: [ 'instagram' ],
followingIns: [ 'instagram' ],
unfollowingIns: [ 'instagram' ],
gettingTwitterUserId: [ 'twitter' ],
followingTwitterUser: [ 'twitter' ],
unfollowingTwitterUser: [ 'twitter' ],
joiningReddit: [ 'reddit', 'subreddit' ],
leavingReddit: [ 'reddit', 'subreddit' ],
followingRedditUser: [ 'reddit', 'user' ],
unfollowingRedditUser: [ 'reddit', 'user' ],
followingYtbChannel: [ 'youtube', 'channel' ],
unfollowingYtbChannel: [ 'youtube', 'channel' ],
likingYtbVideo: [ 'youtube', 'video' ],
unlikingYtbVideo: [ 'youtube', 'video' ],
gettingVkId: [ 'vk' ],
joiningVkGroup: [ 'vk' ],
leavingVkGroup: [ 'vk' ],
joiningVkPublic: [ 'vk' ],
leavingVkPublic: [ 'vk' ],
sendingVkWall: [ 'vk' ],
deletingVkWall: [ 'vk' ]
};
const urlConfig = typeMap[type];
if (!urlConfig || !text) {
return null;
}
const [platform, subType] = urlConfig;
const urlGenerator = urlGenerators[platform];
if (typeof urlGenerator === 'function') {
const url = urlGenerator(text, id);
const displayText = platform === 'announcement' ? id || '' : text;
return createBaseElement(`${I18n(type)}[${generateLink(url, displayText)}]...`);
}
if (subType && typeof urlGenerator === 'object') {
const subGenerator = urlGenerator[subType];
if (typeof subGenerator === 'function') {
const displayText = type.includes('RedditUser') ? text.replace('u_', '') : text;
return createBaseElement(`${I18n(type)}[${generateLink(subGenerator(text), displayText)}]...`);
}
}
return null;
};
const createSpecialElement = (type, text, html, id) => {
switch (type) {
case 'retweetting':
case 'unretweetting':
return createBaseElement(`${I18n(type)}${text}...`);
case 'visitingLink':
return createBaseElement(`${I18n('visitingLink')}[${generateLink(text || '', text || '')}]...`);
case 'verifyingInsAuth':
case 'text':
return createBaseElement(I18n(text || ''));
case 'html':
return $(text || html || '');
case 'whiteList':
return $(`<li><font class="warning">${I18n('skipTask')}[${text}(${id})](${I18n('whiteList')})</font></li>`);
case 'globalOptionsSkip':
return $(`<li>${I18n('skipTaskOption')}<font class="warning">${text}</font></li>`);
default:
return createBaseElement(`${I18n('unKnown')}:${type}(${text})...`);
}
};
const echoLog = ({type: type, text: text, html: html, id: id, before: before}) => {
const emptyStatus = {
success: () => emptyStatus,
error: () => emptyStatus,
warning: () => emptyStatus,
info: () => emptyStatus,
view: () => emptyStatus,
remove: () => emptyStatus
};
try {
let ele;
if (!type && !text && !html) {
ele = createBaseElement('');
} else if (text && !type) {
ele = createBaseElement(text);
} else if (html && !type) {
ele = $(html);
} else if (type) {
const platformElement = createPlatformElement(type, text, id);
ele = platformElement || createSpecialElement(type, text, html, id);
} else {
ele = createBaseElement('');
}
if (before) {
if (before in ICONS) {
const iconKey = before;
const svgContent = ICONS[iconKey];
const base64Svg = btoa(svgContent);
ele.prepend(`<font class="before-icon" style="background-image: url('data:image/svg+xml;base64,${base64Svg}')"></font>`);
} else {
ele.prepend(`<font class="before">${before}</font>`);
}
} else {
const base64Svg = btoa(ICONS['[AutoTask]']);
ele.prepend(`<font class="before-icon" style="background-image: url('data:image/svg+xml;base64,${base64Svg}')"></font>`);
}
ele.addClass('card-text');
$('#auto-task-info').append(ele);
ele[0]?.scrollIntoView();
const font = ele.find('font.log-status');
const status = {
font: font,
success(text = 'Success', html = false) {
this.font?.attr('class', '').addClass('success');
html ? this.font?.html(text) : this.font?.text(text);
return this;
},
error(text = 'Error', html = false) {
this.font?.attr('class', '').addClass('error');
html ? this.font?.html(text) : this.font?.text(text);
return this;
},
warning(text = 'Warning', html = false) {
this.font?.attr('class', '').addClass('warning');
html ? this.font?.html(text) : this.font?.text(text);
return this;
},
info(text = 'Info', html = false) {
this.font?.attr('class', '').addClass('info');
html ? this.font?.html(text) : this.font?.text(text);
return this;
},
view() {
this.font?.[0].scrollIntoView();
return this;
},
remove() {
this.font?.parent().remove();
return this;
}
};
return status;
} catch (error) {
throwError(error, 'echoLog');
return emptyStatus;
}
};
const unique = array => {
try {
return Array.from(new Set(array));
} catch (error) {
throwError(error, 'unique');
return [];
}
};
const delay = (time = 1e3) => new Promise((resolve => setTimeout((() => resolve(true)), time)));
const getRedirectLink = async (link, redirectOnce = false) => {
try {
if (!link) {
return null;
}
const redirectLinksCache = GM_getValue('redirectLinks') || {};
const cachedLink = redirectLinksCache[link];
if (cachedLink) {
debug('使用缓存的重定向链接', {
original: link,
cached: cachedLink
});
return cachedLink;
}
const {data: data} = await httpRequest({
url: link,
method: 'GET',
redirect: redirectOnce ? 'manual' : 'follow'
});
if (data?.finalUrl) {
redirectLinksCache[link] = data.finalUrl;
GM_setValue('redirectLinks', redirectLinksCache);
debug('获取新的重定向链接', {
original: link,
final: data.finalUrl
});
return data.finalUrl;
}
debug('未找到重定向链接', {
link: link
});
return null;
} catch (error) {
throwError(error, 'getRedirectLink');
return null;
}
};
const visitLink = async (link, options) => {
try {
debug('开始访问链接', {
link: link,
options: options
});
const logStatus = echoLog({
type: 'visitLink',
text: link
});
const {result: result, statusText: statusText, status: status} = await httpRequest({
url: link,
method: 'GET',
...options
});
if (result === 'Success') {
debug('链接访问成功', {
link: link
});
logStatus.success();
return true;
}
debug('链接访问失败', {
link: link,
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
} catch (error) {
throwError(error, 'visitLink');
return false;
}
};
const getUrlQuery = url => {
try {
debug('开始解析URL查询参数', {
url: url || window.location.href
});
const searchParams = url ? new URL(url, window.location.origin).searchParams : new URLSearchParams(window.location.search);
const query = {};
for (const [key, value] of searchParams.entries()) {
query[key] = value;
}
debug('URL查询参数解析结果', query);
return query;
} catch (error) {
throwError(error, 'getUrlQuery');
return {};
}
};
const stringToColour = str => {
try {
let hash = 0;
for (let i = 0; i < str.length; i++) {
hash = str.charCodeAt(i) + ((hash << 5) - hash);
}
const rgb = Array.from({
length: 3
}, ((_, i) => {
const value = hash >> i * 8 & 255;
return value.toString(16).padStart(2, '0');
}));
const color = `#${rgb.join('')}`;
return color;
} catch (error) {
throwError(error, 'stringToColour');
return '#ffffff';
}
};
const getAllLocalStorageAsObjects = localStorage => {
try {
debug('开始将所有LocalStorage转换为对象');
const result = {};
for (let i = 0; i < localStorage.length; i++) {
const key = localStorage.key(i);
if (!key) {
continue;
}
const value = localStorage.getItem(key);
try {
result[key] = JSON.parse(value);
} catch (error) {
result[key] = value;
console.error(error);
}
}
debug('所有LocalStorage转换为对象完成', result);
return result;
} catch (error) {
debug('将所有LocalStorage转换为对象失败', {
error: error
});
throwError(error, 'getAllLocalStorageAsObjects');
return {};
}
};
class Social {
tasks;
getRealParams(name, links, doTask, link2param) {
try {
debug('开始获取实际参数', {
name: name,
linksCount: links.length,
doTask: doTask
});
let realParams = [];
if (links.length > 0) {
debug('处理链接参数');
const convertedLinks = links.map((link => link2param(link))).filter((link => link !== undefined));
debug('链接参数处理结果', {
convertedLinksCount: convertedLinks.length
});
realParams = [ ...realParams, ...convertedLinks ];
}
if (!doTask && this.tasks[name]?.length) {
debug('处理任务参数', {
taskCount: this.tasks[name].length
});
realParams = [ ...realParams, ...this.tasks[name] ];
}
const uniqueParams = unique(realParams);
debug('参数处理完成', {
originalCount: realParams.length,
uniqueCount: uniqueParams.length
});
return uniqueParams;
} catch (error) {
debug('获取实际参数时发生错误', {
error: error
});
throwError(error, 'Social.getRealParams');
return [];
}
}
}
class Discord extends Social {
tasks;
whiteList;
#auth=GM_getValue('discordAuth') || {};
#cache=GM_getValue('discordCache') || {};
#xContextPropertiesCache=GM_getValue('discordXContextPropertiesCache') || {};
#initialized=false;
constructor() {
super();
const defaultTasksTemplate = {
servers: []
};
this.tasks = defaultTasksTemplate;
this.whiteList = {
...defaultTasksTemplate,
...GM_getValue('whiteList')?.discord || {}
};
}
async init(action) {
try {
debug('开始初始化Discord模块', {
action: action
});
if (!GM_getValue('dontRemindDiscordAgain')) {
debug('显示Discord重要提示对话框');
const result = await Swal.fire({
title: I18n('discordImportantNotice'),
text: I18n('discordImportantNoticeText'),
showCancelButton: true,
showDenyButton: true,
confirmButtonText: I18n('continueDiscordTask'),
cancelButtonText: I18n('skipDiscordTask'),
denyButtonText: I18n('continueAndDontRemindAgain')
}).then((({isConfirmed: isConfirmed, isDenied: isDenied}) => {
if (isConfirmed) {
return true;
}
if (isDenied) {
GM_setValue('dontRemindDiscordAgain', true);
return true;
}
return false;
}));
if (!result) {
this.#initialized = false;
return 'skip';
}
}
if (action === 'do' && !globalOptions.doTask.discord.servers || action === 'undo' && !globalOptions.undoTask.discord.servers) {
this.#initialized = false;
debug('检测到用户已禁用Discord任务,跳过初始化');
return 'skip';
}
if (this.#initialized) {
debug('Discord模块已初始化');
return true;
}
if (!this.#auth.auth || !this.#auth.xSuperProperties) {
debug('未找到Discord授权信息,尝试更新授权');
if (await this.#updateAuth()) {
this.#initialized = true;
return true;
}
return false;
}
const isVerified = await this.#verifyAuth();
if (isVerified) {
debug('Discord授权验证成功');
echoLog({
before: '[Discord]'
}).success(I18n('initSuccess', 'Discord'));
this.#initialized = true;
return true;
}
GM_setValue('discordAuth', {
auth: null
});
debug('Discord授权验证失败,尝试重新获取授权');
if (await this.#updateAuth()) {
echoLog({
before: '[Discord]'
}).success(I18n('initSuccess', 'Discord'));
this.#initialized = true;
return true;
}
echoLog({
before: '[Discord]'
}).error(I18n('initFailed', 'Discord'));
return false;
} catch (error) {
throwError(error, 'Discord.init');
return false;
}
}
async #verifyAuth() {
try {
debug('开始验证Discord授权');
const logStatus = echoLog({
text: I18n('verifyingAuth', 'Discord'),
before: '[Discord]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://discord.com/api/v6/users/@me',
method: 'HEAD',
headers: {
authorization: this.#auth.auth
}
});
if (result !== 'Success') {
debug('Discord授权验证请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('Discord授权验证状态码错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('Discord授权验证成功');
logStatus.success();
return true;
} catch (error) {
throwError(error, 'Discord.verifyAuth');
return false;
}
}
async #updateAuth() {
try {
debug('开始更新Discord授权');
const logStatus = echoLog({
text: I18n('updatingAuth', 'Discord'),
before: '[Discord]'
});
return await new Promise((resolve => {
const newTab = GM_openInTab('https://discord.com/channels/@me', {
active: true,
insert: true,
setParent: true
});
newTab.name = 'ATv4_discordAuth';
newTab.onclose = async () => {
const {auth: auth, xSuperProperties: xSuperProperties} = GM_getValue('discordAuth');
if (auth && xSuperProperties) {
debug('成功获取新的Discord授权');
this.#auth = {
auth: auth,
xSuperProperties: xSuperProperties
};
logStatus.success();
resolve(await this.#verifyAuth());
} else {
debug('获取Discord授权失败');
logStatus.error('Error: Update discord auth failed!');
resolve(false);
}
};
}));
} catch (error) {
throwError(error, 'Discord.updateAuth');
return false;
}
}
async #joinServer(inviteId) {
try {
debug('开始加入Discord服务器', {
inviteId: inviteId
});
const logStatus = echoLog({
type: 'joiningDiscordServer',
text: inviteId,
before: '[Discord]'
});
const xContextProperties = await this.#getXContextProperties(inviteId);
if (!xContextProperties) {
debug('获取加群参数失败,无法加入服务器', {
inviteId: inviteId
});
logStatus.error('Error: Failed to get xContextProperties');
return false;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://discord.com/api/v9/invites/${inviteId}`,
method: 'POST',
dataType: 'json',
headers: {
'content-type': 'application/json',
authorization: this.#auth.auth,
origin: 'https://discord.com',
referer: `https://discord.com/invite/${inviteId}`,
'x-super-properties': this.#auth.xSuperProperties,
'x-context-properties': xContextProperties
},
data: '{"session_id":null}'
});
if (result !== 'Success' || data?.status !== 200) {
debug('加入Discord服务器失败', {
result: result,
statusText: statusText,
status: status
});
if (status === 400) {
debug('加入Discord服务器失败,状态码为400,需完成人机验证');
logStatus.error(I18n('captchaNeeded'));
return false;
}
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
const guild = data.response?.guild?.id;
if (!guild) {
debug('获取服务器ID失败');
logStatus.error('Error: Failed to get guild ID');
return false;
}
debug('成功加入Discord服务器', {
guild: guild
});
logStatus.success();
this.#setCache(inviteId, guild);
this.tasks.servers = unique([ ...this.tasks.servers, inviteId ]);
return true;
} catch (error) {
throwError(error, 'Discord.joinServer');
return false;
}
}
async #leaveServer(inviteId) {
try {
debug('开始退出Discord服务器', {
inviteId: inviteId
});
if (this.whiteList.servers.includes(inviteId)) {
debug('服务器在白名单中,跳过退出操作', {
inviteId: inviteId
});
echoLog({
type: 'whiteList',
text: 'Discord.leaveServer',
id: inviteId,
before: '[Discord]'
});
return true;
}
const guild = await this.#getGuild(inviteId);
if (!guild) {
debug('获取服务器ID失败,无法退出服务器', {
inviteId: inviteId
});
return false;
}
const logStatus = echoLog({
type: 'leavingDiscordServer',
text: guild,
before: '[Discord]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://discord.com/api/v9/users/@me/guilds/${guild}`,
method: 'DELETE',
headers: {
authorization: this.#auth.auth,
'x-super-properties': this.#auth.xSuperProperties
}
});
if (result !== 'Success' || data?.status !== 204) {
debug('退出Discord服务器失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
debug('成功退出Discord服务器', {
guild: guild
});
logStatus.success();
return true;
} catch (error) {
throwError(error, 'Discord.leaveServer');
return false;
}
}
async #getXContextProperties(inviteId) {
try {
debug('开始获取Discord加群参数', {
inviteId: inviteId
});
const logStatus = echoLog({
type: 'gettingDiscordXContextProperties',
text: inviteId,
before: '[Discord]'
});
const cachedXContextProperties = this.#xContextPropertiesCache[inviteId];
if (cachedXContextProperties) {
debug('从缓存中获取到加群参数', {
inviteId: inviteId,
cachedXContextProperties: cachedXContextProperties
});
logStatus.success();
return cachedXContextProperties;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://discord.com/api/v9/invites/${inviteId}?with_counts=true&with_expiration=true&with_permissions=true`,
responseType: 'json',
method: 'GET',
headers: {
authorization: this.#auth.auth,
'x-super-properties': this.#auth.xSuperProperties
}
});
if (result !== 'Success' || data?.status !== 200) {
debug('获取加群参数失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
const guild = data.response?.guild?.id;
if (!guild) {
debug('加群参数中未找到ID', {
inviteId: inviteId
});
logStatus.error('Error: Failed to get guild ID');
return false;
}
const xContextProperties = {
location: 'Accept Invite Page',
location_guild_id: data.response?.guild?.id,
location_channel_id: data.response?.channel?.id,
location_channel_type: data.response?.channel?.type
};
debug('成功获取加群参数', xContextProperties);
logStatus.success();
this.#setXContextPropertiesCache(inviteId, window.btoa(JSON.stringify(xContextProperties)));
this.#setCache(inviteId, guild);
return window.btoa(JSON.stringify(xContextProperties));
} catch (error) {
throwError(error, 'Discord.getXContextProperties');
return false;
}
}
async #getGuild(inviteId) {
try {
debug('开始获取Discord服务器ID', {
inviteId: inviteId
});
const logStatus = echoLog({
type: 'gettingDiscordGuild',
text: inviteId,
before: '[Discord]'
});
const cachedGuild = this.#cache[inviteId];
if (cachedGuild) {
debug('从缓存中获取到服务器ID', {
inviteId: inviteId,
cachedGuild: cachedGuild
});
logStatus.success();
return cachedGuild;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://discord.com/api/v9/invites/${inviteId}`,
responseType: 'json',
method: 'GET',
headers: {
authorization: this.#auth.auth,
'x-super-properties': this.#auth.xSuperProperties
}
});
if (result !== 'Success' || data?.status !== 200) {
debug('获取服务器信息失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
const guild = data.response?.guild?.id;
if (!guild) {
debug('服务器信息中未找到ID', {
inviteId: inviteId
});
logStatus.error('Error: Failed to get guild ID');
return false;
}
debug('成功获取服务器ID', {
inviteId: inviteId,
guild: guild
});
logStatus.success();
this.#setCache(inviteId, guild);
return guild;
} catch (error) {
throwError(error, 'Discord.getGuild');
return false;
}
}
async toggle({doTask: doTask = true, serverLinks: serverLinks = []}) {
try {
debug('开始处理Discord服务器任务', {
doTask: doTask,
serverLinksCount: serverLinks.length
});
if (!this.#initialized) {
debug('Discord模块未初始化');
echoLog({
text: I18n('needInit'),
before: '[Discord]'
});
return false;
}
if (doTask && !globalOptions.doTask.discord.servers || !doTask && !globalOptions.undoTask.discord.servers) {
debug('根据全局选项跳过Discord服务器任务', {
doTask: doTask
});
echoLog({
type: 'globalOptionsSkip',
text: 'discord.servers',
before: '[Discord]'
});
return true;
}
const realServers = this.getRealParams('servers', serverLinks, doTask, (link => link.match(/invite\/(.+)/)?.[1]));
debug('处理后的服务器列表', {
count: realServers.length,
servers: realServers
});
if (realServers.length === 0) {
debug('没有需要处理的服务器');
return true;
}
const results = [];
for (const server of realServers) {
results.push(doTask ? this.#joinServer(server) : this.#leaveServer(server));
await delay(1e3);
}
return await Promise.allSettled(results).then((() => true));
} catch (error) {
throwError(error, 'Discord.toggleServers');
return false;
}
}
#setCache(inviteId, guild) {
try {
debug('设置Discord服务器缓存', {
inviteId: inviteId,
guild: guild
});
this.#cache[inviteId] = guild;
GM_setValue('discordCache', this.#cache);
debug('Discord服务器缓存设置成功');
} catch (error) {
debug('设置Discord服务器缓存失败', {
error: error
});
throwError(error, 'Discord.setCache');
}
}
#setXContextPropertiesCache(inviteId, xContextProperties) {
try {
debug('设置Discord加群参数缓存', {
inviteId: inviteId,
xContextProperties: xContextProperties
});
this.#xContextPropertiesCache[inviteId] = xContextProperties;
GM_setValue('discordXContextPropertiesCache', this.#xContextPropertiesCache);
} catch (error) {
debug('设置Discord加群参数缓存失败', {
error: error
});
throwError(error, 'Discord.setXContextPropertiesCache');
}
}
}
class Reddit extends Social {
tasks;
whiteList;
#auth;
#initialized=false;
constructor() {
super();
const defaultTasksTemplate = {
reddits: []
};
debug('初始化Reddit实例');
this.tasks = defaultTasksTemplate;
this.whiteList = {
...defaultTasksTemplate,
...GM_getValue('whiteList')?.reddit || {}
};
}
async init() {
try {
debug('开始初始化Reddit模块');
if (this.#initialized) {
debug('Reddit模块已初始化');
return true;
}
const isVerified = await this.#updateAuth();
if (isVerified) {
debug('Reddit授权验证成功');
echoLog({
before: '[Reddit]'
}).success(I18n('initSuccess', 'Reddit'));
this.#initialized = true;
return true;
}
debug('Reddit初始化失败');
echoLog({
before: '[Reddit]'
}).error(I18n('initFailed', 'Reddit'));
return false;
} catch (error) {
debug('Reddit初始化发生错误', {
error: error
});
throwError(error, 'Reddit.init');
return false;
}
}
async #useBeta() {
try {
debug('开始切换Reddit为新版');
const logStatus = echoLog({
text: I18n('changingRedditVersion'),
before: '[Reddit]'
});
return await new Promise((resolve => {
const newTab = GM_openInTab('https://www.reddit.com/', {
active: true,
insert: true,
setParent: true
});
newTab.name = 'ATv4_redditAuth';
newTab.onclose = async () => {
debug('新版Reddit标签页已关闭');
logStatus.success();
resolve(await this.#updateAuth(true));
};
}));
} catch (error) {
debug('切换Reddit版本时发生错误', {
error: error
});
throwError(error, 'Reddit.useBeta');
return false;
}
}
async #updateAuth(beta = false) {
try {
debug('开始更新Reddit授权', {
beta: beta
});
const logStatus = echoLog({
text: I18n('updatingAuth', 'Reddit'),
before: '[Reddit]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://www.reddit.com/',
method: 'GET',
nochche: true,
headers: {
'Cache-Control': 'no-cache'
}
});
if (result !== 'Success') {
debug('获取Reddit页面失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.responseText.includes('www.reddit.com/login/')) {
debug('需要登录Reddit');
logStatus.error(`Error:${I18n('loginReddit')}`, true);
return false;
}
if (data?.status !== 200) {
debug('Reddit页面状态码错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
if (data.responseText.includes('redesign-beta-optin-btn') && !beta) {
debug('检测到旧版Reddit,需要切换到新版');
return await this.#useBeta();
}
const accessToken = data.responseText.match(/"accessToken":"(.*?)","expires":"(.*?)"/)?.[1];
if (!accessToken) {
debug('未找到Reddit访问令牌');
logStatus.error('Error: Parameter "accessToken" not found!');
return false;
}
debug('成功获取Reddit访问令牌');
this.#auth = {
token: accessToken
};
logStatus.success();
return true;
} catch (error) {
debug('更新Reddit授权时发生错误', {
error: error
});
throwError(error, 'Reddit.updateAuth');
return false;
}
}
async #toggleTask({name: name, doTask: doTask = true}) {
try {
debug('开始处理Reddit任务', {
name: name,
doTask: doTask
});
if (!doTask && this.whiteList.reddits.includes(name)) {
debug('Reddit在白名单中,跳过取消订阅', {
name: name
});
echoLog({
type: 'whiteList',
text: 'Reddit.undoTask',
id: name,
before: '[Reddit]'
});
return true;
}
let type = doTask ? 'joiningReddit' : 'leavingReddit';
if (/^u_/.test(name)) {
type = doTask ? 'followingRedditUser' : 'unfollowingRedditUser';
}
debug('任务类型', {
type: type,
name: name
});
const logStatus = echoLog({
type: type,
text: name,
before: '[Reddit]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://oauth.reddit.com/api/subscribe?redditWebClient=desktop2x&app=desktop2x-client-production&raw_json=1&gilding_detail=1',
method: 'POST',
headers: {
authorization: `Bearer ${this.#auth.token}`,
'content-type': 'application/x-www-form-urlencoded'
},
data: $.param({
action: doTask ? 'sub' : 'unsub',
sr_name: name,
api_type: 'json'
})
});
if (result !== 'Success') {
debug('Reddit任务请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('Reddit任务状态码错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('Reddit任务处理成功', {
name: name,
doTask: doTask
});
logStatus.success();
if (doTask) {
this.tasks.reddits = unique([ ...this.tasks.reddits, name ]);
}
return true;
} catch (error) {
debug('处理Reddit任务时发生错误', {
error: error
});
throwError(error, 'Reddit.toggleTask');
return false;
}
}
async toggle({doTask: doTask = true, redditLinks: redditLinks = []}) {
try {
debug('开始处理Reddit链接任务', {
doTask: doTask,
redditLinksCount: redditLinks.length
});
if (!this.#initialized) {
debug('Reddit模块未初始化');
echoLog({
text: I18n('needInit'),
before: '[Reddit]'
});
return false;
}
if (doTask && !globalOptions.doTask.reddit.reddits || !doTask && !globalOptions.undoTask.reddit.reddits) {
debug('根据全局选项跳过Reddit任务', {
doTask: doTask
});
echoLog({
type: 'globalOptionsSkip',
text: 'reddit.reddits',
before: '[Reddit]'
});
return true;
}
const realReddits = this.getRealParams('reddits', redditLinks, doTask, (link => {
const name = link.match(/https?:\/\/www\.reddit\.com\/r\/([^/]*)/)?.[1];
const userName = link.match(/https?:\/\/www\.reddit\.com\/user\/([^/]*)/)?.[1];
if (userName) {
return name || userName;
}
return name;
}));
debug('处理后的Reddit列表', {
count: realReddits.length,
reddits: realReddits
});
if (realReddits.length === 0) {
debug('没有需要处理的Reddit链接');
return true;
}
const prom = [];
for (const name of realReddits) {
prom.push(this.#toggleTask({
name: name,
doTask: doTask
}));
await delay(1e3);
}
return await Promise.all(prom).then((() => true));
} catch (error) {
debug('处理Reddit链接任务时发生错误', {
error: error
});
throwError(error, 'Reddit.toggle');
return false;
}
}
}
class Twitch extends Social {
tasks;
whiteList;
#auth=GM_getValue('twitchAuth') || {};
#cache=GM_getValue('twitchCache') || {};
#initialized=false;
#integrityToken;
constructor() {
super();
const defaultTasksTemplate = {
channels: []
};
debug('初始化Twitch实例');
this.tasks = defaultTasksTemplate;
this.whiteList = {
...defaultTasksTemplate,
...GM_getValue('whiteList')?.twitch || {}
};
}
async init() {
try {
debug('开始初始化Twitch模块');
if (this.#initialized) {
debug('Twitch模块已初始化');
return true;
}
if (!this.#auth.authToken || !this.#auth.clientId || !this.#auth.clientVersion || !this.#auth.deviceId || !this.#auth.clientSessionId) {
if (await this.#updateAuth()) {
this.#initialized = true;
return true;
}
return false;
}
const isVerified = await this.#verifyAuth(true);
if (isVerified) {
debug('Twitch授权验证成功');
echoLog({
before: '[Twitch]'
}).success(I18n('initSuccess', 'Twitch'));
this.#initialized = true;
return true;
}
GM_setValue('twitchAuth', null);
if (await this.#updateAuth()) {
debug('Twitch重新授权成功');
echoLog({
before: '[Twitch]'
}).success(I18n('initSuccess', 'Twitch'));
this.#initialized = true;
return true;
}
debug('Twitch初始化失败');
echoLog({
before: '[Twitch]'
}).error(I18n('initFailed', 'Twitch'));
return false;
} catch (error) {
debug('Twitch初始化发生错误', {
error: error
});
throwError(error, 'Twitch.init');
return false;
}
}
async #verifyAuth(isFirst) {
try {
debug('开始验证Twitch授权');
const logStatus = echoLog({
text: I18n('verifyingAuth', 'Twitch'),
before: '[Twitch]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://gql.twitch.tv/gql',
method: 'POST',
dataType: 'json',
headers: {
Authorization: `OAuth ${this.#auth.authToken}`,
'Client-Id': this.#auth.clientId
},
data: '[{"operationName":"FrontPageNew_User","variables":{"limit":1},"extensions":{"persistedQuery":{"version":1,' + '"sha256Hash":"64bd07a2cbaca80699d62636d966cf6395a5d14a1f0a14282067dcb28b13eb11"}}}]'
});
if (result !== 'Success') {
debug('Twitch授权验证请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200 || !data.response?.[0]?.data?.currentUser) {
debug('Twitch授权验证状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
await this.#integrity(isFirst);
debug('Twitch授权验证成功');
logStatus.success();
return true;
} catch (error) {
debug('Twitch授权验证发生错误', {
error: error
});
throwError(error, 'Twitch.verifyAuth');
return false;
}
}
async #integrity(isFirst = true, ct = '') {
try {
debug('开始检查Twitch完整性', {
isFirst: isFirst,
ct: ct
});
const logStatus = echoLog({
text: I18n('checkingTwitchIntegrity'),
before: '[Twitch]'
});
if (isFirst && (!this.#auth.authToken || !this.#auth.clientId || !this.#auth.clientVersion || !this.#auth.deviceId || !this.#auth.clientSessionId)) {
return await this.#updateAuth(false);
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://gql.twitch.tv/integrity',
method: 'POST',
dataType: 'json',
anonymous: true,
headers: {
Origin: 'https://www.twitch.tv',
Referer: 'https://www.twitch.tv/',
Authorization: `OAuth ${this.#auth.authToken}`,
'Client-Id': this.#auth.clientId,
'Client-Version': this.#auth.clientVersion,
'X-Device-Id': this.#auth.deviceId,
'Client-Session-Id': this.#auth.clientSessionId,
'x-kpsdk-ct': ct
}
});
if (result !== 'Success') {
debug('Twitch完整性检查请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (!ct && data?.responseHeaders?.['x-kpsdk-ct']) {
debug('需要重新检查Twitch完整性');
return await this.#integrity(isFirst, data.responseHeaders['x-kpsdk-ct']);
}
if (data?.status !== 200 || !data.response?.token) {
debug('Twitch完整性检查状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
this.#integrityToken = data.response.token;
debug('Twitch完整性检查成功');
logStatus.success();
return true;
} catch (error) {
debug('Twitch完整性检查发生错误', {
error: error
});
throwError(error, 'Twitch.integrity');
return false;
}
}
async #updateAuth(isFirst = true) {
try {
debug('开始更新Twitch授权', {
isFirst: isFirst
});
const logStatus = echoLog({
text: I18n('updatingAuth', 'Twitch'),
before: '[Twitch]'
});
return await new Promise((resolve => {
const newTab = GM_openInTab('https://www.twitch.tv/', {
active: true,
insert: true,
setParent: true
});
newTab.name = 'ATv4_twitchAuth';
newTab.onclose = async () => {
const auth = GM_getValue('twitchAuth');
if (auth) {
debug('成功获取新的Twitch授权');
this.#auth = auth;
logStatus.success();
resolve(await this.#verifyAuth(isFirst));
} else {
debug('获取Twitch授权失败');
logStatus.error('Error: Update twitch auth failed!');
resolve(false);
}
};
}));
} catch (error) {
debug('更新Twitch授权时发生错误', {
error: error
});
throwError(error, 'Twitch.updateAuth');
return false;
}
}
async #toggleChannel({name: name, doTask: doTask = true}) {
try {
debug('开始处理Twitch频道任务', {
name: name,
doTask: doTask
});
if (!doTask && this.whiteList.channels.includes(name)) {
debug('Twitch频道在白名单中,跳过取消关注', {
name: name
});
echoLog({
type: 'whiteList',
text: 'Twitch.unfollowChannel',
id: name,
before: '[Twitch]'
});
return true;
}
const channelId = await this.#getChannelId(name);
if (!channelId) {
return false;
}
const logStatus = echoLog({
type: `${doTask ? '' : 'un'}followingTwitchChannel`,
text: name,
before: '[Twitch]'
});
const followData = `[{"operationName":"FollowButton_FollowUser","variables":{"input":{"disableNotifications":false,"targetID":"${channelId}` + '"}},"extensions":{"persistedQuery":{"version":1,"sha256Hash":"800e7346bdf7e5278a3c1d3f21b2b56e2639928f86815677a7126b093b2fdd08"}}}]';
const unfollowData = `[{"operationName":"FollowButton_UnfollowUser","variables":{"input":{"targetID":"${channelId}"}},` + '"extensions":{"persistedQuery":{"version":1,"sha256Hash":"f7dae976ebf41c755ae2d758546bfd176b4eeb856656098bb40e0a672ca0d880"}}}]';
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://gql.twitch.tv/gql',
method: 'POST',
dataType: 'json',
anonymous: true,
headers: {
Origin: 'https://www.twitch.tv',
Referer: 'https://www.twitch.tv/',
Authorization: `OAuth ${this.#auth.authToken}`,
'Client-Id': this.#auth.clientId,
'Client-Version': this.#auth.clientVersion,
'X-Device-Id': this.#auth.deviceId,
'Client-Session-Id': this.#auth.clientSessionId,
'Client-Integrity': this.#integrityToken
},
data: doTask ? followData : unfollowData
});
if (result !== 'Success') {
debug('Twitch频道操作请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200 || data.response?.[0] && data.response[0].errors) {
debug('Twitch频道操作状态错误', {
status: data?.status,
statusText: data?.statusText,
errors: data?.response?.[0].errors
});
logStatus.error(`Error:${data?.response?.[0].errors?.[0]?.message || `${data?.statusText}(${data?.status})`}`);
return false;
}
debug('Twitch频道操作成功', {
name: name,
doTask: doTask
});
logStatus.success();
if (doTask) {
this.tasks.channels = unique([ ...this.tasks.channels, name ]);
}
return true;
} catch (error) {
debug('处理Twitch频道任务时发生错误', {
error: error
});
throwError(error, 'Twitch.toggleChannel');
return false;
}
}
async #getChannelId(name) {
try {
debug('开始获取Twitch频道ID', {
name: name
});
const logStatus = echoLog({
type: 'gettingTwitchChannelId',
text: name,
before: '[Twitch]'
});
const cachedChannelId = this.#cache[name];
if (cachedChannelId) {
debug('从缓存获取到Twitch频道ID', {
name: name,
id: cachedChannelId
});
logStatus.success();
return cachedChannelId;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://gql.twitch.tv/gql',
method: 'POST',
headers: {
Authorization: `OAuth ${this.#auth.authToken}`,
'Client-Id': this.#auth.clientId
},
responseType: 'json',
data: `[{"operationName":"ActiveWatchParty","variables":{"channelLogin":"${name}"},` + '"extensions":{"persistedQuery":{"version":1,"sha256Hash":"4a8156c97b19e3a36e081cf6d6ddb5dbf9f9b02ae60e4d2ff26ed70aebc80a30"}}}]'
});
if (result !== 'Success') {
debug('获取Twitch频道ID请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('获取Twitch频道ID状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const newChannelId = data.response?.[0]?.data?.user?.id;
if (!newChannelId) {
debug('未找到Twitch频道ID', {
name: name
});
logStatus.error(`Error:${data?.statusText || 'Unknown'}(${data?.status || 'Unknown'})`);
return false;
}
debug('成功获取Twitch频道ID', {
name: name,
id: newChannelId
});
this.#setCache(name, newChannelId);
logStatus.success();
return newChannelId;
} catch (error) {
debug('获取Twitch频道ID时发生错误', {
error: error
});
throwError(error, 'Twitch.getChannelId');
return false;
}
}
async toggle({doTask: doTask = true, channelLinks: channelLinks = []}) {
try {
debug('开始处理Twitch链接任务', {
doTask: doTask,
channelLinksCount: channelLinks.length
});
if (!this.#initialized) {
debug('Twitch模块未初始化');
echoLog({
text: I18n('needInit'),
before: '[Twitch]'
});
return false;
}
const prom = [];
if (doTask && !globalOptions.doTask.twitch.channels || !doTask && !globalOptions.undoTask.twitch.channels) {
debug('根据全局选项跳过Twitch任务', {
doTask: doTask
});
echoLog({
type: 'globalOptionsSkip',
text: 'twitch.channels',
before: '[Twitch]'
});
} else {
const realChannels = this.getRealParams('channels', channelLinks, doTask, (link => link.match(/https:\/\/(www\.)?twitch\.tv\/(.+)/)?.[2]));
debug('处理后的Twitch频道列表', {
count: realChannels.length,
channels: realChannels
});
if (realChannels.length > 0) {
for (const channel of realChannels) {
prom.push(this.#toggleChannel({
name: channel,
doTask: doTask
}));
await delay(1e3);
}
}
}
return Promise.all(prom).then((() => true));
} catch (error) {
debug('处理Twitch链接任务时发生错误', {
error: error
});
throwError(error, 'Twitch.toggle');
return false;
}
}
#setCache(name, id) {
try {
debug('设置Twitch频道ID缓存', {
name: name,
id: id
});
this.#cache[name] = id;
GM_setValue('twitchCache', this.#cache);
} catch (error) {
debug('设置Twitch频道ID缓存时发生错误', {
error: error
});
throwError(error, 'Twitch.setCache');
}
}
}
const encodeSha256 = async data => {
const encoder = new TextEncoder;
const dataBuffer = encoder.encode(data);
const hashBuffer = await crypto.subtle.digest('SHA-256', dataBuffer);
return Array.from(new Uint8Array(hashBuffer));
};
const encodeBase64 = data => {
let binary = '';
const bytes = new Uint8Array(data);
const len = bytes.byteLength;
for (let i = 0; i < len; i++) {
binary += String.fromCharCode(bytes[i]);
}
return btoa(binary).replace(/=/g, '');
};
const decodeBase64 = data => {
const binaryString = atob(data);
const len = binaryString.length;
const bytes = new Uint8Array(len);
for (let i = 0; i < len; i++) {
bytes[i] = binaryString.charCodeAt(i);
}
return Array.from(bytes);
};
const generateTransactionId = async (method, path, key, animationKey) => {
const DEFAULT_KEYWORD = 'obfiowerehiring';
const ADDITIONAL_RANDOM_NUMBER = 3;
const timeNow = Math.floor((Date.now() - 1682924400 * 1e3) / 1e3);
const timeNowBytes = [ timeNow & 255, timeNow >> 8 & 255, timeNow >> 16 & 255, timeNow >> 24 & 255 ];
const data = `${method}!${path}!${timeNow}${DEFAULT_KEYWORD}${animationKey}`;
const hashBytes = await encodeSha256(data);
const keyBytes = decodeBase64(key);
const randomNum = Math.floor(Math.random() * 256);
const bytesArr = [ ...keyBytes, ...timeNowBytes, ...hashBytes.slice(0, 16), ADDITIONAL_RANDOM_NUMBER ];
const out = new Uint8Array(bytesArr.length + 1);
out[0] = randomNum;
bytesArr.forEach(((item, index) => {
out[index + 1] = item ^ randomNum;
}));
return encodeBase64(out);
};
const url = 'https://raw.githubusercontent.com/fa0311/x-client-transaction-id-pair-dict/refs/heads/main/pair.json';
const getTID = async () => {
const res = await fetch(url);
const json = await res.json();
return async (method, path) => {
const randomPair = json[Math.floor(Math.random() * json.length)];
const {animationKey: animationKey, verification: verification} = randomPair;
const tid = await generateTransactionId(method, path, verification, animationKey);
return tid;
};
};
const parseResponseHeaders = headerStr => {
const headers = {};
if (!headerStr) {
return headers;
}
headerStr.split('\r\n').forEach((line => {
if (line) {
const parts = line.split(':');
const key = parts.shift()?.trim();
const value = parts.join(':').trim();
if (key) {
if (key.toLowerCase() === 'set-cookie') {
if (headers[key]) {
if (Array.isArray(headers[key])) {
headers[key].push(value);
} else {
headers[key] = [ headers[key], value ];
}
} else {
headers[key] = value;
}
} else {
headers[key] = value;
}
}
}
}));
return headers;
};
const axiosGM = function(config) {
const finalConfig = {
...axiosGM.defaults,
...config
};
const retries = finalConfig.retry ?? 0;
const retryDelay = finalConfig.retryDelay ?? 0;
const requestAttempt = attempt => new Promise(((resolve, reject) => {
GM_xmlhttpRequest({
method: finalConfig.method ? finalConfig.method.toUpperCase() : 'GET',
url: finalConfig.url,
headers: finalConfig.headers,
data: finalConfig.data,
responseType: finalConfig.responseType || 'json',
timeout: finalConfig.timeout,
fetch: finalConfig.fetch ?? true,
onload(response) {
const axiosResponse = {
data: response.response || response.responseText,
status: response.status,
statusText: response.statusText,
headers: parseResponseHeaders(response.responseHeaders),
config: finalConfig,
request: response
};
resolve(axiosResponse);
},
onerror(error) {
if (attempt < retries) {
setTimeout((() => {
requestAttempt(attempt + 1).then(resolve).catch(reject);
}), retryDelay);
} else {
reject(error);
}
},
ontimeout() {
if (attempt < retries) {
setTimeout((() => {
requestAttempt(attempt + 1).then(resolve).catch(reject);
}), retryDelay);
} else {
reject('Error: timeout');
}
}
});
}));
return requestAttempt(0);
};
axiosGM.defaults = {};
axiosGM.get = function(url, config = {}) {
return axiosGM({
...config,
url: url,
method: 'GET'
});
};
axiosGM.post = function(url, data, config = {}) {
return axiosGM({
...config,
url: url,
data: data,
method: 'POST'
});
};
axiosGM.head = function(url, config = {}) {
return axiosGM({
...config,
url: url,
method: 'HEAD'
});
};
axiosGM.create = function(instanceDefaults = {}) {
const instance = config => {
const mergedConfig = {
...axiosGM.defaults,
...instanceDefaults,
...config
};
return axiosGM(mergedConfig);
};
instance.defaults = {
...axiosGM.defaults,
...instanceDefaults
};
instance.get = function(url, config = {}) {
return instance({
...config,
url: url,
method: 'GET'
});
};
instance.post = function(url, data, config = {}) {
return instance({
...config,
url: url,
data: data,
method: 'POST'
});
};
instance.head = function(url, config = {}) {
return instance({
...config,
url: url,
method: 'HEAD'
});
};
instance.create = axiosGM.create;
return instance;
};
const getFwdForSdkUrl = async () => {
const rawHtml = await axiosGM({
url: 'https://x.com',
method: 'GET'
});
return [ ...rawHtml.data.matchAll(/"(loader\.FwdForSdk)":"([^"]+?)"/g) ];
};
const fwdForSdkExpoter = async url => {
const {data: data} = await axiosGM.get(url);
const regex = /Uint8Array\(\w\)\.set\(\[(.*?)\]\)/;
if (!regex.test(data)) {
return false;
}
const json = `[${data.match(regex)?.[1]}]`;
const obj = JSON.parse(json);
return new Uint8Array(obj);
};
const getWasmData = async () => {
const fwdForSdkUrl = await getFwdForSdkUrl();
for (const url of fwdForSdkUrl) {
const sdkData = await fwdForSdkExpoter(`https://abs.twimg.com/responsive-web/client-web/${url[1]}.${url[2]}a.js`);
if (sdkData) {
return sdkData;
}
}
return false;
};
const getFwdForSdk = async () => {
debug('开始获取 XFwdForSdk');
const wasmData = await getWasmData();
debug('获取 wasmData 成功', {
wasmData: wasmData
});
const go = new Go;
const wasmModule = await WebAssembly.instantiate(wasmData, {
...go.importObject,
env: {
...go.importObject.env,
memory: new WebAssembly.Memory({
initial: 10
}),
table: new WebAssembly.Table({
initial: 0,
element: 'anyfunc'
})
}
});
debug('初始化 wasmModule 成功');
go.run(wasmModule.instance);
debug('运行 wasmModule 成功');
const {str: str, expiryTimeMillis: expiryTimeMillis} = await globalThis.getForwardedForStr();
debug('获取 XFwdForSdk 成功', {
str: str,
expiryTimeMillis: expiryTimeMillis
});
return {
str: str,
expiryTimeMillis: parseInt(expiryTimeMillis, 10)
};
};
const generateSecCHUA = () => {
if (navigator.userAgentData && navigator.userAgentData.brands) {
return navigator.userAgentData.brands.map((brand => `"${brand.brand}";v="${brand.version}"`)).join(', ');
}
return '"Google Chrome";v="125", "Chromium";v="125", "Not-A.Brand";v="99"';
};
class Twitter extends Social {
tasks;
whiteList;
#verifyId=globalOptions.other.twitterVerifyId;
#auth=GM_getValue('twitterAuth') || {};
#cache=GM_getValue('twitterCache') || {};
#initialized=false;
#getTID;
#FwdForSdk;
#headers={};
constructor() {
super();
const defaultTasksTemplate = {
users: [],
retweets: [],
likes: []
};
debug('初始化Twitter实例');
this.tasks = defaultTasksTemplate;
this.whiteList = {
...defaultTasksTemplate,
...GM_getValue('whiteList')?.twitter || {}
};
this.#headers = {
authorization: 'Bearer AAAAAAAAAAAAAAAAAAAAANRILgAAAAAAnNwIzUejRCOuH5E6I8xnZz4puTs%3D1Zv7ttfk8LF81IUq16cHjhLTvJu4FA33AGWWjCpTnA',
'X-Twitter-Auth-Type': 'OAuth2Session',
'X-Twitter-Active-User': 'yes',
'sec-fetch-dest': 'empty',
'sec-fetch-mode': 'cors',
'sec-fetch-site': 'same-origin',
'sec-ch-ua-platform': '"Windows"',
'sec-ch-ua-mobile': '?0',
'sec-ch-ua': generateSecCHUA()
};
}
async init() {
try {
debug('开始初始化Twitter模块');
if (this.#initialized) {
debug('Twitter模块已初始化');
return true;
}
debug('获取Twitter授权信息');
if (!await this.#updateAuth()) {
return false;
}
debug('创建Twitter会话和SDK');
this.#getTID = await getTID();
this.#FwdForSdk = await getFwdForSdk();
const isVerified = await this.#verifyAuth();
if (isVerified) {
debug('Twitter授权验证成功');
echoLog({
before: '[Twitter]'
}).success(I18n('initSuccess', 'Twitter'));
this.#initialized = true;
return true;
}
debug('Twitter授权失效,尝试重新获取');
GM_setValue('twitterAuth', null);
if (await this.#updateAuth()) {
debug('Twitter重新授权成功');
echoLog({
before: '[Twitter]'
}).success(I18n('initSuccess', 'Twitter'));
this.#initialized = true;
return true;
}
debug('Twitter初始化失败');
echoLog({
before: '[Twitter]'
}).error(I18n('initFailed', 'Twitter'));
return false;
} catch (error) {
debug('Twitter初始化发生错误', {
error: error
});
throwError(error, 'Twitter.init');
return false;
}
}
async #verifyAuth() {
try {
debug('开始验证Twitter授权');
return await this.#toggleUser({
name: 'verify',
doTask: true,
verify: true
});
} catch (error) {
debug('Twitter授权验证发生错误', {
error: error
});
throwError(error, 'Twitter.verifyAuth');
return false;
}
}
async #updateAuth() {
try {
debug('开始更新Twitter授权');
const logStatus = echoLog({
text: I18n('updatingAuth', 'Twitter'),
before: '[Twitter]'
});
return await new Promise((resolve => {
GM_cookie.list({
url: 'https://x.com/settings/account'
}, (async (cookies, error) => {
if (!error) {
const ct0 = cookies.find((cookie => cookie.name === 'ct0'))?.value;
const isLogin = cookies.find((cookie => cookie.name === 'twid'))?.value;
if (isLogin && ct0) {
debug('成功获取Twitter授权信息');
GM_setValue('twitterAuth', {
ct0: ct0
});
this.#auth = {
ct0: ct0
};
this.#headers['x-csrf-token'] = ct0;
this.#headers['x-twitter-client-language'] = cookies.find((cookie => cookie.name === 'lang'))?.value || 'en';
logStatus.success();
resolve(true);
} else {
debug('获取Twitter授权失败:未登录');
logStatus.error(I18n('needLogin'));
resolve(false);
}
} else {
debug('获取Twitter授权失败', {
error: error
});
logStatus.error('Error: Update twitter auth failed!');
resolve(false);
}
}));
}));
} catch (error) {
debug('更新Twitter授权时发生错误', {
error: error
});
throwError(error, 'Twitter.updateToken');
return false;
}
}
async #toggleUser({name: name, doTask: doTask = true, verify: verify = false, retry: retry = false}) {
try {
debug('开始处理Twitter用户任务', {
name: name,
doTask: doTask,
verify: verify,
retry: retry
});
if (!doTask && !verify && this.whiteList.users.includes(name)) {
debug('Twitter用户在白名单中,跳过取消关注', {
name: name
});
echoLog({
type: 'whiteList',
text: 'Twitter.unfollowUser',
id: name,
before: '[Twitter]'
});
return true;
}
const userId = verify ? this.#verifyId : await this.userName2id(name);
if (!userId) {
return false;
}
const logStatus = verify ? echoLog({
text: I18n('verifyingAuth', 'Twitter'),
before: '[Twitter]'
}) : echoLog({
type: `${doTask ? '' : 'un'}followingTwitterUser`,
text: name,
before: '[Twitter]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://x.com/i/api/1.1/friendships/${doTask ? 'create' : 'destroy'}.json`,
method: 'POST',
headers: {
...this.#headers,
'Content-Type': 'application/x-www-form-urlencoded',
'x-client-transaction-id': await this.#getTID('POST', `/i/api/1.1/friendships/${doTask ? 'create' : 'destroy'}.json`),
'x-xp-forwarded-for': this.#FwdForSdk.str
},
responseType: 'json',
data: $.param({
include_profile_interstitial_type: 1,
include_blocking: 1,
include_blocked_by: 1,
include_followed_by: 1,
include_want_retweets: 1,
include_mute_edge: 1,
include_can_dm: 1,
include_can_media_tag: 1,
skip_status: 1,
id: userId
})
});
if (result !== 'Success') {
debug('Twitter用户操作请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 200) {
debug('Twitter用户操作成功', {
name: name,
doTask: doTask
});
logStatus.success();
if (doTask && !verify) {
this.tasks.users = unique([ ...this.tasks.users, name ]);
}
return true;
}
if (verify && data?.status === 403) {
if (data.response?.errors?.[0]?.code === 158) {
debug('Twitter授权验证成功(已关注)');
logStatus.success();
return true;
}
if (data.response?.errors?.[0]?.code === 353 && !retry && data.responseHeaders?.['set-cookie']) {
const newCt0 = data.responseHeaders['set-cookie']?.find((cookie => cookie.includes('ct0=')))?.split(';')?.at(0)?.split('=')?.at(-1);
if (newCt0) {
debug('获取到新的Twitter授权Token,重试操作');
this.#auth.ct0 = newCt0;
GM_setValue('twitterAuth', this.#auth);
logStatus.warning(I18n('retry'));
return this.#toggleUser({
name: name,
doTask: doTask,
verify: verify,
retry: true
});
}
}
}
debug('Twitter用户操作失败', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
} catch (error) {
debug('处理Twitter用户任务时发生错误', {
error: error
});
throwError(error, 'Twitter.toggleUser');
return false;
}
}
async userName2id(name) {
try {
debug('开始获取Twitter用户ID', {
name: name
});
const logStatus = echoLog({
type: 'gettingTwitterUserId',
text: name,
before: '[Twitter]'
});
const cachedUserId = this.#cache[name];
if (cachedUserId) {
debug('从缓存获取到Twitter用户ID', {
name: name,
id: cachedUserId
});
logStatus.success();
return cachedUserId;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://x.com/i/api/graphql/jUKA--0QkqGIFhmfRZdWrQ/UserByScreenName' + `?variables=%7B%22screen_name%22%3A%22${name}%22%7D&features=%7B%22responsive_web_grok_bio_auto_translation_is_enabled%22%3Afalse%2C%22hidden_profile_subscriptions_enabled%22%3Atrue%2C%22payments_enabled%22%3Afalse%2C%22profile_label_improvements_pcf_label_in_post_enabled%22%3Atrue%2C%22rweb_tipjar_consumption_enabled%22%3Atrue%2C%22verified_phone_label_enabled%22%3Afalse%2C%22subscriptions_verification_info_is_identity_verified_enabled%22%3Atrue%2C%22subscriptions_verification_info_verified_since_enabled%22%3Atrue%2C%22highlights_tweets_tab_ui_enabled%22%3Atrue%2C%22responsive_web_twitter_article_notes_tab_enabled%22%3Atrue%2C%22subscriptions_feature_can_gift_premium%22%3Atrue%2C%22creator_subscriptions_tweet_preview_api_enabled%22%3Atrue%2C%22responsive_web_graphql_skip_user_profile_image_extensions_enabled%22%3Afalse%2C%22responsive_web_graphql_timeline_navigation_enabled%22%3Atrue%7D&fieldToggles=%7B%22withAuxiliaryUserLabels%22%3Atrue%7D`,
method: 'GET',
headers: {
...this.#headers,
'content-type': 'application/json',
referer: `https://x.com/${name}`,
'x-client-transaction-id': await this.#getTID('GET', '/i/api/graphql/jUKA--0QkqGIFhmfRZdWrQ/UserByScreenName' + `?variables=%7B%22screen_name%22%3A%22${name}%22%7D&features=%7B%22responsive_web_grok_bio_auto_translation_is_enabled%22%3Afalse%2C%22hidden_profile_subscriptions_enabled%22%3Atrue%2C%22payments_enabled%22%3Afalse%2C%22profile_label_improvements_pcf_label_in_post_enabled%22%3Atrue%2C%22rweb_tipjar_consumption_enabled%22%3Atrue%2C%22verified_phone_label_enabled%22%3Afalse%2C%22subscriptions_verification_info_is_identity_verified_enabled%22%3Atrue%2C%22subscriptions_verification_info_verified_since_enabled%22%3Atrue%2C%22highlights_tweets_tab_ui_enabled%22%3Atrue%2C%22responsive_web_twitter_article_notes_tab_enabled%22%3Atrue%2C%22subscriptions_feature_can_gift_premium%22%3Atrue%2C%22creator_subscriptions_tweet_preview_api_enabled%22%3Atrue%2C%22responsive_web_graphql_skip_user_profile_image_extensions_enabled%22%3Afalse%2C%22responsive_web_graphql_timeline_navigation_enabled%22%3Atrue%7D&fieldToggles=%7B%22withAuxiliaryUserLabels%22%3Atrue%7D`),
'x-xp-forwarded-for': this.#FwdForSdk.str
},
responseType: 'json'
});
if (result !== 'Success') {
debug('获取Twitter用户ID请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('获取Twitter用户ID状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
let response = data.response || (typeof data.responseText === 'object' ? data.responseText : null);
if (!response) {
try {
response = JSON.parse(data.responseText);
} catch (error) {
response = null;
}
}
const fetchedUserId = response?.data?.user?.result?.rest_id;
if (!fetchedUserId) {
debug('未找到Twitter用户ID', {
name: name
});
logStatus.error(`Error:${data.statusText}(${data.status})`);
return false;
}
debug('成功获取Twitter用户ID', {
name: name,
id: fetchedUserId
});
this.#setCache(name, fetchedUserId);
logStatus.success();
return fetchedUserId;
} catch (error) {
debug('获取Twitter用户ID时发生错误', {
error: error
});
throwError(error, 'Twitter.getUserId');
return false;
}
}
async #toggleRetweet({retweetId: retweetId, doTask: doTask = true, retry: retry = false}) {
try {
debug('开始处理Twitter转推任务', {
retweetId: retweetId,
doTask: doTask,
retry: retry
});
if (!doTask && this.whiteList.retweets.includes(retweetId)) {
debug('Twitter转推在白名单中,跳过取消', {
retweetId: retweetId
});
echoLog({
type: 'whiteList',
text: 'Twitter.unretweet',
id: retweetId,
before: '[Twitter]'
});
return true;
}
const logStatus = echoLog({
type: `${doTask ? '' : 'un'}retweetting`,
text: retweetId,
before: '[Twitter]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://x.com/i/api/graphql/${doTask ? 'ojPdsZsimiJrUGLR1sjUtA/CreateRetweet' : 'iQtK4dl5hBmXewYZuEOKVw/DeleteRetweet'}`,
method: 'POST',
headers: {
...this.#headers,
'Content-Type': 'application/json',
origin: 'https://x.com',
referer: 'https://x.com/home',
'x-client-transaction-id': await this.#getTID('POST', `/i/api/graphql/${doTask ? 'ojPdsZsimiJrUGLR1sjUtA/CreateRetweet' : 'iQtK4dl5hBmXewYZuEOKVw/DeleteRetweet'}`),
'x-xp-forwarded-for': this.#FwdForSdk.str
},
data: `{"variables":{"${doTask ? '' : 'source_'}tweet_id":"${retweetId}","dark_request":false},"queryId":"${doTask ? 'ojPdsZsimiJrUGLR1sjUtA' : 'iQtK4dl5hBmXewYZuEOKVw'}"}`,
responseType: 'json'
});
if (result !== 'Success') {
debug('Twitter转推操作请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 403 && data.response?.errors?.[0]?.code === 353 && !retry && data.responseHeaders?.['set-cookie']) {
const newCt0 = data.responseHeaders['set-cookie']?.find((cookie => cookie.includes('ct0=')))?.split(';')?.at(0)?.split('=')?.at(-1);
if (newCt0) {
debug('获取到新的Twitter授权Token,重试操作');
this.#auth.ct0 = newCt0;
GM_setValue('twitterAuth', this.#auth);
logStatus.warning(I18n('retry'));
return this.#toggleRetweet({
retweetId: retweetId,
doTask: doTask,
retry: true
});
}
}
if (data?.status !== 200 && !(data?.status === 403 && data.response?.errors?.[0]?.code === 327)) {
debug('Twitter转推操作状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
if (data.response?.errors && data.response?.errors?.[0]?.code !== 327) {
debug('Twitter转推操作出错', {
error: data.response?.errors?.[0]?.message
});
logStatus.error(`Error:${data.response?.errors?.[0]?.message}`);
return false;
}
debug('Twitter转推操作成功', {
retweetId: retweetId,
doTask: doTask
});
logStatus.success();
if (doTask) {
this.tasks.retweets = unique([ ...this.tasks.retweets, retweetId ]);
}
return true;
} catch (error) {
debug('处理Twitter转推任务时发生错误', {
error: error
});
throwError(error, 'Twitter.toggleRetweet');
return false;
}
}
async toggle({doTask: doTask = true, userLinks: userLinks = [], retweetLinks: retweetLinks = []}) {
try {
debug('开始处理Twitter链接任务', {
doTask: doTask,
userLinksCount: userLinks.length,
retweetLinksCount: retweetLinks.length
});
if (!this.#initialized) {
debug('Twitter模块未初始化');
echoLog({
text: I18n('needInit'),
before: '[Twitter]'
});
return false;
}
if (doTask && !globalOptions.doTask.twitter.users || !doTask && !globalOptions.undoTask.twitter.users) {
debug('根据全局选项跳过Twitter用户任务', {
doTask: doTask
});
echoLog({
type: 'globalOptionsSkip',
text: 'twitter.users',
before: '[Twitter]'
});
} else {
const realUsers = this.getRealParams('users', userLinks, doTask, (link => link.match(/https:\/\/x\.com\/([^/]+)/)?.[1] || link.match(/https:\/\/twitter\.com\/([^/]+)/)?.[1]));
debug('处理后的Twitter用户列表', {
count: realUsers.length,
users: realUsers
});
if (realUsers.length > 0) {
for (const user of realUsers) {
if (Date.now() > this.#FwdForSdk.expiryTimeMillis) {
debug('Twitter SDK过期,重新获取', {
expiryTimeMillis: this.#FwdForSdk.expiryTimeMillis
});
this.#FwdForSdk = await getFwdForSdk();
}
await this.#toggleUser({
name: user,
doTask: doTask
});
await delay(1e3);
}
}
}
if (doTask && !globalOptions.doTask.twitter.retweets || !doTask && !globalOptions.undoTask.twitter.retweets) {
debug('根据全局选项跳过Twitter转推任务', {
doTask: doTask
});
echoLog({
type: 'globalOptionsSkip',
text: 'twitter.retweets',
before: '[Twitter]'
});
} else {
const realRetweets = this.getRealParams('retweets', retweetLinks, doTask, (link => link.match(/https:\/\/x\.com\/.*?\/status\/([\d]+)/)?.[1] || link.match(/https:\/\/twitter\.com\/.*?\/status\/([\d]+)/)?.[1]));
debug('处理后的Twitter转推列表', {
count: realRetweets.length,
retweets: realRetweets
});
if (realRetweets.length > 0) {
for (const retweet of realRetweets) {
if (Date.now() > this.#FwdForSdk.expiryTimeMillis) {
debug('Twitter SDK过期,重新获取');
this.#FwdForSdk = await getFwdForSdk();
}
await this.#toggleRetweet({
retweetId: retweet,
doTask: doTask
});
await delay(1e3);
}
}
}
return true;
} catch (error) {
debug('处理Twitter链接任务时发生错误', {
error: error
});
throwError(error, 'Twitter.toggle');
return false;
}
}
#setCache(name, id) {
try {
debug('设置Twitter用户ID缓存', {
name: name,
id: id
});
this.#cache[name] = id;
GM_setValue('twitterCache', this.#cache);
} catch (error) {
debug('设置Twitter用户ID缓存时发生错误', {
error: error
});
throwError(error, 'Twitter.setCache');
}
}
}
class Vk extends Social {
tasks;
whiteList;
#username='';
#cache=GM_getValue('vkCache') || {};
#initialized=false;
constructor() {
super();
const defaultTasksTemplate = {
names: []
};
debug('初始化Vk实例');
this.tasks = defaultTasksTemplate;
this.whiteList = {
...defaultTasksTemplate,
...GM_getValue('whiteList')?.vk || {}
};
}
async init() {
try {
debug('开始初始化Vk模块');
if (this.#initialized) {
debug('Vk模块已初始化');
return true;
}
const isVerified = await this.#verifyAuth();
if (isVerified) {
debug('Vk授权验证成功');
echoLog({
before: '[Vk]'
}).success(I18n('initSuccess', 'Vk'));
this.#initialized = true;
return true;
}
debug('Vk初始化失败');
echoLog({
before: '[Vk]'
}).error(I18n('initFailed', 'Vk'));
return false;
} catch (error) {
debug('Vk初始化发生错误', {
error: error
});
throwError(error, 'Vk.init');
return false;
}
}
async #verifyAuth() {
try {
debug('开始验证Vk授权');
const logStatus = echoLog({
text: I18n('verifyAuth', 'Vk'),
before: '[Vk]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://vk.com/im',
method: 'GET'
});
if (result !== 'Success') {
debug('Vk授权验证请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.finalUrl.includes('vk.com/login')) {
debug('Vk授权验证失败:需要登录');
logStatus.error(`Error:${I18n('loginVk')}`, true);
return false;
}
if (data?.status !== 200) {
debug('Vk授权验证状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
this.#username = data.responseText.match(/TopNavBtn__profileLink" href="\/(.*?)"/)?.[1] || '';
debug('Vk授权验证成功');
logStatus.success();
return true;
} catch (error) {
debug('Vk授权验证发生错误', {
error: error
});
throwError(error, 'Vk.verifyAuth');
return false;
}
}
async #toggleGroup(name, dataParam, doTask = true) {
try {
debug('开始处理Vk群组任务', {
name: name,
doTask: doTask
});
const logStatus = echoLog({
type: doTask ? 'joiningVkGroup' : 'leavingVkGroup',
text: name,
before: '[Vk]'
});
if (dataParam.groupAct === 'enter' && !doTask || dataParam.groupAct === 'leave' && doTask) {
debug('Vk群组操作已完成,跳过', {
name: name,
doTask: doTask
});
logStatus.success();
return true;
}
const reqData = {
act: doTask ? 'enter' : 'leave',
al: 1,
gid: dataParam.groupId,
hash: dataParam.groupHash
};
if (doTask) {
reqData.context = '_';
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://vk.com/al_groups.php',
method: 'POST',
headers: {
origin: 'https://vk.com',
referer: `https://vk.com/${name}`,
'content-type': 'application/x-www-form-urlencoded'
},
data: $.param(reqData)
});
if (result !== 'Success') {
debug('Vk群组操作请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('Vk群组操作状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('Vk群组操作成功', {
name: name,
doTask: doTask
});
logStatus.success();
if (doTask) {
this.tasks.names = unique([ ...this.tasks.names, name ]);
}
return true;
} catch (error) {
debug('处理Vk群组任务时发生错误', {
error: error
});
throwError(error, 'Vk.toggleGroup');
return false;
}
}
async #togglePublic(name, dataParam, doTask = true) {
try {
debug('开始处理Vk公共页面任务', {
name: name,
doTask: doTask
});
const logStatus = echoLog({
type: doTask ? 'joiningVkPublic' : 'leavingVkPublic',
text: name,
before: '[Vk]'
});
if (dataParam.publicJoined && doTask || !dataParam.publicJoined && !doTask) {
debug('Vk公共页面操作已完成,跳过', {
name: name,
doTask: doTask
});
logStatus.success();
return true;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://vk.com/al_public.php',
method: 'POST',
headers: {
origin: 'https://vk.com',
referer: `https://vk.com/${name}`,
'content-type': 'application/x-www-form-urlencoded'
},
data: $.param({
act: doTask ? 'a_enter' : 'a_leave',
al: 1,
pid: dataParam.publicPid,
hash: dataParam.publicHash
})
});
if (result !== 'Success') {
debug('Vk公共页面操作请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('Vk公共页面操作状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('Vk公共页面操作成功', {
name: name,
doTask: doTask
});
logStatus.success();
if (doTask) {
this.tasks.names = unique([ ...this.tasks.names, name ]);
}
return true;
} catch (error) {
debug('处理Vk公共页面任务时发生错误', {
error: error
});
throwError(error, 'Vk.togglePublic');
return false;
}
}
async #toggleLikeWall(name, dataParam, doTask = true) {
try {
debug('开始处理Vk点赞任务', {
name: name,
doTask: doTask
});
const logStatus = echoLog({
type: doTask ? 'likingVkPublic' : 'unlikingVkPublic',
text: name,
before: '[Vk]'
});
const postData = {
act: 'a_set_reaction',
al: 1,
event_subtype: 'post_modal',
from: 'wall_page',
hash: dataParam.hash,
object: dataParam.object,
track_code: dataParam.trackCode,
wall: 2
};
if (doTask) {
postData.reaction_id = 0;
}
const {result: resultR, statusText: statusTextR, status: statusR, data: dataR} = await httpRequest({
url: 'https://vk.com/like.php?act=a_set_reaction',
method: 'POST',
headers: {
origin: 'https://vk.com',
referer: `https://vk.com/${name}`,
'content-type': 'application/x-www-form-urlencoded'
},
data: $.param(postData)
});
if (resultR !== 'Success') {
debug('Vk点赞操作请求失败', {
result: resultR,
statusText: statusTextR,
status: statusR
});
logStatus.error(`${resultR}:${statusTextR}(${statusR})`);
return false;
}
if (dataR?.status !== 200) {
debug('Vk点赞操作状态错误', {
status: dataR?.status,
statusText: dataR?.statusText
});
logStatus.error(`Error:${dataR?.statusText}(${dataR?.status})`);
return false;
}
if (dataR.response?.payload?.[1]?.[1]?.like_my !== true) {
debug('Vk点赞操作验证失败');
logStatus.error(`Error:${dataR?.statusText}(${dataR?.status})`);
return false;
}
debug('Vk点赞操作成功', {
name: name,
doTask: doTask
});
logStatus.success();
return true;
} catch (error) {
debug('处理Vk点赞任务时发生错误', {
error: error
});
throwError(error, 'Vk.sendWall');
return false;
}
}
async #sendWall(name) {
try {
debug('开始处理Vk转发任务', {
name: name
});
const logStatus = echoLog({
type: 'sendingVkWall',
text: name,
before: '[Vk]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://vk.com/like.php',
method: 'POST',
headers: {
origin: 'https://vk.com',
referer: `https://vk.com/${name}`,
'content-type': 'application/x-www-form-urlencoded'
},
data: $.param({
act: 'publish_box',
al: 1,
object: name
})
});
if (result !== 'Success') {
debug('Vk转发操作请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('Vk转发操作状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const hash = data.responseText.match(/shHash:[\s]*'(.*?)'/)?.[1];
if (!hash) {
debug('获取Vk转发hash失败');
logStatus.error('Error: Get "hash" failed');
return false;
}
const {result: resultR, statusText: statusTextR, status: statusR, data: dataR} = await httpRequest({
url: 'https://vk.com/like.php',
method: 'POST',
headers: {
origin: 'https://vk.com',
referer: `https://vk.com/${name}`,
'content-type': 'application/x-www-form-urlencoded'
},
data: $.param({
Message: '',
act: 'a_do_publish',
al: 1,
close_comments: 0,
friends_only: 0,
from: 'box',
hash: hash,
list: '',
mark_as_ads: 0,
mute_notifications: 0,
object: name,
ret_data: 1,
to: 0
})
});
if (resultR !== 'Success') {
debug('Vk转发确认请求失败', {
result: resultR,
statusText: statusTextR,
status: statusR
});
logStatus.error(`${resultR}:${statusTextR}(${statusR})`);
return false;
}
if (dataR?.status !== 200) {
debug('Vk转发确认状态错误', {
status: dataR?.status,
statusText: dataR?.statusText
});
logStatus.error(`Error:${dataR?.statusText}(${dataR?.status})`);
return false;
}
const jsonData = JSON.parse(dataR.responseText?.replace('\x3c!--', '') || '{}');
if (jsonData?.payload?.[1]?.[1]?.share_my !== true) {
debug('Vk转发确认验证失败');
logStatus.error(`Error:${dataR?.statusText}(${dataR?.status})`);
return false;
}
debug('Vk转发操作成功', {
name: name
});
logStatus.success();
const postId = jsonData?.payload?.[1]?.[1]?.post_id;
const ownerId = jsonData?.payload?.[1]?.[1]?.owner_id;
if (postId && ownerId) {
this.#setCache(name, `${ownerId}_${postId}`);
}
this.tasks.names = unique([ ...this.tasks.names, name ]);
return true;
} catch (error) {
debug('处理Vk转发任务时发生错误', {
error: error
});
throwError(error, 'Vk.sendWall');
return false;
}
}
async #deleteWall(name, dataParams) {
try {
debug('开始处理Vk删除墙任务', {
name: name
});
const logStatus = echoLog({
type: 'deletingVkWall',
text: name,
before: '[Vk]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://vk.com/al_wall.php?act=delete',
method: 'POST',
headers: {
origin: 'https://vk.com',
referer: `https://vk.com/${this.#username}?w=wall${this.#cache[name]}%2Fall`,
'content-type': 'application/x-www-form-urlencoded'
},
data: $.param({
act: 'delete',
al: 1,
confirm: 0,
from: 'wkview',
hash: dataParams.wallHash,
post: this.#cache[name]
})
});
if (result !== 'Success') {
debug('Vk删除墙请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('Vk删除墙状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const jsonData = JSON.parse(data.responseText?.replace('\x3c!--', '') || '{}');
if (!jsonData?.payload?.[1]?.[1]) {
debug('Vk删除墙验证失败');
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('Vk删除墙操作成功', {
name: name
});
logStatus.success();
return true;
} catch (error) {
debug('处理Vk删除墙任务时发生错误', {
error: error
});
throwError(error, 'Vk.deleteWall');
return false;
}
}
async #getId(name, doTask) {
try {
debug('开始获取Vk ID', {
name: name,
doTask: doTask
});
let url = `https://vk.com/${name}`;
if (/^wall-/.test(name)) {
if (doTask) {
return {
type: 'sendWall'
};
}
if (!this.#cache[name]) {
return {
type: 'unSupport'
};
}
url = `https://vk.com/${this.#username}?w=wall${this.#cache[name]}`;
}
const logStatus = echoLog({
type: 'gettingVkId',
text: name,
before: '[Vk]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: url,
method: 'GET'
});
if (result !== 'Success') {
debug('获取Vk ID请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('获取Vk ID状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const [, groupAct, groupId, , groupHash] = data.responseText.match(/Groups.(enter|leave)\(.*?,.*?([\d]+?), ('|')(.*?)('|')/) || [];
const publicHash = data.responseText.match(/"enterHash":"(.*?)"/)?.[1];
const publicPid = data.responseText.match(/"public_id":([\d]+?),/)?.[1];
const publicJoined = !data.responseText.includes('Public.subscribe');
if (groupAct && groupId && groupHash) {
debug('获取到Vk群组ID', {
groupAct: groupAct,
groupId: groupId,
groupHash: groupHash
});
logStatus.success();
return {
groupAct: groupAct,
groupId: groupId,
groupHash: groupHash,
type: 'group'
};
}
if (publicHash && publicPid) {
debug('获取到Vk公共页面ID', {
publicHash: publicHash,
publicPid: publicPid,
publicJoined: publicJoined
});
logStatus.success();
return {
publicHash: publicHash,
publicPid: publicPid,
publicJoined: publicJoined,
type: 'public'
};
}
if (name.includes('action=like')) {
const hash = data.responseText.match(/data-reaction-hash="(.*?)"/)?.[1];
const trackCode = data.responseText.match(/data-post-track-code="(.*?)"/)?.[1];
const object = name.match(/wall-[\w_]+/)?.[0];
if (hash && trackCode && object) {
debug('获取到Vk点赞ID', {
hash: hash,
trackCode: trackCode,
object: object
});
logStatus.success();
return {
type: 'likeWall',
hash: hash,
trackCode: trackCode,
object: object
};
}
}
if (data.responseText.includes('wall.deletePost') && !doTask) {
const wallHash = data.responseText.match(/wall\.deletePost\(this, '.*?', '(.*?)'\)/)?.[1];
if (wallHash) {
debug('获取到Vk删除墙ID', {
wallHash: wallHash
});
logStatus.success();
return {
type: 'deleteWall',
wallHash: wallHash
};
}
}
if (name.includes('wall') && doTask) {
debug('获取到Vk墙ID');
logStatus.success();
return {
type: 'sendWall'
};
}
debug('未找到Vk ID参数');
logStatus.error('Error: Parameters not found!');
return false;
} catch (error) {
debug('获取Vk ID时发生错误', {
error: error
});
throwError(error, 'Vk.getId');
return false;
}
}
async #toggleVk({name: name, doTask: doTask = true}) {
try {
debug('开始处理Vk任务', {
name: name,
doTask: doTask
});
if (!doTask && this.whiteList.names.includes(name)) {
debug('Vk任务在白名单中,跳过', {
name: name
});
echoLog({
type: 'whiteList',
text: 'Vk.undoTask',
id: name,
before: '[Vk]'
});
return true;
}
const formatName = name.replace(/\/$/, '');
const data = await this.#getId(formatName, doTask);
if (!data) {
return false;
}
switch (data.type) {
case 'group':
return await this.#toggleGroup(formatName, data, doTask);
case 'public':
return await this.#togglePublic(formatName, data, doTask);
case 'likeWall':
return await this.#toggleLikeWall(formatName, data, doTask);
case 'sendWall':
return doTask ? await this.#sendWall(formatName) : true;
case 'deleteWall':
return doTask ? true : await this.#deleteWall(formatName, data);
default:
debug('未知的Vk任务类型', {
type: data.type
});
return false;
}
} catch (error) {
debug('处理Vk任务时发生错误', {
error: error
});
throwError(error, 'Vk.toggleVk');
return false;
}
}
async toggle({doTask: doTask = true, nameLinks: nameLinks = []}) {
try {
debug('开始处理Vk链接任务', {
doTask: doTask,
nameLinksCount: nameLinks.length
});
if (!this.#initialized) {
debug('Vk模块未初始化');
echoLog({
text: I18n('needInit'),
before: '[Vk]'
});
return false;
}
const prom = [];
if (doTask && !globalOptions.doTask.vk.names || !doTask && !globalOptions.undoTask.vk.names) {
debug('根据全局选项跳过Vk任务', {
doTask: doTask
});
echoLog({
type: 'globalOptionsSkip',
text: 'vk.names',
before: '[Vk]'
});
} else {
const realNames = this.getRealParams('names', nameLinks, doTask, (link => link.match(/https:\/\/vk\.com\/([^/]+)/)?.[1]));
debug('处理后的Vk链接列表', {
count: realNames.length,
names: realNames
});
if (realNames.length > 0) {
for (const name of realNames) {
prom.push(this.#toggleVk({
name: name,
doTask: doTask
}));
await delay(1e3);
}
}
}
return Promise.all(prom).then((() => true));
} catch (error) {
debug('处理Vk链接任务时发生错误', {
error: error
});
throwError(error, 'Vk.toggle');
return false;
}
}
#setCache(name, postId) {
try {
debug('设置Vk缓存', {
name: name,
postId: postId
});
this.#cache[name] = postId;
GM_setValue('vkCache', this.#cache);
} catch (error) {
debug('设置Vk缓存时发生错误', {
error: error
});
throwError(error, 'Vk.setCache');
}
}
}
const getInfo = async function(link, type) {
try {
debug('开始获取YouTube信息', {
link: link,
type: type
});
const logStatus = echoLog({
text: I18n('gettingYtbToken'),
before: '[Youtube]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: link,
method: 'GET'
});
if (result !== 'Success') {
debug('获取YouTube信息请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return {};
}
if (data?.status !== 200) {
debug('获取YouTube信息状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return {};
}
if (data.responseText.includes('accounts.google.com/ServiceLogin?service=youtube')) {
debug('获取YouTube信息失败:需要登录');
logStatus.error(`Error:${I18n('loginYtb')}`, true);
return {
needLogin: true
};
}
const apiKey = data.responseText.match(/"INNERTUBE_API_KEY":"(.*?)"/)?.[1];
const context = (data.responseText.match(/\(\{"INNERTUBE_CONTEXT":([\w\W]*?)\}\)/) || data.responseText.match(/"INNERTUBE_CONTEXT":([\w\W]*?\}),"INNERTUBE/))?.[1] || '{}';
const {client: client, request: request} = JSON.parse(context);
if (!apiKey || !client || !request) {
debug('获取YouTube信息失败:缺少必要参数');
logStatus.error('Error: Parameter "apiKey" not found!');
return {};
}
client.hl = 'en';
if (type === 'channel') {
const channelId = data.responseText.match(/"channelId":"(.+?)"/)?.[1];
if (!channelId) {
debug('获取YouTube频道ID失败');
logStatus.error('Error: Get "channelId" failed!');
return {};
}
debug('成功获取YouTube频道信息', {
channelId: channelId
});
logStatus.success();
return {
params: {
apiKey: apiKey,
client: client,
request: request,
channelId: channelId
}
};
}
if (type === 'likeVideo') {
const videoId = data.responseText.match(/<link rel="shortlinkUrl" href="https:\/\/youtu\.be\/(.*?)">/)?.[1];
const likeParams = data.responseText.match(/"likeParams":"(.*?)"/)?.[1];
if (!videoId) {
debug('获取YouTube视频ID失败');
logStatus.error('Error: Get "videoId" failed!');
return {};
}
debug('成功获取YouTube视频信息', {
videoId: videoId
});
logStatus.success();
return {
params: {
apiKey: apiKey,
client: client,
request: request,
videoId: videoId,
likeParams: likeParams
}
};
}
debug('未知的YouTube信息类型', {
type: type
});
logStatus.error('Error: Unknown type');
return {};
} catch (error) {
debug('获取YouTube信息时发生错误', {
error: error
});
throwError(error, 'Youtube.getInfo');
return {};
}
};
class Youtube extends Social {
tasks;
whiteList;
#auth=GM_getValue('youtubeAuth') || {};
#initialized=false;
#verifyChannel=`https://www.youtube.com/channel/${globalOptions.other.youtubeVerifyChannel}`;
constructor() {
super();
const defaultTasksTemplate = {
channels: [],
likes: []
};
debug('初始化YouTube实例');
this.tasks = defaultTasksTemplate;
this.whiteList = {
...defaultTasksTemplate,
...GM_getValue('whiteList')?.youtube || {}
};
}
async init() {
try {
debug('开始初始化YouTube模块');
if (this.#initialized) {
debug('YouTube模块已初始化');
return true;
}
if (!this.#auth.PAPISID) {
debug('YouTube授权信息不完整,需要更新授权');
if (await this.#updateAuth()) {
this.#initialized = true;
return true;
}
return false;
}
const isVerified = await this.#verifyAuth();
if (isVerified) {
debug('YouTube授权验证成功');
echoLog({
before: '[Youtube]'
}).success(I18n('initSuccess', 'Youtube'));
this.#initialized = true;
return true;
}
debug('YouTube授权失效,尝试重新获取');
GM_setValue('youtubeAuth', null);
if (await this.#updateAuth()) {
debug('YouTube重新授权成功');
echoLog({
before: '[Youtube]'
}).success(I18n('initSuccess', 'Youtube'));
this.#initialized = true;
return true;
}
debug('YouTube初始化失败');
echoLog({
before: '[Youtube]'
}).error(I18n('initFailed', 'Youtube'));
return false;
} catch (error) {
debug('YouTube初始化发生错误', {
error: error
});
throwError(error, 'Youtube.init');
return false;
}
}
async #verifyAuth() {
try {
debug('开始验证YouTube授权');
return await this.#toggleChannel({
link: this.#verifyChannel,
doTask: true,
verify: true
});
} catch (error) {
debug('YouTube授权验证发生错误', {
error: error
});
throwError(error, 'Youtube.verifyAuth');
return false;
}
}
async #updateAuth() {
try {
debug('开始更新YouTube授权');
const logStatus = echoLog({
text: I18n('updatingAuth', 'Youtube'),
before: '[Youtube]'
});
return await new Promise((resolve => {
GM_cookie.list({
url: 'https://www.youtube.com/@YouTube'
}, (async (cookies, error) => {
if (!error) {
const PAPISID = cookies.find((cookie => cookie.name === '__Secure-3PAPISID'))?.value;
if (PAPISID) {
debug('成功获取YouTube新授权信息');
GM_setValue('youtubeAuth', {
PAPISID: PAPISID
});
this.#auth = {
PAPISID: PAPISID
};
logStatus.success();
resolve(await this.#verifyAuth());
} else {
debug('获取YouTube授权失败:未登录');
logStatus.error(I18n('needLogin'));
resolve(false);
}
} else {
debug('获取YouTube授权失败', {
error: error
});
logStatus.error('Error: Update youtube auth failed!');
resolve(false);
}
}));
}));
} catch (error) {
debug('更新YouTube授权时发生错误', {
error: error
});
throwError(error, 'Youtube.updateAuth');
return false;
}
}
#getInfo(link, type) {
debug('调用获取YouTube信息方法', {
link: link,
type: type
});
return getInfo(link, type);
}
async #toggleChannel({link: link, doTask: doTask = true, verify: verify = false}) {
try {
debug('开始处理YouTube频道任务', {
link: link,
doTask: doTask,
verify: verify
});
const {params: params, needLogin: needLogin} = await this.#getInfo(link, 'channel');
const {apiKey: apiKey, client: client, request: request, channelId: channelId} = params || {};
if (needLogin) {
debug('YouTube频道操作失败:需要登录');
echoLog({
html: I18n('loginYtb'),
before: '[Youtube]'
});
return false;
}
if (!(apiKey && client && request && channelId)) {
debug('YouTube频道操作失败:获取参数失败');
echoLog({
text: '"getYtbToken" failed',
before: '[Youtube]'
});
return false;
}
if (!doTask && !verify && this.whiteList.channels.includes(channelId)) {
debug('YouTube频道在白名单中,跳过取消订阅', {
channelId: channelId
});
echoLog({
type: 'whiteList',
text: 'Youtube.unfollowChannel',
id: channelId,
before: '[Youtube]'
});
return true;
}
const logStatus = verify ? echoLog({
text: I18n('verifyingAuth', 'Youtube'),
before: '[Youtube]'
}) : echoLog({
type: doTask ? 'followingYtbChannel' : 'unfollowingYtbChannel',
text: channelId,
before: '[Youtube]'
});
const nowTime = parseInt(String((new Date).getTime() / 1e3), 10);
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://www.youtube.com/youtubei/v1/subscription/${doTask ? '' : 'un'}subscribe?key=${apiKey}&prettyPrint=false`,
method: 'POST',
headers: {
origin: 'https://www.youtube.com',
referer: `https://www.youtube.com/channel/${channelId}`,
'content-type': 'application/json',
'x-goog-authuser': '0',
'x-goog-visitor-id': client?.visitorData,
'x-origin': 'https://www.youtube.com',
authorization: `SAPISIDHASH ${nowTime}_${sha1(`${nowTime} ${this.#auth.PAPISID} https://www.youtube.com`)}`
},
data: JSON.stringify({
context: {
client: client,
request: {
sessionId: request?.sessionId,
internalExperimentFlags: [],
consistencyTokenJars: []
},
user: {}
},
channelIds: [ channelId ],
params: doTask ? 'EgIIAhgA' : 'CgIIAhgA'
})
});
if (result !== 'Success') {
debug('YouTube频道操作请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('YouTube频道操作状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const isSubscribed = doTask && (/"subscribed":true/.test(data.responseText) || data.responseText.includes('The subscription already exists'));
const isUnsubscribed = !doTask && /"subscribed":false/.test(data.responseText);
const isVerified = verify && data.responseText.includes('You may not subscribe to yourself');
if (isSubscribed || isUnsubscribed || isVerified) {
debug('YouTube频道操作成功', {
doTask: doTask,
verify: verify
});
logStatus.success();
if (doTask && !verify) {
this.tasks.channels = unique([ ...this.tasks.channels, link ]);
}
return true;
}
debug('YouTube频道操作失败,需要更新授权');
logStatus.error(I18n('tryUpdateYtbAuth'), true);
return false;
} catch (error) {
debug('处理YouTube频道任务时发生错误', {
error: error
});
throwError(error, 'Youtube.toggleChannel');
return false;
}
}
async #toggleLikeVideo({link: link, doTask: doTask = true}) {
try {
debug('开始处理YouTube视频点赞任务', {
link: link,
doTask: doTask
});
const {params: params, needLogin: needLogin} = await this.#getInfo(link, 'likeVideo');
const {apiKey: apiKey, client: client, request: request, videoId: videoId, likeParams: likeParams} = params || {};
if (needLogin) {
debug('YouTube视频点赞失败:需要登录');
echoLog({
html: `${I18n('loginYtb')}`,
before: '[Youtube]'
});
return false;
}
if (!(apiKey && client && request && videoId && likeParams)) {
debug('YouTube视频点赞失败:获取参数失败');
echoLog({
text: '"getYtbToken" failed',
before: '[Youtube]'
});
return false;
}
if (!doTask && this.whiteList.likes.includes(videoId)) {
debug('YouTube视频在白名单中,跳过取消点赞', {
videoId: videoId
});
echoLog({
type: 'whiteList',
text: 'Youtube.unlikeVideo',
id: videoId,
before: '[Youtube]'
});
return true;
}
const logStatus = echoLog({
type: doTask ? 'likingYtbVideo' : 'unlikingYtbVideo',
text: videoId,
before: '[Youtube]'
});
const nowTime = parseInt(String((new Date).getTime() / 1e3), 10);
const likeVideoData = {
context: {
client: client,
request: {
sessionId: request.sessionId,
internalExperimentFlags: [],
consistencyTokenJars: []
},
user: {}
},
target: {
videoId: videoId
}
};
if (doTask && !likeParams) {
debug('YouTube视频点赞失败:缺少likeParams参数');
logStatus.error('Empty likeParams');
return false;
}
if (doTask) {
likeVideoData.params = likeParams;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://www.youtube.com/youtubei/v1/like/${doTask ? '' : 'remove'}like?key=${apiKey}`,
method: 'POST',
headers: {
origin: 'https://www.youtube.com',
referer: `https://www.youtube.com/watch?v=${videoId}`,
'content-type': 'application/json',
'x-goog-authuser': '0',
'x-goog-visitor-id': client.visitorData,
'x-origin': 'https://www.youtube.com',
authorization: `SAPISIDHASH ${nowTime}_${sha1(`${nowTime} ${this.#auth.PAPISID} https://www.youtube.com`)}`
},
data: JSON.stringify(likeVideoData)
});
if (result !== 'Success') {
debug('YouTube视频点赞请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('YouTube视频点赞状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const isLiked = doTask && data.responseText.includes('Added to Liked videos');
const isUnliked = !doTask && (data.responseText.includes('Removed from Liked videos') || data.responseText.includes('Dislike removed'));
if (isLiked || isUnliked) {
debug('YouTube视频点赞操作成功', {
doTask: doTask
});
logStatus.success();
if (doTask) {
this.tasks.likes = unique([ ...this.tasks.likes, link ]);
}
return true;
}
debug('YouTube视频点赞失败,需要更新授权');
logStatus.error(I18n('tryUpdateYtbAuth'), true);
return false;
} catch (error) {
debug('处理YouTube视频点赞任务时发生错误', {
error: error
});
throwError(error, 'Youtube.toggleLikeVideo');
return false;
}
}
async toggle({doTask: doTask = true, channelLinks: channelLinks = [], videoLinks: videoLinks = []}) {
try {
debug('开始处理YouTube链接任务', {
doTask: doTask,
channelLinksCount: channelLinks.length,
videoLinksCount: videoLinks.length
});
if (!this.#initialized) {
debug('YouTube模块未初始化');
echoLog({
text: I18n('needInit'),
before: '[Youtube]'
});
return false;
}
const prom = [];
const shouldProcessChannels = doTask && globalOptions.doTask.youtube.channels || !doTask && globalOptions.undoTask.youtube.channels;
const shouldProcessVideos = doTask && globalOptions.doTask.youtube.likes || !doTask && globalOptions.undoTask.youtube.likes;
if (!shouldProcessChannels) {
debug('根据全局选项跳过YouTube频道任务', {
doTask: doTask
});
echoLog({
type: 'globalOptionsSkip',
text: 'youtube.channels',
before: '[Youtube]'
});
} else {
const realChannels = this.getRealParams('channels', channelLinks, doTask, (link => {
if (/^https:\/\/(www\.)?google\.com.*?\/url\?.*?url=https:\/\/www\.youtube\.com\/.*/.test(link)) {
return link.match(/url=(https:\/\/www\.youtube\.com\/.*)/)?.[1];
}
return link;
}));
debug('处理后的YouTube频道链接列表', {
count: realChannels.length,
channels: realChannels
});
if (realChannels.length > 0) {
for (const channel of realChannels) {
prom.push(this.#toggleChannel({
link: channel,
doTask: doTask
}));
await delay(1e3);
}
}
}
if (!shouldProcessVideos) {
debug('根据全局选项跳过YouTube视频任务', {
doTask: doTask
});
echoLog({
type: 'globalOptionsSkip',
text: 'youtube.likes',
before: '[Youtube]'
});
} else {
const realLikes = this.getRealParams('likes', videoLinks, doTask, (link => {
if (/^https:\/\/(www\.)?google\.com.*?\/url\?.*?url=https:\/\/www\.youtube\.com\/.*/.test(link)) {
return link.match(/url=(https:\/\/www\.youtube\.com\/.*)/)?.[1];
}
return link;
}));
debug('处理后的YouTube视频链接列表', {
count: realLikes.length,
videos: realLikes
});
if (realLikes.length > 0) {
for (const video of realLikes) {
prom.push(this.#toggleLikeVideo({
link: video,
doTask: doTask
}));
await delay(1e3);
}
}
}
return Promise.all(prom).then((() => true));
} catch (error) {
debug('处理YouTube链接任务时发生错误', {
error: error
});
throwError(error, 'Youtube.toggle');
return false;
}
}
}
class SteamASF {
#asfOptions;
#botName='asf';
#groupInfo;
#steamWebApiKey;
#steamId;
constructor({AsfIpcUrl: AsfIpcUrl, AsfIpcPassword: AsfIpcPassword, AsfBotname: AsfBotname, steamWebApiKey: steamWebApiKey}) {
debug('初始化SteamASF实例', {
AsfIpcUrl: AsfIpcUrl,
AsfBotname: AsfBotname
});
const asfCommandsUrl = new URL('/Api/Command/', AsfIpcUrl);
this.#asfOptions = {
url: asfCommandsUrl.href,
method: 'POST',
responseType: 'json',
headers: {
accept: 'application/json',
'Content-Type': 'application/json',
Host: asfCommandsUrl.host,
Origin: asfCommandsUrl.origin,
Referer: asfCommandsUrl.href,
Authentication: AsfIpcPassword
}
};
if (AsfBotname) {
this.#botName = AsfBotname;
}
if (steamWebApiKey) {
this.#steamWebApiKey = steamWebApiKey;
}
debug('SteamASF实例初始化完成', {
botName: this.#botName
});
}
async init() {
try {
debug('开始初始化ASF');
const logStatus = echoLog({
text: I18n('initingASF'),
before: '[ASF]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
...this.#asfOptions,
data: '{"Command":"!stats"}'
});
if (result !== 'Success') {
debug('ASF初始化请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.response?.Success === true && data.response.Message === 'OK' && data.response.Result) {
debug('ASF初始化成功');
logStatus.success();
return true;
}
if (data?.response?.Result || data?.response?.Message) {
debug('ASF初始化失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(data?.response?.Result || data.response.Message);
return false;
}
debug('ASF初始化失败', {
statusText: data?.statusText,
status: data?.status
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
} catch (error) {
debug('ASF初始化发生错误', {
error: error
});
throwError(error, 'SteamASF.init');
return false;
}
}
async joinGroup(groupName) {
try {
debug('开始加入Steam组', {
groupName: groupName
});
const logStatus = echoLog({
type: 'joiningSteamGroup',
text: groupName,
before: '[ASF]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
...this.#asfOptions,
data: JSON.stringify({
Command: `!JOINGROUP ${this.#botName} ${groupName}`
})
});
if (result !== 'Success') {
debug('加入Steam组请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 200 && [ '已加入', '已申请', 'Joined', 'Applied', 'Присоединился', 'costs' ].find((text => data.response?.Result?.includes(text)))) {
debug('成功加入Steam组', {
groupName: groupName
});
logStatus.success();
return true;
}
debug('加入Steam组失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
return false;
} catch (error) {
debug('加入Steam组时发生错误', {
error: error,
groupName: groupName
});
throwError(error, 'SteamASF.joinGroup');
return false;
}
}
joinOfficialGroup=this.joinGroup;
leaveOfficialGroup=this.leaveGroup;
async leaveGroup(groupName) {
try {
debug('开始退出Steam组', {
groupName: groupName
});
if (!this.#groupInfo) {
debug('未找到组信息,尝试获取组ID');
if (!await this.#getGroupId()) {
return false;
}
}
const groupId = await this.#groupInfo[groupName];
if (!groupId) {
debug('未找到组ID', {
groupName: groupName
});
return false;
}
const logStatus = echoLog({
type: 'leavingSteamGroup',
text: groupName,
before: '[ASF]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
...this.#asfOptions,
data: JSON.stringify({
Command: `!LEAVEGROUP ${this.#botName} ${groupId}`
})
});
if (result !== 'Success') {
debug('退出Steam组请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 200 && [ '成功', 'Success', 'Успех' ].find((text => data.response?.Result?.includes(text)))) {
debug('成功退出Steam组', {
groupName: groupName
});
logStatus.success();
return true;
}
debug('退出Steam组失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
return false;
} catch (error) {
debug('退出Steam组时发生错误', {
error: error,
groupName: groupName
});
throwError(error, 'SteamASF.leaveGroup');
return false;
}
}
async #getGroupId() {
try {
debug('开始获取Steam组ID列表');
const logStatus = echoLog({
type: 'gettingSteamGroupId',
text: 'All',
before: '[ASF]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
...this.#asfOptions,
data: JSON.stringify({
Command: `!GROUPLIST ${this.#botName}`
})
});
if (result !== 'Success') {
debug('获取Steam组ID列表请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 200 && data.response?.Result?.includes('|')) {
this.#groupInfo = Object.fromEntries(data.response.Result.split('\n').map((line => {
const [, name, id] = line.trim().split('|');
if (name && id) {
return [ name, id ];
}
return null;
})).filter((ele => ele)));
debug('成功获取Steam组ID列表', {
groupCount: Object.keys(this.#groupInfo).length
});
logStatus.success();
return true;
}
debug('获取Steam组ID列表失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
return false;
} catch (error) {
debug('获取Steam组ID列表时发生错误', {
error: error
});
throwError(error, 'SteamASF.getGroupID');
return false;
}
}
async addToWishlist(gameId) {
try {
debug('开始添加游戏到愿望单', {
gameId: gameId
});
const logStatus = echoLog({
type: 'addingToWishlist',
text: gameId,
before: '[ASF]'
});
const gameStatus = await this.#checkGame(gameId);
if (gameStatus.wishlist === true) {
debug('游戏已在愿望单中', {
gameId: gameId
});
logStatus.success();
return true;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
...this.#asfOptions,
data: JSON.stringify({
Command: `!ADDWISHLIST ${this.#botName} ${gameId}`
})
});
if (result !== 'Success') {
debug('添加游戏到愿望单请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 200 && [ '成功', 'Success', 'Успех' ].find((text => data.response?.Result?.includes(text)))) {
debug('成功添加游戏到愿望单', {
gameId: gameId
});
logStatus.success();
return true;
}
debug('添加游戏到愿望单失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
return false;
} catch (error) {
debug('添加游戏到愿望单时发生错误', {
error: error,
gameId: gameId
});
throwError(error, 'SteamASF.addToWishlist');
return false;
}
}
async removeFromWishlist(gameId) {
try {
debug('开始从愿望单移除游戏', {
gameId: gameId
});
const logStatus = echoLog({
type: 'removingFromWishlist',
text: gameId,
before: '[ASF]'
});
const gameStatus = await this.#checkGame(gameId);
if (gameStatus.wishlist === false) {
debug('游戏已不在愿望单中', {
gameId: gameId
});
logStatus.success();
return true;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
...this.#asfOptions,
data: JSON.stringify({
Command: `!REMOVEWISHLIST ${this.#botName} ${gameId}`
})
});
if (result !== 'Success') {
debug('从愿望单移除游戏请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 200 && [ '成功', 'Success', 'Успех' ].find((text => data.response?.Result?.includes(text)))) {
debug('成功从愿望单移除游戏', {
gameId: gameId
});
logStatus.success();
return true;
}
debug('从愿望单移除游戏失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
return false;
} catch (error) {
debug('从愿望单移除游戏时发生错误', {
error: error,
gameId: gameId
});
throwError(error, 'SteamASF.removeFromWishlist');
return false;
}
}
async toggleFollowGame(gameId, doTask) {
try {
debug('开始处理游戏关注状态', {
gameId: gameId,
doTask: doTask
});
const logStatus = echoLog({
type: `${doTask ? '' : 'un'}followingGame`,
text: gameId,
before: '[ASF]'
});
const gameStatus = await this.#checkGame(gameId);
if (doTask && gameStatus.followed === true || !doTask && gameStatus.followed === false) {
debug('游戏关注状态已符合要求', {
gameId: gameId,
doTask: doTask,
currentStatus: gameStatus.followed
});
logStatus.success();
return true;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
...this.#asfOptions,
data: JSON.stringify({
Command: `!${doTask ? '' : 'UN'}FOLLOWGAME ${this.#botName} ${gameId}`
})
});
if (result !== 'Success') {
debug('处理游戏关注状态请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 200 && [ '成功', 'Success', 'Успех' ].find((text => data.response?.Result?.includes(text)))) {
debug('成功处理游戏关注状态', {
gameId: gameId,
doTask: doTask
});
logStatus.success();
return true;
}
debug('处理游戏关注状态失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
return false;
} catch (error) {
debug('处理游戏关注状态时发生错误', {
error: error,
gameId: gameId,
doTask: doTask
});
throwError(error, 'SteamASF.toggleFollowGame');
return false;
}
}
async #checkGame(gameId) {
try {
debug('开始检查游戏状态', {
gameId: gameId
});
const {result: result, data: data} = await httpRequest({
...this.#asfOptions,
data: JSON.stringify({
Command: `!CHECK ${this.#botName} ${gameId}`
})
});
if (result !== 'Success') {
debug('检查游戏状态请求失败', {
result: result
});
return {};
}
if (data?.status !== 200 || !data.response?.Result?.includes(gameId)) {
debug('检查游戏状态响应无效', {
status: data?.status
});
return {};
}
const matchedResult = data.response.Result.split('\n').find((result => result.includes(gameId)))?.split('|');
if (!matchedResult || matchedResult.length <= 3) {
debug('未找到游戏状态信息', {
gameId: gameId
});
return {};
}
const status = {
wishlist: matchedResult.at(-3).trim() === '√' || matchedResult.at(-2).trim() === '√',
followed: matchedResult.at(-1).trim() === '√'
};
debug('成功获取游戏状态', {
gameId: gameId,
status: status
});
return status;
} catch (error) {
debug('检查游戏状态时发生错误', {
error: error,
gameId: gameId
});
throwError(error, 'SteamASF.checkGame');
return {};
}
}
async toggleCurator(curatorId, doTask = true) {
try {
debug('开始处理鉴赏家关注状态', {
curatorId: curatorId,
doTask: doTask
});
const logStatus = echoLog({
type: doTask ? 'followingCurator' : 'unfollowingCurator',
text: curatorId,
before: '[ASF]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
...this.#asfOptions,
data: JSON.stringify({
Command: `!${doTask ? '' : 'UN'}FOLLOWCURATOR ${this.#botName} ${curatorId}`
})
});
if (result !== 'Success') {
debug('处理鉴赏家关注状态请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 200 && [ '成功', 'Success', 'Успех' ].find((text => data.response?.Result?.includes(text)))) {
debug('成功处理鉴赏家关注状态', {
curatorId: curatorId,
doTask: doTask
});
logStatus.success();
return true;
}
if (data?.status === 200) {
debug('处理鉴赏家关注状态失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(I18n('curatorLimitNotice'));
return false;
}
debug('处理鉴赏家关注状态失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
return false;
} catch (error) {
debug('处理鉴赏家关注状态时发生错误', {
error: error,
curatorId: curatorId,
doTask: doTask
});
throwError(error, 'Steam.toggleCurator');
return false;
}
}
async addLicense(id) {
try {
debug('开始添加许可证', {
id: id
});
const [type, ids] = id.split('-');
const idsArr = ids.split(',');
if (type === 'appid') {
const logStatus = echoLog({
type: 'addingFreeLicense',
text: ids,
before: '[ASF]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
...this.#asfOptions,
data: JSON.stringify({
Command: `!addlicense ${this.#botName} ${idsArr.map((id => `app/${id}`)).join(',')}`
})
});
if (result !== 'Success') {
debug('添加应用许可证请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 200 && [ 'AlreadyPurchased', 'OK' ].find((text => data.response?.Result?.includes(text)))) {
debug('成功添加应用许可证', {
ids: ids
});
logStatus.success();
return true;
}
debug('添加应用许可证失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
return false;
}
if (type === 'subid') {
const logStatus = echoLog({
type: 'addingFreeLicenseSubid',
text: ids,
before: '[ASF]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
...this.#asfOptions,
data: JSON.stringify({
Command: `!addlicense ${this.#botName} ${idsArr.map((id => `sub/${id}`)).join(',')}`
})
});
if (result !== 'Success') {
debug('添加订阅许可证请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 200 && data.response?.Result) {
const resultLines = data.response.Result.split('\n');
debug('处理订阅许可证结果', {
resultLines: resultLines
});
idsArr.forEach((subid => {
const targetLine = resultLines.find((text => text.includes(subid)));
if (targetLine && [ '成功', 'Success', 'Успех' ].find((text => targetLine.includes(text)))) {
debug('成功添加订阅许可证', {
subid: subid
});
echoLog({
before: '[ASF]'
}).success(targetLine);
} else {
debug('添加订阅许可证失败', {
subid: subid,
targetLine: targetLine
});
echoLog({
before: '[ASF]'
}).error(targetLine);
}
}));
return true;
}
debug('添加订阅许可证失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
return false;
}
debug('无效的许可证类型', {
type: type
});
return false;
} catch (error) {
debug('添加许可证时发生错误', {
error: error,
id: id
});
throwError(error, 'SteamASF.addLicense');
return false;
}
}
async requestPlayTestAccess(id) {
try {
debug('开始请求游戏试玩权限', {
id: id
});
const logStatus = echoLog({
type: 'requestingPlayTestAccess',
text: id,
before: '[ASF]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
...this.#asfOptions,
data: JSON.stringify({
Command: `!REQUESTACCESS ${this.#botName} ${id}`
})
});
if (result !== 'Success') {
debug('请求游戏试玩权限请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 200 && [ '成功', 'Success', 'Успех' ].find((text => data.response?.Result?.includes(text)))) {
debug('成功请求游戏试玩权限', {
id: id
});
logStatus.success();
return true;
}
debug('请求游戏试玩权限失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
return false;
} catch (error) {
debug('请求游戏试玩权限时发生错误', {
error: error,
id: id
});
throwError(error, 'SteamASF.requestPlayTestAccess');
return false;
}
}
async playGames(ids) {
try {
debug('开始挂游戏时长', {
ids: ids
});
const logStatus = echoLog({
text: I18n('playingGames', ids),
before: '[ASF]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
...this.#asfOptions,
data: JSON.stringify({
Command: `!play ${this.#botName} ${ids}`
})
});
if (result !== 'Success') {
debug('挂游戏时长请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 200 && [ '正在运行', '正在掛', 'Playing', 'Играет' ].find((text => data.response?.Result?.includes(text)))) {
debug('成功开始挂游戏时长', {
ids: ids
});
logStatus.success();
return true;
}
debug('开始挂游戏时长失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
return false;
} catch (error) {
debug('挂游戏时长时发生错误', {
error: error,
ids: ids
});
throwError(error, 'SteamASF.playGames');
return false;
}
}
async getSteamIdASF() {
try {
debug('开始获取Steam ID');
const logStatus = echoLog({
text: I18n('gettingSteamId'),
before: '[ASF]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
...this.#asfOptions,
data: JSON.stringify({
Command: `!steamid ${this.#botName}`
})
});
if (result !== 'Success' || data?.status !== 200) {
debug('获取Steam ID请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return '';
}
if (data.response?.Result) {
const steamId = data.response.Result.trim()?.split(/\s+/)?.at(-1);
if (steamId) {
debug('成功获取Steam ID', steamId);
logStatus.success();
return steamId;
}
}
debug('获取Steam ID失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
return '';
} catch (error) {
debug('获取Steam ID时发生错误', {
error: error
});
throwError(error, 'SteamASF.getSteamIdASF');
return '';
}
}
async getSteamIdWeb() {
try {
debug('开始获取Steam ID');
const logStatus = echoLog({
text: I18n('gettingSteamId'),
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://store.steampowered.com',
method: 'GET',
headers: {
host: 'store.steampowered.com'
}
});
if (result !== 'Success' || data?.status !== 200) {
debug('获取Steam ID请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return '';
}
const steamId = data.responseText.match(/steamid":"(\d+)/)?.[1];
if (steamId) {
debug('成功获取Steam ID', steamId);
logStatus.success();
return steamId;
}
debug('获取Steam ID失败', {
data: data
});
logStatus.error(`${result}:${statusText}(${status})`);
return '';
} catch (error) {
debug('获取Steam ID时发生错误', {
error: error
});
throwError(error, 'SteamASF.getSteamIdWeb');
return '';
}
}
async getSteamId() {
const steamId = await this.getSteamIdWeb();
if (steamId) {
return steamId;
}
return this.getSteamIdASF();
}
async checkPlayStatus(ids) {
try {
debug('开始检查挂游戏时长状态');
if (!this.#steamWebApiKey) {
debug('未设置Steam Web API Key');
return 'skip';
}
if (!this.#steamId) {
const steamId = await this.getSteamId();
if (!steamId) {
debug('未获取到Steam ID');
return 'skip';
}
this.#steamId = steamId;
}
const logStatus = echoLog({
text: I18n('checkingPlayStatus'),
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://api.steampowered.com/ISteamUser/GetPlayerSummaries/v0002/?key=${this.#steamWebApiKey}&steamids=${this.#steamId}`,
method: 'GET',
headers: {
'Content-Type': 'application/json'
}
});
if (result !== 'Success') {
debug('检查挂游戏时长状态请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 200) {
debug('挂游戏时长状态正常', {
data: data
});
const playedIds = new Set(data.responseText?.match(/\d+/g));
const neededIds = new Set(ids.match(/\d+/g));
if (neededIds.intersection(playedIds).size > 0) {
logStatus.success();
return true;
}
logStatus.warning(I18n('noPlayStatus'));
return false;
}
debug('挂游戏时长状态异常', {
data: data
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
} catch (error) {
debug('检查挂游戏时长状态时发生错误', {
error: error
});
throwError(error, 'SteamASF.checkPlayStatus');
return false;
}
}
async stopPlayGames() {
try {
debug('开始停止挂游戏时长');
const logStatus = echoLog({
text: I18n('stoppingPlayGames'),
before: '[ASF]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
...this.#asfOptions,
data: JSON.stringify({
Command: `!resume ${this.#botName}`
})
});
if (result !== 'Success') {
debug('停止挂游戏时长请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status === 200 && [ '已经恢复', '已恢复', '已經繼續', '已繼續', 'resumed', 'возобновлён' ].find((text => data.response?.Result?.includes(text)))) {
debug('成功停止挂游戏时长');
logStatus.success();
return true;
}
debug('停止挂游戏时长失败', {
result: data?.response?.Result,
message: data?.response?.Message
});
logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
return false;
} catch (error) {
debug('停止挂游戏时长时发生错误', {
error: error
});
throwError(error, 'SteamASF.stopPlayGames');
return false;
}
}
async #unsupportted(name) {
try {
debug('尝试使用不支持的功能', {
name: name
});
echoLog({
before: '[ASF]'
}).warning(I18n('ASFNotSupportted', name));
return false;
} catch (error) {
debug('处理不支持的功能时发生错误', {
error: error,
name: name
});
throwError(error, 'SteamASF.unsupportted');
return false;
}
}
async toggleForum() {
return this.#unsupportted('toggleForum');
}
async toggleFavoriteWorkshop() {
return this.#unsupportted('toggleFavoriteWorkshop');
}
async voteUpWorkshop() {
return this.#unsupportted('voteUpWorkshop');
}
async likeAnnouncement() {
return this.#unsupportted('likeAnnouncement');
}
}
class SteamWeb {
#cache={
...{
group: {},
officialGroup: {},
forum: {},
workshop: {},
curator: {}
},
...GM_getValue('steamCache')
};
#auth={};
#storeInitialized=false;
#communityInitialized=false;
#area='CN';
#oldArea;
#areaStatus='end';
constructor() {
debug('初始化SteamWeb实例');
}
async init(type = 'all') {
try {
debug('开始初始化SteamWeb', {
type: type
});
const initStoreResult = await this.initStore();
debug('Steam商店初始化完成', {
initStoreResult: initStoreResult
});
if (type === 'store') {
return initStoreResult;
}
const initCommunityResult = await this.initCommunity(initStoreResult);
debug('Steam社区初始化完成', {
initCommunityResult: initCommunityResult
});
return initCommunityResult;
} catch (error) {
debug('SteamWeb初始化发生错误', {
error: error,
type: type
});
throwError(error, 'SteamWeb.init');
return false;
}
}
async initStore() {
try {
debug('开始初始化Steam商店');
if (this.#storeInitialized) {
return true;
}
let storeInitialized = await this.#updateStoreAuth();
if (!storeInitialized) {
storeInitialized = await this.#updateStoreAuthTab();
}
this.#storeInitialized = storeInitialized;
if (!this.#storeInitialized) {
echoLog({
before: '[Web]'
}).error(I18n('initFailed', 'Steam'));
return false;
}
echoLog({
before: '[Web]'
}).success(I18n('initSuccess', 'SteamStore'));
debug('Steam商店初始化完成');
return true;
} catch (error) {
debug('Steam商店初始化发生错误', {
error: error
});
throwError(error, 'SteamWeb.initStore');
return false;
}
}
async initCommunity(initStoreResult) {
try {
debug('开始初始化Steam社区');
if (this.#communityInitialized) {
return true;
}
let communityInitialized = await this.#updateCommunityAuth(initStoreResult);
if (!communityInitialized) {
communityInitialized = await this.#updateCommunityAuthTab();
GM_setValue('steamCommunityAuth', null);
}
this.#communityInitialized = communityInitialized;
if (!this.#communityInitialized) {
echoLog({
before: '[Web]'
}).error(I18n('initFailed', 'Steam'));
return false;
}
echoLog({
before: '[Web]'
}).success(I18n('initSuccess', 'SteamCommunity'));
debug('Steam社区初始化完成');
return true;
} catch (error) {
debug('Steam社区初始化发生错误', {
error: error
});
throwError(error, 'SteamWeb.initCommunity');
return false;
}
}
async #refreshToken(type = 'steamStore') {
try {
debug('开始刷新令牌', {
type: type
});
const host = {
steamStore: 'store.steampowered.com',
steamCommunity: 'steamcommunity.com'
};
const logStatus = echoLog({
text: I18n('refreshingToken', I18n(type)),
before: '[Web]'
});
debug('准备刷新令牌请求数据');
const formData = new FormData;
formData.append('redir', `https://${host[type]}/`);
debug('发送刷新令牌请求');
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://login.steampowered.com/jwt/ajaxrefresh',
method: 'POST',
responseType: 'json',
headers: {
Host: 'login.steampowered.com',
Origin: `https://${host[type]}`,
Referer: `https://${host[type]}/`
},
data: formData
});
debug('收到刷新令牌响应', {
result: result,
status: status,
statusText: statusText
});
if (result !== 'Success') {
debug('请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (!data?.response?.success) {
debug('响应不成功', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('开始设置新令牌');
if (!await this.#setToken(data.response, type)) {
debug('设置新令牌失败');
logStatus.error('Error');
return false;
}
debug('成功刷新令牌');
logStatus.success();
return true;
} catch (error) {
debug('刷新令牌时发生错误', {
error: error
});
throwError(error, 'SteamWeb.refreshToken');
return false;
}
}
async #setToken(param, type) {
try {
const host = {
steamStore: 'store.steampowered.com',
steamCommunity: 'steamcommunity.com'
};
debug('开始设置Steam令牌', {
type: type
});
const logStatus = echoLog({
text: I18n('settingToken', I18n(type)),
before: '[Web]'
});
debug('准备表单数据');
const formData = new FormData;
formData.append('steamID', param.steamID);
formData.append('nonce', param.nonce);
formData.append('redir', param.redir);
formData.append('auth', param.auth);
debug('表单数据准备完成', {
steamID: param.steamID,
nonce: param.nonce,
redir: param.redir
});
debug('发送设置令牌请求');
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://${host[type]}/login/settoken`,
method: 'POST',
headers: {
Accept: 'application/json, text/plain, */*',
Host: host[type],
Origin: `https://${host[type]}`
},
data: formData
});
debug('收到设置令牌响应', {
result: result,
status: status,
statusText: statusText
});
if (result !== 'Success') {
debug('请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('响应状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('成功设置令牌');
logStatus.success();
return true;
} catch (error) {
debug('设置令牌时发生错误', {
error: error,
type: type
});
throwError(error, 'SteamWeb.setToken');
return false;
}
}
async #updateStoreAuth(retry = false) {
try {
debug('开始更新Steam商店身份验证');
const logStatus = echoLog({
text: I18n('updatingAuth', I18n('steamStore')),
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://store.steampowered.com/',
method: 'GET',
headers: {
Accept: 'text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9',
'Sec-Fetch-Dest': 'document',
'Sec-Fetch-Mode': 'navigate',
'Upgrade-Insecure-Requests': '1'
},
redirect: 'manual'
});
debug('收到Steam商店身份验证响应', {
result: result,
statusText: statusText,
status: status
});
if (data?.status !== 200) {
if (![ 301, 302 ].includes(data?.status)) {
debug('Steam商店身份验证状态错误', {
status: data?.status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (!await this.#refreshToken('steamStore')) {
debug('Steam商店身份验证刷新失败');
logStatus.error(`Error:${I18n('needLoginSteamStore')}`, true);
return false;
}
if (retry) {
debug('Steam商店身份验证重试失败');
logStatus.error(`Error:${I18n('needLoginSteamStore')}`, true);
return false;
}
debug('Steam商店身份验证重试中');
logStatus.warning(I18n('retry'));
return this.#updateStoreAuth(true);
}
if (!data.responseText.includes('data-miniprofile=')) {
if (await this.#refreshToken('steamStore')) {
debug('Steam商店身份验证需要重试');
logStatus.warning(I18n('retry'));
if (retry) {
debug('Steam商店身份验证重试次数超限');
logStatus.error(`Error:${I18n('needLoginSteamStore')}`, true);
return false;
}
return this.#updateStoreAuth(true);
}
debug('Steam商店身份验证失败:需要登录');
logStatus.error(`Error:${I18n('needLoginSteamStore')}`, true);
return false;
}
const storeSessionID = data.responseText.match(/g_sessionID = "(.+?)";/)?.[1];
if (!storeSessionID) {
debug('Steam商店身份验证失败:获取sessionID失败');
logStatus.error('Error: Get "sessionID" failed');
return false;
}
this.#auth.storeSessionID = storeSessionID;
debug('Steam商店身份验证更新成功', {
storeSessionID: storeSessionID
});
logStatus.success();
return true;
} catch (error) {
debug('更新Steam商店身份验证时发生错误', {
error: error
});
throwError(error, 'SteamWeb.updateStoreAuth');
return false;
}
}
async #updateStoreAuthTab() {
try {
debug('开始通过新标签页更新Steam商店身份验证');
const logStatus = echoLog({
text: I18n('updatingAuth', I18n('steamStoreTab')),
before: '[Web]'
});
return await new Promise((resolve => {
GM_deleteValue('steamStoreAuth');
GM_setValue('ATv4_updateStoreAuth', true);
const newTab = GM_openInTab('https://store.steampowered.com/', {
active: true,
setParent: true
});
debug('打开Steam商店新标签页');
newTab.name = 'ATv4_updateStoreAuth';
const listenerId = GM_addValueChangeListener('steamStoreAuth', ((key, oldValue, newValue) => {
debug('监听到Steam商店身份验证值变化', {
oldValue: oldValue,
newValue: newValue
});
GM_removeValueChangeListener(listenerId);
GM_deleteValue('ATv4_updateStoreAuth');
newTab?.close();
window.focus();
if (newValue && JSON.stringify(newValue) !== JSON.stringify(oldValue)) {
this.#auth.storeSessionID = newValue.storeSessionID;
debug('Steam商店身份验证更新成功', {
storeSessionID: newValue.storeSessionID
});
logStatus.success();
resolve(true);
return;
}
debug('Steam商店身份验证更新失败');
logStatus.error('Failed');
resolve(false);
}));
newTab.onclose = () => {
debug('Steam商店新标签页已关闭');
GM_deleteValue('ATv4_updateStoreAuth');
};
}));
} catch (error) {
debug('通过新标签页更新Steam商店身份验证时发生错误', {
error: error
});
throwError(error, 'SteamWeb.updateStoreAuthTab');
return false;
}
}
async #updateCommunityAuthTab() {
try {
debug('开始通过新标签页更新Steam社区身份验证');
const logStatus = echoLog({
text: I18n('updatingAuth', I18n('steamCommunityTab')),
before: '[Web]'
});
return await new Promise((resolve => {
GM_deleteValue('steamCommunityAuth');
GM_setValue('ATv4_updateCommunityAuth', true);
const newTab = GM_openInTab('https://steamcommunity.com/my', {
active: true,
setParent: true
});
debug('打开Steam社区新标签页');
newTab.name = 'ATv4_updateCommunityAuth';
const listenerId = GM_addValueChangeListener('steamCommunityAuth', ((key, oldValue, newValue) => {
debug('监听到Steam社区身份验证值变化', {
oldValue: oldValue,
newValue: newValue
});
GM_removeValueChangeListener(listenerId);
GM_deleteValue('ATv4_updateCommunityAuth');
newTab?.close();
window.focus();
if (newValue && JSON.stringify(newValue) !== JSON.stringify(oldValue)) {
this.#auth.steam64Id = newValue.steam64Id;
this.#auth.communitySessionID = newValue.communitySessionID;
debug('Steam社区身份验证更新成功', {
steam64Id: newValue.steam64Id,
communitySessionID: newValue.communitySessionID
});
logStatus.success();
resolve(true);
return;
}
debug('Steam社区身份验证更新失败');
logStatus.error('Failed');
resolve(false);
}));
newTab.onclose = () => {
debug('Steam社区新标签页已关闭');
GM_deleteValue('ATv4_updateCommunityAuth');
};
}));
} catch (error) {
debug('通过新标签页更新Steam社区身份验证时发生错误', {
error: error
});
throwError(error, 'SteamWeb.updateCommunityAuthTab');
return false;
}
}
async #updateCommunityAuth(initStoreResult, retry = false) {
try {
debug('开始更新Steam社区身份验证');
const logStatus = echoLog({
text: I18n('gettingUserInfo'),
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://steamcommunity.com/my',
method: 'GET',
headers: {
Accept: 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7',
Host: 'steamcommunity.com',
'Sec-Fetch-Dest': 'document',
'Sec-Fetch-Mode': 'navigate'
},
redirect: 'follow'
});
debug('收到Steam社区身份验证响应', {
result: result,
statusText: statusText,
status: status
});
if (data?.status !== 200) {
debug('Steam社区身份验证状态错误', {
status: data?.status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data.finalUrl.includes('https://steamcommunity.com/login/home')) {
if (initStoreResult) {
if (await this.#refreshToken('steamCommunity')) {
debug('Steam社区身份验证需要重试');
logStatus.warning(I18n('retry'));
if (retry) {
debug('Steam社区身份验证重试次数超限');
logStatus.error(`Error:${I18n('needLoginSteamCommunity')}`, true);
return false;
}
return this.#updateCommunityAuth(initStoreResult, retry);
}
}
debug('Steam社区身份验证失败:需要登录');
logStatus.error(`Error:${I18n('needLoginSteamCommunity')}`, true);
return false;
}
const steam64Id = data.responseText.match(/g_steamID = "(.+?)";/)?.[1];
const communitySessionID = data.responseText.match(/g_sessionID = "(.+?)";/)?.[1];
if (!steam64Id || !communitySessionID) {
debug('Steam社区身份验证失败:获取身份信息失败');
logStatus.error('Error: Get "sessionID" failed');
return false;
}
this.#auth.steam64Id = steam64Id;
this.#auth.communitySessionID = communitySessionID;
debug('Steam社区身份验证更新成功', {
steam64Id: steam64Id,
communitySessionID: communitySessionID
});
logStatus.success();
return true;
} catch (error) {
debug('更新Steam社区身份验证时发生错误', {
error: error
});
throwError(error, 'SteamWeb.updateCommunityAuth');
return false;
}
}
async #getAreaInfo() {
try {
debug('开始获取Steam地区信息');
const logStatus = echoLog({
text: I18n('gettingAreaInfo'),
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://store.steampowered.com/cart/',
method: 'GET'
});
debug('获取地区信息请求结果', {
result: result,
statusText: statusText,
status: status
});
if (result !== 'Success' || data?.status !== 200) {
debug('获取地区信息失败', {
result: result,
status: data?.status,
statusText: data?.statusText
});
logStatus.error(result === 'Success' ? `Error:${data?.statusText}(${data?.status})` : `${result}:${statusText}(${status})`);
return {};
}
const cartConfigRaw = data.responseText.match(/data-cart_config="(.*?)"/)?.[1];
debug('cartConfigRaw提取结果', {
cartConfigRaw: cartConfigRaw
});
const temp = document.createElement('div');
temp.innerHTML = cartConfigRaw || '{}';
const cartConfigStr = temp.textContent || temp.innerText;
let cartConfig;
try {
cartConfig = JSON.parse(cartConfigStr);
debug('cartConfig解析成功', {
cartConfig: cartConfig
});
} catch (error) {
debug('cartConfig解析失败', {
error: error
});
logStatus.error('Error: get country info filed');
console.error(error);
return {};
}
if (!cartConfig.rgUserCountryOptions) {
debug('未找到可更换地区');
logStatus.warning('Warning: Area cannot be changed');
return {};
}
const userInfoRaw = data.responseText.match(/data-userinfo="(.*?)"/)?.[1];
debug('userInfoRaw提取结果', {
userInfoRaw: userInfoRaw
});
const temp1 = document.createElement('div');
temp1.innerHTML = userInfoRaw || '{}';
const userInfoStr = temp1.textContent || temp1.innerText;
let userInfo;
try {
userInfo = JSON.parse(userInfoStr);
debug('userInfo解析成功', {
userInfo: userInfo
});
} catch (error) {
debug('userInfo解析失败', {
error: error
});
logStatus.error('Error: get country info filed');
console.error(error);
return {};
}
const currentArea = userInfo.country_code;
const areas = Object.keys(cartConfig.rgUserCountryOptions).filter((area => area !== 'help'));
debug('地区信息提取', {
currentArea: currentArea,
areas: areas
});
if (!currentArea || areas.length === 0) {
debug('未获取到当前地区或可更换地区为空', {
currentArea: currentArea,
areas: areas
});
logStatus.error('Error: get country info filed');
return {};
}
this.#area = currentArea;
debug('获取地区信息成功', {
currentArea: currentArea,
areas: areas
});
logStatus.success();
return {
currentArea: currentArea,
areas: areas
};
} catch (error) {
debug('获取地区信息时发生异常', {
error: error
});
throwError(error, 'SteamWeb.getAreaInfo');
return {};
}
}
async #changeArea(area) {
try {
debug('开始更换Steam地区', {
area: area
});
if (this.#areaStatus === 'waiting') {
debug('当前地区状态为waiting,等待状态改变');
await new Promise((resolve => {
const checker = setInterval((() => {
if (this.#areaStatus !== 'waiting') {
clearInterval(checker);
resolve(true);
}
}));
}));
}
if (this.#area === area || !area && this.#area !== 'CN') {
debug('无需更换地区', {
currentArea: this.#area,
targetArea: area
});
return true;
}
this.#areaStatus = 'waiting';
let aimedArea = area;
if (!aimedArea) {
debug('未指定目标地区,自动获取可用地区');
const {currentArea: currentArea, areas: areas} = await this.#getAreaInfo();
debug('获取到地区信息', {
currentArea: currentArea,
areas: areas
});
if (!currentArea || !areas) {
debug('获取地区信息失败', {
currentArea: currentArea,
areas: areas
});
this.#areaStatus = 'error';
return false;
}
if (currentArea !== 'CN') {
debug('当前地区不是CN,无需更换', {
currentArea: currentArea
});
this.#areaStatus = 'skip';
echoLog({
text: I18n('notNeededChangeArea'),
before: '[Web]'
});
return 'skip';
}
const anotherArea = areas.filter((area => area && area !== 'CN'));
debug('可更换的其他地区', {
anotherArea: anotherArea
});
if (!anotherArea || anotherArea.length === 0) {
debug('没有可用的其他地区');
this.#areaStatus = 'noAnotherArea';
echoLog({
text: I18n('noAnotherArea'),
before: '[Web]'
});
return false;
}
[aimedArea] = anotherArea;
debug('选定目标地区', {
aimedArea: aimedArea
});
}
const logStatus = echoLog({
text: I18n('changingArea', aimedArea),
before: '[Web]'
});
debug('发送更换地区请求', {
aimedArea: aimedArea
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://store.steampowered.com/country/setcountry',
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: $.param({
cc: aimedArea,
sessionid: this.#auth.storeSessionID
})
});
debug('更换地区请求结果', {
result: result,
statusText: statusText,
status: status
});
if (result !== 'Success' || data?.status !== 200 || data.responseText !== 'true') {
debug('更换地区失败', {
result: result,
status: data?.status,
statusText: data?.statusText,
responseText: data?.responseText
});
this.#areaStatus = 'error';
logStatus.error(result === 'Success' ? `Error:${data?.statusText}(${data?.status})` : `${result}:${statusText}(${status})`);
return 'CN';
}
const {currentArea: currentArea} = await this.#getAreaInfo();
debug('更换后获取到的当前地区', {
currentArea: currentArea
});
if (currentArea) {
this.#area = currentArea;
if (!this.#oldArea) {
this.#oldArea = currentArea;
}
}
if (currentArea !== aimedArea) {
debug('更换后当前地区与目标地区不符', {
currentArea: currentArea,
aimedArea: aimedArea
});
this.#areaStatus = 'error';
logStatus.error('Error: change country filed');
return 'CN';
}
this.#areaStatus = 'success';
debug('更换地区成功', {
currentArea: currentArea
});
logStatus.success();
return currentArea;
} catch (error) {
debug('更换地区时发生异常', {
error: error
});
this.#areaStatus = 'error';
throwError(error, 'SteamWeb.changeArea');
return false;
}
}
async joinGroup(groupName) {
try {
debug('开始加入Steam组', {
groupName: groupName
});
const logStatus = echoLog({
type: 'joiningSteamGroup',
text: groupName,
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://steamcommunity.com/groups/${groupName}`,
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: $.param({
action: 'join',
sessionID: this.#auth.communitySessionID
})
});
if (result !== 'Success') {
debug('加入Steam组请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200 || data.responseText.includes('grouppage_join_area')) {
debug('加入Steam组失败', {
status: data?.status
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('成功加入Steam组', {
groupName: groupName
});
logStatus.success();
return true;
} catch (error) {
debug('加入Steam组时发生错误', {
error: error,
groupName: groupName
});
throwError(error, 'SteamWeb.joinGroup');
return false;
}
}
async leaveGroup(groupName) {
try {
debug('开始退出Steam组', {
groupName: groupName
});
const groupId = await this.#getGroupId(groupName);
if (!groupId) {
return false;
}
const logStatus = echoLog({
type: 'leavingSteamGroup',
text: groupName,
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://steamcommunity.com/profiles/${this.#auth.steam64Id}/home_process`,
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: $.param({
sessionID: this.#auth.communitySessionID,
action: 'leaveGroup',
groupId: groupId
})
});
if (result !== 'Success') {
debug('退出Steam组请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200 || !data.finalUrl.includes('groups')) {
debug('退出Steam组失败', {
status: data?.status
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const hasGroupLink = $(data.responseText.replace(/<img.*?>/g, '').toLowerCase()).find(`a[href='https://steamcommunity.com/groups/${groupName.toLowerCase()}']`).length > 0;
if (hasGroupLink) {
debug('Error: Group link still exists');
return false;
}
debug('成功退出Steam组', {
groupName: groupName
});
logStatus.success();
return true;
} catch (error) {
debug('退出Steam组时发生错误', {
error: error,
groupName: groupName
});
throwError(error, 'SteamWeb.leaveGroup');
return false;
}
}
async #getGroupId(groupName) {
try {
debug('开始获取Steam组ID', {
groupName: groupName
});
const logStatus = echoLog({
type: 'gettingSteamGroupId',
text: groupName,
before: '[Web]'
});
const cachedGroupId = this.#cache.group[groupName];
if (cachedGroupId) {
debug('从缓存中获取到组ID', {
groupName: groupName,
cachedGroupId: cachedGroupId
});
logStatus.success();
return cachedGroupId;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://steamcommunity.com/groups/${groupName}`,
method: 'GET',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
}
});
debug('获取组ID请求结果', {
result: result,
statusText: statusText,
status: status
});
if (result !== 'Success') {
debug('获取组ID请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('获取组ID响应状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const matchedGroupId = data.responseText.match(/OpenGroupChat\( '([0-9]+)'/)?.[1];
debug('正则提取组ID结果', {
matchedGroupId: matchedGroupId
});
if (!matchedGroupId) {
debug('未能提取到组ID', {
groupName: groupName
});
logStatus.error(`Error:${data.statusText}(${data.status})`);
return false;
}
this.#setCache('group', groupName, matchedGroupId);
debug('组ID已缓存', {
groupName: groupName,
matchedGroupId: matchedGroupId
});
logStatus.success();
return matchedGroupId;
} catch (error) {
debug('获取组ID时发生异常', {
error: error,
groupName: groupName
});
throwError(error, 'SteamWeb.getGroupID');
return false;
}
}
async joinOfficialGroup(gameId) {
try {
debug('开始加入Steam官方组', {
gameId: gameId
});
const logStatus = echoLog({
type: 'joiningSteamOfficialGroup',
text: gameId,
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://steamcommunity.com/games/${gameId}?action=join&sessionID=${this.#auth.communitySessionID}`,
method: 'GET',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
}
});
if (result !== 'Success') {
debug('加入Steam官方组请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200 || data.responseText.includes('id="publicGroupJoin"')) {
debug('加入Steam官方组失败', {
status: data?.status
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const groupId = data.responseText.match(/steam:\/\/friends\/joinchat\/([0-9]+)/)?.[1];
if (groupId) {
this.#setCache('officialGroup', gameId, groupId);
}
debug('成功加入Steam官方组', {
gameId: gameId
});
logStatus.success();
return true;
} catch (error) {
debug('加入Steam官方组时发生错误', {
error: error,
gameId: gameId
});
throwError(error, 'SteamWeb.joinOfficialGroup');
return false;
}
}
async leaveOfficialGroup(gameId) {
try {
debug('开始退出Steam官方组', {
gameId: gameId
});
const groupId = await this.#getOfficialGroupId(gameId);
if (!groupId) {
return false;
}
const logStatus = echoLog({
type: 'leavingSteamOfficialGroup',
text: gameId,
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://steamcommunity.com/profiles/${this.#auth.steam64Id}/home_process`,
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: $.param({
sessionID: this.#auth.communitySessionID,
action: 'leaveGroup',
groupId: groupId
})
});
if (result !== 'Success') {
debug('退出Steam官方组请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('退出Steam官方组失败', {
status: data?.status
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const {result: resultR, statusText: statusTextR, status: statusR, data: dataR} = await httpRequest({
url: `https://steamcommunity.com/games/${gameId}`,
method: 'GET',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
}
});
if (resultR !== 'Success') {
debug('退出Steam官方组时发生错误', {
error: resultR,
status: statusR,
statusText: statusTextR
});
logStatus.error(`${resultR}:${statusTextR}(${statusR})`);
return false;
}
if (dataR?.status !== 200 || !dataR.responseText.includes('id="publicGroupJoin"')) {
debug('退出Steam官方组失败', {
status: dataR?.status
});
logStatus.error(`Error:${dataR?.statusText}(${dataR?.status})`);
return false;
}
debug('成功退出Steam官方组', {
gameId: gameId
});
logStatus.success();
return true;
} catch (error) {
debug('退出Steam官方组时发生错误', {
error: error,
gameId: gameId
});
throwError(error, 'SteamWeb.leaveOfficialGroup');
return false;
}
}
async #getOfficialGroupId(gameId) {
try {
debug('开始获取Steam官方群组ID', {
gameId: gameId
});
const logStatus = echoLog({
type: 'gettingSteamOfficialGroupId',
text: gameId,
before: '[Web]'
});
const cachedGroupId = this.#cache.officialGroup[gameId];
if (cachedGroupId) {
debug('从缓存中获取到官方群组ID', {
gameId: gameId,
cachedGroupId: cachedGroupId
});
logStatus.success();
return cachedGroupId;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://steamcommunity.com/games/${gameId}`,
method: 'GET',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
}
});
debug('获取官方群组ID请求结果', {
result: result,
statusText: statusText,
status: status
});
if (result !== 'Success') {
debug('获取官方群组ID请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('获取官方群组ID响应状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const matchedGroupId = data.responseText.match(/steam:\/\/friends\/joinchat\/([0-9]+)/)?.[1];
debug('正则提取官方群组ID结果', {
matchedGroupId: matchedGroupId
});
if (!matchedGroupId) {
debug('未能提取到官方群组ID', {
gameId: gameId
});
logStatus.error(`Error:${data.statusText}(${data.status})`);
return false;
}
this.#setCache('officialGroup', gameId, matchedGroupId);
debug('官方群组ID已缓存', {
gameId: gameId,
matchedGroupId: matchedGroupId
});
logStatus.success();
return matchedGroupId;
} catch (error) {
debug('获取官方群组ID时发生异常', {
error: error,
gameId: gameId
});
throwError(error, 'SteamWeb.getGroupID');
return false;
}
}
async addToWishlist(gameId) {
try {
debug('开始添加游戏到愿望单', {
gameId: gameId
});
const logStatus = echoLog({
type: 'addingToWishlist',
text: gameId,
before: '[Web]'
});
const {result: result, data: data} = await httpRequest({
url: 'https://store.steampowered.com/api/addtowishlist',
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: $.param({
sessionid: this.#auth.storeSessionID,
appid: gameId
}),
dataType: 'json'
});
if (result === 'Success' && data?.status === 200 && data.response?.success === true) {
debug('成功添加游戏到愿望单', {
gameId: gameId
});
logStatus.success();
return true;
}
const {result: resultR, statusText: statusTextR, status: statusR, data: dataR} = await httpRequest({
url: `https://store.steampowered.com/app/${gameId}`,
method: 'GET'
});
if (resultR !== 'Success') {
debug('添加游戏到愿望单请求失败', {
result: resultR,
status: statusR,
statusText: statusTextR
});
logStatus.error(`${resultR}:${statusTextR}(${statusR})`);
return false;
}
if (dataR?.status !== 200) {
debug('添加游戏到愿望单失败', {
status: dataR?.status
});
logStatus.error(`Error:${dataR?.statusText}(${dataR?.status})`);
return false;
}
if (this.#area === 'CN' && dataR.responseText.includes('id="error_box"')) {
debug('changeAreaNotice');
if (!await this.#changeArea()) {
return false;
}
return await this.addToWishlist(gameId);
}
if (dataR.responseText.includes('class="queue_actions_ctn"') && dataR.responseText.includes('class="already_in_library"')) {
debug('成功添加游戏到愿望单', {
gameId: gameId
});
logStatus.success();
return true;
}
if (dataR.responseText.includes('class="queue_actions_ctn"') && dataR.responseText.includes('id="add_to_wishlist_area_success" style="display: none;') || !dataR.responseText.includes('class="queue_actions_ctn"')) {
debug('添加游戏到愿望单失败', {
status: dataR.statusText
});
logStatus.error(`Error:${dataR.statusText}(${dataR.status})`);
return false;
}
debug('成功添加游戏到愿望单', {
gameId: gameId
});
logStatus.success();
return true;
} catch (error) {
debug('添加游戏到愿望单时发生错误', {
error: error,
gameId: gameId
});
throwError(error, 'SteamWeb.addToWishlist');
return false;
}
}
async removeFromWishlist(gameId) {
try {
debug('开始从愿望单移除游戏', {
gameId: gameId
});
const logStatus = echoLog({
type: 'removingFromWishlist',
text: gameId,
before: '[Web]'
});
const {result: result, data: data} = await httpRequest({
url: 'https://store.steampowered.com/api/removefromwishlist',
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: $.param({
sessionid: this.#auth.storeSessionID,
appid: gameId
}),
dataType: 'json'
});
if (result === 'Success' && data?.status === 200 && data.response?.success === true) {
debug('成功从愿望单移除游戏', {
gameId: gameId
});
logStatus.success();
return true;
}
const {result: resultR, statusText: statusTextR, status: statusR, data: dataR} = await httpRequest({
url: `https://store.steampowered.com/app/${gameId}`,
method: 'GET'
});
if (resultR !== 'Success') {
debug('从愿望单移除游戏请求失败', {
result: resultR,
status: statusR,
statusText: statusTextR
});
logStatus.error(`${resultR}:${statusTextR}(${statusR})`);
return false;
}
if (dataR?.status !== 200) {
debug('从愿望单移除游戏失败', {
status: dataR?.status
});
logStatus.error(`Error:${dataR?.statusText}(${dataR?.status})`);
return false;
}
if (this.#area === 'CN' && dataR.responseText.includes('id="error_box"')) {
debug('changeAreaNotice');
const result = await this.#changeArea();
if (!result || result === 'CN' || result === 'skip') {
return false;
}
return await this.removeFromWishlist(gameId);
}
if (dataR.responseText.includes('class="queue_actions_ctn"') && (dataR.responseText.includes('ds_owned_flag ds_flag') || dataR.responseText.includes('add_to_wishlist_area'))) {
debug('成功从愿望单移除游戏', {
gameId: gameId
});
logStatus.success();
return true;
}
debug('从愿望单移除游戏请求失败', {
result: resultR,
status: statusR,
statusText: statusTextR
});
logStatus.error(`Error:${dataR.statusText}(${dataR.status})`);
return false;
} catch (error) {
debug('从愿望单移除游戏时发生错误', {
error: error,
gameId: gameId
});
throwError(error, 'SteamWeb.removeFromWishlist');
return false;
}
}
async toggleFollowGame(gameId, doTask) {
try {
debug('开始处理游戏关注状态', {
gameId: gameId,
doTask: doTask
});
const logStatus = echoLog({
type: `${doTask ? '' : 'un'}followingGame`,
text: gameId,
before: '[Web]'
});
const requestData = {
sessionid: this.#auth.storeSessionID,
appid: gameId
};
if (!doTask) {
requestData.unfollow = '1';
}
const {result: result, data: data} = await httpRequest({
url: 'https://store.steampowered.com/explore/followgame/',
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: $.param(requestData)
});
if (result === 'Success' && data?.status === 200 && data.responseText === 'true') {
debug('成功处理游戏关注状态', {
gameId: gameId,
doTask: doTask
});
logStatus.success();
return true;
}
const followed = await this.#isFollowedGame(gameId);
if (this.#area === 'CN' && followed === 'areaLocked') {
debug('changeAreaNotice');
if (!await this.#changeArea()) {
return false;
}
return await this.toggleFollowGame(gameId, doTask);
}
if (doTask === followed) {
debug('成功处理游戏关注状态', {
gameId: gameId,
doTask: doTask
});
logStatus.success();
return true;
}
debug('处理游戏关注状态请求失败', {
result: result
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
} catch (error) {
debug('处理游戏关注状态时发生错误', {
error: error,
gameId: gameId,
doTask: doTask
});
throwError(error, 'SteamWeb.toggleFollowGame');
return false;
}
}
async #isFollowedGame(gameId) {
try {
debug('开始判断Steam游戏是否已关注', {
gameId: gameId
});
const {result: result, data: data} = await httpRequest({
url: `https://store.steampowered.com/app/${gameId}`,
method: 'GET'
});
debug('获取游戏页面请求结果', {
result: result,
status: data?.status
});
if (result !== 'Success') {
debug('请求失败', {
result: result
});
return false;
}
if (data?.status !== 200) {
debug('响应状态错误', {
status: data?.status
});
return false;
}
if (this.#area === 'CN' && data.responseText.includes('id="error_box"')) {
debug('地区锁定,返回areaLocked', {
area: this.#area
});
return 'areaLocked';
}
const isFollowed = $(data.responseText.replace(/<img.*?>/g, '')).find('.queue_control_button.queue_btn_follow>.btnv6_blue_hoverfade.btn_medium.queue_btn_active').css('display') !== 'none';
debug('关注状态判断结果', {
isFollowed: isFollowed
});
return isFollowed;
} catch (error) {
debug('判断游戏关注状态时发生异常', {
error: error,
gameId: gameId
});
throwError(error, 'SteamWeb.isFollowedGame');
return false;
}
}
async toggleForum(gameId, doTask = true) {
try {
debug('开始处理论坛订阅状态', {
gameId: gameId,
doTask: doTask
});
const forumId = await this.#getForumId(gameId);
if (!forumId) {
return false;
}
const logStatus = echoLog({
type: `${doTask ? '' : 'un'}subscribingForum`,
text: gameId,
before: '[Web]'
});
const [id, feature] = forumId.split('_');
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://steamcommunity.com/forum/${id}/General/${doTask ? '' : 'un'}subscribe/${feature || '0'}/`,
method: 'POST',
responseType: 'json',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: $.param({
sessionid: this.#auth.communitySessionID
})
});
if (result !== 'Success') {
debug('处理论坛订阅状态请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return true;
}
if (data?.status !== 200 || data.response?.success !== 1 && data.response?.success !== 29) {
debug('处理论坛订阅状态失败', {
status: data?.status
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return true;
}
debug('成功处理论坛订阅状态', {
gameId: gameId,
doTask: doTask
});
logStatus.success();
return true;
} catch (error) {
debug('处理论坛订阅状态时发生错误', {
error: error,
gameId: gameId,
doTask: doTask
});
throwError(error, 'SteamWeb.toggleForum');
return false;
}
}
async #getForumId(gameId) {
try {
debug('开始获取Steam论坛ID', {
gameId: gameId
});
const logStatus = echoLog({
type: 'gettingForumId',
text: gameId,
before: '[Web]'
});
const cachedForumId = this.#cache.forum[gameId];
if (cachedForumId) {
debug('从缓存中获取到论坛ID', {
gameId: gameId,
cachedForumId: cachedForumId
});
logStatus.success();
return cachedForumId;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://steamcommunity.com/app/${gameId}/discussions/`,
method: 'GET'
});
debug('获取论坛ID请求结果', {
result: result,
statusText: statusText,
status: status
});
if (result !== 'Success') {
debug('获取论坛ID请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('获取论坛ID响应状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const matchedForumId = data.responseText?.match(/General_([\d]+(_[\d]+)?)/)?.[1];
debug('正则提取论坛ID结果', {
matchedForumId: matchedForumId
});
if (!matchedForumId) {
debug('未能提取到论坛ID', {
gameId: gameId
});
logStatus.error(`Error:${data.statusText}(${data.status})`);
return false;
}
this.#setCache('forum', gameId, matchedForumId);
debug('论坛ID已缓存', {
gameId: gameId,
matchedForumId: matchedForumId
});
logStatus.success();
return matchedForumId;
} catch (error) {
debug('获取论坛ID时发生异常', {
error: error,
gameId: gameId
});
throwError(error, 'SteamWeb.getForumId');
return false;
}
}
async toggleFavoriteWorkshop(id, doTask = true) {
try {
debug('开始处理创意工坊收藏状态', {
id: id,
doTask: doTask
});
const appid = await this.#getWorkshopAppId(id);
if (!appid) {
return false;
}
const logStatus = echoLog({
type: doTask ? 'favoritingWorkshop' : 'unfavoritingWorkshop',
text: id,
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://steamcommunity.com/sharedfiles/${doTask ? '' : 'un'}favorite`,
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: $.param({
id: id,
appid: appid,
sessionid: this.#auth.communitySessionID
})
});
if (result !== 'Success') {
debug('处理创意工坊收藏状态请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200 || data.responseText) {
debug('处理创意工坊收藏状态失败', {
status: data?.status
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('成功处理创意工坊收藏状态', {
id: id,
doTask: doTask
});
logStatus.success();
return true;
} catch (error) {
debug('处理创意工坊收藏状态时发生错误', {
error: error,
id: id,
doTask: doTask
});
throwError(error, 'SteamWeb.toggleFavoriteWorkshop');
return false;
}
}
async #getWorkshopAppId(id) {
try {
debug('开始获取Steam创意工坊AppId', {
id: id
});
const logStatus = echoLog({
type: 'gettingWorkshopAppId',
text: id,
before: '[Web]'
});
const cachedAppId = this.#cache.workshop[id];
if (cachedAppId) {
debug('从缓存中获取到AppId', {
id: id,
cachedAppId: cachedAppId
});
logStatus.success();
return cachedAppId;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://steamcommunity.com/sharedfiles/filedetails/?id=${id}`,
method: 'GET'
});
debug('获取创意工坊AppId请求结果', {
result: result,
statusText: statusText,
status: status
});
if (result !== 'Success') {
debug('获取创意工坊AppId请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('获取创意工坊AppId响应状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const matchedAppId = data.responseText.match(/<input type="hidden" name="appid" value="([\d]+?)" \/>/)?.[1];
debug('正则提取AppId结果', {
matchedAppId: matchedAppId
});
if (!matchedAppId) {
debug('未能提取到AppId', {
id: id
});
logStatus.error('Error: getWorkshopAppId failed');
return false;
}
debug('AppId已缓存', {
id: id,
matchedAppId: matchedAppId
});
return matchedAppId;
} catch (error) {
debug('获取创意工坊AppId时发生异常', {
error: error,
id: id
});
throwError(error, 'SteamWeb.getWorkshopAppId');
return false;
}
}
async voteUpWorkshop(id) {
try {
debug('开始点赞创意工坊物品', {
id: id
});
const logStatus = echoLog({
type: 'votingUpWorkshop',
text: id,
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://steamcommunity.com/sharedfiles/voteup',
method: 'POST',
responseType: 'json',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: $.param({
id: id,
sessionid: this.#auth.communitySessionID
})
});
if (result !== 'Success') {
debug('点赞创意工坊物品请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return true;
}
if (data?.status !== 200 || data.response?.success !== 1) {
debug('点赞创意工坊物品失败', {
status: data?.status
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return true;
}
debug('成功点赞创意工坊物品', {
id: id
});
logStatus.success();
return true;
} catch (error) {
debug('点赞创意工坊物品时发生错误', {
error: error,
id: id
});
throwError(error, 'SteamWeb.voteUpWorkshop');
return false;
}
}
async toggleCurator(curatorId, doTask = true) {
try {
debug('开始处理鉴赏家关注状态', {
curatorId: curatorId,
doTask: doTask
});
const logStatus = echoLog({
type: doTask ? 'followingCurator' : 'unfollowingCurator',
text: curatorId,
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://store.steampowered.com/curators/ajaxfollow',
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: $.param({
clanid: curatorId,
sessionid: this.#auth.storeSessionID,
follow: doTask
}),
dataType: 'json'
});
if (result !== 'Success') {
debug('处理鉴赏家关注状态请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.response?.success?.success === 25) {
debug('处理鉴赏家关注状态失败', {
status: data?.status,
success: data?.response?.success,
message: data?.response?.msg
});
logStatus.error(I18n('curatorLimitNotice'));
return false;
}
if (data?.status !== 200 || data.response?.success?.success !== 1) {
debug('处理鉴赏家关注状态失败', {
status: data?.status,
success: data?.response?.success
});
logStatus.error(`Error:${data?.statusText}(${data?.response?.success}` || `${data?.status})`);
return false;
}
debug('成功处理鉴赏家关注状态', {
curatorId: curatorId,
doTask: doTask
});
logStatus.success();
return true;
} catch (error) {
debug('处理鉴赏家关注状态时发生错误', {
error: error,
curatorId: curatorId,
doTask: doTask
});
throwError(error, 'SteamWeb.toggleCurator');
return false;
}
}
async #getAnnouncementParams(appId, viewId) {
try {
debug('开始获取Steam公告参数', {
appId: appId,
viewId: viewId
});
const logStatus = echoLog({
type: 'gettingAnnouncementParams',
text: appId,
id: viewId,
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://store.steampowered.com/events/ajaxgetpartnerevent?appid=${appId}&announcement_gid=${viewId}&lang_list=6_0&last_modified_time=0&origin=https:%2F%2Fstore.steampowered.com&for_edit=false`,
method: 'GET',
responseType: 'json',
headers: {
Host: 'store.steampowered.com',
Referer: `https://store.steampowered.com/news/app/${appId}/view/${viewId}`
}
});
debug('获取公告参数请求结果', {
result: result,
statusText: statusText,
status: status
});
if (result !== 'Success') {
debug('获取公告参数请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return {};
}
if (data?.status !== 200 || data?.response?.success !== 1) {
debug('获取公告参数响应状态错误', {
status: data?.status,
statusText: data?.statusText,
response: data?.response
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return {};
}
const {clanid: clanid, gid: gid} = data.response.event?.announcement_body || {};
debug('公告参数提取', {
clanid: clanid,
gid: gid
});
if (!clanid) {
debug('未能提取到clanid', {
appId: appId,
viewId: viewId
});
logStatus.error(`Error:${data.statusText}(${data.status})`);
return {};
}
logStatus.success();
debug('获取公告参数成功', {
clanId: clanid,
gid: gid
});
return {
clanId: clanid,
gid: gid
};
} catch (error) {
debug('获取公告参数时发生异常', {
error: error,
appId: appId,
viewId: viewId
});
throwError(error, 'SteamWeb.likeAnnouncement');
return {};
}
}
async likeAnnouncement(id) {
try {
debug('开始点赞公告', {
id: id
});
const [appId, viewId] = id.split('/');
if (!(appId && viewId)) {
echoLog({
before: '[Web]'
}).error(`${I18n('missParams')}(id)`);
return false;
}
const {clanId: clanId, gid: gid} = await this.#getAnnouncementParams(appId, viewId);
if (!clanId) {
return false;
}
const logStatus = echoLog({
type: 'likingAnnouncement',
text: appId,
id: viewId,
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://store.steampowered.com/updated/ajaxrateupdate/${gid || viewId}`,
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
Host: 'store.steampowered.com',
Origin: 'https://store.steampowered.com',
Referer: `https://store.steampowered.com/news/app/${appId}/view/${viewId}`
},
data: $.param({
sessionid: this.#auth.storeSessionID,
voteup: 1,
clanid: clanId,
ajax: 1
}),
dataType: 'json'
});
if (result !== 'Success') {
debug('点赞公告请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200 || data.response.success !== 1) {
debug('点赞公告失败', {
status: data?.status
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('成功点赞公告', {
id: id
});
logStatus.success();
return true;
} catch (error) {
debug('点赞公告时发生错误', {
error: error,
id: id
});
throwError(error, 'SteamWeb.likeAnnouncement');
return false;
}
}
async #appid2subid(id) {
try {
debug('开始将AppId转换为SubId', {
id: id
});
const logStatus = echoLog({
type: 'gettingSubid',
text: id,
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://store.steampowered.com/app/${id}`,
method: 'GET',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
}
});
debug('获取App页面请求结果', {
result: result,
statusText: statusText,
status: status
});
if (result !== 'Success') {
debug('获取App页面请求失败', {
result: result
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('获取App页面响应状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
if (data.responseText.includes('ds_owned_flag ds_flag') || data.responseText.includes('class="already_in_library"')) {
debug('App已拥有', {
id: id
});
logStatus.success(I18n('owned'));
return true;
}
if (this.#area === 'CN' && data.responseText.includes('id="error_box"')) {
debug('地区锁定,尝试更换地区', {
area: this.#area
});
logStatus.warning(I18n('changeAreaNotice'));
const result = await this.#changeArea();
if (!result || result === 'CN' || result === 'skip') {
debug('更换地区失败或未更换', {
result: result
});
return false;
}
return await this.#appid2subid(id);
}
let subid = data.responseText.match(/name="subid" value="([\d]+?)"/)?.[1];
debug('正则提取subid结果1', {
subid: subid
});
if (subid) {
logStatus.success();
return subid;
}
subid = data.responseText.match(/AddFreeLicense\(\s*(\d+)/)?.[1];
debug('正则提取subid结果2', {
subid: subid
});
if (subid) {
logStatus.success();
return subid;
}
debug('未能提取到subid', {
id: id
});
logStatus.error(`Error:${I18n('noSubid')}`);
return false;
} catch (error) {
debug('AppId转SubId时发生异常', {
error: error,
id: id
});
throwError(error, 'SteamWeb.appid2subid');
return false;
}
}
async #getLicenses() {
try {
debug('开始获取Steam用户许可证信息');
const logStatus = echoLog({
text: I18n('gettingLicenses'),
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://store.steampowered.com/dynamicstore/userdata/?t=${(new Date).getTime()}`,
method: 'GET',
responseType: 'json'
});
debug('获取许可证请求结果', {
result: result,
statusText: statusText,
status: status
});
if (result !== 'Success') {
debug('获取许可证请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('获取许可证响应状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('获取到的许可证列表', {
licenses: data.response?.rgOwnedPackages
});
logStatus.remove();
return data.response?.rgOwnedPackages;
} catch (error) {
debug('获取许可证时发生异常', {
error: error
});
throwError(error, 'SteamWeb.getLicenses');
return false;
}
}
async addLicense(id) {
try {
debug('开始添加许可证', {
id: id
});
const [type, ids] = id.split('-');
debug('解析许可证ID', {
type: type,
ids: ids
});
if (type !== 'appid' && type !== 'subid') {
debug('无效的许可证类型', {
type: type
});
return false;
}
if (type === 'appid') {
debug('处理appid类型许可证', {
ids: ids
});
const subid = await this.#appid2subid(ids);
debug('appid转换为subid结果', {
appid: ids,
subid: subid
});
if (!subid) {
debug('appid转换失败', {
appid: ids
});
return false;
}
if (subid === true) {
debug('appid已拥有', {
appid: ids
});
return true;
}
const logStatus = echoLog({
type: 'addingFreeLicense',
text: ids,
before: '[Web]'
});
debug('开始添加免费许可证', {
subid: subid
});
if (!await this.#addFreeLicense(subid, logStatus)) {
debug('添加免费许可证失败', {
subid: subid
});
return false;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://store.steampowered.com/app/${ids}`,
method: 'GET'
});
debug('验证许可证添加状态', {
result: result,
status: status,
statusText: statusText
});
if (result !== 'Success') {
debug('验证请求失败', {
result: result,
status: status,
statusText: statusText
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('验证响应状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
if (!data.responseText.includes('ds_owned_flag ds_flag') && !data.responseText.includes('class="already_in_library"')) {
debug('未找到游戏拥有标记', {
status: data.status,
statusText: data.statusText
});
logStatus.error(`Error:${data.statusText}(${data.status})`);
return false;
}
debug('appid许可证添加成功', {
appid: ids
});
logStatus.success();
return true;
}
if (this.#area === 'CN') {
debug('当前区域为CN,尝试更改区域', {
currentArea: this.#area
});
echoLog({
before: '[Web]'
}).success(I18n('tryChangeAreaNotice'));
await this.#changeArea();
}
const logStatusArr = {};
const idsArr = ids.split(',');
debug('处理subid类型许可证', {
idsArr: idsArr
});
for (const subid of idsArr) {
debug('开始处理单个subid', {
subid: subid
});
const logStatus = echoLog({
type: 'addingFreeLicense',
text: subid,
before: '[Web]'
});
if (!await this.#addFreeLicense(subid, logStatus)) {
debug('添加subid许可证失败', {
subid: subid
});
return false;
}
logStatusArr[subid] = logStatus;
}
const licenses = await this.#getLicenses();
debug('获取许可证列表', {
licenses: licenses
});
if (!licenses) {
debug('获取许可证列表失败');
return false;
}
for (const subid of idsArr) {
const hasLicense = licenses.includes(parseInt(subid, 10));
debug('验证许可证添加状态', {
subid: subid,
hasLicense: hasLicense
});
if (hasLicense) {
logStatusArr[subid].success();
} else {
logStatusArr[subid].error();
}
}
debug('所有subid许可证处理完成', {
idsArr: idsArr
});
return true;
} catch (error) {
debug('添加许可证过程发生错误', {
error: error,
id: id
});
throwError(error, 'SteamWeb.addLicense');
return false;
}
}
async #addFreeLicense(id, logStatusPre) {
try {
debug('开始添加免费Steam游戏许可证', {
id: id
});
const logStatus = logStatusPre || echoLog({
type: 'addingFreeLicenseSubid',
text: id,
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://store.steampowered.com/freelicense/addfreelicense/${id}`,
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
Host: 'store.steampowered.com',
Origin: 'https://store.steampowered.com',
Referer: 'https://store.steampowered.com/account/licenses/'
},
data: $.param({
ajax: true,
sessionid: this.#auth.storeSessionID
}),
dataType: 'json'
});
debug('添加免费许可证请求结果', {
result: result,
statusText: statusText,
status: status
});
if (result !== 'Success') {
debug('添加免费许可证请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('添加免费许可证响应状态错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
if (this.#area === 'CN' && data.responseText.includes('id="error_box"')) {
debug('地区锁定,尝试更换地区', {
area: this.#area
});
logStatus.warning(I18n('changeAreaNotice'));
const result = await this.#changeArea();
if (!result || [ 'CN', 'skip' ].includes(result)) {
debug('更换地区失败或未更换', {
result: result
});
return false;
}
return await this.#addFreeLicense(id);
}
debug('成功添加免费许可证', {
id: id
});
logStatus.success();
return true;
} catch (error) {
debug('添加免费许可证时发生异常', {
error: error,
id: id
});
throwError(error, 'SteamWeb.addFreeLicense');
return false;
}
}
async requestPlayTestAccess(id) {
debug('开始请求游戏试玩权限', {
id: id
});
try {
debug('开始请求游戏试玩权限', {
id: id
});
const logStatus = echoLog({
type: 'requestingPlayTestAccess',
text: id,
before: '[Web]'
});
debug('准备发送试玩权限请求', {
id: id
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://store.steampowered.com/ajaxrequestplaytestaccess/${id}`,
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
Host: 'store.steampowered.com',
Origin: 'https://store.steampowered.com',
Referer: `https://store.steampowered.com/app/${id}`
},
data: $.param({
sessionid: this.#auth.storeSessionID
}),
dataType: 'json'
});
debug('收到试玩权限请求响应', {
result: result,
status: status,
statusText: statusText,
responseData: data
});
if (result !== 'Success') {
debug('请求失败', {
result: result,
status: status,
statusText: statusText
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200 || data?.response?.success !== 1) {
debug('响应状态错误', {
status: data?.status,
statusText: data?.statusText,
success: data?.response?.success
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('成功请求游戏试玩权限', {
id: id
});
logStatus.success();
return true;
} catch (error) {
debug('请求游戏试玩权限时发生错误', {
error: error,
id: id
});
throwError(error, 'SteamWeb.requestPlayTestAccess');
return false;
}
}
async resetArea() {
try {
debug('检查区域设置状态', {
currentArea: this.#area,
oldArea: this.#oldArea,
needReset: Boolean(this.#oldArea && this.#area !== this.#oldArea)
});
if (this.#oldArea && this.#area !== this.#oldArea) {
debug('需要重置区域', {
fromArea: this.#area,
toArea: this.#oldArea
});
echoLog({
before: '[Web]'
}).warning(I18n('steamFinishNotice') + this.#oldArea);
const changeResult = await this.#changeArea(this.#oldArea);
debug('区域重置结果', {
success: changeResult,
targetArea: this.#oldArea
});
} else {
debug('无需重置区域', {
currentArea: this.#area,
oldArea: this.#oldArea
});
}
debug('区域重置流程完成');
return true;
} catch (error) {
debug('重置区域时发生错误', {
error: error
});
throwError(error, 'SteamWeb.resetArea');
return false;
}
}
#setCache(type, name, id) {
try {
debug('开始设置缓存', {
type: type,
name: name,
id: id
});
this.#cache[type][name] = id;
GM_setValue('steamCache', this.#cache);
debug('设置缓存成功', {
type: type,
name: name,
id: id
});
} catch (error) {
debug('设置缓存时发生异常', {
error: error,
type: type,
name: name,
id: id
});
throwError(error, 'SteamWeb.setCache');
}
}
}
class Steam extends Social {
tasks;
whiteList;
#cache={
...{
group: {},
officialGroup: {},
forum: {},
workshop: {},
curator: {}
},
...GM_getValue('steamCache')
};
#TaskExecutor=[];
constructor() {
super();
debug('初始化Steam实例');
const defaultTasksTemplate = {
groups: [],
officialGroups: [],
wishlists: [],
follows: [],
forums: [],
workshops: [],
workshopVotes: [],
curators: [],
curatorLikes: [],
announcements: [],
licenses: [],
playtests: [],
playTime: []
};
this.tasks = defaultTasksTemplate;
this.whiteList = {
...defaultTasksTemplate,
...GM_getValue('whiteList')?.steam || {}
};
this.#TaskExecutor = this.#getTaskExecutionOrder(globalOptions.ASF.AsfEnabled, globalOptions.ASF.steamWeb, globalOptions.ASF.preferASF);
debug('Steam实例初始化完成', {
taskExecutorCount: this.#TaskExecutor.length
});
}
async init(type = 'all') {
try {
debug('开始初始化Steam模块', {
type: type
});
for (let i = 0; i < this.#TaskExecutor.length; i++) {
debug(`初始化执行器 ${i + 1}/${this.#TaskExecutor.length}`);
if (!await this.#TaskExecutor[i].init(type)) {
debug(`执行器 ${i + 1} 初始化失败,移除该执行器`);
this.#TaskExecutor.splice(i, 1);
}
}
debug('Steam模块初始化完成', {
remainingExecutors: this.#TaskExecutor.length
});
return this.#TaskExecutor.length > 0;
} catch (error) {
debug('Steam初始化发生错误', {
error: error
});
throwError(error, 'Steam.init');
return false;
}
}
async #joinGroup(groupName) {
try {
debug('开始加入Steam组', {
groupName: groupName
});
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.joinGroup(groupName)) {
debug('成功加入Steam组', {
groupName: groupName
});
this.tasks.groups = unique([ ...this.tasks.groups, groupName ]);
return true;
}
}
debug('加入Steam组失败', {
groupName: groupName
});
return false;
} catch (error) {
debug('加入Steam组时发生错误', {
error: error,
groupName: groupName
});
throwError(error, 'Steam.joinGroup');
return false;
}
}
async #leaveGroup(groupName) {
try {
debug('开始退出Steam组', {
groupName: groupName
});
if (this.whiteList.groups.includes(groupName)) {
debug('Steam组在白名单中,跳过退出', {
groupName: groupName
});
echoLog({
type: 'whiteList',
text: 'Steam.leaveGroup',
id: groupName
});
return true;
}
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.leaveGroup(groupName)) {
debug('成功退出Steam组', {
groupName: groupName
});
return true;
}
}
debug('退出Steam组失败', {
groupName: groupName
});
return false;
} catch (error) {
debug('退出Steam组时发生错误', {
error: error,
groupName: groupName
});
throwError(error, 'Steam.leaveGroup');
return false;
}
}
async #joinOfficialGroup(gameId) {
try {
debug('开始加入Steam官方组', {
gameId: gameId
});
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.joinOfficialGroup(gameId)) {
debug('成功加入Steam官方组', {
gameId: gameId
});
return true;
}
}
debug('加入Steam官方组失败', {
gameId: gameId
});
return false;
} catch (error) {
debug('加入Steam官方组时发生错误', {
error: error,
gameId: gameId
});
throwError(error, 'Steam.joinOfficialGroup');
return false;
}
}
async #leaveOfficialGroup(gameId) {
try {
debug('开始退出Steam官方组', {
gameId: gameId
});
if (this.whiteList.officialGroups.includes(gameId)) {
debug('Steam官方组在白名单中,跳过退出', {
gameId: gameId
});
echoLog({
type: 'whiteList',
text: 'Steam.leaveOfficialGroup',
id: gameId
});
return true;
}
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.leaveOfficialGroup(gameId)) {
debug('成功退出Steam官方组', {
gameId: gameId
});
this.tasks.officialGroups = unique([ ...this.tasks.officialGroups, gameId ]);
return true;
}
}
debug('退出Steam官方组失败', {
gameId: gameId
});
return false;
} catch (error) {
debug('退出Steam官方组时发生错误', {
error: error,
gameId: gameId
});
throwError(error, 'Steam.leaveOfficialGroup');
return false;
}
}
async #addToWishlist(gameId) {
try {
debug('开始添加游戏到愿望单', {
gameId: gameId
});
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.addToWishlist(gameId)) {
debug('成功添加游戏到愿望单', {
gameId: gameId
});
this.tasks.wishlists = unique([ ...this.tasks.wishlists, gameId ]);
return true;
}
}
debug('添加游戏到愿望单失败', {
gameId: gameId
});
return false;
} catch (error) {
debug('添加游戏到愿望单时发生错误', {
error: error,
gameId: gameId
});
throwError(error, 'Steam.addToWishlist');
return false;
}
}
async #removeFromWishlist(gameId) {
try {
debug('开始从愿望单移除游戏', {
gameId: gameId
});
if (this.whiteList.wishlists.includes(gameId)) {
debug('游戏在愿望单白名单中,跳过移除', {
gameId: gameId
});
echoLog({
type: 'whiteList',
text: 'Steam.removeFromWishlist',
id: gameId
});
return true;
}
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.removeFromWishlist(gameId)) {
debug('成功从愿望单移除游戏', {
gameId: gameId
});
return true;
}
}
debug('从愿望单移除游戏失败', {
gameId: gameId
});
return false;
} catch (error) {
debug('从愿望单移除游戏时发生错误', {
error: error,
gameId: gameId
});
throwError(error, 'Steam.removeFromWishlist');
return false;
}
}
async #toggleFollowGame(gameId, doTask) {
try {
debug('开始处理游戏关注状态', {
gameId: gameId,
doTask: doTask
});
if (!doTask && this.whiteList.follows.includes(gameId)) {
debug('游戏在关注白名单中,跳过取关', {
gameId: gameId
});
echoLog({
type: 'whiteList',
text: 'Steam.unfollowGame',
id: gameId
});
return true;
}
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.toggleFollowGame(gameId, doTask)) {
if (doTask) {
debug('成功关注游戏', {
gameId: gameId
});
this.tasks.follows = unique([ ...this.tasks.follows, gameId ]);
} else {
debug('成功取关游戏', {
gameId: gameId
});
}
return true;
}
}
debug('处理游戏关注状态失败', {
gameId: gameId,
doTask: doTask
});
return false;
} catch (error) {
debug('处理游戏关注状态时发生错误', {
error: error,
gameId: gameId,
doTask: doTask
});
throwError(error, 'Steam.toggleFollowGame');
return false;
}
}
async #toggleForum(gameId, doTask = true) {
try {
debug('开始处理论坛订阅状态', {
gameId: gameId,
doTask: doTask
});
if (!doTask && this.whiteList.forums.includes(gameId)) {
debug('论坛在白名单中,跳过取消订阅', {
gameId: gameId
});
echoLog({
type: 'whiteList',
text: 'Steam.unsubscribeForum',
id: gameId
});
return true;
}
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.toggleForum(gameId, doTask)) {
if (doTask) {
debug('成功订阅论坛', {
gameId: gameId
});
this.tasks.forums = unique([ ...this.tasks.forums, gameId ]);
} else {
debug('成功取消订阅论坛', {
gameId: gameId
});
}
return true;
}
}
debug('处理论坛订阅状态失败', {
gameId: gameId,
doTask: doTask
});
return false;
} catch (error) {
debug('处理论坛订阅状态时发生错误', {
error: error,
gameId: gameId,
doTask: doTask
});
throwError(error, 'Steam.toggleForum');
return true;
}
}
async #toggleFavoriteWorkshop(id, doTask = true) {
try {
debug('开始处理创意工坊收藏状态', {
id: id,
doTask: doTask
});
if (!doTask && this.whiteList.workshops.includes(id)) {
debug('创意工坊物品在白名单中,跳过取消收藏', {
id: id
});
echoLog({
type: 'whiteList',
text: 'Steam.unfavoriteWorkshop',
id: id
});
return true;
}
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.toggleFavoriteWorkshop(id)) {
if (doTask) {
debug('成功收藏创意工坊物品', {
id: id
});
this.tasks.workshops = unique([ ...this.tasks.workshops, id ]);
} else {
debug('成功取消收藏创意工坊物品', {
id: id
});
}
return true;
}
}
debug('处理创意工坊收藏状态失败', {
id: id,
doTask: doTask
});
return false;
} catch (error) {
debug('处理创意工坊收藏状态时发生错误', {
error: error,
id: id,
doTask: doTask
});
throwError(error, 'Steam.toggleFavoriteWorkshop');
return false;
}
}
async #voteUpWorkshop(id) {
try {
debug('开始点赞创意工坊物品', {
id: id
});
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.voteUpWorkshop(id)) {
debug('成功点赞创意工坊物品', {
id: id
});
return true;
}
}
debug('点赞创意工坊物品失败', {
id: id
});
return false;
} catch (error) {
debug('点赞创意工坊物品时发生错误', {
error: error,
id: id
});
throwError(error, 'Steam.voteupWorkshop');
return true;
}
}
async #toggleCurator(curatorId, doTask = true) {
try {
debug('开始处理鉴赏家关注状态', {
curatorId: curatorId,
doTask: doTask
});
if (!doTask && this.whiteList.curators.includes(curatorId)) {
debug('鉴赏家在白名单中,跳过取关', {
curatorId: curatorId
});
echoLog({
type: 'whiteList',
text: 'Steam.unfollowCurator',
id: curatorId
});
return true;
}
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.toggleCurator(curatorId, doTask)) {
if (doTask) {
debug('成功关注鉴赏家', {
curatorId: curatorId
});
this.tasks.curators = unique([ ...this.tasks.curators, curatorId ]);
} else {
debug('成功取关鉴赏家', {
curatorId: curatorId
});
}
return true;
}
}
debug('处理鉴赏家关注状态失败', {
curatorId: curatorId,
doTask: doTask
});
return false;
} catch (error) {
debug('处理鉴赏家关注状态时发生错误', {
error: error,
curatorId: curatorId,
doTask: doTask
});
throwError(error, 'Steam.toggleCurator');
return false;
}
}
async getCuratorId(path, name) {
try {
debug('开始获取鉴赏家ID', {
path: path,
name: name
});
const logStatus = echoLog({
type: 'gettingCuratorId',
text: `${path}/${name}`,
before: '[Web]'
});
const curatorId = this.#cache.curator[`${path}/${name}`];
if (curatorId) {
debug('从缓存获取到鉴赏家ID', {
path: path,
name: name,
curatorId: curatorId
});
logStatus.success();
return curatorId;
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://store.steampowered.com/${path}/${name}`,
method: 'GET',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
}
});
if (result === 'Success') {
if (data?.status === 200) {
const curatorId = data.responseText.match(/g_pagingData.*?"clanid":([\d]+)/)?.[1];
if (curatorId) {
debug('成功获取鉴赏家ID', {
path: path,
name: name,
curatorId: curatorId
});
this.#setCache('curator', `${path}/${name}`, curatorId);
logStatus.success();
return curatorId;
}
debug('未找到鉴赏家ID', {
path: path,
name: name,
status: data.status
});
logStatus.error(`Error:${data.statusText}(${data.status})`);
return false;
}
debug('获取鉴赏家页面失败', {
path: path,
name: name,
status: data?.status
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('请求鉴赏家页面失败', {
path: path,
name: name,
result: result,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
} catch (error) {
debug('获取鉴赏家ID时发生错误', {
error: error,
path: path,
name: name
});
throwError(error, 'SteamWeb.getCuratorID');
return false;
}
}
async #toggleCuratorLike(link, doTask = true) {
try {
debug('开始处理鉴赏家点赞状态', {
link: link,
doTask: doTask
});
const [path, name] = link.split('/');
if (!(path && name)) {
debug('无效的鉴赏家链接', {
link: link
});
echoLog({
text: I18n('errorLink', link),
before: '[Web]'
});
return false;
}
const curatorId = await this.getCuratorId(path, name);
if (curatorId) {
debug('获取到鉴赏家ID,开始处理点赞', {
curatorId: curatorId,
doTask: doTask
});
return await this.#toggleCurator(curatorId, doTask);
}
debug('未获取到鉴赏家ID', {
link: link
});
return false;
} catch (error) {
debug('处理鉴赏家点赞状态时发生错误', {
error: error,
link: link,
doTask: doTask
});
throwError(error, 'Steam.toggleCuratorLike');
return false;
}
}
async #likeAnnouncement(id) {
try {
debug('开始点赞公告', {
id: id
});
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.likeAnnouncement(id)) {
debug('成功点赞公告', {
id: id
});
return true;
}
}
debug('点赞公告失败', {
id: id
});
return false;
} catch (error) {
debug('点赞公告时发生错误', {
error: error,
id: id
});
throwError(error, 'Steam.likeAnnouncement');
return false;
}
}
async #addLicense(id) {
try {
debug('开始添加许可证', {
id: id
});
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.addLicense(id)) {
debug('成功添加许可证', {
id: id
});
return true;
}
}
debug('添加许可证失败', {
id: id
});
return false;
} catch (error) {
debug('添加许可证时发生错误', {
error: error,
id: id
});
throwError(error, 'Steam.addLicense');
return false;
}
}
async #requestPlayTestAccess(id) {
try {
debug('开始请求游戏试玩权限', {
id: id
});
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.requestPlayTestAccess(id)) {
debug('成功请求游戏试玩权限', {
id: id
});
return true;
}
}
debug('请求游戏试玩权限失败', {
id: id
});
return false;
} catch (error) {
debug('请求游戏试玩权限时发生错误', {
error: error,
id: id
});
throwError(error, 'Steam.requestPlayTestAccess');
return false;
}
}
async #getDemoAppid(id) {
try {
debug('开始获取游戏试玩ID', {
id: id
});
const logStatus = echoLog({
type: 'gettingDemoAppid',
text: id,
before: '[Web]'
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://store.steampowered.com/app/${id}`,
method: 'GET',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
Host: 'store.steampowered.com',
Origin: 'https://store.steampowered.com',
Referer: `https://store.steampowered.com/app/${id}`
}
});
if (result === 'Success') {
if (data?.status === 200) {
const demoAppid = data.responseText.match(/steam:\/\/(install|run)\/(\d+)/)?.[2];
debug('成功获取游戏试玩ID', {
id: id,
demoAppid: demoAppid
});
logStatus.success();
return demoAppid || false;
}
debug('获取游戏页面失败', {
id: id,
status: data?.status
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('请求游戏页面失败', {
id: id,
result: result,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
} catch (error) {
debug('获取游戏试玩ID时发生错误', {
error: error,
id: id
});
throwError(error, 'Steam.getDemoAppid');
return false;
}
}
async #playGames(ids, playTime, doTask = true) {
try {
debug('开始处理游戏挂时长', {
ids: ids,
playTime: playTime,
doTask: doTask
});
if (playTime <= 0) {
debug('游戏时长小于等于0,跳过挂时长');
return true;
}
const asf = this.#TaskExecutor.find((e => e instanceof SteamASF));
if (!asf) {
debug('未找到ASF实例');
echoLog({}).warning(I18n('noASFInstance'));
return false;
}
if (!doTask) {
debug('停止挂时长');
return await asf.stopPlayGames();
}
const idsArr = await Promise.all(ids.split(',').map((async id => {
try {
const demoAppid = await this.#getDemoAppid(id);
return demoAppid ? `${id},${demoAppid}` : id;
} catch (error) {
debug('获取游戏试玩ID失败', {
error: error,
id: id
});
return id;
}
})));
const uniqueIds = unique(idsArr.join(',').split(','));
debug('处理后的游戏ID列表', {
uniqueIds: uniqueIds
});
debug('开始尝试入库游戏', {
uniqueIds: uniqueIds
});
await Promise.all(uniqueIds.map((async id => {
for (const taskExecutor of this.#TaskExecutor) {
if (await taskExecutor.addLicense(`appid-${id}`)) {
debug('成功入库游戏', {
id: id
});
return true;
}
}
return false;
})));
await asf.playGames(uniqueIds.join(','));
const status = await asf.checkPlayStatus(uniqueIds.join(','));
if (status !== true) {
await delay(3e3);
await asf.playGames(uniqueIds.join(','));
const status = await asf.checkPlayStatus(uniqueIds.join(','));
if (!status) {
debug('启动游戏失败');
return false;
}
}
const stopPlayTime = Date.now() + (playTime + 10) * 60 * 1e3;
const stopPlayTimeOld = GM_getValue('stopPlayTime', 0) || 0;
GM_setValue('stopPlayTime', Math.max(stopPlayTime, stopPlayTimeOld));
const playedGames = GM_getValue('playedGames', []) || [];
GM_setValue('playedGames', unique([ ...playedGames, ...uniqueIds ]));
const taskLink = GM_getValue('taskLink', []) || [];
GM_setValue('taskLink', unique([ ...taskLink, window.location.href ]));
debug('游戏挂时长状态更新完成');
return true;
} catch (error) {
debug('处理游戏挂时长时发生错误', {
error: error,
ids: ids,
playTime: playTime
});
throwError(error, 'Steam.playGames');
return false;
}
}
async toggle({doTask: doTask = true, groupLinks: groupLinks = [], officialGroupLinks: officialGroupLinks = [], wishlistLinks: wishlistLinks = [], followLinks: followLinks = [], forumLinks: forumLinks = [], workshopLinks: workshopLinks = [], workshopVoteLinks: workshopVoteLinks = [], curatorLinks: curatorLinks = [], curatorLikeLinks: curatorLikeLinks = [], announcementLinks: announcementLinks = [], licenseLinks: licenseLinks = [], playtestLinks: playtestLinks = [], playTimeLinks: playTimeLinks = []}) {
try {
debug('开始处理Steam任务', {
doTask: doTask,
linksCount: {
groups: groupLinks.length,
officialGroups: officialGroupLinks.length,
wishlists: wishlistLinks.length,
follows: followLinks.length,
forums: forumLinks.length,
workshops: workshopLinks.length,
workshopVotes: workshopVoteLinks.length,
curators: curatorLinks.length,
curatorLikes: curatorLikeLinks.length,
announcements: announcementLinks.length,
licenses: licenseLinks.length,
playtests: playtestLinks.length,
playTime: playTimeLinks.length
}
});
const allLinks = [ ...groupLinks, ...officialGroupLinks, ...forumLinks, ...workshopLinks, ...workshopVoteLinks, ...wishlistLinks, ...followLinks, ...curatorLinks, ...curatorLikeLinks, ...announcementLinks, ...licenseLinks, ...playtestLinks, ...playTimeLinks ];
if (allLinks.length > 0 && this.#TaskExecutor.length === 0) {
debug('Steam模块未初始化');
echoLog({
text: I18n('needInit')
});
return false;
}
const tasks = [];
if (this.shouldProcessTask('groups', doTask)) {
debug('开始处理群组任务');
const realGroups = this.getRealParams('groups', groupLinks, doTask, (link => link.match(/groups\/(.+)\/?/)?.[1]?.split('/')?.[0]));
debug('处理后的群组列表', {
count: realGroups.length,
groups: realGroups
});
for (const group of realGroups) {
tasks.push(doTask ? this.#joinGroup(group) : this.#leaveGroup(group));
await delay(1e3);
}
}
if (this.shouldProcessTask('officialGroups', doTask)) {
const realOfficialGroups = this.getRealParams('officialGroups', officialGroupLinks, doTask, (link => link.match(/games\/(.+)\/?/)?.[1]));
for (const officialGroup of realOfficialGroups) {
tasks.push(doTask ? this.#joinOfficialGroup(officialGroup) : this.#leaveOfficialGroup(officialGroup));
await delay(1e3);
}
}
if (this.shouldProcessTask('wishlists', doTask)) {
const realWishlists = this.getRealParams('wishlists', wishlistLinks, doTask, (link => link.match(/app\/([\d]+)/)?.[1]));
for (const game of realWishlists) {
tasks.push(doTask ? this.#addToWishlist(game) : this.#removeFromWishlist(game));
await delay(1e3);
}
}
if (this.shouldProcessTask('follows', doTask)) {
const realFollows = this.getRealParams('follows', followLinks, doTask, (link => link.match(/app\/([\d]+)/)?.[1]));
for (const game of realFollows) {
tasks.push(this.#toggleFollowGame(game, doTask));
await delay(1e3);
}
}
if (this.shouldProcessTask('playTime', doTask)) {
const realGames = this.getRealParams('playTime', playTimeLinks, doTask, (link => `${link.split('-')[0]}-${link.match(/app\/([\d]+)/)?.[1] || ''}`));
if (realGames.length > 0) {
const maxTime = Math.max(...realGames.map((info => parseInt(info.split('-')[0], 10) || 0)));
const games = realGames.filter((info => {
const [time, game] = info.split('-');
return (parseInt(time, 10) || 0) > 0 && game;
})).map((info => info.split('-')[1]));
tasks.push(this.#playGames(games.join(','), maxTime, doTask));
await delay(1e3);
}
}
if (this.shouldProcessTask('forums', doTask)) {
const realForums = this.getRealParams('forums', forumLinks, doTask, (link => link.match(/app\/([\d]+)/)?.[1]));
for (const forum of realForums) {
tasks.push(this.#toggleForum(forum, doTask));
await delay(1e3);
}
}
if (this.shouldProcessTask('workshops', doTask)) {
const realWorkshops = this.getRealParams('workshops', workshopLinks, doTask, (link => link.match(/\?id=([\d]+)/)?.[1]));
for (const workshop of realWorkshops) {
tasks.push(this.#toggleFavoriteWorkshop(workshop, doTask));
await delay(1e3);
}
}
if (doTask && globalOptions.doTask.steam.workshopVotes) {
const realworkshopVotes = this.getRealParams('workshopVotes', workshopVoteLinks, doTask, (link => link.match(/\?id=([\d]+)/)?.[1]));
for (const workshop of realworkshopVotes) {
tasks.push(this.#voteUpWorkshop(workshop));
await delay(1e3);
}
}
if (this.shouldProcessTask('curators', doTask)) {
const realCurators = this.getRealParams('curators', curatorLinks, doTask, (link => link.match(/curator\/([\d]+)/)?.[1]));
const realCuratorLikes = this.getRealParams('curatorLikes', curatorLikeLinks, doTask, (link => link.match(/https?:\/\/store\.steampowered\.com\/(.*?)\/([^/?]+)/)?.slice(1, 3).join('/')));
for (const curator of realCurators) {
tasks.push(this.#toggleCurator(curator, doTask));
await delay(1e3);
}
for (const curatorLike of realCuratorLikes) {
tasks.push(this.#toggleCuratorLike(curatorLike, doTask));
await delay(1e3);
}
}
if (doTask && globalOptions.doTask.steam.announcements) {
const realAnnouncements = this.getRealParams('announcements', announcementLinks, doTask, (link => {
if (link.includes('store.steampowered.com')) {
return link.match(/store\.steampowered\.com\/news\/app\/([\d]+)\/view\/([\d]+)/)?.slice(1, 3).join('/');
}
return link.match(/steamcommunity\.com\/games\/([\d]+)\/announcements\/detail\/([\d]+)/)?.slice(1, 3).join('/');
}));
for (const id of realAnnouncements) {
tasks.push(this.#likeAnnouncement(id));
await delay(1e3);
}
}
if (doTask && globalOptions.doTask.steam.licenses && licenseLinks.length > 0) {
for (const ids of licenseLinks) {
const [type, idsStr] = ids.split('-');
const idsArr = idsStr.split(',');
for (const id of idsArr) {
tasks.push(this.#addLicense(`${type}-${id}`));
await delay(1e3);
}
}
}
if (doTask && globalOptions.doTask.steam.playtests) {
const realPlaytests = this.getRealParams('playtests', playtestLinks, doTask, (link => link.match(/app\/([\d]+)/)?.[1]));
for (const id of realPlaytests) {
tasks.push(this.#requestPlayTestAccess(id));
await delay(1e3);
}
}
debug('开始执行所有任务');
const results = await Promise.all(tasks);
this.#TaskExecutor.find((e => e instanceof SteamWeb))?.resetArea();
debug('所有任务执行完成', {
success: results.every((result => result))
});
return results.every((result => result));
} catch (error) {
debug('处理Steam任务时发生错误', {
error: error
});
throwError(error, 'Steam.toggle');
return false;
}
}
shouldProcessTask(taskType, doTask) {
debug('检查是否处理任务', {
taskType: taskType,
doTask: doTask
});
if (doTask) {
const result = globalOptions.doTask.steam[taskType];
debug('检查doTask配置', {
taskType: taskType,
result: result
});
return globalOptions.doTask.steam[taskType];
}
const undoTaskType = taskType;
return undoTaskType in globalOptions.undoTask.steam && globalOptions.undoTask.steam[undoTaskType];
}
#setCache(type, name, id) {
try {
this.#cache[type][name] = id;
GM_setValue('steamCache', this.#cache);
} catch (error) {
throwError(error, 'SteamWeb.setCache');
}
}
#getTaskExecutionOrder(asfEnabled, steamWebEnabled, preferASF) {
if (!asfEnabled) {
return [ new SteamWeb ];
}
if (!steamWebEnabled) {
return [ new SteamASF(globalOptions.ASF) ];
}
return preferASF ? [ new SteamASF(globalOptions.ASF), new SteamWeb ] : [ new SteamWeb, new SteamASF(globalOptions.ASF) ];
}
}
class Website {
undoneTasks;
socialTasks;
giveawayId;
socialInitialized={
discord: false,
instagram: false,
reddit: false,
twitch: false,
twitter: false,
vk: false,
youtube: false,
steamStore: false,
steamCommunity: false
};
initialized=false;
steamTaskType={
steamStore: false,
steamCommunity: false
};
social={};
async #bind(name, init) {
try {
debug('开始绑定社交媒体', {
name: name
});
const result = await init;
debug('绑定结果', {
name: name,
result: result
});
return {
name: name,
result: result
};
} catch (error) {
debug('绑定失败', {
name: name,
error: error
});
throwError(error, 'Website.bind');
return {
name: name,
result: false
};
}
}
async initSocial(action) {
try {
debug('开始初始化社交媒体', {
action: action
});
const pro = [];
const tasks = action === 'do' ? this.undoneTasks : this.socialTasks;
if (tasks.discord) {
const hasDiscord = Object.values(tasks.discord).reduce(((total, arr) => [ ...total, ...arr ])).length > 0;
debug('检查 Discord 任务', {
hasDiscord: hasDiscord
});
if (hasDiscord && (!this.socialInitialized.discord || !this.social.discord)) {
debug('初始化 Discord');
this.social.discord = new Discord;
pro.push(this.#bind('discord', this.social.discord.init(action)));
}
}
if (tasks.reddit) {
const hasReddit = Object.values(tasks.reddit).reduce(((total, arr) => [ ...total, ...arr ])).length > 0;
debug('检查 Reddit 任务', {
hasReddit: hasReddit
});
if (hasReddit && (!this.socialInitialized.reddit || !this.social.reddit)) {
debug('初始化 Reddit');
this.social.reddit = new Reddit;
pro.push(this.#bind('reddit', this.social.reddit.init()));
}
}
if (tasks.twitch) {
const hasTwitch = Object.values(tasks.twitch).reduce(((total, arr) => [ ...total, ...arr ])).length > 0;
debug('检查 Twitch 任务', {
hasTwitch: hasTwitch
});
if (hasTwitch && (!this.socialInitialized.twitch || !this.social.twitch)) {
debug('初始化 Twitch');
this.social.twitch = new Twitch;
pro.push(this.#bind('twitch', this.social.twitch.init()));
}
}
if (tasks.twitter) {
const hasTwitter = Object.values(tasks.twitter).reduce(((total, arr) => [ ...total, ...arr ])).length > 0;
debug('检查 Twitter 任务', {
hasTwitter: hasTwitter
});
if (hasTwitter && (!this.socialInitialized.twitter || !this.social.twitter)) {
debug('初始化 Twitter');
this.social.twitter = new Twitter;
pro.push(this.#bind('twitter', this.social.twitter.init()));
}
}
if (tasks.vk) {
const hasVk = Object.values(tasks.vk).reduce(((total, arr) => [ ...total, ...arr ])).length > 0;
debug('检查 VK 任务', {
hasVk: hasVk
});
if (hasVk && (!this.socialInitialized.vk || !this.social.vk)) {
debug('初始化 VK');
this.social.vk = new Vk;
pro.push(this.#bind('vk', this.social.vk.init()));
}
}
if (tasks.youtube) {
const hasYoutube = Object.values(tasks.youtube).reduce(((total, arr) => [ ...total, ...arr ])).length > 0;
debug('检查 YouTube 任务', {
hasYoutube: hasYoutube
});
if (hasYoutube && (!this.socialInitialized.youtube || !this.social.youtube)) {
debug('初始化 YouTube');
this.social.youtube = new Youtube;
pro.push(this.#bind('youtube', this.social.youtube.init()));
}
}
if (tasks.steam) {
const steamLength = Object.values(tasks.steam).reduce(((total, arr) => [ ...total, ...arr ])).length;
debug('检查 Steam 任务', {
steamLength: steamLength
});
if (steamLength > 0) {
if (!this.social.steam) {
debug('创建 Steam 实例');
this.social.steam = new Steam;
}
const steamCommunityLength = Object.keys(tasks.steam).map((type => [ 'groupLinks', 'officialGroupLinks', 'forumLinks', 'workshopLinks', 'workshopVoteLinks' ].includes(type) ? tasks.steam?.[type]?.length || 0 : 0)).reduce(((total, number) => total + number), 0);
debug('Steam 社区任务数量', {
steamCommunityLength: steamCommunityLength
});
if (steamLength - steamCommunityLength > 0) {
this.steamTaskType.steamStore = true;
if (!this.socialInitialized.steamStore) {
debug('初始化 Steam 商店');
pro.push(this.#bind('steamStore', this.social.steam.init('store')));
}
}
if (steamCommunityLength > 0) {
if (!this.socialInitialized.steamCommunity) {
this.steamTaskType.steamCommunity = true;
debug('初始化 Steam 社区');
pro.push(this.#bind('steamCommunity', this.social.steam.init('community')));
}
}
}
}
if (tasks.links && tasks.links.length > 0) {
debug('初始化链接访问', {
linksCount: tasks.links.length
});
this.social.visitLink = visitLink;
}
debug('等待所有社交媒体初始化完成');
return await Promise.all(pro).then((result => {
let checked = true;
for (const data of result) {
if (data.result) {
debug('社交媒体初始化成功', {
name: data.name
});
this.socialInitialized[data.name] = data.result;
} else {
debug('社交媒体初始化失败', {
name: data.name
});
checked = false;
}
}
debug('社交媒体初始化完成', {
allSuccess: checked
});
return checked;
}));
} catch (error) {
debug('初始化社交媒体失败', {
error: error
});
throwError(error, 'Website.initSocial');
return false;
}
}
uniqueTasks(allTasks) {
try {
debug('开始去重任务');
const result = {};
for (const [social, types] of Object.entries(allTasks)) {
debug('处理社交媒体任务', {
social: social
});
result[social] = {};
for (const [type, tasks] of Object.entries(types)) {
debug('处理任务类型', {
social: social,
type: type
});
result[social][type] = unique(tasks);
}
}
debug('任务去重完成');
return result;
} catch (error) {
debug('任务去重失败', {
error: error
});
throwError(error, 'Website.uniqueTasks');
return allTasks;
}
}
async toggleTask(action) {
try {
debug('开始切换任务状态', {
action: action
});
if (!this.initialized && !this.init()) {
debug('初始化失败');
return false;
}
if (!await this.classifyTask(action)) {
debug('任务分类失败');
return false;
}
debug('初始化社交媒体');
await this.initSocial(action);
const pro = [];
const doTask = action === 'do';
const tasks = doTask ? this.undoneTasks : this.socialTasks;
if (this.socialInitialized.discord === true && this.social.discord) {
debug('处理 Discord 任务');
pro.push(this.social.discord.toggle({
doTask: doTask,
...tasks.discord
}));
}
if (this.socialInitialized.reddit === true && this.social.reddit) {
debug('处理 Reddit 任务');
pro.push(this.social.reddit.toggle({
doTask: doTask,
...tasks.reddit
}));
}
if (this.socialInitialized.twitch === true && this.social.twitch) {
debug('处理 Twitch 任务');
pro.push(this.social.twitch.toggle({
doTask: doTask,
...tasks.twitch
}));
}
if (this.socialInitialized.twitter === true && this.social.twitter) {
debug('处理 Twitter 任务');
pro.push(this.social.twitter.toggle({
doTask: doTask,
...tasks.twitter
}));
}
if (this.socialInitialized.vk === true && this.social.vk) {
debug('处理 VK 任务');
pro.push(this.social.vk.toggle({
doTask: doTask,
...tasks.vk
}));
}
if (this.socialInitialized.youtube === true && this.social.youtube) {
debug('处理 YouTube 任务');
pro.push(this.social.youtube.toggle({
doTask: doTask,
...tasks.youtube
}));
}
if ((this.steamTaskType.steamCommunity ? this.socialInitialized.steamCommunity === true : true) && (this.steamTaskType.steamStore ? this.socialInitialized.steamStore === true : true) && this.social.steam) {
debug('处理 Steam 任务');
pro.push(this.social.steam.toggle({
doTask: doTask,
...tasks.steam
}));
}
if (this.social.visitLink && tasks.links && doTask) {
debug('处理链接任务', {
linksCount: tasks.links.length
});
for (const link of tasks.links) {
pro.push(this.social.visitLink(link));
}
}
if (doTask && tasks.extra && this.extraDoTask) {
const hasExtra = Object.values(tasks.extra).reduce(((total, arr) => [ ...total, ...arr ])).length > 0;
if (hasExtra) {
debug('处理额外任务');
pro.push(this.extraDoTask(tasks.extra));
}
}
debug('等待所有任务完成');
await Promise.all(pro);
debug('所有任务完成');
echoLog({}).success(I18n('allTasksComplete'));
return true;
} catch (error) {
debug('切换任务失败', {
error: error
});
throwError(error, 'Website.toggleTask');
return false;
}
}
async doTask() {
try {
debug('开始执行任务');
const result = await this.toggleTask('do');
debug('任务执行完成', {
success: result
});
return result;
} catch (error) {
debug('执行任务失败', {
error: error
});
throwError(error, 'Website.doTask');
return false;
}
}
async undoTask() {
try {
debug('开始撤销任务');
const result = await this.toggleTask('undo');
debug('任务撤销完成', {
success: result
});
return result;
} catch (error) {
debug('撤销任务失败', {
error: error
});
throwError(error, 'Website.undoTask');
return false;
}
}
}
const defaultTasksTemplate$6 = {
steam: {
groupLinks: [],
wishlistLinks: [],
curatorLinks: [],
followLinks: [],
playTimeLinks: []
},
discord: {
serverLinks: []
},
vk: {
nameLinks: []
},
youtube: {
channelLinks: []
},
extra: {
website: []
}
};
const defaultTasks$8 = JSON.stringify(defaultTasksTemplate$6);
class FreeAnyWhere extends Website {
name='FreeAnyWhere';
tasks=[];
socialTasks=JSON.parse(defaultTasks$8);
undoneTasks=JSON.parse(defaultTasks$8);
games;
buttons=[ 'doTask', 'undoTask', 'verifyTask', 'getKey' ];
static test() {
const isMatch = window.location.host === 'freeanywhere.net';
debug('检查网站匹配', {
host: window.location.host,
isMatch: isMatch
});
return isMatch;
}
async init() {
try {
debug('初始化 FreeAnyWhere', {
url: window.location.href
});
const logStatus = echoLog({
text: I18n('initing')
});
debug('检测登录状态');
if ($('div.header__login a[href*=logout]').length === 0) {
debug('未登录,准备跳转到登录页面');
window.open('https://freeanywhere.net/game.php?steam_login', '_self');
logStatus.warning(I18n('needLogin'));
return false;
}
debug('检测是否为登录页面');
if (window.location.href.includes('/login')) {
logStatus.warning(I18n('needLogin'));
return false;
}
if (!await this.#checkLeftKey()) {
debug('检查剩余密钥失败');
echoLog({}).warning(I18n('checkLeftKeyFailed'));
}
const giveawayIdSuccess = this.#getGiveawayId();
debug('获取抽奖ID结果', {
success: giveawayIdSuccess,
id: this.giveawayId
});
this.initialized = true;
logStatus.success();
return true;
} catch (error) {
debug('初始化失败', {
error: error
});
throwError(error, 'Freeanywhere.init');
return false;
}
}
async classifyTask(action) {
try {
debug('开始分类任务', {
action: action
});
const logStatus = echoLog({
text: I18n('getTasksInfo')
});
if (action === 'undo') {
debug('获取已保存的任务信息');
this.socialTasks = GM_getValue(`fawTasks-${this.giveawayId}`)?.tasks || JSON.parse(defaultTasks$8);
}
const tasks = $('div.game__content-tasks__task').map(((index, element) => ({
id: $(element).attr('data-id'),
social: $(element).find('div.task-img img').attr('alt'),
link: $(element).find('div.task-link a').attr('href'),
title: $(element).find('div.task-link').text().trim(),
type: $(element).attr('data-type'),
data: $(element).attr('data-data'),
isSuccess: $(element).hasClass('done')
}))).toArray();
debug('获取到的任务列表', {
tasksCount: tasks.length,
tasks: tasks
});
if (tasks.length === 0) {
logStatus.success();
return false;
}
if (action === 'verify') {
this.tasks = [];
}
for (const task of tasks) {
await this.#processTask(task, action);
}
logStatus.success();
this.undoneTasks = this.uniqueTasks(this.undoneTasks);
this.socialTasks = this.uniqueTasks(this.socialTasks);
debug('任务分类结果', {
undoneTasks: this.undoneTasks,
socialTasks: this.socialTasks
});
GM_setValue(`fawTasks-${this.giveawayId}`, {
tasks: this.socialTasks,
time: (new Date).getTime()
});
return true;
} catch (error) {
debug('任务分类失败', {
error: error
});
throwError(error, 'Freeanywhere.classifyTask');
return false;
}
}
async #processTask(task, action) {
try {
debug('处理任务', {
task: task,
action: action
});
const {id: id, social: social, title: title, type: type, link: link, data: data, isSuccess: isSuccess} = task;
const taskInfo = {
id: id,
title: title,
social: social,
type: type,
data: data
};
if (action === 'verify' && !isSuccess) {
debug('添加到验证任务列表', taskInfo);
this.tasks.push(taskInfo);
return;
}
debug('处理特定类型任务', {
type: type,
action: action,
isSuccess: isSuccess
});
switch (type) {
case 'steam_account_verify':
case 'site_email_verify':
debug('跳过任务', {
type: type
});
break;
case 'steam_game_sub':
if (action === 'undo' && link) {
this.socialTasks.steam.followLinks.push(link);
}
if (action === 'do' && !isSuccess && link) {
this.undoneTasks.steam.followLinks.push(link);
}
break;
case 'steam_game_wishlist':
if (action === 'undo' && link) {
this.socialTasks.steam.wishlistLinks.push(link);
}
if (action === 'do' && !isSuccess && link) {
this.undoneTasks.steam.wishlistLinks.push(link);
}
break;
case 'steam_group_sub':
if (action === 'undo' && link) {
this.socialTasks.steam.groupLinks.push(link);
}
if (action === 'do' && !isSuccess && link) {
this.undoneTasks.steam.groupLinks.push(link);
}
break;
case 'site_visit':
if (action === 'do' && !isSuccess) {
this.undoneTasks.extra.website.push(`id=${id}&type=${type}&task=true`);
}
break;
case 'vk_community_sub':
if (action === 'undo' && link) {
this.socialTasks.vk.nameLinks.push(link);
}
if (action === 'do' && !isSuccess && link) {
this.undoneTasks.vk.nameLinks.push(link);
}
break;
case 'vk_post_like':
if (action === 'undo' && link) {
this.socialTasks.vk.nameLinks.push(`${link}&action=like`);
}
if (action === 'do' && !isSuccess && link) {
this.undoneTasks.vk.nameLinks.push(`${link}&action=like`);
}
break;
case 'discord_server_sub':
if (action === 'undo' && link) {
this.socialTasks.discord.serverLinks.push(link);
}
if (action === 'do' && !isSuccess && link) {
this.undoneTasks.discord.serverLinks.push(link);
}
break;
case 'youtube_channel_sub':
if (action === 'undo' && link) {
this.socialTasks.youtube.channelLinks.push(link);
}
if (action === 'do' && !isSuccess && link) {
this.undoneTasks.youtube.channelLinks.push(link);
}
break;
case 'steam_game_playtime':
if (action === 'undo' && link) {
this.socialTasks.steam.playTimeLinks.push(`${title.match(/(\d+)\s*min/)?.[1] || '0'}-${link}`);
}
if (action === 'do' && !isSuccess && link) {
this.undoneTasks.steam.playTimeLinks.push(`${title.match(/(\d+)\s*min/)?.[1] || '0'}-${link}`);
}
break;
case 'telegram_channel_sub':
debug('跳过 Telegram 任务');
echoLog({}).warning(`${I18n('tgTaskNotice')}`);
break;
case 'none':
debug('跳过未连接的任务', {
type: type
});
echoLog({}).warning(`${I18n('notConnect', type)}`);
break;
default:
debug('未知任务类型', {
type: type
});
echoLog({}).warning(`${I18n('unKnownTaskType', type)}`);
break;
}
} catch (error) {
debug('处理任务失败', {
error: error
});
throwError(error, 'FreeAnyWhere.processTask');
}
}
async verifyTask() {
try {
debug('开始验证任务');
if (!this.initialized && !await this.init()) {
debug('未初始化');
return false;
}
if (this.tasks.length === 0 && !await this.classifyTask('verify')) {
debug('任务列表为空', this.tasks);
return false;
}
debug('开始验证任务列表', {
tasks: this.tasks
});
const pro = [];
for (const task of this.tasks) {
pro.push(this.#verify(task));
await delay(1e3);
}
const result = await Promise.allSettled(pro);
debug('任务验证结果', {
result: result
});
echoLog({}).success(I18n('allTasksComplete'));
if (result.every((item => item.status === 'fulfilled' && item.value === true))) {
return !!await this.getKey(true);
}
return false;
} catch (error) {
debug('验证任务失败', {
error: error
});
throwError(error, 'Freeanywhere.verifyTask');
return false;
}
}
async extraDoTask({website: website}) {
try {
debug('执行额外任务', {
website: website
});
const promises = website.map((link => this.#doVisitWebsite(link)));
const results = await Promise.allSettled(promises);
debug('额外任务执行结果', {
results: results
});
return true;
} catch (error) {
debug('执行额外任务失败', {
error: error
});
throwError(error, 'FreeAnyWhere.extraDoTask');
return false;
}
}
async #doVisitWebsite(link) {
try {
debug('访问网站', {
link: link
});
const logStatus = echoLog({
text: I18n('visitingLink')
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://freeanywhere.net/php/task_site_visit_done.php',
method: 'POST',
headers: {
'content-type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: link
});
if (result !== 'Success') {
debug('访问失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.responseText.indexOf('bad') !== -1 || data?.responseText.length > 50) {
debug('访问响应异常', {
responseText: data?.responseText
});
logStatus.error(data?.responseText);
return false;
}
debug('访问成功');
logStatus.success();
return true;
} catch (error) {
debug('访问网站失败', {
error: error
});
throwError(error, 'FreeAnyWhere.doVisitWebsite');
return false;
}
}
async getKey(initialized) {
try {
debug('开始获取密钥', {
initialized: initialized
});
if (!initialized && !this.initialized && !await this.init()) {
debug('未初始化');
return false;
}
const logStatus = echoLog({
text: I18n('gettingKey')
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://freeanywhere.net/php/user_get_key.php',
method: 'POST'
});
if (result !== 'Success') {
debug('获取密钥失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.responseText.indexOf('bad') !== -1 || data?.responseText.length > 50) {
debug('密钥响应异常', {
responseText: data?.responseText
});
logStatus.error(data?.responseText);
return false;
}
debug('获取密钥成功', {
key: data.responseText
});
logStatus.success();
echoLog({}).success(data.responseText);
return data.responseText;
} catch (error) {
debug('获取密钥失败', {
error: error
});
throwError(error, 'FreeAnyWhere.getKey');
return false;
}
}
async #verify(task) {
try {
if ($('.task-check-extension').length > 0) {
return this.#verifyWithExtension(task);
}
return this.#verifyWithoutExtension(task);
} catch (error) {
debug('验证任务失败', {
error: error
});
throwError(error, 'Freeanywhere.verify');
return false;
}
}
async #verifyWithExtension(task) {
try {
await this.#updateUserData();
debug('验证任务', {
task: task
});
const logStatus = echoLog({
text: `${I18n('verifyingTask')}${task.title.trim()}...`
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://freeanywhere.net/php/extension/user_task_update.php',
method: 'POST',
headers: {
'content-type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: `id=${task.id}&type=${task.type}${task.data && task.data !== 'none' ? `&data=${task.data}` : ''}`
});
if (result !== 'Success' || !data?.responseText) {
debug('验证请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
const response = data.responseText.trim();
if (response !== 'good') {
debug('验证响应异常', {
response: response,
statusText: data?.statusText,
status: data?.status
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('验证成功');
logStatus.success();
return true;
} catch (error) {
debug('验证任务失败', {
error: error
});
throwError(error, 'Freeanywhere.verifyWithExtension');
return false;
}
}
async #verifyWithoutExtension(task) {
try {
debug('验证任务', {
task: task
});
const logStatus = echoLog({
text: `${I18n('verifyingTask')}${task.title.trim()}...`
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://freeanywhere.net/php/user_task_update.php',
method: 'POST',
headers: {
'content-type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: `id=${task.id}&type=${task.type}${task.data && task.data !== 'none' ? `&data=${task.data}` : ''}`
});
if (result !== 'Success' || !data?.responseText) {
debug('验证请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
const response = data.responseText.trim();
if (response !== 'good') {
debug('验证响应异常', {
response: response,
statusText: data?.statusText,
status: data?.status
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('验证成功');
logStatus.success();
return true;
} catch (error) {
debug('验证任务失败', {
error: error
});
throwError(error, 'Freeanywhere.verifyWithoutExtension');
return false;
}
}
async #updateUserData() {
try {
let postData = '';
const userData = GM_getValue('FAW_STORAGE') || {};
if (Object.keys(userData).length === 0 || !userData.tasks || !userData.user || !userData.games || !userData.settings) {
if (!this.games) {
await this.#userGamesGet();
}
if (!this.games) {
debug('获取用户游戏失败');
return false;
}
postData = `extension=${encodeURIComponent(JSON.stringify({
games: this.games,
settings: {
game_update: Math.floor(Date.now() / 1e3)
},
tasks: {},
user: {
avatar: $('header.games_for_farm_site').attr('data-avatar'),
lang: $('header.games_for_farm_site').attr('data-lang'),
name: $('header.games_for_farm_site').attr('data-name'),
steam: $('header.games_for_farm_site').attr('data-steam')
}
}))}`;
} else {
postData = `extension=${encodeURIComponent(JSON.stringify(userData))}`;
}
debug('更新用户数据');
const logStatus = echoLog({
text: `${I18n('updatingUserData')}`
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://freeanywhere.net/php/extension/user_data_update.php',
method: 'POST',
headers: {
'content-type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: postData
});
if (data?.status !== 200) {
debug('验证请求失败', {
result: result,
statusText: statusText,
status: status,
data: data
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
debug('验证成功');
logStatus.success();
return true;
} catch (error) {
debug('验证任务失败', {
error: error
});
throwError(error, 'Freeanywhere.updateUserData');
return false;
}
}
async #userGamesGet() {
try {
debug('获取用户游戏');
const logStatus = echoLog({
text: `${I18n('gettingUserGames')}`
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://freeanywhere.net/php/extension/user_games_get.php',
method: 'POST',
headers: {
'content-type': 'application/x-www-form-urlencoded; charset=UTF-8'
},
data: `steam=${$('header.games_for_farm_site').attr('data-steam')}`,
dataType: 'json'
});
if (result !== 'Success' || data?.status !== 200 || !data?.responseText) {
debug('验证请求失败', {
result: result,
statusText: statusText,
status: status,
data: data
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
debug('验证成功');
this.games = data.response;
logStatus.success();
return true;
} catch (error) {
debug('验证任务失败', {
error: error
});
throwError(error, 'Freeanywhere.userGamesGet');
return false;
}
}
async #checkLeftKey() {
try {
debug('检查剩余密钥');
if (!globalOptions.other.checkLeftKey) {
debug('跳过密钥检查');
return true;
}
const giveawayStatus = $('div.card-info__lable-info').text()?.includes('Giveaway ended');
debug('Giveaway状态', {
giveawayStatus: giveawayStatus
});
if (!giveawayStatus) {
return true;
}
debug('没有剩余密钥,显示确认对话框');
const {value: value} = await Swal.fire({
icon: 'warning',
title: I18n('notice'),
text: I18n('giveawayEnded'),
confirmButtonText: I18n('confirm'),
cancelButtonText: I18n('cancel'),
showCancelButton: true
});
if (value) {
debug('用户确认关闭窗口');
window.close();
}
return true;
} catch (error) {
debug('检查剩余密钥失败', {
error: error
});
throwError(error, 'FreeAnyWhere.checkLeftKey');
return false;
}
}
#getGiveawayId() {
try {
debug('开始获取抽奖ID');
const giveawayId = $('link[rel="canonical"]').attr('href')?.match(/n=([\d]+)/)?.[1];
if (giveawayId) {
this.giveawayId = giveawayId;
debug('获取抽奖ID成功', {
giveawayId: giveawayId
});
return true;
}
debug('获取抽奖ID失败');
echoLog({}).error(I18n('getFailed', 'GiveawayId'));
return false;
} catch (error) {
debug('获取抽奖ID出错', {
error: error
});
throwError(error, 'FreeAnyWhere.getGiveawayId');
return false;
}
}
}
const defaultTasks$7 = {
steam: {
groupLinks: [],
wishlistLinks: [],
curatorLinks: [],
curatorLikeLinks: [],
followLinks: [],
forumLinks: [],
announcementLinks: [],
workshopVoteLinks: [],
playtestLinks: [],
playTimeLinks: []
},
discord: {
serverLinks: []
},
vk: {
nameLinks: []
},
twitch: {
channelLinks: []
},
reddit: {
redditLinks: []
},
youtube: {
channelLinks: [],
likeLinks: []
},
twitter: {
userLinks: [],
retweetLinks: []
}
};
class GiveawaySu extends Website {
name='GiveawaySu';
socialTasks=defaultTasks$7;
undoneTasks=defaultTasks$7;
buttons=[ 'doTask', 'undoTask' ];
static test() {
const url = window.location.href;
const isMatch = /^https?:\/\/giveaway\.su\/giveaway\/view\/[\d]+/.test(url);
debug('检查网站匹配', {
url: url,
isMatch: isMatch
});
return isMatch;
}
async after() {
try {
debug('开始执行后续操作');
if (!this.#checkLogin()) {
debug('登录检查失败');
echoLog({}).warning(I18n('checkLoginFailed'));
}
if (!await this.#checkLeftKey()) {
debug('检查剩余密钥失败');
echoLog({}).warning(I18n('checkLeftKeyFailed'));
}
debug('显示网站通知');
echoLog({}).warning(I18n('gsNotice'));
} catch (error) {
debug('后续操作失败', {
error: error
});
throwError(error, 'Giveawaysu.after');
}
}
init() {
try {
debug('初始化 GiveawaySu');
const logStatus = echoLog({
text: I18n('initing')
});
if ($('a.steam-login').length > 0) {
debug('发现未登录状态,重定向到 Steam 登录');
window.open('/steam/redirect', '_self');
logStatus.warning(I18n('needLogin'));
return false;
}
const giveawayIdResult = this.#getGiveawayId();
if (!giveawayIdResult) {
debug('获取抽奖ID失败');
return false;
}
this.initialized = true;
debug('初始化完成');
logStatus.success();
return true;
} catch (error) {
debug('初始化失败', {
error: error
});
throwError(error, 'Giveawaysu.init');
return false;
}
}
async classifyTask(action) {
try {
debug('开始分类任务', {
action: action
});
const logStatus = echoLog({
text: I18n('getTasksInfo')
});
if (action === 'undo') {
debug('恢复已保存的任务信息');
this.socialTasks = GM_getValue(`gasTasks-${this.giveawayId}`)?.tasks || defaultTasks$7;
return true;
}
const tasks = $('#actions tr');
if (!tasks.length) {
debug('未找到任务');
logStatus.warning(I18n('noTasks'));
return true;
}
debug('检查并处理 Discord 和 Twitch 绑定');
if ($('div.bind-discord').is(':visible')) {
debug('点击 Discord 绑定按钮');
$('div.bind-discord a')[0]?.click();
}
if ($('div.bind-twitch').is(':visible')) {
debug('点击 Twitch 绑定按钮');
$('div.bind-twitch a')[0]?.click();
}
const processTask = async task => {
const td = $(task).find('td:not(".hidden")');
const colorfulTask = td.eq(1).find('a:not([data-trigger="link"])');
const colorlessTask = td.eq(2).find('a:not([data-trigger="link"])');
const taskDes = colorfulTask.length > 0 ? colorfulTask : colorlessTask;
if (!taskDes.length) {
debug('跳过无效任务');
return true;
}
const taskIcon = td.eq(0).find('i').attr('class') || '';
const taskName = taskDes.text().trim();
const taskHref = taskDes.attr('href');
debug('处理任务', {
taskIcon: taskIcon,
taskName: taskName,
taskHref: taskHref
});
if (taskIcon.includes('ban') || /disable adblock/gi.test(taskName)) {
debug('跳过禁用任务');
return true;
}
if (!taskHref) {
debug('任务链接为空');
return false;
}
try {
debug('获取重定向链接');
const taskLink = await getRedirectLink(taskHref);
if (!taskLink) {
debug('获取重定向链接失败');
return false;
}
debug('分类任务', {
taskLink: taskLink,
taskIcon: taskIcon,
taskName: taskName
});
this.#classifyTaskByType(taskLink, taskIcon, taskName);
return true;
} catch (error) {
debug('获取重定向链接失败', {
error: error
});
throwError(error, 'Giveawaysu.classifyTask->getRedirectLink');
return false;
}
};
debug('开始处理所有任务');
const results = await Promise.all(Array.from(tasks).map(processTask));
const success = results.some((result => result));
if (!success) {
debug('所有任务处理失败');
logStatus.error(I18n('allTasksFailed'));
return false;
}
debug('任务处理完成');
logStatus.success();
this.undoneTasks = this.uniqueTasks(this.undoneTasks);
this.socialTasks = this.undoneTasks;
debug('保存任务信息');
GM_setValue(`gasTasks-${this.giveawayId}`, {
tasks: this.socialTasks,
time: (new Date).getTime()
});
return true;
} catch (error) {
debug('任务分类失败', {
error: error
});
throwError(error, 'Giveawaysu.classifyTask');
return false;
}
}
static TASK_PATTERNS={
wishlist: /wishlist.*game|add.*wishlist/gim,
follow: /follow.*button/gim,
twitter: /(on twitter)|(Follow.*on.*Facebook)/gim,
vkGroup: /join.*vk.*group/gim,
youtubeVideo: /(watch|like).*video/gim,
youtubeChannel: /subscribe.*youtube.*channel/gim,
watchArt: /watch.*art/gim,
reddit: /subscribe.*subreddit|follow.*reddit/gim,
twitchChannel: /follow.*twitch.*channel/gim,
instagram: /follow.*instagram/gim,
discord: /join.*discord/gim,
playtest: /request.*playtest/gim,
steamForum: /subscribe.*steam.*forum/gim,
curator: /(follow|subscribe).*curator/gim,
curatorLink: /^https?:\/\/store\.steampowered\.com\/curator\//,
announcement: /like.*announcement/gim,
steamGroup: /join/gi
};
#classifyTaskByType(taskLink, taskIcon, taskName) {
try {
debug('开始分类任务', {
taskLink: taskLink,
taskIcon: taskIcon,
taskName: taskName
});
const {TASK_PATTERNS: TASK_PATTERNS} = GiveawaySu;
if (taskIcon.includes('steam') && TASK_PATTERNS.steamGroup.test(taskName)) {
debug('添加 Steam 组任务');
this.undoneTasks.steam.groupLinks.push(taskLink);
return;
}
if (TASK_PATTERNS.announcement.test(taskName)) {
debug('添加 Steam 公告任务');
this.undoneTasks.steam.announcementLinks.push(taskLink);
return;
}
if (TASK_PATTERNS.curator.test(taskName) && TASK_PATTERNS.curatorLink.test(taskLink)) {
debug('添加 Steam 鉴赏家关注任务');
this.undoneTasks.steam.curatorLinks.push(taskLink);
return;
}
if (taskIcon.includes('steam') && /follow|subscribe/gim.test(taskName)) {
debug('添加 Steam 鉴赏家点赞任务');
this.undoneTasks.steam.curatorLikeLinks.push(taskLink);
return;
}
if (TASK_PATTERNS.steamForum.test(taskName)) {
debug('添加 Steam 论坛任务');
this.undoneTasks.steam.forumLinks.push(taskLink);
return;
}
if (taskIcon.includes('thumbs-up') && /^https?:\/\/steamcommunity\.com\/sharedfiles\/filedetails\/\?id=[\d]+/.test(taskLink)) {
debug('添加 Steam 创意工坊投票任务');
this.undoneTasks.steam.workshopVoteLinks.push(taskLink);
return;
}
if (taskIcon.includes('plus') && TASK_PATTERNS.playtest.test(taskName)) {
debug('添加 Steam 游戏测试任务');
this.undoneTasks.steam.playtestLinks.push(taskLink);
return;
}
if (taskIcon.includes('discord') || TASK_PATTERNS.discord.test(taskName)) {
debug('添加 Discord 服务器任务');
this.undoneTasks.discord.serverLinks.push(taskLink);
return;
}
if (taskIcon.includes('instagram') || TASK_PATTERNS.instagram.test(taskName)) {
debug('跳过 Instagram 任务');
return;
}
if (taskIcon.includes('twitch') || TASK_PATTERNS.twitchChannel.test(taskName)) {
debug('添加 Twitch 频道任务');
this.undoneTasks.twitch.channelLinks.push(taskLink);
return;
}
if (taskIcon.includes('reddit') || TASK_PATTERNS.reddit.test(taskName)) {
debug('添加 Reddit 任务');
this.undoneTasks.reddit.redditLinks.push(taskLink);
return;
}
if (TASK_PATTERNS.watchArt.test(taskName)) {
debug('添加创意工坊物品任务');
this.undoneTasks.steam.workshopVoteLinks.push(taskLink);
return;
}
if (TASK_PATTERNS.youtubeChannel.test(taskName)) {
debug('添加 YouTube 频道任务');
this.undoneTasks.youtube.channelLinks.push(taskLink);
return;
}
if (TASK_PATTERNS.youtubeVideo.test(taskName) || (taskIcon.includes('youtube') || taskIcon.includes('thumbs-up')) && TASK_PATTERNS.youtubeVideo.test(taskName)) {
debug('添加 YouTube 视频任务');
this.undoneTasks.youtube.likeLinks.push(taskLink);
return;
}
if (taskIcon.includes('vk') || TASK_PATTERNS.vkGroup.test(taskName)) {
debug('添加 VK 任务');
this.undoneTasks.vk.nameLinks.push(taskLink);
return;
}
if (TASK_PATTERNS.twitter.test(taskName)) {
debug('跳过 Twitter 任务');
return;
}
if (TASK_PATTERNS.wishlist.test(taskName)) {
debug('添加 Steam 愿望单任务');
this.undoneTasks.steam.wishlistLinks.push(taskLink);
}
if (TASK_PATTERNS.follow.test(taskName)) {
debug('添加 Steam 关注任务');
this.undoneTasks.steam.followLinks.push(taskLink);
return;
}
debug('未识别的任务类型', {
taskLink: taskLink,
taskIcon: taskIcon,
taskName: taskName
});
} catch (error) {
debug('任务分类失败', {
error: error
});
throwError(error, 'Giveawaysu.classifyTaskByType');
}
}
#checkLogin() {
try {
debug('检查登录状态');
if (!globalOptions.other.checkLogin) {
debug('跳过登录检查');
return true;
}
const needLogin = $('a.steam-login').length > 0;
if (needLogin) {
debug('未登录,重定向到 Steam 登录');
window.open('/steam/redirect', '_self');
}
debug('登录检查完成', {
needLogin: needLogin
});
return true;
} catch (error) {
debug('登录检查失败', {
error: error
});
throwError(error, 'Giveawaysu.checkLogin');
return false;
}
}
async #checkLeftKey() {
try {
debug('检查剩余密钥');
if (!globalOptions.other.checkLeftKey) {
debug('跳过密钥检查');
return true;
}
const isEnded = $('.giveaway-ended').length > 0;
const hasNoKeys = $('.giveaway-key').length === 0;
debug('检查抽奖状态', {
isEnded: isEnded,
hasNoKeys: hasNoKeys
});
if (!(isEnded && hasNoKeys)) {
return true;
}
debug('没有剩余密钥,显示确认对话框');
const {value: value} = await Swal.fire({
icon: 'warning',
title: I18n('notice'),
text: I18n('noKeysLeft'),
confirmButtonText: I18n('confirm'),
cancelButtonText: I18n('cancel'),
showCancelButton: true
});
if (value) {
debug('用户确认关闭窗口');
window.close();
}
return true;
} catch (error) {
debug('检查剩余密钥失败', {
error: error
});
throwError(error, 'Giveawaysu.checkLeftKey');
return false;
}
}
#getGiveawayId() {
try {
debug('从URL获取抽奖ID');
const giveawayId = window.location.href.match(/\/view\/([\d]+)/)?.[1];
if (giveawayId) {
this.giveawayId = giveawayId;
debug('获取抽奖ID成功', {
giveawayId: giveawayId
});
return true;
}
debug('获取抽奖ID失败');
echoLog({
text: I18n('getFailed', 'GiveawayId')
});
return false;
} catch (error) {
debug('获取抽奖ID出错', {
error: error
});
throwError(error, 'Giveawaysu.getGiveawayId');
return false;
}
}
}
class Indiedb {
name='Indiedb';
buttons=[ 'doTask' ];
static test() {
const {host: host} = window.location;
const isMatch = host === 'www.indiedb.com';
debug('检查网站匹配', {
host: host,
isMatch: isMatch
});
return isMatch;
}
async after() {
try {
debug('开始执行后续操作');
if (!this.#checkLogin()) {
debug('检查登录失败');
echoLog({}).warning(I18n('checkLoginFailed'));
}
if (!await this.#checkLeftKey()) {
debug('检查剩余密钥失败');
echoLog({}).warning(I18n('checkLeftKeyFailed'));
}
} catch (error) {
debug('后续操作失败', {
error: error
});
throwError(error, 'Indiedb.after');
}
}
async doTask() {
try {
debug('开始执行任务');
if (!await this.#join()) {
debug('加入抽奖失败');
return false;
}
return await this.#do();
} catch (error) {
debug('执行任务失败', {
error: error
});
throwError(error, 'Indiedb.doTask');
return false;
}
}
async #join() {
try {
debug('开始加入抽奖');
if ($('a.buttonenter:contains(Register to join)').length > 0) {
debug('需要登录');
echoLog({}).error(I18n('needLogin'));
return false;
}
const currentoption = $('a.buttonenter.buttongiveaway');
const buttonText = currentoption.text();
debug('检查按钮状态', {
buttonText: buttonText
});
if (/success/gim.test($('a.buttonenter.buttongiveaway').text())) {
debug('已成功加入抽奖');
return true;
}
if (!/join giveaway/gim.test(buttonText)) {
debug('需要加入抽奖');
echoLog({}).warning(I18n('needJoinGiveaway'));
return false;
}
const logStatus = echoLog({
text: `${I18n('joiningGiveaway')}...`
});
debug('发送加入请求');
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: currentoption.attr('href'),
method: 'POST',
data: 'ajax=t',
dataType: 'json',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
Accept: 'application/json, text/javascript, */*; q=0.01',
Origin: window.location.origin,
referer: window.location.href
}
});
if (result !== 'Success') {
debug('请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('尝试备用加入方法');
if (await this.#join2()) {
debug('备用加入方法成功');
logStatus.success('Success');
return true;
}
debug('加入失败', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
if (!data.response?.success) {
debug('响应失败', {
text: data.response?.text
});
logStatus.error(`Error${data.response?.text ? `:${data.response?.text}` : ''}`);
return false;
}
debug('加入成功');
currentoption.addClass('buttonentered').text('Success - Giveaway joined');
$('#giveawaysjoined').slideDown();
$('#giveawaysrecommend').slideDown();
logStatus.success(`Success${data.response?.text ? `:${data.response?.text}` : ''}`);
return true;
} catch (error) {
debug('加入抽奖失败', {
error: error
});
throwError(error, 'Indiedb.join');
return false;
}
}
async #join2() {
try {
debug('开始备用加入方法');
return await new Promise((resolve => {
const targetNode = document.getElementById('giveawaysjoined');
const config = {
attributes: true
};
const observer = new MutationObserver((() => {
if ($('#giveawaysjoined').is(':visible')) {
debug('检测到加入成功');
resolve(true);
observer.disconnect();
}
}));
observer.observe(targetNode, config);
debug('点击加入按钮');
$('a.buttonenter.buttongiveaway')[0]?.click();
setTimeout((() => {
debug('加入超时');
resolve(false);
observer.disconnect();
}), 3e4);
}));
} catch (error) {
debug('备用加入方法失败', {
error: error
});
throwError(error, 'Indiedb.join2');
return false;
}
}
async #do() {
try {
debug('开始执行任务');
const id = $('script').map(((index, script) => {
if (!/\$\(document\)/gim.test(script.innerHTML)) {
return null;
}
return [ script.innerHTML.match(/"\/[\d]+"/gim)?.[0]?.match(/[\d]+/)?.[0], script.innerHTML.match(/"\/newsletter\/ajax\/subscribeprofile\/optin\/[\d]+"/gim)?.[0]?.match(/[\d]+/)?.[0] ];
}));
if (id.length < 2) {
debug('获取任务ID失败');
echoLog({}).error(I18n('getFailed', 'TaskId'));
return false;
}
const pro = [];
const tasks = $('#giveawaysjoined a[class*=promo]');
debug('找到任务', {
count: tasks.length
});
for (const task of tasks) {
const promo = $(task);
if (promo.hasClass('buttonentered')) {
debug('跳过已完成任务');
continue;
}
const taskText = promo.parents('p').text();
debug('处理任务', {
taskText: taskText
});
const status = echoLog({
text: `${I18n('doing')}:${taskText}...`
});
if (/the-challenge-of-adblock/gim.test(promo.attr('href'))) {
debug('跳过未知任务类型');
status.error(`Error:${I18n('unKnownTaskType')}`);
continue;
}
if (/facebookpromo|twitterpromo|visitpromo/gim.test(task.className)) {
let text = '';
if (promo.hasClass('facebookpromo')) {
text = 'facebookpromo';
} else if (promo.hasClass('twitterpromo')) {
text = 'twitterpromo';
} else {
text = 'visitpromo';
}
debug('处理社交媒体任务', {
type: text
});
pro.push(this.#handleSocialPromo(text, id[0], status, promo));
} else if (promo.hasClass('emailoptinpromo')) {
debug('处理邮件订阅任务');
pro.push(this.#handleEmailPromo(id[1], status, promo));
} else if (promo.hasClass('watchingpromo')) {
debug('处理关注任务');
pro.push(this.#handleWatchingPromo(promo, status));
} else {
debug('处理默认任务');
pro.push(this.#handleDefaultPromo(promo, status));
}
}
await Promise.all(pro);
debug('所有任务完成');
echoLog({}).success(I18n('allTasksComplete'));
return true;
} catch (error) {
debug('执行任务失败', {
error: error
});
throwError(error, 'Indiedb.do');
return false;
}
}
async #handleSocialPromo(text, id, status, promo) {
try {
debug('处理社交媒体任务', {
text: text,
id: id
});
return await new Promise((resolve => {
$.ajax({
type: 'POST',
url: urlPath(`/giveaways/ajax/${text}/${id}`),
timeout: 6e4,
dataType: 'json',
data: {
ajax: 't'
},
error(response, error, exception) {
debug('请求失败', {
response: response,
error: error,
exception: exception
});
if (window.DEBUG) {
console.log('%cAuto-Task[Debug]:', 'color:red', {
response: response,
error: error,
exception: exception
});
}
status.error('Error:An error has occurred performing the action requested. Please try again shortly.');
resolve(true);
},
success(response) {
if (response.success) {
debug('任务完成', {
response: response
});
status.success(`Success:${response.text}`);
promo.addClass('buttonentered').closest('p').html(promo.closest('p').find('span').html());
} else {
debug('任务失败', {
response: response
});
status.error(`Error:${response.text}`);
}
resolve(true);
}
});
}));
} catch (error) {
debug('处理社交媒体任务失败', {
error: error
});
throwError(error, 'Indiedb.handleSocialPromo');
return false;
}
}
async #handleEmailPromo(id, status, promo) {
try {
debug('处理邮件订阅任务', {
id: id
});
return await new Promise((resolve => {
$.ajax({
type: 'POST',
url: urlPath(`/newsletter/ajax/subscribeprofile/optin/${id}`),
timeout: 6e4,
dataType: 'json',
data: {
ajax: 't',
emailsystoggle: 4
},
error(response, error, exception) {
debug('请求失败', {
response: response,
error: error,
exception: exception
});
if (window.DEBUG) {
console.log('%cAuto-Task[Debug]:', 'color:red', {
response: response,
error: error,
exception: exception
});
}
status.error('Error:An error has occurred performing the action requested. Please try again shortly.');
resolve(true);
},
success(response) {
if (response.success) {
debug('任务完成', {
response: response
});
status.success(`Success:${response.text}`);
promo.toggleClass('buttonentered').closest('p').html(promo.closest('p').find('span').html());
} else {
debug('任务失败', {
response: response
});
status.error(`Error:${response.text}`);
}
resolve(true);
}
});
}));
} catch (error) {
debug('处理邮件订阅任务失败', {
error: error
});
throwError(error, 'Indiedb.handleEmailPromo');
return false;
}
}
async #handleWatchingPromo(promo, status) {
try {
debug('处理关注任务');
return await new Promise((resolve => {
const href = promo.attr('href');
if (!href) {
debug('无效的链接');
status.error('Error: Invalid href');
resolve(true);
return;
}
const data = getUrlQuery(href);
data.ajax = 't';
const [baseUrl] = href.split(/[?#]/);
if (!baseUrl) {
debug('无效的URL');
status.error('Error: Invalid URL');
resolve(true);
return;
}
debug('发送请求', {
url: baseUrl,
data: data
});
$.ajax({
type: 'POST',
url: urlPath(baseUrl),
timeout: 6e4,
dataType: 'json',
data: data,
error(response, error, exception) {
debug('请求失败', {
response: response,
error: error,
exception: exception
});
if (window.DEBUG) {
console.log('%cAuto-Task[Debug]:', 'color:red', {
response: response,
error: error,
exception: exception
});
}
status.error('Error:An error has occurred performing the action requested. Please try again shortly.');
resolve(true);
},
success(response) {
if (response.success) {
debug('任务完成', {
response: response
});
status.success(`Success:${response.text}`);
promo.toggleClass('buttonentered').closest('p').html(promo.closest('p').find('span').html());
} else {
debug('任务失败', {
response: response
});
status.error(`Error:${response.text}`);
}
resolve(true);
}
});
}));
} catch (error) {
debug('处理关注任务失败', {
error: error
});
throwError(error, 'Indiedb.handleWatchingPromo');
return false;
}
}
async #handleDefaultPromo(promo, status) {
try {
debug('处理默认任务');
return await new Promise((resolve => {
const href = promo.attr('href');
if (!href) {
debug('无效的链接');
status.error('Error: Invalid href');
resolve(true);
return;
}
debug('发送请求', {
url: href
});
$.ajax({
type: 'POST',
url: urlPath(href),
timeout: 6e4,
dataType: 'json',
data: {
ajax: 't'
},
error(response, error, exception) {
debug('请求失败', {
response: response,
error: error,
exception: exception
});
if (window.DEBUG) {
console.log('%cAuto-Task[Debug]:', 'color:red', {
response: response,
error: error,
exception: exception
});
}
status.error('Error:An error has occurred performing the action requested. Please try again shortly.');
resolve(true);
},
success(response) {
if (response.success) {
debug('任务完成', {
response: response
});
status.success(`Success:${response.text}`);
promo.toggleClass('buttonentered').closest('p').html(promo.closest('p').find('span').html());
} else {
debug('任务失败', {
response: response
});
status.error(`Error:${response.text}`);
}
resolve(true);
}
});
}));
} catch (error) {
debug('处理默认任务失败', {
error: error
});
throwError(error, 'Indiedb.handleDefaultPromo');
return false;
}
}
#checkLogin() {
try {
debug('检查登录状态');
if (!globalOptions.other.checkLogin) {
debug('跳过登录检查');
return true;
}
if ($('a.buttonenter:contains(Register to join)').length > 0) {
debug('未登录,重定向到登录页面');
window.open('/members/login', '_self');
}
debug('登录检查完成');
return true;
} catch (error) {
debug('检查登录失败', {
error: error
});
throwError(error, 'Indiedb.checkLogin');
return false;
}
}
async #checkLeftKey() {
try {
debug('检查剩余密钥');
if (!globalOptions.other.checkLeftKey) {
debug('跳过密钥检查');
return true;
}
const hasEndedButton = $('a.buttonenter:contains("next time"), a.buttonenter:contains("Giveaway is closed")').length > 0;
debug('检查抽奖状态', {
hasEndedButton: hasEndedButton
});
if (!hasEndedButton) {
return true;
}
debug('抽奖已结束,显示确认对话框');
const {value: value} = await Swal.fire({
icon: 'warning',
title: I18n('notice'),
text: I18n('giveawayEnded'),
confirmButtonText: I18n('confirm'),
cancelButtonText: I18n('cancel'),
showCancelButton: true
});
if (value) {
debug('用户确认关闭窗口');
window.close();
}
return true;
} catch (error) {
debug('检查剩余密钥失败', {
error: error
});
throwError(error, 'Indiedb.checkLeftKey');
return false;
}
}
}
const defaultTasksTemplate$5 = {
steam: {
groupLinks: [],
officialGroupLinks: [],
wishlistLinks: [],
curatorLinks: []
},
discord: {
serverLinks: []
},
extra: {
videoTasks: []
},
links: []
};
const defaultTasks$6 = JSON.stringify(defaultTasksTemplate$5);
class Keyhub extends Website {
name='Keyhub';
socialTasks=JSON.parse(defaultTasks$6);
undoneTasks=JSON.parse(defaultTasks$6);
buttons=[ 'doTask', 'undoTask' ];
static test() {
const {host: host} = window.location;
const isMatch = host === 'key-hub.eu';
debug('检查网站匹配', {
host: host,
isMatch: isMatch
});
return isMatch;
}
async after() {
try {
debug('开始执行后续操作');
if (!this.#checkLogin()) {
debug('检查登录失败');
echoLog({}).warning(I18n('checkLoginFailed'));
}
if (!await this.#checkLeftKey()) {
debug('检查剩余密钥失败');
echoLog({}).warning(I18n('checkLeftKeyFailed'));
}
debug('隐藏 NSFW 内容');
$('.NSFW').hide();
} catch (error) {
debug('后续操作失败', {
error: error
});
throwError(error, 'Keyhub.after');
}
}
init() {
try {
debug('开始初始化');
const logStatus = echoLog({
text: I18n('initing')
});
if ($('a[href*="/connect/steam"]').length > 0) {
debug('需要登录 Steam');
window.open('/connect/steam', '_self');
logStatus.warning(I18n('needLogin'));
return false;
}
if (!this.#getGiveawayId()) {
debug('获取抽奖ID失败');
return false;
}
debug('隐藏 VPN 覆盖层');
$('#VPNoverlay').hide();
$('#mainArticleSection').show();
this.initialized = true;
debug('初始化完成');
logStatus.success();
return true;
} catch (error) {
debug('初始化失败', {
error: error
});
throwError(error, 'Keyhub.init');
return false;
}
}
async classifyTask(action) {
try {
debug('开始分类任务', {
action: action
});
const logStatus = echoLog({
text: I18n('getTasksInfo')
});
if (action === 'undo') {
debug('恢复已保存的任务信息');
this.socialTasks = GM_getValue(`khTasks-${this.giveawayId}`)?.tasks || JSON.parse(defaultTasks$6);
}
const tasks = $('.task:not(".googleads")').filter(((index, element) => action === 'do' ? $(element).find('i.fa-check-circle:visible').length === 0 : true)).find('a');
debug('找到任务', {
count: tasks.length
});
for (const task of tasks) {
let link = $(task).attr('href');
const taskDes = $(task).text().trim();
debug('处理任务', {
taskDes: taskDes,
link: link
});
if (!link) {
debug('跳过无链接任务');
continue;
}
if (/\/away\?data=/.test(link) || /steamcommunity\.com\/gid\//.test(link)) {
debug('获取重定向链接');
link = await getRedirectLink(link) || link;
}
if (/https?:\/\/key-hub\.eu\/connect\/discord/.test(link)) {
debug('处理 Discord 连接任务');
GM_openInTab(link, {
active: true
});
continue;
}
if (/steamcommunity\.com\/groups\//.test(link)) {
debug('处理 Steam 组任务');
if (action === 'undo') {
this.socialTasks.steam.groupLinks.push(link);
}
if (action === 'do') {
this.undoneTasks.steam.groupLinks.push(link);
}
continue;
}
if (/steamcommunity\.com\/games\/[\d]+/.test(link)) {
debug('处理 Steam 官方组任务');
if (action === 'undo') {
this.socialTasks.steam.officialGroupLinks.push(link);
}
if (action === 'do') {
this.undoneTasks.steam.officialGroupLinks.push(link);
}
continue;
}
if (/store\.steampowered\.com\/app\//.test(link) && /wishlist/gim.test(taskDes)) {
debug('处理 Steam 愿望单任务');
if (action === 'undo') {
this.socialTasks.steam.wishlistLinks.push(link);
}
if (action === 'do') {
this.undoneTasks.steam.wishlistLinks.push(link);
}
continue;
}
if (/store\.steampowered\.com\/curator\//.test(link)) {
debug('处理 Steam 鉴赏家任务');
if (action === 'undo') {
this.socialTasks.steam.curatorLinks.push(link);
}
if (action === 'do') {
this.undoneTasks.steam.curatorLinks.push(link);
}
continue;
}
if (/^https?:\/\/discord\.com\/invite\//.test(link)) {
debug('处理 Discord 服务器任务');
if (action === 'undo') {
this.socialTasks.discord.serverLinks.push(link);
}
if (action === 'do') {
this.undoneTasks.discord.serverLinks.push(link);
}
continue;
}
if (/^javascript:videoTask.+/.test(link)) {
debug('处理视频任务');
if (action === 'do') {
const taskData = link.match(/javascript:videoTask\('.+?','(.+?)'/)?.[1];
if (taskData) {
debug('添加视频任务', {
taskData: taskData
});
this.undoneTasks.extra.videoTasks.push(taskData);
}
}
continue;
}
if (this.#isSkippableLink(link)) {
debug('跳过可忽略的链接', {
link: link
});
continue;
}
debug('未知任务类型', {
taskDes: taskDes,
link: link
});
echoLog({}).warning(`${I18n('unKnownTaskType')}: ${taskDes}(${link})`);
}
debug('任务分类完成');
logStatus.success();
this.undoneTasks = this.uniqueTasks(this.undoneTasks);
this.socialTasks = this.uniqueTasks(this.socialTasks);
if (window.DEBUG) {
console.log('%cAuto-Task[Debug]:', 'color:blue', JSON.stringify(this));
}
debug('保存任务信息');
GM_setValue(`khTasks-${this.giveawayId}`, {
tasks: this.socialTasks,
time: (new Date).getTime()
});
return true;
} catch (error) {
debug('任务分类失败', {
error: error
});
throwError(error, 'Keyhub.classifyTask');
return false;
}
}
#isSkippableLink(link) {
return /^https?:\/\/www\.instagram\.com\/.*/.test(link) || /^https?:\/\/twitter\.com\/.*/.test(link) || /^https?:\/\/www\.twitch\.tv\/.*/.test(link) || /^https?:\/\/www\.facebook\.com\/.*/.test(link) || /^https?:\/\/www\.youtube\.com\/.*/.test(link) || /^https?:\/\/store\.steampowered\.com\/developer\//.test(link) || /^https?:\/\/.*?\.itch\.io\/.*/.test(link) || /^https?:\/\/key-hub\.eu.*/.test(link) || /^https?:\/\/store\.steampowered\.com\/app\/.*/.test(link) || /^https?:\/\/qr\.streamelements\.com\/.*/.test(link) || /^https?:\/\/store\.steampowered\.com\/news\/app\/.*/.test(link);
}
async #doScriptTask(data) {
try {
debug('执行脚本任务', {
data: data
});
const logStatus = echoLog({
text: I18n('doingKeyhubTask')
});
const {result: result, statusText: statusText, status: status, data: response} = await httpRequest({
url: `/away?data=${data}`,
method: 'GET',
headers: {
origin: 'https://key-hub.eu',
referer: 'https://key-hub.eu/'
}
});
if (result !== 'Success') {
debug('请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (response?.status !== 200) {
debug('响应错误', {
status: response?.status,
statusText: response?.statusText
});
logStatus.error(`Error:${response?.statusText}(${response?.status})`);
return false;
}
debug('任务完成');
logStatus.success();
return true;
} catch (error) {
debug('执行脚本任务失败', {
error: error
});
throwError(error, 'Keyhub.doScriptTask');
return false;
}
}
async extraDoTask({videoTasks: videoTasks}) {
try {
debug('开始执行额外任务', {
count: videoTasks.length
});
const pro = [];
for (const data of videoTasks) {
pro.push(this.#doScriptTask(data));
}
return Promise.all(pro).then((() => {
debug('所有额外任务完成');
return true;
}));
} catch (error) {
debug('执行额外任务失败', {
error: error
});
throwError(error, 'Keyhub.extraDoTask');
return false;
}
}
#getGiveawayId() {
try {
debug('获取抽奖ID');
const giveawayId = window.location.href.match(/giveaway\/([\d]+)/)?.[1];
if (giveawayId) {
this.giveawayId = giveawayId;
debug('获取抽奖ID成功', {
giveawayId: giveawayId
});
return true;
}
debug('获取抽奖ID失败');
echoLog({}).error(I18n('getFailed', 'GiveawayId'));
return false;
} catch (error) {
debug('获取抽奖ID出错', {
error: error
});
throwError(error, 'Keyhub.getGiveawayId');
return false;
}
}
async #checkLeftKey() {
try {
debug('检查剩余密钥');
if (!globalOptions.other.checkLeftKey) {
debug('跳过密钥检查');
return true;
}
const leftKey = $('#keysleft').text().trim();
debug('检查剩余密钥数量', {
leftKey: leftKey
});
if (leftKey !== '0') {
return true;
}
debug('没有剩余密钥,显示确认对话框');
const {value: value} = await Swal.fire({
icon: 'warning',
title: I18n('notice'),
text: I18n('noKeysLeft'),
confirmButtonText: I18n('confirm'),
cancelButtonText: I18n('cancel'),
showCancelButton: true
});
if (value) {
debug('用户确认关闭窗口');
window.close();
}
return true;
} catch (error) {
debug('检查剩余密钥失败', {
error: error
});
throwError(error, 'Keyhub.checkLeftKey');
return false;
}
}
#checkLogin() {
try {
debug('检查登录状态');
if (!globalOptions.other.checkLogin) {
debug('跳过登录检查');
return true;
}
if ($('a[href*="/connect/steam"]').length > 0) {
debug('未登录,重定向到 Steam 登录页面');
window.open('/connect/steam', '_self');
}
debug('登录检查完成');
return true;
} catch (error) {
debug('检查登录失败', {
error: error
});
throwError(error, 'Keyhub.checkLogin');
return false;
}
}
}
const defaultTasksTemplate$4 = {
steam: {
groupLinks: [],
wishlistLinks: [],
curatorLinks: [],
curatorLikeLinks: []
},
twitter: {
userLinks: []
},
vk: {
nameLinks: []
},
discord: {
serverLinks: []
}
};
const defaultTasks$5 = JSON.stringify(defaultTasksTemplate$4);
class Givekey extends Website {
name='Givekey';
tasks=[];
socialTasks=JSON.parse(defaultTasks$5);
undoneTasks=JSON.parse(defaultTasks$5);
userId;
buttons=[ 'doTask', 'undoTask', 'verifyTask' ];
static test() {
const url = window.location.host;
const isMatch = url === 'givekey.ru';
debug('检查网站匹配', {
url: url,
isMatch: isMatch
});
return isMatch;
}
async after() {
try {
debug('开始执行后续操作');
await new Promise((resolve => {
const checker = setInterval((() => {
if ($('#navbarDropdown').length > 0) {
debug('导航栏元素已加载');
clearInterval(checker);
resolve(true);
}
}), 500);
}));
if (!await this.#checkLeftKey()) {
debug('检查剩余密钥失败');
echoLog({}).warning(I18n('checkLeftKeyFailed'));
}
} catch (error) {
debug('后续操作失败', {
error: error
});
throwError(error, 'Givekey.after');
}
}
init() {
try {
debug('初始化 Givekey');
const logStatus = echoLog({
text: I18n('initing')
});
if ($('a[href*="/auth/steam"]').length > 0) {
debug('未登录,重定向到 Steam 登录页面');
window.open('/auth/steam', '_self');
logStatus.warning(I18n('needLogin'));
return false;
}
if (!this.#getGiveawayId()) {
debug('获取抽奖ID失败');
return false;
}
const userId = $('meta[name="user-id"]').attr('content');
if (!userId) {
debug('获取用户ID失败');
logStatus.error(I18n('getFailed', I18n('userId')));
return false;
}
this.userId = userId;
this.initialized = true;
debug('初始化完成', {
userId: userId
});
logStatus.success();
return true;
} catch (error) {
debug('初始化失败', {
error: error
});
throwError(error, 'Givekey.init');
return false;
}
}
async classifyTask(action) {
try {
debug('开始分类任务', {
action: action
});
const logStatus = echoLog({
text: I18n('getTasksInfo')
});
if (action === 'undo') {
debug('恢复已保存的任务信息');
this.socialTasks = GM_getValue(`gkTasks-${this.giveawayId}`)?.tasks || JSON.parse(defaultTasks$5);
}
const tasks = $('.card-body:has("button") .row');
debug('找到任务元素', {
count: tasks.length
});
for (const task of tasks) {
const taskEle = $(task);
const button = taskEle.find('button');
const isSuccess = /Complete/i.test(button.text().trim());
debug('处理任务', {
isSuccess: isSuccess
});
if (isSuccess && action !== 'undo') {
debug('跳过已完成的任务');
continue;
}
const checkButton = taskEle.find('#task_check');
const taskId = checkButton.attr('data-id');
if (taskId) {
debug('添加任务ID', {
taskId: taskId
});
this.tasks.push(taskId);
}
if (action === 'verify') {
continue;
}
const taskLink = taskEle.find('a');
let href = taskLink.attr('href');
if (!href) {
debug('任务链接为空');
continue;
}
const text = taskLink.text().trim();
if (!text) {
debug('任务描述为空');
continue;
}
if (/^https?:\/\/givekey\.ru\/giveaway\/[\d]+\/execution_task/.test(href)) {
debug('获取重定向链接', {
href: href
});
href = await getRedirectLink(href);
}
if (!href) {
debug('获取重定向链接失败');
continue;
}
const icon = taskEle.find('i');
await this.#classifyTaskByType(href, text, icon, isSuccess, action);
}
debug('任务分类完成');
logStatus.success();
this.tasks = unique(this.tasks);
this.undoneTasks = this.uniqueTasks(this.undoneTasks);
this.socialTasks = this.uniqueTasks(this.socialTasks);
debug('保存任务信息');
GM_setValue(`gkTasks-${this.giveawayId}`, {
tasks: this.socialTasks,
time: (new Date).getTime()
});
return true;
} catch (error) {
debug('任务分类失败', {
error: error
});
throwError(error, 'Givekey.classifyTask');
return false;
}
}
async verifyTask() {
try {
debug('开始验证任务');
if (!this.initialized && !this.init()) {
debug('初始化失败');
return false;
}
if (this.tasks.length === 0 && !await this.classifyTask('verify')) {
debug('任务分类失败');
return false;
}
echoLog({}).warning(I18n('giveKeyNoticeBefore'));
const taskLength = this.tasks.length;
debug('开始验证任务', {
taskCount: taskLength
});
for (let i = 0; i < taskLength; i++) {
await this.#verify(this.tasks[i]);
if (i < taskLength - 1) {
debug('等待15秒');
await delay(15e3);
}
}
debug('所有任务验证完成');
echoLog({}).success(I18n('allTasksComplete'));
echoLog({
html: `<li><font class="warning">${I18n('giveKeyNoticeAfter')}</font></li>`
});
return true;
} catch (error) {
debug('任务验证失败', {
error: error
});
throwError(error, 'Givekey.verifyTask');
return false;
}
}
async #verify(task) {
try {
debug('验证任务', {
taskId: task
});
const logStatus = echoLog({
html: `<li>${I18n('verifyingTask')}${task}...<font></font></li>`
});
const csrfToken = $('meta[name="csrf-token"]').attr('content');
if (!csrfToken) {
debug('CSRF token 未找到');
logStatus.error('CSRF token not found');
return false;
}
debug('发送验证请求');
const response = await $.ajax({
url: 'https://givekey.ru/giveaway/task',
method: 'POST',
data: `id=${task}&user_id=${this.userId}`,
dataType: 'json',
headers: {
'X-CSRF-TOKEN': csrfToken
}
});
if (!response) {
debug('未收到响应');
logStatus.error('No response received');
return false;
}
debug('处理响应', {
response: response
});
if (response.btn) {
$(`button[data-id=${this.userId}]`).html(response.btn);
}
if (response.status === 'ok') {
$(`.task_check_${response.id}`).html(`<button class="btn btn-success mb-2 btn-block" disabled>${response.btn}</button>`);
debug('任务验证成功');
logStatus.success();
return true;
}
if (response.status === 'end') {
debug('获得密钥');
logStatus.success();
echoLog({}).success(response.key);
return true;
}
debug('验证失败', {
error: response.msg
});
logStatus.error(`Error:${response.msg}`);
return false;
} catch (error) {
debug('验证过程出错', {
error: error
});
throwError(error, 'Givekey.verify');
return false;
}
}
#getGiveawayId() {
try {
debug('从URL获取抽奖ID');
const giveawayId = window.location.href.match(/giveaway\/([\d]+)/)?.[1];
if (giveawayId) {
this.giveawayId = giveawayId;
debug('获取抽奖ID成功', {
giveawayId: giveawayId
});
return true;
}
debug('获取抽奖ID失败');
echoLog({
text: I18n('getFailed', 'GiveawayId')
});
return false;
} catch (error) {
debug('获取抽奖ID出错', {
error: error
});
throwError(error, 'Givekey.getGiveawayId');
return false;
}
}
async #checkLeftKey() {
try {
debug('检查剩余密钥');
if (!globalOptions.other.checkLeftKey) {
debug('跳过密钥检查');
return true;
}
const keysCount = $('#keys_count').text();
debug('检查密钥数量', {
keysCount: keysCount
});
if (keysCount) {
return true;
}
debug('没有剩余密钥,显示确认对话框');
const {value: value} = await Swal.fire({
icon: 'warning',
title: I18n('notice'),
text: I18n('noKeysLeft'),
confirmButtonText: I18n('confirm'),
cancelButtonText: I18n('cancel'),
showCancelButton: true
});
if (value) {
debug('用户确认关闭窗口');
window.close();
}
return true;
} catch (error) {
debug('检查剩余密钥失败', {
error: error
});
throwError(error, 'Givekey.checkLeftKey');
return false;
}
}
async #classifyTaskByType(href, text, icon, isSuccess, action) {
try {
debug('开始分类任务类型', {
href: href,
text: text,
isSuccess: isSuccess,
action: action
});
if (/^https?:\/\/vk\.com\//.test(href)) {
debug('添加 VK 任务');
this.socialTasks.vk.nameLinks.push(href);
if (action === 'do' && !isSuccess) {
this.undoneTasks.vk.nameLinks.push(href);
}
return;
}
if (/^https?:\/\/steamcommunity\.com\/groups/.test(href)) {
debug('添加 Steam 组任务');
this.socialTasks.steam.groupLinks.push(href);
if (action === 'do' && !isSuccess) {
this.undoneTasks.steam.groupLinks.push(href);
}
return;
}
if (/^https?:\/\/store\.steampowered\.com\/app\//.test(href)) {
debug('添加 Steam 愿望单任务');
this.socialTasks.steam.wishlistLinks.push(href);
if (action === 'do' && !isSuccess) {
this.undoneTasks.steam.wishlistLinks.push(href);
}
return;
}
if (/Subscribe/gi.test(text) && icon.hasClass('fa-steam-square')) {
if (/^https?:\/\/store\.steampowered\.com\/curator\//.test(href)) {
debug('添加 Steam 鉴赏家关注任务');
this.socialTasks.steam.curatorLinks.push(href);
if (action === 'do' && !isSuccess) {
this.undoneTasks.steam.curatorLinks.push(href);
}
} else {
debug('添加 Steam 鉴赏家点赞任务');
this.socialTasks.steam.curatorLikeLinks.push(href);
if (action === 'do' && !isSuccess) {
this.undoneTasks.steam.curatorLikeLinks.push(href);
}
}
return;
}
if (/^https?:\/\/twitter\.com\//.test(href) && /Subscribe/gi.test(text)) {
debug('添加 Twitter 关注任务');
this.socialTasks.twitter.userLinks.push(href);
if (action === 'do' && !isSuccess) {
this.undoneTasks.twitter.userLinks.push(href);
}
return;
}
if (icon.hasClass('fa-discord') || /^https?:\/\/discord\.com\/invite\//.test(href)) {
debug('添加 Discord 服务器任务');
this.socialTasks.discord.serverLinks.push(href);
if (action === 'do' && !isSuccess) {
this.undoneTasks.discord.serverLinks.push(href);
}
return;
}
debug('未识别的任务类型', {
href: href,
text: text
});
echoLog({}).warning(`${I18n('unKnownTaskType')}: ${text}(${href})`);
} catch (error) {
debug('任务类型分类失败', {
error: error
});
throwError(error, 'Givekey.classifyTaskByType');
}
}
}
class GiveeClub extends GiveawaySu {
name='GiveeClub';
buttons=[ 'doTask', 'undoTask', 'verifyTask' ];
static test() {
const url = window.location.href;
const isMatch = /^https?:\/\/givee\.club\/.*?\/event\/[\d]+/.test(url);
debug('检查网站匹配', {
url: url,
isMatch: isMatch
});
return isMatch;
}
async after() {
try {
debug('开始执行后续操作');
if (!this.#checkLogin()) {
debug('登录检查失败');
echoLog({}).warning(I18n('checkLoginFailed'));
}
if (!await this.#checkLeftKey()) {
debug('检查剩余密钥失败');
echoLog({}).warning(I18n('checkLeftKeyFailed'));
}
} catch (error) {
debug('后续操作失败', {
error: error
});
throwError(error, 'GiveeClub.after');
}
}
init() {
try {
debug('初始化 GiveeClub');
const logStatus = echoLog({
text: I18n('initing')
});
if (!this.#checkLogin()) {
debug('登录检查失败');
logStatus.warning(I18n('needLogin'));
return false;
}
const giveawayIdResult = this.#getGiveawayId();
if (!giveawayIdResult) {
debug('获取抽奖ID失败');
return false;
}
this.initialized = true;
debug('初始化完成');
logStatus.success();
return true;
} catch (error) {
debug('初始化失败', {
error: error
});
throwError(error, 'GiveeClub.init');
return false;
}
}
async classifyTask(action) {
try {
debug('开始分类任务', {
action: action
});
const logStatus = echoLog({
text: I18n('getTasksInfo')
});
if (action === 'undo') {
debug('恢复已保存的任务信息');
this.socialTasks = GM_getValue(`gcTasks-${this.giveawayId}`)?.tasks || defaultTasks$7;
return true;
}
debug('初始化未完成任务列表');
this.undoneTasks = defaultTasks$7;
const tasks = $('.event-actions tr');
const processTask = async task => {
const taskDes = $(task).find('.event-action-label a');
const taskIcon = $(task).find('.event-action-icon i').attr('class') || '';
const taskName = taskDes.text().trim();
const taskType = $(task).find('button[data-type]')?.attr('data-type') || '';
const taskFinished = $(task).find('.event-action-buttons .btn-success')?.length;
const appId = taskDes.attr('data-steam-wishlist-appid');
debug('处理任务', {
taskName: taskName,
taskType: taskType,
taskIcon: taskIcon,
taskFinished: taskFinished,
appId: appId
});
if (taskIcon.includes('ban') || /AdBlock/i.test(taskName) || taskIcon.includes('envelope') || taskFinished) {
debug('跳过无效或已完成任务');
return true;
}
const taskHref = taskDes.attr('href');
if (!taskHref) {
debug('任务链接为空');
return false;
}
try {
debug('获取重定向链接', {
taskHref: taskHref
});
const taskLink = await getRedirectLink(taskHref, taskType.includes('steam'));
if (!taskLink) {
debug('获取重定向链接失败');
return false;
}
if (taskType === 'steam.game.wishlist' && appId) {
debug('添加 Steam 愿望单任务', {
appId: appId
});
this.undoneTasks.steam.wishlistLinks.push(`https://store.steampowered.com/app/${appId}`);
return true;
}
debug('分类任务', {
taskLink: taskLink,
taskType: taskType
});
this.#classifyTaskByType(taskLink, taskType, taskIcon, taskName, taskDes);
return true;
} catch (error) {
debug('获取重定向链接失败', {
error: error
});
throwError(error, 'GiveeClub.classifyTask->getRedirectLink');
return false;
}
};
debug('开始处理所有任务');
await Promise.all(Array.from(tasks).map(processTask));
debug('任务处理完成');
logStatus.success();
this.undoneTasks = this.uniqueTasks(this.undoneTasks);
this.socialTasks = this.undoneTasks;
debug('保存任务信息');
GM_setValue(`gcTasks-${this.giveawayId}`, {
tasks: this.socialTasks,
time: (new Date).getTime()
});
return true;
} catch (error) {
debug('任务分类失败', {
error: error
});
throwError(error, 'GiveeClub.classifyTask');
return false;
}
}
#classifyTaskByType(taskLink, taskType, taskIcon, taskName, taskDes) {
try {
debug('开始分类任务', {
taskLink: taskLink,
taskType: taskType,
taskIcon: taskIcon,
taskName: taskName
});
if (taskType === 'steam.group.join' && /^https?:\/\/steamcommunity\.com\/groups/.test(taskLink)) {
debug('添加 Steam 组任务');
this.undoneTasks.steam.groupLinks.push(taskLink);
return;
}
if (/like.*announcement/gi.test(taskName)) {
debug('添加 Steam 公告任务');
this.undoneTasks.steam.announcementLinks.push(taskLink);
return;
}
if (taskType === 'steam.game.wishlist' && /^https?:\/\/store\.steampowered\.com\/app\//.test(taskLink)) {
debug('添加 Steam 愿望单任务');
this.undoneTasks.steam.wishlistLinks.push(taskLink);
return;
}
if (taskType === 'steam.game.wishlist' && taskDes.attr('data-steam-wishlist-appid')) {
debug('添加 Steam 愿望单任务(通过 appId)');
this.undoneTasks.steam.wishlistLinks.push(`https://store.steampowered.com/app/${taskDes.attr('data-steam-wishlist-appid')}`);
return;
}
if (taskType === 'steam.game.follow' && /^https?:\/\/store\.steampowered\.com\/app\//.test(taskLink)) {
debug('添加 Steam 游戏关注任务');
this.undoneTasks.steam.followLinks.push(taskLink);
return;
}
if (/^https?:\/\/store\.steampowered\.com\/curator\//.test(taskLink)) {
debug('添加 Steam 鉴赏家关注任务');
this.undoneTasks.steam.curatorLinks.push(taskLink);
return;
}
if (taskIcon.includes('steam') && /follow|subscribe/gim.test(taskName)) {
debug('添加 Steam 鉴赏家点赞任务');
this.undoneTasks.steam.curatorLikeLinks.push(taskLink);
return;
}
if (/subscribe.*steam.*forum/gim.test(taskName)) {
debug('添加 Steam 论坛任务');
this.undoneTasks.steam.forumLinks.push(taskLink);
return;
}
if (taskType === 'steam.game.playtime' && /^https?:\/\/store\.steampowered\.com\/app\//.test(taskLink)) {
const time = taskDes.text().match(/(\d+)(?:\.\d+)?/gim)?.[0] || '0';
debug('添加 Steam 游戏时长任务', {
time: time
});
this.undoneTasks.steam.playTimeLinks.push(`${time}-${taskLink}`);
return;
}
if (taskIcon.includes('discord')) {
debug('添加 Discord 服务器任务');
this.undoneTasks.discord.serverLinks.push(taskLink);
return;
}
if (taskIcon.includes('instagram')) {
debug('跳过 Instagram 任务');
return;
}
if (taskIcon.includes('twitch')) {
debug('添加 Twitch 频道任务');
this.undoneTasks.twitch.channelLinks.push(taskLink);
return;
}
if (taskIcon.includes('reddit')) {
debug('添加 Reddit 任务');
this.undoneTasks.reddit.redditLinks.push(taskLink);
return;
}
if (/watch.*art/gim.test(taskName)) {
debug('添加创意工坊物品任务');
this.undoneTasks.steam.workshopVoteLinks.push(taskLink);
return;
}
if (/subscribe.*youtube.*channel/gim.test(taskName)) {
debug('添加 YouTube 频道任务');
this.undoneTasks.youtube.channelLinks.push(taskLink);
return;
}
if (/(watch|like).*youtube.*video/gim.test(taskName) || (taskIcon.includes('youtube') || taskIcon.includes('thumbs-up')) && /(watch|like).*video/gim.test(taskName)) {
debug('添加 YouTube 视频任务');
this.undoneTasks.youtube.likeLinks.push(taskLink);
return;
}
if (taskIcon.includes('vk') || /join.*vk.*group/gim.test(taskName)) {
debug('添加 VK 任务');
this.undoneTasks.vk.nameLinks.push(taskLink);
return;
}
if (taskIcon.includes('twitter')) {
if (/https?:\/\/(twitter|x)\.com\/[^/]+\/?$/gim.test(taskLink)) {
debug('添加 Twitter 用户关注任务');
this.undoneTasks.twitter.userLinks.push(taskLink);
return;
}
if (/https?:\/\/(twitter|x)\.com\/[^/]+?\/status\/[\d]+/gim.test(taskLink)) {
debug('添加 Twitter 转发任务');
this.undoneTasks.twitter.retweetLinks.push(taskLink);
return;
}
}
if (/(on twitter)|(Follow.*on.*Facebook)/gim.test(taskName)) {
debug('跳过 Twitter/Facebook 任务');
return;
}
if (/follow.*button/gim.test(taskName)) {
debug('添加 Steam 关注任务');
this.undoneTasks.steam.followLinks.push(taskLink);
return;
}
debug('未识别的任务类型', {
taskLink: taskLink,
taskType: taskType,
taskIcon: taskIcon,
taskName: taskName
});
} catch (error) {
debug('任务分类失败', {
error: error
});
throwError(error, 'GiveeClub.classifyTaskByType');
return;
}
}
async verifyTask() {
try {
debug('开始验证任务');
const logStatus = echoLog({
text: I18n('giveeClubVerifyNotice')
});
const taskButtons = $('.event-actions tr button').has('i.glyphicon-refresh').not('[data-type="user.adblock"]');
debug('找到需要验证的任务按钮', {
count: taskButtons.length
});
for (const button of taskButtons) {
debug('点击验证按钮', {
type: $(button).attr('data-type')
});
button.click();
if ($(button).attr('data-type') !== 'steam.game.wishlist') {
debug('等待1秒');
await delay(1e3);
}
}
debug('任务验证完成');
logStatus.warning(I18n('giveeClubVerifyFinished'));
return true;
} catch (error) {
debug('任务验证失败', {
error: error
});
throwError(error, 'Givekey.verifyTask');
return false;
}
}
#checkLogin() {
try {
debug('检查登录状态');
if (!globalOptions.other.checkLogin) {
debug('跳过登录检查');
return true;
}
const needLogin = $('a[href*="/account/auth"]').length > 0;
if (needLogin) {
debug('未登录,重定向到登录页面');
window.open($('a[href*="/account/auth"]').attr('href'), '_self');
}
debug('登录检查完成', {
needLogin: needLogin
});
return true;
} catch (error) {
debug('登录检查失败', {
error: error
});
throwError(error, 'GiveeClub.checkLogin');
return false;
}
}
#getGiveawayId() {
try {
debug('从URL获取抽奖ID');
const giveawayId = window.location.href.match(/\/event\/([\d]+)/)?.[1];
if (giveawayId) {
this.giveawayId = giveawayId;
debug('获取抽奖ID成功', {
giveawayId: giveawayId
});
return true;
}
debug('获取抽奖ID失败');
echoLog({
text: I18n('getFailed', 'GiveawayId')
});
return false;
} catch (error) {
debug('获取抽奖ID出错', {
error: error
});
throwError(error, 'GiveeClub.getGiveawayId');
return false;
}
}
async #checkLeftKey() {
try {
debug('检查剩余密钥');
if (!globalOptions.other.checkLeftKey) {
debug('跳过密钥检查');
return true;
}
const isEnded = $('.event-ended').length > 0;
const hasNoWinner = $('.event-winner').length === 0;
debug('检查抽奖状态', {
isEnded: isEnded,
hasNoWinner: hasNoWinner
});
if (!(isEnded && hasNoWinner)) {
return true;
}
debug('没有剩余密钥,显示确认对话框');
const {value: value} = await Swal.fire({
icon: 'warning',
title: I18n('notice'),
text: I18n('giveawayEnded'),
confirmButtonText: I18n('confirm'),
cancelButtonText: I18n('cancel'),
showCancelButton: true
});
if (value) {
debug('用户确认关闭窗口');
window.close();
}
return true;
} catch (error) {
debug('检查剩余密钥失败', {
error: error
});
throwError(error, 'GiveeClub.checkLeftKey');
return false;
}
}
}
const defaultOptions$1 = {
maxPoint: '99999999'
};
class OpiumPulses {
name='OpiumPulses';
options={
...defaultOptions$1,
...GM_getValue('OpiumPulsesOptions')
};
maxPoints=99999999;
myPoints=0;
buttons=[ 'doFreeTask', 'doPointTask' ];
static test() {
const {host: host} = window.location;
const isMatch = host === 'www.opiumpulses.com';
debug('检查网站匹配', {
host: host,
isMatch: isMatch
});
return isMatch;
}
async after() {
try {
debug('开始执行后续操作');
if (!this.#checkLogin()) {
debug('检查登录失败');
echoLog({}).warning(I18n('checkLoginFailed'));
}
debug('解析最大积分', {
maxPoint: this.options.maxPoint
});
this.maxPoints = parseInt(this.options.maxPoint, 10);
} catch (error) {
debug('后续操作失败', {
error: error
});
throwError(error, 'OpiumPulses.after');
}
}
async doFreeTask() {
try {
debug('开始执行免费任务');
this.#toggleTask('FREE');
} catch (error) {
debug('执行免费任务失败', {
error: error
});
throwError(error, 'OpiumPulses.doFreeTask');
}
}
async doPointTask() {
try {
debug('开始执行积分任务');
const pointsText = $('.page-header__nav-func-user-nav-items.points-items').text();
const pointsMatch = pointsText.match(/[\d]+/gim)?.[0] || '0';
this.myPoints = parseInt(pointsMatch, 10);
debug('获取当前积分', {
pointsText: pointsText,
pointsMatch: pointsMatch,
myPoints: this.myPoints
});
this.#toggleTask('points');
} catch (error) {
debug('执行积分任务失败', {
error: error
});
throwError(error, 'OpiumPulses.doPointTask');
}
}
async #toggleTask(type) {
try {
debug('开始切换任务', {
type: type
});
const items = $(`.giveaways-page-item:contains('${type}'):not(:contains('ENTERED'))`);
debug('找到未参与的抽奖项目', {
count: items.length
});
for (const item of items) {
const pointsText = $(item).find('.giveaways-page-item-header-points').text();
const needPoints = parseInt(pointsText.match(/[\d]+/gim)?.[0] || '999999', 10);
const name = $(item).find('.giveaways-page-item-footer-name').text().trim();
debug('处理抽奖项目', {
name: name,
needPoints: needPoints
});
if (type === 'points') {
if (needPoints > this.myPoints) {
debug('积分不足', {
needPoints: needPoints,
myPoints: this.myPoints
});
echoLog({}).warning(`${I18n('noPoints')}: ${name}`);
continue;
}
if (!needPoints) {
debug('获取所需积分失败');
echoLog({}).warning(`${I18n('getNeedPointsFailed')}: ${name}`);
continue;
}
if (needPoints > this.maxPoints) {
debug('超过最大积分限制', {
needPoints: needPoints,
maxPoints: this.maxPoints
});
continue;
}
}
const logStatus = echoLog({
text: `${I18n('joiningLottery')}<a href="${$(item).find('a.giveaways-page-item-img-btn-more').attr('href')}" target="_blank">${name}</a>...`
});
const aElement = $(item).find('a.giveaways-page-item-img-btn-enter:contains(\'enter\')');
if (aElement?.attr('onclick')?.includes('checkUser')) {
const giveawayId = aElement.attr('onclick')?.match(/[\d]+/)?.[0];
if (giveawayId) {
debug('执行用户检查', {
giveawayId: giveawayId
});
checkUser(giveawayId);
}
}
if (!aElement.attr('href')) {
debug('无效的链接');
logStatus.error('Error: No "href".');
continue;
}
debug('发送加入请求', {
url: aElement.attr('href')
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: aElement.attr('href'),
method: 'GET'
});
if (result !== 'Success') {
debug('请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
continue;
}
debug('发送最终请求', {
url: data?.finalUrl
});
const {result: result0, statusText: statusText0, status: status0, data: data0} = await httpRequest({
url: data?.finalUrl,
method: 'GET'
});
if (!data0?.responseText) {
debug('响应无效', {
result: result0,
statusText: statusText0,
status: status0
});
logStatus.error(`${result0}:${statusText0}(${status0})`);
continue;
}
if (/You're not eligible to enter/gim.test(data0.responseText)) {
debug('用户不符合参与条件');
logStatus.error('You\'re not eligible to enter');
continue;
}
if (!/You've entered this giveaway/gim.test(data0.responseText)) {
debug('加入抽奖失败', {
result: result0,
statusText: statusText0,
status: status0
});
logStatus.error(`${result0}:${statusText0}(${status0})`);
continue;
}
debug('加入抽奖成功');
logStatus.success();
if (type === 'points') {
const points = data0.responseText.match(/Points:[\s]*?([\d]+)/)?.[1];
if (points) {
debug('更新用户积分', {
points: points
});
this.myPoints = parseInt(points, 10);
}
}
}
debug('任务处理完成');
echoLog({
text: '-----END-----'
});
} catch (error) {
debug('切换任务失败', {
error: error
});
throwError(error, 'OpiumPulses.toggleTask');
}
}
init() {
debug('初始化完成');
return true;
}
classifyTask() {
debug('任务分类完成');
return true;
}
#checkLogin() {
try {
debug('检查登录状态');
if (!globalOptions.other.checkLogin) {
debug('跳过登录检查');
return true;
}
if ($('a[href*="/site/login"]').length > 1) {
debug('未登录,重定向到登录页面');
window.open('/site/login', '_self');
}
debug('登录检查完成');
return true;
} catch (error) {
debug('检查登录失败', {
error: error
});
throwError(error, 'OpiumPulses.checkLogin');
return false;
}
}
}
const leftKeyChecker = {
async classify(link) {
try {
debug('开始分类链接', {
link: link
});
let result = false;
if (/^https?:\/\/giveaway\.su\/giveaway\/view\/[\d]+/.test(link)) {
debug('匹配到 giveaway.su 链接');
result = await this.giveawaySu(link);
} else if (/^https?:\/\/givee\.club\/[\w]+?\/event\/[\d]+/.test(link)) {
debug('匹配到 givee.club 链接');
result = await this.giveeClub(link);
} else if (/^https?:\/\/gleam\.io\/.+?\/.+/.test(link)) {
debug('匹配到 gleam.io 链接');
result = await this.gleam(link);
} else if (/^https?:\/\/www\.indiedb\.com\/giveaways\/.+/.test(link)) {
debug('匹配到 indiedb.com 链接');
result = await this.indieDb(link);
} else if (/^https?:\/\/key-hub\.eu\/giveaway\/[\d]+/.test(link)) {
debug('匹配到 key-hub.eu 链接');
result = await this.keyhub(link);
} else if (/^https?:\/\/opquests\.com\/quests\/[\d]+/.test(link)) {
debug('匹配到 opquests.com 链接');
result = await this.opquests(link);
} else if (/^https?:\/\/itch\.io\/s\/[\d]+?\/.*/.test(link)) {
debug('匹配到 itch.io 链接');
result = await this.itch(link);
} else if (/^https?:\/\/freeanywhere\.net\/game\?n=[\d]+/.test(link)) {
debug('匹配到 freeanywhere.net 链接');
result = await this.freeanywhere(link);
} else {
debug('未匹配到支持的链接格式');
}
debug('链接分类完成', {
result: result
});
return result;
} catch (error) {
debug('链接分类出错', {
error: error
});
throwError(error, 'leftKeyChecker.classify');
return false;
}
},
async giveawaySu(link) {
try {
debug('开始检查 giveaway.su 链接', {
link: link
});
const {result: result, data: data} = await httpRequest({
url: link,
method: 'GET'
});
if (result !== 'Success' || data?.status !== 200) {
debug('请求失败', {
result: result,
status: data?.status
});
return false;
}
if (data.responseText.includes('class="steam-login"')) {
debug('检测到未登录状态');
return false;
}
if (data.responseText.includes('class="giveaway-ended"')) {
debug('检测到抽奖已结束');
return 'Ended';
}
debug('检测到抽奖进行中');
return 'Active';
} catch (error) {
debug('检查 giveaway.su 链接出错', {
error: error
});
throwError(error, 'leftKeyChecker.giveawaySu');
return false;
}
},
async giveeClub(link) {
try {
debug('开始检查 givee.club 链接', {
link: link
});
const {result: result, data: data} = await httpRequest({
url: link,
method: 'GET'
});
if (result !== 'Success' || data?.status !== 200) {
debug('请求失败', {
result: result,
status: data?.status
});
return false;
}
if (data.responseText.includes('class="event-winner"')) {
debug('检测到已中奖');
return 'Won';
}
if (data.responseText.includes('class="event-ended"')) {
debug('检测到活动已结束');
return 'Ended';
}
debug('检测到活动进行中');
return 'Active';
} catch (error) {
debug('检查 givee.club 链接出错', {
error: error
});
throwError(error, 'leftKeyChecker.giveeClub');
return false;
}
},
async gleam(link) {
try {
debug('开始检查 gleam.io 链接', {
link: link
});
const {result: result, data: data} = await httpRequest({
url: link,
method: 'GET'
});
if (result !== 'Success' || data?.status !== 200) {
debug('请求失败', {
result: result,
status: data?.status
});
return false;
}
if (/incentives":{"[\d]+?":\[".+?"\]/.test(data.responseText)) {
debug('检测到已中奖');
return 'Won';
}
const campaignDiv = data.responseText.match(/<div class='popup-blocks-container'[\w\W]+?'>/)?.[0];
if (!campaignDiv) {
debug('未找到活动信息');
return false;
}
const campaignString = $(campaignDiv).attr('ng-init')?.match(/initCampaign\(([\w\W]+?)\)$/)?.[1];
if (!campaignString) {
debug('未找到活动初始化数据');
return false;
}
const {campaign: campaign} = JSON.parse(campaignString);
debug('解析活动数据', {
campaign: campaign
});
if (campaign.banned) {
debug('检测到活动已被禁止');
return 'Banned';
}
if (campaign.finished) {
debug('检测到活动已结束');
return 'Ended';
}
if (campaign.paused) {
debug('检测到活动已暂停');
return 'Paused';
}
if ((new Date).getTime() < campaign.starts_at * 1e3) {
debug('检测到活动未开始');
return 'NotStart';
}
debug('检测到活动进行中');
return 'Active';
} catch (error) {
debug('检查 gleam.io 链接出错', {
error: error
});
throwError(error, 'leftKeyChecker.gleam');
return false;
}
},
async indieDb(link) {
try {
debug('开始检查 indiedb.com 链接', {
link: link
});
const {result: result, data: data} = await httpRequest({
url: link,
method: 'GET'
});
if (result !== 'Success' || data?.status !== 200) {
debug('请求失败', {
result: result,
status: data?.status
});
return false;
}
if (data.responseText.includes('Congrats you WON')) {
debug('检测到已中奖');
return 'Won';
}
if (data.responseText.includes('Giveaway is closed') || data.responseText.includes('next time')) {
debug('检测到抽奖已结束');
return 'Ended';
}
debug('检测到抽奖进行中');
return 'Active';
} catch (error) {
debug('检查 indiedb.com 链接出错', {
error: error
});
throwError(error, 'leftKeyChecker.indieDb');
return false;
}
},
async keyhub(link) {
try {
debug('开始检查 key-hub.eu 链接', {
link: link
});
const {result: result, data: data} = await httpRequest({
url: link,
method: 'GET'
});
if (result !== 'Success' || data?.status !== 200) {
debug('请求失败', {
result: result,
status: data?.status
});
return false;
}
const keysleft = data.responseText.match(/<span id="keysleft">([\d]+?)<\/span>/)?.[1];
if (!keysleft) {
debug('未找到剩余密钥信息');
return false;
}
debug('检测到剩余密钥数量', {
keysleft: keysleft
});
if (keysleft === '0') {
debug('检测到密钥已用完');
return 'Ended';
}
debug('检测到活动进行中');
return `Active(${keysleft})`;
} catch (error) {
debug('检查 key-hub.eu 链接出错', {
error: error
});
throwError(error, 'leftKeyChecker.keyhub');
return false;
}
},
async opquests(link) {
try {
debug('开始检查 opquests.com 链接', {
link: link
});
const {result: result, data: data} = await httpRequest({
url: link,
method: 'GET'
});
if (data?.status === 404) {
debug('检测到活动不存在');
return 'Ended';
}
if (result !== 'Success' || data?.status !== 200) {
debug('请求失败', {
result: result,
status: data?.status
});
return false;
}
const keysleft = data.responseText.match(/<div class="">[\s]*?([\d]+?)[\s]*?of/)?.[1];
if (!keysleft) {
debug('未找到剩余密钥信息');
return false;
}
debug('检测到剩余密钥数量', {
keysleft: keysleft
});
if (keysleft === '0') {
debug('检测到密钥已用完');
return 'Ended';
}
debug('检测到活动进行中');
return `Active(${keysleft})`;
} catch (error) {
debug('检查 opquests.com 链接出错', {
error: error
});
throwError(error, 'leftKeyChecker.opquests');
return false;
}
},
async itch(link) {
try {
debug('开始检查 itch.io 链接', {
link: link
});
const {result: result, data: data} = await httpRequest({
url: link,
method: 'GET'
});
if (result !== 'Success' || data?.status !== 200) {
debug('请求失败', {
result: result,
status: data?.status
});
return false;
}
const endDate = data.responseText.match(/{"start_date":"[0-9A-Z-:]+?".*?"end_date":"([0-9A-Z-:]+?)".*?}/)?.[1];
if (!endDate) {
debug('未找到结束日期信息');
return false;
}
debug('检测到活动结束日期', {
endDate: endDate
});
if ((new Date).getTime() > new Date(endDate).getTime()) {
debug('检测到活动已结束');
return 'Ended';
}
const formattedEndDate = dayjs(endDate).format('YYYY-MM-DD HH:mm:ss');
debug('检测到活动进行中', {
formattedEndDate: formattedEndDate
});
return `Active(${formattedEndDate})`;
} catch (error) {
debug('检查 itch.io 链接出错', {
error: error
});
throwError(error, 'leftKeyChecker.itch');
return false;
}
},
async freeanywhere(link) {
try {
debug('开始检查 freeanywhere.net 链接', {
link: link
});
const {result: result, data: data} = await httpRequest({
url: link,
method: 'GET'
});
if (result !== 'Success' || data?.status !== 200) {
debug('请求失败', {
result: result,
status: data?.status
});
return false;
}
const giveawayStatus = data.responseText.includes('Giveaway ended');
if (giveawayStatus) {
debug('检测到活动已结束');
return 'Ended';
}
debug('检测到活动进行中');
return 'Active';
} catch (error) {
debug('检查 freeanywhere.net 链接出错', {
error: error
});
throwError(error, 'leftKeyChecker.freeanywhere');
return false;
}
}
};
const defaultTasksTemplate$3 = {
steam: {
groupLinks: [],
wishlistLinks: [],
curatorLinks: [],
curatorLikeLinks: [],
followLinks: [],
forumLinks: [],
announcementLinks: [],
workshopVoteLinks: [],
licenseLinks: []
},
discord: {
serverLinks: []
},
vk: {
nameLinks: []
},
twitch: {
channelLinks: []
},
reddit: {
redditLinks: []
},
twitter: {
userLinks: [],
retweetLinks: []
},
youtube: {
channelLinks: [],
likeLinks: []
}
};
const defaultTasks$4 = JSON.stringify(defaultTasksTemplate$3);
class Keylol extends Website {
name='Keylol';
socialTasks=JSON.parse(defaultTasks$4);
undoneTasks=JSON.parse(defaultTasks$4);
buttons=[ 'doTask', 'undoTask', 'selectAll', 'selectNone', 'invertSelect' ];
static CONFIG={
LINK_PATTERNS: {
DISCORD: /^https?:\/\/discord\.com\/invite\/.+/,
REDDIT: /^https?:\/\/www\.reddit\.com\/(r|user)\/.+/,
INSTAGRAM: /^https:\/\/www\.instagram\.com\/.+/,
TWITTER: /^https:\/\/(twitter|x)\.com\/.+/,
TWITTER_RETWEET: /https:\/\/(twitter|x)\.com\/.*?\/status\/[\d]+/,
TWITCH: /^https:\/\/(www\.)?twitch\.tv\/.+/,
VK: /^https:\/\/vk\.com\/.+/,
STEAM_CURATOR: /curator\/[\d]+/,
STEAM_PUBLISHER: /(publisher|developer|franchise)\/.+/,
STEAM_NEWS: /news(hub)?\/app\/[\d]+\/view\/[\d]+/,
STEAM_APP: /app\/[\d]+/,
STEAM_GROUP: /groups\/.+/,
STEAM_ANNOUNCEMENT: /announcements\/detail\/[\d]+/,
YOUTUBE: /youtube\.com/
},
SELECTORS: {
MAIN_POST: {
KEYLOL: '#postlist>div[id^="post_"]:first',
DEFAULT: 'div.container'
}
}
};
static test() {
const {host: host} = window.location;
const link = $('.subforum_left_title_left_up a').eq(3).attr('href');
const isMatch = host === 'keylol.com' && (!!link?.includes('319') || !!link?.includes('234'));
debug('检查网站匹配', {
host: host,
link: link,
isMatch: isMatch
});
return isMatch;
}
init() {
debug('初始化 Keylol');
return true;
}
after() {
try {
debug('开始处理页面链接');
const mainPost = $(this.name === 'Keylol' ? Keylol.CONFIG.SELECTORS.MAIN_POST.KEYLOL : Keylol.CONFIG.SELECTORS.MAIN_POST.DEFAULT);
const allLinks = mainPost.find('a');
debug('找到所有链接', {
count: allLinks.length
});
allLinks.each(((_, link) => {
const $link = $(link);
const href = $link.attr('href');
if (!href) {
return;
}
this.#classifyAndProcessLink($link, href);
}));
debug('开始处理抽奖链接');
this.#processGiveawayLinks(mainPost);
if (this.name === 'Keylol') {
debug('开始处理 Keylol 特定链接');
this.#processKeylolSpecificLinks(mainPost);
}
debug('设置 MutationObserver');
this.#setupMutationObserver();
} catch (error) {
debug('处理页面链接失败', {
error: error
});
throwError(error, 'keylol.after');
}
}
#classifyAndProcessLink($link, href) {
debug('分类处理链接', {
href: href
});
const {LINK_PATTERNS: LINK_PATTERNS} = Keylol.CONFIG;
switch (true) {
case LINK_PATTERNS.DISCORD.test(href):
debug('发现 Discord 链接');
this.#addBtn($link[0], 'discord', 'serverLinks', href);
break;
case LINK_PATTERNS.REDDIT.test(href):
debug('发现 Reddit 链接');
this.#addBtn($link[0], 'reddit', 'redditLinks', href);
break;
case LINK_PATTERNS.TWITTER.test(href):
if (LINK_PATTERNS.TWITTER_RETWEET.test(href)) {
debug('发现 Twitter 转发链接');
this.#addBtn($link[0], 'twitter', 'retweetLinks', href);
} else {
debug('发现 Twitter 用户链接');
this.#addBtn($link[0], 'twitter', 'userLinks', href);
}
break;
case LINK_PATTERNS.TWITCH.test(href):
debug('发现 Twitch 链接');
this.#addBtn($link[0], 'twitch', 'channelLinks', href);
break;
case LINK_PATTERNS.VK.test(href):
debug('发现 VK 链接');
this.#addBtn($link[0], 'vk', 'nameLinks', href);
break;
case href.includes('store.steampowered.com'):
debug('发现 Steam 商店链接');
this.#processSteamStoreLink($link[0], href);
break;
case href.includes('steamcommunity.com'):
debug('发现 Steam 社区链接');
this.#processSteamCommunityLink($link[0], href);
break;
case LINK_PATTERNS.YOUTUBE.test(href):
debug('发现 YouTube 链接');
this.#addBtn($link[0], 'youtube', 'channelLinks', href);
this.#addBtn($link[0], 'youtube', 'likeLinks', href);
break;
}
}
#processSteamStoreLink(element, href) {
debug('处理 Steam 商店链接', {
href: href
});
const {LINK_PATTERNS: LINK_PATTERNS} = Keylol.CONFIG;
if (LINK_PATTERNS.STEAM_CURATOR.test(href)) {
debug('发现 Steam 鉴赏家链接');
this.#addBtn(element, 'steam', 'curatorLinks', href);
} else if (LINK_PATTERNS.STEAM_PUBLISHER.test(href)) {
debug('发现 Steam 发行商链接');
this.#addBtn(element, 'steam', 'curatorLikeLinks', href);
} else if (LINK_PATTERNS.STEAM_NEWS.test(href)) {
debug('发现 Steam 新闻链接');
this.#addBtn(element, 'steam', 'announcementLinks', href);
} else if (LINK_PATTERNS.STEAM_APP.test(href)) {
debug('发现 Steam 应用链接');
this.#addBtn(element, 'steam', 'followLinks', href);
this.#addBtn(element, 'steam', 'wishlistLinks', href);
}
}
#processSteamCommunityLink(element, href) {
debug('处理 Steam 社区链接', {
href: href
});
const {LINK_PATTERNS: LINK_PATTERNS} = Keylol.CONFIG;
if (LINK_PATTERNS.STEAM_GROUP.test(href)) {
debug('发现 Steam 组链接');
this.#addBtn(element, 'steam', 'groupLinks', href);
} else if (LINK_PATTERNS.STEAM_ANNOUNCEMENT.test(href)) {
debug('发现 Steam 公告链接');
this.#addBtn(element, 'steam', 'announcementLinks', href);
}
}
#processGiveawayLinks(mainPost) {
debug('开始处理抽奖链接');
const giveawayLinks = mainPost.find('a[href*="giveaway.su/giveaway/view/"],' + 'a[href*="givee.club/"],' + 'a[href*="gleam.io/"],' + 'a[href*="www.indiedb.com/giveaways/"],' + 'a[href*="key-hub.eu/giveaway/"],' + 'a[href*="opquests.com/quests/"],' + 'a[href*="freeanywhere.net/game?n="],' + 'a[href*="itch.io/s/"]:visible');
debug('找到抽奖链接', {
count: giveawayLinks.length
});
giveawayLinks.each(((_, link) => {
const href = $(link).attr('href');
if (!href) {
return;
}
debug('检查抽奖链接状态', {
href: href
});
leftKeyChecker.classify(href).then((status => {
if (!status) {
return;
}
const statusClass = /^Active/.test(status) ? 'active' : 'not-active';
const statusTitle = /^Active/.test(status) ? I18n('Active') : I18n(status);
debug('更新抽奖链接状态', {
href: href,
status: status,
statusClass: statusClass
});
$(`a[href="${href}"]`).after(`<font class="auto-task-giveaway-status ${statusClass}" title="${statusTitle}">${status}</font>`);
})).catch((error => {
debug('检查抽奖链接状态失败', {
href: href,
error: error
});
throwError(error, 'keylol.after -> leftKeyChecker');
}));
}));
}
#processKeylolSpecificLinks(mainPost) {
debug('开始处理 Keylol 特定链接');
const asfLinks = mainPost.find('a[href^="#asf"]:visible');
debug('找到 ASF 链接', {
count: asfLinks.length
});
asfLinks.each(((_, link) => {
const href = $(link).attr('href');
if (!href) {
return;
}
debug('处理 ASF 链接', {
href: href
});
const $link = $(`a[href="${href}"]`);
$link.after('<span style="color: #ccc; margin: 0 -5px 0 5px"> | </span>');
this.#addBtn($link.next()[0], 'steam', 'licenseLinks', `appid-${href.replace('#asf', '')}`);
}));
const steamDbLinks = mainPost.find('a[href*="steamdb.info/sub/"]:visible');
debug('找到 SteamDB 链接', {
count: steamDbLinks.length
});
steamDbLinks.each(((_, link) => {
const href = $(link).attr('href');
if (!href) {
return;
}
const subid = href.match(/^https:\/\/steamdb\.info\/sub\/([\d]+)/)?.[1];
if (!subid) {
return;
}
debug('处理 SteamDB 链接', {
href: href,
subid: subid
});
this.#addBtn(link, 'steam', 'licenseLinks', `subid-${subid}`);
}));
const asfBlocks = mainPost.find('.blockcode:contains("addlicense"):visible');
debug('找到 ASF 代码块', {
count: asfBlocks.length
});
asfBlocks.each(((_, block) => {
const appid = [ ...block.innerText.matchAll(/a(pp)?\/([\d]+)/g) ].map((matched => matched?.[2])).filter((id => id));
if (appid.length > 0) {
debug('处理 ASF 代码块 appid', {
appid: appid
});
this.#addBtn($(block).children('em')[0], 'steam', 'licenseLinks', `appid-${appid.join(',')}`);
}
const subid = block.innerText.match(/[\d]+/g)?.filter((matched => !appid.includes(matched)));
if (subid?.length) {
debug('处理 ASF 代码块 subid', {
subid: subid
});
this.#addBtn($(block).children('em')[0], 'steam', 'licenseLinks', `subid-${subid.join(',')}`);
}
}));
}
#setupMutationObserver() {
debug('设置 MutationObserver');
if ($('#threadindex').length > 0) {
const [elementTargetNode] = $('#postlist').children('div[id^="post_"]');
const elementObserver = new MutationObserver((() => {
debug('检测到 DOM 变化,重新处理页面');
elementObserver.disconnect();
this.after();
}));
elementObserver.observe(elementTargetNode, {
childList: true
});
debug('MutationObserver 设置完成');
}
}
classifyTask(action) {
try {
debug('开始分类任务', {
action: action
});
this.socialTasks = JSON.parse(defaultTasks$4);
this.undoneTasks = JSON.parse(defaultTasks$4);
const selectedBtns = $('.auto-task-keylol[selected="selected"]:visible').get();
debug('找到选中的按钮', {
count: selectedBtns.length
});
for (const btn of selectedBtns) {
const social = btn.getAttribute('data-social');
const type = btn.getAttribute('data-type');
const link = btn.getAttribute('data-link');
debug('处理任务按钮', {
social: social,
type: type,
link: link
});
if (!(social && type && link)) {
debug('跳过无效任务按钮');
continue;
}
if (!(social in this.undoneTasks)) {
debug('跳过未知社交平台', {
social: social
});
continue;
}
if (action === 'do' && type in this.undoneTasks[social]) {
debug('添加到未完成任务', {
social: social,
type: type,
link: link
});
this.undoneTasks[social][type].push(link);
}
if (action === 'undo' && type in this.socialTasks[social]) {
debug('添加到社交任务', {
social: social,
type: type,
link: link
});
this.socialTasks[social][type].push(link);
}
}
this.undoneTasks = this.uniqueTasks(this.undoneTasks);
this.socialTasks = this.uniqueTasks(this.socialTasks);
debug('任务分类完成', {
undoneTasks: this.undoneTasks,
socialTasks: this.socialTasks
});
return true;
} catch (error) {
debug('任务分类失败', {
error: error
});
throwError(error, 'Keylol.classifyTask');
return false;
}
}
selectAll() {
try {
debug('选择所有可见任务');
const tasks = $('.auto-task-keylol:visible');
tasks.attr('selected', 'selected');
debug('选择完成', {
count: tasks.length
});
} catch (error) {
debug('选择所有任务失败', {
error: error
});
throwError(error, 'Keylol.selectAll');
}
}
selectNone() {
try {
debug('取消选择所有可见任务');
const tasks = $('.auto-task-keylol:visible');
tasks.removeAttr('selected');
debug('取消选择完成', {
count: tasks.length
});
} catch (error) {
debug('取消选择所有任务失败', {
error: error
});
throwError(error, 'Keylol.selectNone');
}
}
invertSelect() {
try {
debug('反转选择所有可见任务');
const tasks = $('.auto-task-keylol:visible');
tasks.each(((_, element) => {
const $element = $(element);
if ($element.attr('selected')) {
$element.removeAttr('selected');
} else {
$element.attr('selected', 'selected');
}
}));
debug('反转选择完成', {
count: tasks.length
});
} catch (error) {
debug('反转选择任务失败', {
error: error
});
throwError(error, 'Keylol.invertSelect');
}
}
#addBtn(before, social, linkType, link) {
try {
debug('添加任务按钮', {
social: social,
linkType: linkType,
link: link
});
if (!before || !social || !linkType || !link) {
debug('跳过无效按钮参数');
return;
}
const button = $('<a>', {
href: 'javascript:void(0);',
class: 'auto-task-keylol',
target: '_self',
'data-social': social,
'data-type': linkType,
'data-link': link,
text: linkType.replace('Links', ''),
onclick: 'this.getAttribute("selected") ? this.removeAttribute("selected") : this.setAttribute("selected", "selected")'
});
$(before).after(button);
debug('按钮添加成功');
} catch (error) {
debug('添加按钮失败', {
error: error,
social: social,
linkType: linkType
});
throwError(error, `keylol.addBtn: ${social}/${linkType}`);
}
}
}
const defaultTasks$3 = {
steam: {
groupLinks: [],
wishlistLinks: [],
followLinks: [],
curatorLikeLinks: [],
playTimeLinks: []
},
twitter: {
userLinks: [],
retweetLinks: []
},
discord: {
serverLinks: []
}
};
class Opquests extends Website {
name='Opquests';
undoneTasks={
...defaultTasks$3
};
buttons=[ 'doTask', 'verifyTask', 'getKey' ];
static test() {
const {host: host} = window.location;
const isMatch = host === 'opquests.com';
debug('检查网站匹配', {
host: host,
isMatch: isMatch
});
return isMatch;
}
async after() {
try {
debug('开始执行后续操作');
if (!this.#checkLogin()) {
debug('检查登录失败');
echoLog({}).warning(I18n('checkLoginFailed'));
}
const opquestsVerifyTasks = GM_getValue('opquestsVerifyTasks') || [];
debug('获取待验证任务', {
count: opquestsVerifyTasks.length
});
if (opquestsVerifyTasks.length > 0) {
const taskId = opquestsVerifyTasks.pop();
debug('处理任务', {
taskId: taskId
});
GM_setValue('opquestsVerifyTasks', opquestsVerifyTasks);
const [verifyBtn] = $(`#task_id[value="${taskId}"]`).parent().find('button[type="button"],button[type="submit"]').has('i.fa-check');
if (verifyBtn) {
debug('点击验证按钮');
verifyBtn.click();
return;
}
debug('未找到验证按钮,继续处理下一个任务');
this.after();
return;
}
if (GM_getValue('opquestsVerifyTasks')) {
debug('清除验证任务缓存');
GM_deleteValue('opquestsVerifyTasks');
}
} catch (error) {
debug('后续操作失败', {
error: error
});
throwError(error, 'Opquests.after');
}
}
init() {
try {
debug('开始初始化');
const logStatus = echoLog({
text: I18n('initing')
});
if ($('a[href*="/auth/redirect"]').length > 0) {
debug('需要登录');
window.open('/auth/redirect', '_self');
logStatus.warning(I18n('needLogin'));
return false;
}
if (!this.#getGiveawayId()) {
debug('获取抽奖ID失败');
return false;
}
this.initialized = true;
debug('初始化完成');
logStatus.success();
return true;
} catch (error) {
debug('初始化失败', {
error: error
});
throwError(error, 'Opquests.init');
return false;
}
}
async classifyTask(action) {
try {
debug('开始分类任务', {
action: action
});
if (action === 'undo') {
debug('不支持撤销操作');
echoLog({
text: I18n('cannotUndo')
});
return false;
}
const logStatus = echoLog({
text: I18n('getTasksInfo')
});
const tasks = $('.w-full:contains("Validate") .items-center');
debug('找到任务', {
count: tasks.length
});
for (const task of tasks) {
const link = $(task).find('a:contains("Open")').attr('href');
if (!link) {
debug('跳过无链接任务');
continue;
}
const taskDes = $(task).find('div').eq(1).text().trim();
debug('处理任务', {
taskDes: taskDes,
link: link
});
if (/steamcommunity\.com\/groups\//.test(link)) {
debug('添加 Steam 组任务');
this.undoneTasks.steam.groupLinks.push(link);
continue;
}
if (/store\.steampowered\.com\/app\//.test(link)) {
if (/wishlist/gim.test(taskDes)) {
debug('添加 Steam 愿望单任务');
this.undoneTasks.steam.wishlistLinks.push(link);
} else if (/follow/gim.test(taskDes)) {
debug('添加 Steam 关注任务');
this.undoneTasks.steam.followLinks.push(link);
} else if (/play/gim.test(taskDes)) {
const time = parseInt(taskDes.replace(/\s/gim, '').match(/(\d+)hours/im)?.[1] || '0', 10) * 60;
debug('添加 Steam 游戏时长任务', {
time: time
});
this.undoneTasks.steam.playTimeLinks.push(`${time}-${link}`);
}
continue;
}
if (/store\.steampowered\.com\/(publisher|developer|curator)\//.test(link) && /follow/gim.test(taskDes)) {
debug('添加 Steam 鉴赏家关注任务');
this.undoneTasks.steam.curatorLikeLinks.push(link);
continue;
}
if (link.includes('//x.com/')) {
if (/follow/gim.test(taskDes)) {
debug('添加 Twitter 关注任务');
this.undoneTasks.twitter.userLinks.push(link);
continue;
}
if (link.includes('status') && /Repost/gim.test(taskDes)) {
debug('添加 Twitter 转发任务');
this.undoneTasks.twitter.retweetLinks.push(link);
continue;
}
}
if (link.includes('//discord.com/')) {
if (/join/gim.test(taskDes)) {
debug('添加 Discord 加入任务');
this.undoneTasks.discord.serverLinks.push(link);
continue;
}
}
if (/clash\.gg/.test(link)) {
debug('跳过不支持的 Clash.gg 任务');
echoLog({}).warning(`${I18n('unSupporttedTaskType')}: ${taskDes}(${link})`);
continue;
}
debug('未知任务类型');
echoLog({}).warning(`${I18n('unKnownTaskType')}: ${taskDes}(${link})`);
}
debug('任务分类完成');
logStatus.success();
this.undoneTasks = this.uniqueTasks(this.undoneTasks);
if (window.DEBUG) {
console.log('%cAuto-Task[Debug]:', 'color:blue', JSON.stringify(this));
}
return true;
} catch (error) {
debug('任务分类失败', {
error: error
});
throwError(error, 'Opquests.classifyTask');
return false;
}
}
async verifyTask() {
try {
debug('开始验证任务');
if (!this.initialized) {
debug('未初始化,执行初始化');
this.init();
}
const tasks = $.makeArray($('.items-center').has('input[name="task_id"]')).map((ele => $(ele).find('input[name="task_id"]').val()));
debug('获取待验证任务', {
count: tasks.length
});
GM_setValue('opquestsVerifyTasks', tasks);
await this.#confirm();
debug('执行后续操作');
this.after();
return true;
} catch (error) {
debug('验证任务失败', {
error: error
});
throwError(error, 'Opquests.verifyTask');
return false;
}
}
async #confirm() {
try {
debug('开始确认任务');
const logStatus = echoLog({
html: `<li>${I18n('confirmingTask')}...<font></font></li>`
});
debug('发送确认请求');
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://opquests.com/quests/${this.giveawayId}?confirm=1`,
method: 'GET',
nochche: true,
headers: {
origin: 'https://opquests.com',
referer: `https://opquests.com/warning?id=${this.giveawayId}`
}
});
if (result !== 'Success') {
debug('请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('响应错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('确认成功');
logStatus.success();
return true;
} catch (error) {
debug('确认任务失败', {
error: error
});
throwError(error, 'Opquests.confirm');
return false;
}
}
async getKey(isButton) {
try {
if ($('[name="task_id"]').length > 0) {
debug('有任务未完成,不获取密钥');
echoLog({}).warning(I18n('taskNotFinished'));
return false;
}
debug('开始获取密钥', {
isButton: isButton
});
const logStatus = echoLog({
text: I18n('gettingKey')
});
debug('发送获取密钥请求');
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://opquests.com/keys',
method: 'GET'
});
if (result !== 'Success') {
debug('请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (!data?.responseText) {
debug('响应无效', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const questTitle = $('h1.font-bold').text().trim().replace(' Quest', '');
const key = $(data.responseText).find(`div.items-center:contains("${questTitle}")`).find('div.font-bold').next().text();
debug('查找密钥', {
questTitle: questTitle,
hasKey: !!key
});
if (!key) {
debug('未找到密钥');
logStatus.error('Error: Key was not found');
if (isButton) {
debug('重定向到密钥页面');
window.open('https://opquests.com/keys', '_self');
}
return false;
}
debug('获取密钥成功');
logStatus.success();
echoLog({}).success(key);
return true;
} catch (error) {
debug('获取密钥失败', {
error: error
});
throwError(error, 'Opquests.getKey');
return false;
}
}
#getGiveawayId() {
try {
debug('开始获取抽奖ID');
const giveawayId = window.location.href.match(/quests\/([\d]+)/)?.[1];
if (giveawayId) {
this.giveawayId = giveawayId;
debug('获取抽奖ID成功', {
giveawayId: giveawayId
});
return true;
}
debug('获取抽奖ID失败');
echoLog({}).error(I18n('getFailed', 'GiveawayId'));
return false;
} catch (error) {
debug('获取抽奖ID出错', {
error: error
});
throwError(error, 'Opquests.getGiveawayId');
return false;
}
}
#checkLogin() {
try {
debug('检查登录状态');
if (!globalOptions.other.checkLogin) {
debug('跳过登录检查');
return true;
}
if ($('a[href*="/auth/redirect"]').length > 0) {
debug('未登录,重定向到登录页面');
window.open('/auth/redirect', '_self');
}
debug('登录检查完成');
return true;
} catch (error) {
debug('检查登录失败', {
error: error
});
throwError(error, 'Opquests.checkLogin');
return false;
}
}
}
const defaultTasksTemplate$2 = {
steam: {
groupLinks: [],
wishlistLinks: [],
followLinks: [],
curatorLinks: [],
curatorLikeLinks: [],
playTimeLinks: []
},
twitter: {
userLinks: [],
retweetLinks: []
},
twitch: {
channelLinks: []
},
discord: {
serverLinks: []
},
youtube: {
channelLinks: []
},
extra: {
gleam: []
}
};
const defaultTasks$2 = JSON.stringify(defaultTasksTemplate$2);
class Gleam extends Website {
name='Gleam';
undoneTasks=JSON.parse(defaultTasks$2);
socialTasks=JSON.parse(defaultTasks$2);
buttons=[ 'doTask', 'undoTask', 'verifyTask' ];
static test() {
const {host: host} = window.location;
const isMatch = host === 'gleam.io';
debug('检查网站匹配', {
host: host,
isMatch: isMatch
});
return isMatch;
}
before() {
try {
debug('重写全局对话框函数');
unsafeWindow.confirm = () => {};
unsafeWindow.alert = () => {};
unsafeWindow.prompt = () => {};
} catch (error) {
debug('重写全局对话框函数失败', {
error: error
});
throwError(error, 'Gleam.before');
}
}
async after() {
try {
debug('开始执行后续操作');
if (window.location.search.includes('8b07d23f4bfa65f9')) {
debug('检测到特殊查询参数,开始处理任务');
const checkComplete = setInterval((() => {
if ($('.entry-content .entry-method i.fa-check').length > 0) {
debug('任务已完成,关闭窗口');
clearInterval(checkComplete);
window.close();
}
}));
await this.verifyTask();
echoLog({}).warning(I18n('gleamTaskNotice'));
} else if (!await this.#checkLeftKey()) {
debug('检查剩余密钥失败');
echoLog({}).warning(I18n('checkLeftKeyFailed'));
}
} catch (error) {
debug('后续操作失败', {
error: error
});
throwError(error, 'Gleam.after');
}
}
init() {
try {
debug('初始化 Gleam');
const logStatus = echoLog({
text: I18n('initing')
});
if (!this.#getGiveawayId()) {
debug('获取抽奖ID失败');
return false;
}
this.initialized = true;
debug('初始化完成');
logStatus.success();
return true;
} catch (error) {
debug('初始化失败', {
error: error
});
throwError(error, 'Gleam.init');
return false;
}
}
async classifyTask(action) {
try {
debug('开始分类任务', {
action: action
});
const logStatus = echoLog({
text: I18n('getTasksInfo')
});
if (action === 'undo') {
debug('恢复已保存的任务信息');
this.socialTasks = GM_getValue(`gleamTasks-${this.giveawayId}`)?.tasks || JSON.parse(defaultTasks$2);
}
const tasks = $('.entry-content .entry-method');
debug('找到任务元素', {
count: tasks.length
});
for (const task of tasks) {
const $task = $(task);
if (action === 'do' && $task.find('i.fa-question').length === 0) {
debug('跳过已完成的任务');
continue;
}
const socialIcon = $task.find('.icon-wrapper i');
const taskInfo = $task.find('.user-links');
const taskText = taskInfo.text().trim();
const expandInfo = $task.find('.expandable');
const aElements = expandInfo.find('a.btn');
debug('处理任务', {
taskText: taskText
});
if (aElements.length > 0) {
debug('处理可点击元素', {
count: aElements.length
});
for (const element of aElements) {
const $element = $(element);
const href = $element.attr('href');
$element.removeAttr('href')[0].click();
$element.attr('href', href);
}
}
if (socialIcon.hasClass('fa-twitter') || socialIcon.hasClass('fa-x-twitter')) {
const link = $task.find('a[href^="https://twitter.com/"],a[href^="https://x.com/"]').attr('href');
if (!link) {
continue;
}
if (/follow/gi.test(taskText)) {
if (action === 'undo') {
this.socialTasks.twitter.userLinks.push(link);
}
if (action === 'do') {
this.undoneTasks.twitter.userLinks.push(link);
}
continue;
}
if (/retweet/gim.test(taskText)) {
if (action === 'undo') {
this.socialTasks.twitter.retweetLinks.push(link);
}
if (action === 'do') {
this.undoneTasks.twitter.retweetLinks.push(link);
}
continue;
}
}
if (socialIcon.hasClass('fa-twitch') && /follow/gim.test(taskText)) {
const link = $task.find('a[href^="https://twitch.tv/"]').attr('href');
if (!link) {
continue;
}
if (action === 'undo') {
this.socialTasks.twitch.channelLinks.push(link);
}
if (action === 'do') {
this.undoneTasks.twitch.channelLinks.push(link);
}
continue;
}
if (socialIcon.hasClass('fa-discord') && /join/gim.test(taskText)) {
let link = $task.find('a[href^="https://discord.com/invite/"]').attr('href');
if (!link) {
const ggLink = $task.find('a[href^="https://discord.gg/"]').attr('href')?.match(/discord\.gg\/([^/]+)/)?.[1];
if (!ggLink) {
continue;
}
link = `https://discord.com/invite/${ggLink}`;
}
if (action === 'undo') {
this.socialTasks.discord.serverLinks.push(link);
}
if (action === 'do') {
this.undoneTasks.discord.serverLinks.push(link);
}
continue;
}
if (socialIcon.hasClass('fa-external-link-square-alt')) {
continue;
}
if (socialIcon.hasClass('fa-youtube') && /subscribe/gim.test(taskText)) {
const link = $task.find('a[href^="https://www.youtube.com/channel/"]').attr('href');
if (!link) {
continue;
}
if (action === 'undo') {
this.socialTasks.youtube.channelLinks.push(link);
}
if (action === 'do') {
this.undoneTasks.youtube.channelLinks.push(link);
}
continue;
}
if (socialIcon.attr('class')?.includes('steam')) {
if (/join.*group/gi.test(taskText)) {
const link = $task.find('a[href^="https://steamcommunity.com/groups/"]').attr('href');
if (!link) {
continue;
}
if (action === 'undo') {
this.socialTasks.steam.groupLinks.push(link);
}
if (action === 'do') {
this.undoneTasks.steam.groupLinks.push(link);
}
continue;
}
if (/follow.*curator/gi.test(taskText)) {
const link = $task.find('a[href^="https://store.steampowered.com/curator/"]').attr('href');
if (!link) {
continue;
}
if (action === 'undo') {
this.socialTasks.steam.curatorLinks.push(link);
}
if (action === 'do') {
this.undoneTasks.steam.curatorLinks.push(link);
}
continue;
}
if (/play.*hours/gi.test(taskText)) {
const link = $task.find('a[href^="https://steamcommunity.com/app/"],a[href^="https://store.steampowered.com/app/"]').attr('href');
if (!link) {
continue;
}
if (action === 'undo') {
this.socialTasks.steam.playTimeLinks.push(link);
}
if (action === 'do') {
this.undoneTasks.steam.playTimeLinks.push(link);
}
continue;
}
if (/Sign up/gi.test(taskText)) {
continue;
}
}
if (socialIcon.hasClass('fa-bullhorn') && /Complete|Increase/gi.test(taskText)) {
if (action !== 'do') {
continue;
}
const gleamLink = await this.#getGleamLink(taskText);
if (!gleamLink) {
continue;
}
this.undoneTasks.extra.gleam.push(gleamLink);
continue;
}
if (socialIcon.hasClass('fa-question') || socialIcon.hasClass('fa-reddit') || socialIcon.hasClass('fa-instagram') || socialIcon.hasClass('fa-facebook-f') || socialIcon.hasClass('fa-telegram-plane') || socialIcon.hasClass('fa-telegram') || socialIcon.hasClass('fa-vk') || socialIcon.hasClass('fa-envelope') || socialIcon.hasClass('fa-gift') || socialIcon.hasClass('fa-square-up-right') || socialIcon.hasClass('fa-gamepad-modern') || socialIcon.hasClass('fa-dollar-sign') || socialIcon.hasClass('fa-tiktok') || socialIcon.hasClass('fa-gamepad-alt') || socialIcon.hasClass('fa-bag-shopping') || socialIcon.hasClass('fa-swords') || socialIcon.hasClass('fa-shield') && taskText.includes('one of our giveaways') || socialIcon.hasClass('fa-shield') && taskText.includes('Check out') || socialIcon.hasClass('fa-shield') && taskText.includes('vloot.io')) {
continue;
}
echoLog({}).warning(`${I18n('unKnownTaskType')}: ${taskText}`);
}
debug('任务分类完成');
logStatus.success();
this.undoneTasks = this.uniqueTasks(this.undoneTasks);
this.socialTasks = this.uniqueTasks(this.socialTasks);
debug('保存任务信息');
GM_setValue(`gleamTasks-${this.giveawayId}`, {
tasks: this.socialTasks,
time: (new Date).getTime()
});
return true;
} catch (error) {
debug('任务分类失败', {
error: error
});
throwError(error, 'Gleam.classifyTask');
return false;
}
}
async extraDoTask({gleam: gleam}) {
try {
debug('开始执行额外任务', {
count: gleam.length
});
const pro = [];
for (const link of gleam) {
pro.push(this.#doGleamTask(link));
}
return Promise.all(pro).then((() => true));
} catch (error) {
debug('执行额外任务失败', {
error: error
});
throwError(error, 'Gleam.extraDoTask');
return false;
}
}
async #checkCampaign() {
try {
debug('检测人机验证');
let logStatus;
if ($('[campaign-key="campaign.key"]').length > 0) {
logStatus = echoLog({
text: I18n('campaign')
});
debug('检测到人机验证');
await delay(3e3);
logStatus.warning(I18n('retry'));
await this.#checkCampaign();
return true;
}
logStatus?.success();
return false;
} catch (error) {
debug('检测人机验证失败', {
error: error
});
throwError(error, 'Gleam.checkCampaign');
return false;
}
}
async verifyTask() {
try {
debug('开始验证任务');
echoLog({
text: `${I18n('verifyingTask')}...`
});
const tasks = $('.entry-content .entry-method');
unsafeWindow._OxA = '_OxA';
for (const task of tasks) {
const campaign = await this.#checkCampaign();
if (campaign) {
return this.verifyTask();
}
const $task = $(task);
if ($task.find('i.fa-check').length > 0) {
debug('跳过已完成的任务');
continue;
}
debug('处理任务验证');
const taskInfo = $task.find('.user-links');
taskInfo[0].click();
const aElements = $task.find('.expandable').find('a.btn');
if (aElements.length > 0) {
debug('处理可点击元素', {
count: aElements.length
});
for (const element of aElements) {
const $element = $(element);
const href = $element.attr('href');
$element.removeAttr('href')[0].click();
$element.attr('href', href);
}
}
debug('处理计时器');
unsafeWindow.$hookTimer?.setSpeed(1e3);
const visitBtn = $task.find('.expandable').find('span:contains(more seconds),button:contains(more seconds)').filter(':visible');
if (visitBtn.length > 0 && unsafeWindow.$hookTimer) {
debug('处理访问按钮');
const newTab = GM_openInTab('', {
active: true
});
await delay(1e3);
newTab?.close();
window.focus();
}
await delay(3e3);
unsafeWindow.$hookTimer?.setSpeed(1);
const expandInfo = $task.find('.expandable');
const [input] = expandInfo.find('input');
if (input) {
debug('处理输入框');
const evt = new Event('input', {
bubbles: true,
cancelable: true,
composed: true
});
const valuelimit = [ ...expandInfo.text().matchAll(/"(.+?)"/g) ].at(-1)?.[1];
input.value = valuelimit || 'vloot';
input.dispatchEvent(evt);
await delay(1e3);
}
await this.#checkSync();
const continueBtn = $task.find('.expandable').find('span:contains(Continue),button:contains(Continue),a:contains(Continue)');
for (const button of continueBtn) {
debug('点击继续按钮');
button.click();
await delay(500);
await this.#checkSync();
}
}
debug('任务验证完成');
echoLog({
text: I18n('verifiedGleamTasks')
});
return true;
} catch (error) {
debug('任务验证失败', {
error: error
});
throwError(error, 'Gleam.verifyTask');
return false;
}
}
async #checkSync() {
try {
debug('开始检查同步状态');
return await new Promise((resolve => {
const checker = setInterval((() => {
if ($('.entry-content .entry-method i.fa-sync').length === 0) {
debug('同步完成');
clearInterval(checker);
resolve(true);
}
}), 500);
}));
} catch (error) {
debug('检查同步状态失败', {
error: error
});
throwError(error, 'Gleam.checkSync');
return false;
}
}
async #doGleamTask(link) {
try {
debug('执行 Gleam 任务', {
link: link
});
const logStatus = echoLog({
text: I18n('doingGleamTask')
});
return await new Promise((resolve => {
GM_openInTab(`${link}?8b07d23f4bfa65f9`, {
active: true,
insert: true,
setParent: true
}).onclose = () => {
debug('任务完成');
logStatus.success();
resolve(true);
};
}));
} catch (error) {
debug('执行 Gleam 任务失败', {
error: error
});
throwError(error, 'Gleam.doGleamTask');
return false;
}
}
#getGiveawayId() {
try {
debug('获取抽奖ID');
const giveawayId = window.location.pathname;
if (giveawayId) {
this.giveawayId = giveawayId;
debug('获取抽奖ID成功', {
giveawayId: giveawayId
});
return true;
}
debug('获取抽奖ID失败');
echoLog({
text: I18n('getFailed', 'GiveawayId')
});
return false;
} catch (error) {
debug('获取抽奖ID出错', {
error: error
});
throwError(error, 'Gleam.getGiveawayId');
return false;
}
}
async #getGleamLink(title) {
try {
debug('获取 Gleam 链接', {
title: title
});
const logStatus = echoLog({
text: I18n('gettingGleamLink')
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: 'https://www.vloot.io/api/v1/giveaways',
method: 'GET',
responseType: 'json'
});
if (result === 'Success') {
if (data?.status === 200 && data?.response?.Success === true && data?.response?.Data) {
const {link: link} = data.response.Data.find((giveaway => title.replace(/[\s]/g, '').toLowerCase().includes(giveaway.title.replace(/[\s]/g, '').toLowerCase()))) || {};
if (link) {
debug('获取链接成功', {
link: link
});
logStatus.success();
return link;
}
debug('获取链接失败');
logStatus.error(`Error:${I18n('getLinkFailed')}`);
return false;
}
debug('API响应错误', {
status: data?.status,
statusText: data?.statusText
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('请求失败', {
result: result,
status: status,
statusText: statusText
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
} catch (error) {
debug('获取 Gleam 链接失败', {
error: error
});
throwError(error, 'Gleam.getGleamLink');
return false;
}
}
async #checkLeftKey() {
try {
debug('检查剩余密钥');
if (!globalOptions.other.checkLeftKey) {
debug('跳过密钥检查');
return true;
}
const campaignString = $('div.popup-blocks-container').attr('ng-init')?.match(/initCampaign\(([\w\W]+?)\)$/)?.[1];
if (!campaignString) {
debug('未找到活动配置信息');
return false;
}
const {campaign: campaign, incentive: incentive} = JSON.parse(campaignString);
const controllerString = $('div.campaign.reward').attr('ng-init')?.match(/initContestant\(([\w\W]+?)\);/)?.[1];
let ownedKey = false;
if (controllerString) {
if (JSON.parse(controllerString).contestant?.claims?.incentives?.[incentive.id]?.length) {
debug('用户已拥有密钥');
ownedKey = true;
}
}
const isGiveawayInvalid = campaign.banned || campaign.finished && !ownedKey || campaign.paused || (new Date).getTime() < campaign.starts_at * 1e3;
debug('检查抽奖状态', {
banned: campaign.banned,
finished: campaign.finished,
ownedKey: ownedKey,
paused: campaign.paused,
notStarted: (new Date).getTime() < campaign.starts_at * 1e3
});
if (!isGiveawayInvalid) {
return true;
}
debug('抽奖无效,显示确认对话框');
const {value: value} = await Swal.fire({
icon: 'warning',
title: I18n('notice'),
text: I18n('giveawayNotWork'),
confirmButtonText: I18n('confirm'),
cancelButtonText: I18n('cancel'),
showCancelButton: true
});
if (value) {
debug('用户确认关闭窗口');
window.close();
}
return true;
} catch (error) {
debug('检查剩余密钥失败', {
error: error
});
throwError(error, 'Gleam.checkLeftKey');
return false;
}
}
}
const defaultOptions = {
username: '',
email: ''
};
class SweepWidget extends Website {
name='SweepWidget';
options={
...defaultOptions,
...GM_getValue('SweepWidgetOptions')
};
buttons=[ 'doTask' ];
static test() {
const {host: host} = window.location;
const isMatch = /^sweepwidget\.com$/.test(host);
debug('检查网站匹配', {
host: host,
isMatch: isMatch
});
return /^https?:\/\/sweepwidget\.com\/view\/[\d]+/.test(window.location.href);
}
async after() {
try {
debug('开始执行后续操作');
if (!this.#checkLogin()) {
debug('检查登录失败');
echoLog({}).warning(I18n('checkLoginFailed'));
}
} catch (error) {
debug('后续操作失败', {
error: error
});
throwError(error, 'SweepWidget.after');
}
}
init() {
try {
debug('开始初始化');
const logStatus = echoLog({
text: I18n('initing')
});
if (!this.#checkLogin()) {
debug('需要登录');
logStatus.warning(I18n('needLogin'));
return false;
}
if (!this.#getGiveawayId()) {
debug('获取抽奖ID失败');
return false;
}
this.initialized = true;
debug('初始化完成');
logStatus.success();
return true;
} catch (error) {
debug('初始化失败', {
error: error
});
throwError(error, 'SweepWidget.init');
return false;
}
}
classifyTask() {
debug('任务分类完成');
return true;
}
async doTask() {
try {
debug('开始执行任务');
if ($('#unlock_rewards_main_wrapper').length === 0) {
debug('未显示奖励界面,尝试登录');
if ($('input[name="sw__login_name"]:visible').length > 0) {
debug('填写用户名', {
username: this.options.username
});
$('input[name="sw__login_name"]').val(this.options.username);
}
if ($('input[name="sw__login_email"]:visible').length > 0) {
debug('填写邮箱', {
email: this.options.email
});
$('input[name="sw__login_email"]').val(this.options.email);
}
if ($('#sw_login_button:visible').length > 0) {
debug('点击登录按钮');
$('#sw_login_button')[0].click();
}
const isEntered = await this.#checkEnter();
if (!isEntered) {
debug('进入抽奖失败');
return false;
}
}
const logStatus = echoLog({
text: I18n('SweepWidgetNotice')
});
const tasks = $('#sw_inner_entry_methods_l2_wrapper>div.sw_entry');
debug('找到任务列表', {
count: tasks.length
});
for (const task of tasks) {
const $task = $(task);
if ($task.find('i.fa-check:visible').length > 0) {
debug('跳过已完成的任务');
continue;
}
const title = $task.find('.sw_text_inner');
const aElement = $task.find('a.sw_link');
const link = aElement.attr('href');
if (!link) {
debug('跳过无效链接的任务');
continue;
}
debug('处理任务', {
title: title.text(),
link: link
});
title[0].click();
aElement.attr('href', '#a').attr('target', '_self');
aElement[0]?.click();
await delay(300);
aElement.attr('href', link).attr('target', '_blank');
debug('填写测试文本');
$task.find('input[type="text"]').val('test');
const verifyBtn = $task.find('input.sw_verify');
if (verifyBtn.prop('disabled') === true) {
debug('验证按钮被禁用,尝试重新激活');
title[0].click();
await delay(300);
title[0].click();
await delay(300);
}
debug('点击验证按钮');
$task.find('input.sw_verify').removeAttr('disabled')[0]?.click();
await this.#checkFinish($task);
const randomDelay = parseInt(`${Math.random() * (3e3 - 1e3 + 1) + 1e3}`, 10);
debug('等待随机延迟', {
delay: randomDelay
});
await delay(randomDelay);
}
debug('所有任务执行完成');
logStatus.success();
return true;
} catch (error) {
debug('执行任务失败', {
error: error
});
throwError(error, 'SweepWidget.doTask');
return false;
}
}
#checkLogin() {
try {
debug('检查登录状态');
if ($('#twitter_login_button').length > 0) {
debug('点击 Twitter 登录按钮');
$('#twitter_login_button')[0].click();
}
debug('登录检查完成');
return true;
} catch (error) {
debug('检查登录失败', {
error: error
});
throwError(error, 'SweepWidget.checkLogin');
return false;
}
}
#getGiveawayId() {
try {
debug('开始获取抽奖ID');
const giveawayId = window.location.href.match(/\/view\/([\d]+)/)?.[1];
if (!giveawayId) {
debug('获取抽奖ID失败');
echoLog({
text: I18n('getFailed', 'GiveawayId')
});
return false;
}
this.giveawayId = giveawayId;
debug('获取抽奖ID成功', {
giveawayId: giveawayId
});
return true;
} catch (error) {
debug('获取抽奖ID出错', {
error: error
});
throwError(error, 'SweepWidget.getGiveawayId');
return false;
}
}
async #checkEnter() {
try {
debug('开始检查是否进入抽奖');
return new Promise((resolve => {
const checker = setInterval((() => {
if ($('#unlock_rewards_main_wrapper').length === 0) {
debug('等待进入抽奖...');
return;
}
debug('成功进入抽奖');
clearInterval(checker);
resolve(true);
}), 500);
}));
} catch (error) {
debug('检查进入抽奖失败', {
error: error
});
throwError(error, 'SweepWidget.checkEnter');
return false;
}
}
async #checkFinish($task) {
try {
debug('开始检查任务完成状态');
return new Promise((resolve => {
const checker = setInterval((() => {
const isCompleted = $task.find('i.fa-check:visible').length > 0 || $task.find('.sw_entry_input:visible').length === 0;
if (!isCompleted) {
debug('等待任务完成...');
return;
}
debug('任务完成');
clearInterval(checker);
resolve(true);
}), 500);
}));
} catch (error) {
debug('检查任务完成状态失败', {
error: error
});
throwError(error, 'SweepWidget.checkFinish');
return false;
}
}
}
const processFormData = formData => {
debug('开始处理表单数据', {
formDataLength: formData.length
});
const data = {};
formData.forEach((value => {
data[value.name] = value.value;
}));
debug('表单数据处理完成', {
processedData: data
});
return data;
};
const updateGlobalOption = (element, data) => {
const name = $(element).attr('name');
if (!name) {
debug('元素缺少name属性', {
element: element
});
return;
}
debug('开始更新全局选项', {
name: name
});
const keys = name.split('.');
const value = data[name];
const processedValue = value ? value === 'on' ? true : value : value ?? false;
debug('处理选项值', {
keys: keys,
originalValue: value,
processedValue: processedValue
});
if (keys.length === 3) {
globalOptions[keys[0]][keys[1]][keys[2]] = processedValue;
debug('更新三级选项', {
path: keys.join('.'),
value: processedValue
});
} else if (keys.length === 2) {
globalOptions[keys[0]][keys[1]] = processedValue;
debug('更新二级选项', {
path: keys.join('.'),
value: processedValue
});
}
};
const generateFormRow = (type, option, data, isFirstOption, totalOptions) => {
debug('开始生成表单行', {
type: type,
option: option,
isFirstOption: isFirstOption,
totalOptions: totalOptions
});
const backgroundColor = `${stringToColour(type)}44`;
const headerBackgroundColor = `${stringToColour(type)}66`;
if ([ 'other', 'position', 'hotKey', 'ASF' ].includes(type)) {
const header = isFirstOption ? `<th rowspan="${totalOptions}" style="background-color: ${headerBackgroundColor}">${I18n(type)}</th>` : '';
if (typeof data === 'boolean') {
debug('生成布尔类型选项行', {
type: type,
option: option,
value: data
});
return `\n <tr style="background-color: ${backgroundColor}">\n ${header}\n <td>${I18n(option)}</td>\n <td>\n <label>\n <input type="checkbox" name="${type}.${option}"${data ? ' checked="checked"' : ''}/>\n <span><i></i></span>\n </label>\n </td>\n </tr>`;
}
debug('生成文本类型选项行', {
type: type,
option: option,
value: data
});
return `\n <tr style="background-color: ${backgroundColor}">\n ${header}\n <td>${I18n(option)}</td>\n <td>\n <input class="editOption" type="text" name="${type}.${option}" value="${data}"/>\n </td>\n </tr>`;
}
debug('生成社交媒体选项行', {
type: type,
option: option,
dataKeys: Object.keys(data)
});
return Object.entries(data).map((([socialType, value]) => `\n <tr style="background-color: ${stringToColour(option)}66">\n ${isFirstOption ? `<th rowspan="${totalOptions}" style="background-color: ${headerBackgroundColor}">${I18n(type)}</th>` : ''}\n <td>${option}.${I18n(socialType)}</td>\n <td>\n <label>\n <input type="checkbox" name="${type}.${option}.${socialType}"${value ? ' checked="checked"' : ''}/>\n <span><i></i></span>\n </label>\n </td>\n </tr>`)).join('');
};
const generateGlobalOptionsForm = () => {
debug('开始生成全局选项表单');
const formRows = Object.entries(globalOptions).map((([type, data1]) => {
debug('处理选项类型', {
type: type,
optionsCount: Object.keys(data1).length
});
return Object.entries(data1).map((([option, data2], index) => {
const totalOptions = [ 'other', 'position', 'hotKey', 'ASF' ].includes(type) ? Object.keys(data1).length : Object.values(data1).reduce(((acc, cur) => acc + Object.keys(cur).length), 0);
return generateFormRow(type, option, data2, index === 0, totalOptions);
})).join('');
})).join('');
debug('表单生成完成');
return `\n <form id="globalOptionsForm" class="auto-task-form">\n <table class="auto-task-table">\n <thead>\n <tr>\n <td>${I18n('type')}</td>\n <td>${I18n('option')}</td>\n <td>${I18n('value')}</td>\n </tr>\n </thead>\n <tbody>\n ${formRows}\n </tbody>\n </table>\n </form>`;
};
const saveData = () => {
try {
debug('开始保存全局选项数据');
const formData = $('#globalOptionsForm').serializeArray();
debug('获取表单数据', {
formDataLength: formData.length
});
const data = processFormData(formData);
debug('开始更新全局选项');
$.makeArray($('#globalOptionsForm input')).forEach((element => {
updateGlobalOption(element, data);
}));
GM_setValue('globalOptions', globalOptions);
debug('全局选项保存完成');
Swal.fire({
title: I18n('changeGlobalOptionsSuccess'),
icon: 'success'
});
} catch (error) {
debug('保存全局选项时发生错误', {
error: error
});
throwError(error, 'saveData');
}
};
const changeGlobalOptions = showType => {
try {
debug('开始显示全局选项配置界面', {
showType: showType
});
const formHtml = generateGlobalOptionsForm();
if (showType === 'swal') {
debug('使用弹窗显示选项');
Swal.fire({
title: I18n('globalOptions'),
html: formHtml,
showConfirmButton: true,
confirmButtonText: I18n('save'),
showCancelButton: true,
cancelButtonText: I18n('close')
}).then((({isConfirmed: isConfirmed}) => {
if (isConfirmed) {
debug('用户确认保存选项');
saveData();
} else {
debug('用户取消保存选项');
}
}));
} else {
debug('使用页面内显示选项');
$('body').append(`<h2>${I18n('globalOptions')}</h2>${formHtml}`);
}
} catch (error) {
debug('显示全局选项配置界面时发生错误', {
error: error
});
throwError(error, 'changeGlobalOptions');
}
};
const defaultWhiteList = {
discord: {
servers: []
},
instagram: {
users: []
},
twitch: {
channels: []
},
twitter: {
users: [],
retweets: [],
likes: []
},
vk: {
names: []
},
youtube: {
channels: [],
likes: []
},
reddit: {
reddits: []
},
steam: {
groups: [],
officialGroups: [],
wishlists: [],
follows: [],
forums: [],
workshops: [],
curators: [],
workshopVotes: [],
curatorLikes: [],
announcements: [],
licenses: [],
playtests: [],
playTime: []
}
};
const REGEX_PATTERNS = {
DISCORD_INVITE: /invite\/(.+)/,
INSTAGRAM_USER: /https:\/\/www\.instagram\.com\/(.+)?\//,
TWITCH_CHANNEL: /https:\/\/(www\.)?twitch\.tv\/(.+)/,
TWITTER_USER: /https:\/\/twitter\.com\/(.+)/,
TWITTER_STATUS: /https:\/\/twitter\.com\/.*?\/status\/([\d]+)/,
VK_NAME: /https:\/\/vk\.com\/([^/]+)/,
REDDIT_USER: /https?:\/\/www\.reddit\.com\/user\/([^/]*)/,
REDDIT_SUBREDDIT: /https?:\/\/www\.reddit\.com\/r\/([^/]*)/,
STEAM_GROUP: /groups\/(.+)\/?/,
STEAM_APP: /app\/([\d]+)/,
STEAM_WORKSHOP: /\?id=([\d]+)/,
STEAM_CURATOR: /curator\/([\d]+)/,
STEAM_STORE: /https?:\/\/store\.steampowered\.com\/(.*?)\/([^/?]+)/
};
const link2id = async function(type) {
try {
debug('开始从链接获取ID', {
type: type
});
const link = $('#socialLink').val();
let id = '';
switch (type) {
case 'discord.servers':
id = REGEX_PATTERNS.DISCORD_INVITE.exec(link)?.[1] || '';
break;
case 'instagram.users':
id = REGEX_PATTERNS.INSTAGRAM_USER.exec(link)?.[1] || '';
break;
case 'twitch.channels':
id = REGEX_PATTERNS.TWITCH_CHANNEL.exec(link)?.[2] || '';
break;
case 'twitter.users':
id = REGEX_PATTERNS.TWITTER_USER.exec(link)?.[1] || '';
break;
case 'twitter.retweets':
id = REGEX_PATTERNS.TWITTER_STATUS.exec(link)?.[1] || '';
break;
case 'vk.names':
id = REGEX_PATTERNS.VK_NAME.exec(link)?.[1] || '';
break;
case 'youtube.channels':
id = (await getInfo(link, 'channel'))?.params?.channelId || '';
break;
case 'youtube.likes':
id = (await getInfo(link, 'likeVideo'))?.params?.videoId || '';
break;
case 'reddit.reddits':
{
const userMatch = REGEX_PATTERNS.REDDIT_USER.exec(link);
const subredditMatch = REGEX_PATTERNS.REDDIT_SUBREDDIT.exec(link);
id = userMatch?.[1] || subredditMatch?.[1] || '';
break;
}
case 'steam.groups':
id = REGEX_PATTERNS.STEAM_GROUP.exec(link)?.[1] || '';
break;
case 'steam.wishlists':
case 'steam.follows':
case 'steam.forums':
case 'steam.playtests':
case 'steam.playTime':
id = REGEX_PATTERNS.STEAM_APP.exec(link)?.[1] || '';
break;
case 'steam.workshops':
id = REGEX_PATTERNS.STEAM_WORKSHOP.exec(link)?.[1] || '';
break;
case 'steam.curators':
{
if (link.includes('curator')) {
id = REGEX_PATTERNS.STEAM_CURATOR.exec(link)?.[1] || '';
} else {
const storeMatch = REGEX_PATTERNS.STEAM_STORE.exec(link);
if (!storeMatch) {
break;
}
const [, param1, param2] = storeMatch;
const steam = new Steam;
if (await steam.init()) {
id = await steam.getCuratorId(param1, param2) || '';
}
}
break;
}
}
debug('从链接获取ID结果', {
type: type,
id: id
});
return id;
} catch (error) {
debug('从链接获取ID时发生错误', {
error: error
});
throwError(error, 'link2id');
return I18n('getFailed', 'id');
}
};
const disabledType = {
steam: [ 'workshopVotes', 'curatorLikes', 'announcements' ],
twitter: [ 'likes' ]
};
const assignWhiteList = whiteList => {
try {
debug('开始合并白名单');
const newWhiteList = {};
for (const [key, value] of Object.entries(defaultWhiteList)) {
newWhiteList[key] = {
...value,
...whiteList[key]
};
}
debug('白名单合并完成');
return newWhiteList;
} catch (error) {
debug('合并白名单时发生错误', {
error: error
});
throwError(error, 'assignWhiteList');
return defaultWhiteList;
}
};
const whiteListOptions = function(showType) {
try {
debug('开始显示白名单选项', {
showType: showType
});
const whiteList = assignWhiteList(GM_getValue('whiteList') || {});
let whiteListOptionsForm = `<form id="whiteListForm" class="auto-task-form">\n <table class="auto-task-table">\n <thead>\n <tr>\n <td>${I18n('website')}</td>\n <td>${I18n('type')}</td>\n <td>${I18n('edit')}</td>\n </tr>\n </thead>\n <tbody>`;
for (const [social, types] of Object.entries(whiteList)) {
const validTypes = Object.keys(types).filter((type => !disabledType[social]?.includes(type)));
whiteListOptionsForm += validTypes.map(((type, index) => {
const bgColor = `${stringToColour(social)}66`;
return `\n <tr style="background-color: ${bgColor}">\n ${index === 0 ? `<th rowspan="${validTypes.length}" style="background-color: ${bgColor}">${social}</th>` : ''}\n <td>${I18n(type)}</td>\n <td><button type="button" class="editWhiteList" data-value="${social}.${type}">${I18n('edit')}</button></td>\n </tr>`;
})).join('');
}
whiteListOptionsForm += '</tbody></table></form>';
if (showType === 'swal') {} else {
debug('使用页面显示白名单选项');
$('body').append(`<h2>${I18n('whiteList')}</h2>${whiteListOptionsForm}`);
}
$('.editWhiteList').on('click', (function() {
const value = $(this).attr('data-value');
if (!value) {
return;
}
const [social, type] = value.split('.');
const currentList = whiteList[social]?.[type];
if (!currentList) {
debug('未找到白名单配置', {
social: social,
type: type
});
echoLog({}).warning(I18n('whiteListNotFound', value));
return;
}
debug('编辑白名单', {
social: social,
type: type
});
Swal.fire({
title: I18n('changeWhiteListOption', value),
input: 'textarea',
html: `\n <input id="socialLink" class="swal2-input" placeholder="在此处输入链接获取id">\n <button id="link2id" data-type="${value}" class="swal2-confirm swal2-styled">获取id</button>\n <p style="margin-bottom:0 !important;">在下方填写白名单,每行一个</p>\n `,
inputValue: currentList.join('\n'),
showConfirmButton: true,
confirmButtonText: I18n('save'),
showCancelButton: true,
cancelButtonText: I18n('close'),
showDenyButton: true,
denyButtonText: I18n('return')
}).then((({isDenied: isDenied, isConfirmed: isConfirmed, value: value}) => {
if (isDenied) {
debug('返回白名单选项');
if (showType === 'swal') {}
return;
}
if (isConfirmed && value) {
debug('保存白名单更改', {
social: social,
type: type,
value: value
});
whiteList[social][type] = value.split('\n').filter(Boolean);
GM_setValue('whiteList', whiteList);
Swal.fire({
title: I18n('changeWhiteListSuccess'),
icon: 'success'
});
}
}));
$('#link2id').on('click', (async function() {
const type = $(this).attr('data-type');
if (!type) {
return;
}
debug('从链接获取ID按钮点击', {
type: type
});
const id = await link2id(type);
$('#socialLink').val(id);
}));
}));
} catch (error) {
debug('显示白名单选项时发生错误', {
error: error
});
throwError(error, 'whiteListOptions');
}
};
const setGistData = async (token, gistId, fileName, content) => {
try {
debug('开始设置Gist数据', {
gistId: gistId,
fileName: fileName
});
const logStatus = echoLog({
text: I18n('settingData')
});
const contentData = JSON.stringify({
files: {
[fileName]: {
content: JSON.stringify(content)
}
}
});
debug('准备发送的数据', {
contentData: contentData
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://api.github.com/gists/${gistId}`,
headers: {
Accept: 'application/vnd.github.v3+json',
Authorization: `token ${token}`
},
data: contentData,
responseType: 'json',
method: 'POST',
timeout: 3e4
});
if (result !== 'Success') {
debug('设置Gist数据失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
const expectedContent = JSON.stringify(content);
if (data?.status !== 200 || data?.response?.files?.[fileName]?.content !== expectedContent) {
debug('设置Gist数据验证失败', {
status: data?.status,
content: data?.response?.files?.[fileName]?.content
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('设置Gist数据成功');
logStatus.success();
return true;
} catch (error) {
debug('设置Gist数据发生错误', {
error: error
});
throwError(error, 'setGistData');
return false;
}
};
const getGistData = async (token, gistId, fileName, test = false) => {
try {
debug('开始获取Gist数据', {
gistId: gistId,
fileName: fileName,
test: test
});
const logStatus = echoLog({
text: I18n('gettingData')
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://api.github.com/gists/${gistId}`,
headers: {
Accept: 'application/vnd.github.v3+json',
Authorization: `token ${token}`
},
responseType: 'json',
method: 'GET',
timeout: 3e4
});
if (result !== 'Success') {
debug('获取Gist数据失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200) {
debug('获取Gist数据状态码错误', {
status: data?.status
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
const content = data.response?.files?.[fileName]?.content;
if (!content) {
debug('获取的Gist数据为空');
logStatus.error(`Error:${I18n('noRemoteData')}`);
return false;
}
if (test) {
debug('Gist数据测试成功');
logStatus.success();
return true;
}
try {
const formatedContent = JSON.parse(content);
debug('Gist数据解析成功', {
contentLength: Object.keys(formatedContent).length
});
logStatus.success();
return formatedContent;
} catch (error) {
debug('Gist数据解析失败', {
error: error
});
logStatus.error(`Error:${I18n('errorRemoteDataFormat')}`);
console.log('%c%s', 'color:white;background:red', `Auto-Task[Error]: getGistData\n${error.stack}`);
return false;
}
} catch (error) {
debug('获取Gist数据发生错误', {
error: error
});
throwError(error, 'getGistData');
return false;
}
};
const syncOptions = async () => {
try {
debug('开始同步选项配置');
const defaultOptions = {
TOKEN: '',
GIST_ID: '',
FILE_NAME: '',
SYNC_HISTORY: true
};
let syncOptions = GM_getValue('gistOptions') || defaultOptions;
debug('获取已保存的同步选项', syncOptions);
const createForm = options => `\n <div class="gist-options-form">\n <p>\n <label for="github-token">Github Token</label>\n <input\n id="github-token"\n class="swal2-input"\n placeholder="Github Token"\n value="${options.TOKEN}"\n autocomplete="off"\n spellcheck="false"\n />\n </p>\n <p>\n <label for="gist-id">Gist ID</label>\n <input\n id="gist-id"\n class="swal2-input"\n placeholder="Gist ID"\n value="${options.GIST_ID}"\n autocomplete="off"\n spellcheck="false"\n />\n </p>\n <p>\n <label for="file-name">${I18n('fileName')}</label>\n <input\n id="file-name"\n class="swal2-input"\n placeholder="${I18n('fileName')}"\n value="${options.FILE_NAME}"\n autocomplete="off"\n spellcheck="false"\n />\n </p>\n <p class="sync-history-wrapper">\n <label for="sync-history" class="swal2-checkbox-custom">\n <input\n id="sync-history"\n type="checkbox"\n ${options.SYNC_HISTORY ? 'checked="checked"' : ''}\n />\n <span class="swal2-label">${I18n('syncHistory')}</span>\n </label>\n </p>\n <div class="button-group">\n <button id="upload-data" type="button" class="swal2-confirm swal2-styled" onclick="handleUpload()">\n ${I18n('upload2gist')}\n </button>\n <button id="download-data" type="button" class="swal2-confirm swal2-styled" onclick="handleDownload()">\n ${I18n('downloadFromGist')}\n </button>\n </div>\n </div>\n `;
const showConfigDialog = async () => {
debug('显示配置对话框');
const result = await Swal.fire({
title: I18n('gistOptions'),
html: createForm(syncOptions),
focusConfirm: false,
showLoaderOnConfirm: true,
footer: `<a href="https://auto-task-doc.js.org/guide/#%E6%95%B0%E6%8D%AE%E5%90%8C%E6%AD%A5" target="_blank">${I18n('help')}</a>`,
preConfirm: async () => {
const options = {
TOKEN: $('#github-token').val().trim(),
GIST_ID: $('#gist-id').val().trim(),
FILE_NAME: $('#file-name').val().trim(),
SYNC_HISTORY: $('#sync-history').prop('checked')
};
debug('保存新的同步选项', options);
GM_setValue('gistOptions', options);
syncOptions = options;
return await getGistData(options.TOKEN, options.GIST_ID, options.FILE_NAME, true);
},
allowOutsideClick: () => !Swal.isLoading(),
confirmButtonText: I18n('saveAndTest'),
showCancelButton: true,
cancelButtonText: I18n('close')
});
if (result.value) {
debug('配置测试成功');
await Swal.fire({
icon: 'success',
title: I18n('testSuccess'),
timer: 2e3,
timerProgressBar: true
});
await showConfigDialog();
} else if (result.value !== undefined) {
debug('配置测试失败');
await Swal.fire({
icon: 'error',
title: I18n('testFailed'),
timer: 2e3,
timerProgressBar: true
});
await showConfigDialog();
}
};
const handleUpload = async () => {
debug('开始处理数据上传');
const options = GM_getValue('gistOptions');
if (!options?.TOKEN || !options?.GIST_ID || !options?.FILE_NAME) {
debug('配置信息不完整');
await Swal.fire({
icon: 'error',
title: I18n('saveAndTestNotice')
});
await showConfigDialog();
return;
}
debug('显示数据处理提示');
Swal.fire({
icon: 'info',
title: I18n('processingData'),
allowOutsideClick: false
});
const data = {};
const names = GM_listValues();
const syncHistory = $('#sync-history').prop('checked');
debug('开始收集数据', {
namesCount: names.length,
syncHistory: syncHistory
});
for (const name of names) {
if (name === 'gistOptions' || /^[\w]+?Auth$/.test(name)) {
continue;
}
if (!syncHistory && /^[\w]+?Tasks-/.test(name)) {
continue;
}
data[name] = GM_getValue(name);
}
debug('数据收集完成', {
dataKeysCount: Object.keys(data).length
});
Swal.update({
icon: 'info',
title: I18n('updatingData')
});
const success = await setGistData(options.TOKEN, options.GIST_ID, options.FILE_NAME, data);
debug('数据上传完成', {
success: success
});
await Swal.fire({
icon: success ? 'success' : 'error',
title: I18n(success ? 'syncDataSuccess' : 'syncDataFailed'),
timer: 2e3,
timerProgressBar: true
});
};
const handleDownload = async () => {
debug('开始处理数据下载');
const options = GM_getValue('gistOptions');
if (!options?.TOKEN || !options?.GIST_ID || !options?.FILE_NAME) {
debug('配置信息不完整');
await Swal.fire({
icon: 'error',
title: I18n('saveAndTestNotice')
});
await showConfigDialog();
return;
}
debug('显示数据下载提示');
Swal.fire({
icon: 'info',
title: I18n('downloadingData'),
allowOutsideClick: false
});
const data = await getGistData(options.TOKEN, options.GIST_ID, options.FILE_NAME);
if (!data || typeof data === 'boolean') {
debug('未检测到远程数据');
await Swal.fire({
icon: 'error',
title: I18n('checkedNoData')
});
await showConfigDialog();
return;
}
debug('开始保存数据');
Swal.update({
icon: 'info',
title: I18n('savingData')
});
const syncHistory = $('#sync-history').prop('checked');
let savedCount = 0;
for (const [name, value] of Object.entries(data)) {
if (!syncHistory && /^[\w]+?Tasks-/.test(name)) {
continue;
}
GM_setValue(name, value);
savedCount += 1;
}
debug('数据保存完成', {
savedCount: savedCount
});
await Swal.fire({
icon: 'success',
title: I18n('syncDataSuccess'),
timer: 2e3,
timerProgressBar: true
});
};
unsafeWindow.handleUpload = handleUpload;
unsafeWindow.handleDownload = handleDownload;
await showConfigDialog();
} catch (error) {
debug('同步选项发生错误', {
error: error
});
throwError(error, 'syncOptions');
await Swal.fire({
icon: 'error',
title: I18n('error'),
text: error instanceof Error ? error.message : 'Unknown error occurred',
timer: 3e3,
timerProgressBar: true
});
}
};
const VALID_SIDES_X = [ 'right', 'left' ];
const VALID_SIDES_Y = [ 'top', 'bottom' ];
class Setting {
name='Setting';
buttons=[ 'saveGlobalOptions', 'syncData', 'tasksHistory' ];
syncData=syncOptions;
selectors={
body: 'body',
autoTaskInfo: '#auto-task-info',
autoTaskButtons: '#auto-task-buttons',
showButtonDiv: 'div.show-button-div',
positionInputs: 'input[name^="position"]',
hotKeyInputs: 'input[name^="hotKey"]'
};
tasksHistory() {
debug('打开任务历史记录页面');
GM_openInTab('https://auto-task.hclonely.com/history.html', {
active: true
});
}
static test() {
const {host: host, pathname: pathname} = window.location;
const isMatch = [ 'auto-task.hclonely.com', 'auto-task-doc.js.org' ].includes(host) && pathname === '/setting.html';
debug('检查设置页面匹配', {
host: host,
pathname: pathname,
isMatch: isMatch
});
return isMatch;
}
before() {
try {
debug('开始清空页面内容');
$(this.selectors.body).html('').addClass('auto-task-options');
debug('页面内容已清空');
} catch (error) {
debug('清空页面内容失败', {
error: error
});
throwError(error, 'Setting.before');
}
}
async after() {
try {
debug('开始初始化设置页面');
await this.#initializeEnvironment();
this.#initializeGlobalSettings();
this.#setupSocialButtons();
this.#setupPositionHandlers();
this.#setupHotKeyHandlers();
debug('设置页面初始化完成');
} catch (error) {
debug('设置页面初始化失败', {
error: error
});
throwError(error, 'Setting.after');
}
}
saveGlobalOptions() {
try {
debug('开始保存全局选项');
saveData();
debug('全局选项保存完成');
} catch (error) {
debug('保存全局选项失败', {
error: error
});
throwError(error, 'Setting.saveGlobalOptions');
}
}
async #initializeEnvironment() {
try {
debug('开始初始化环境信息');
const userAgent = await browser.getInfo();
debug('获取用户代理信息', {
userAgent: userAgent
});
const environmentHtml = this.#generateEnvironmentHtml(userAgent);
$(this.selectors.body).append(`<h2>${I18n('environment')}</h2>${environmentHtml}`);
debug('环境信息初始化完成');
} catch (error) {
debug('初始化环境信息失败', {
error: error
});
throwError(error, 'Setting.initializeEnvironment');
}
}
#generateEnvironmentHtml(userAgent) {
return `\n <form id="environmentForm" class="auto-task-form">\n <table class="auto-task-table">\n <thead>\n <tr>\n <td>${I18n('type')}</td>\n <td>${I18n('name')}</td>\n <td>${I18n('version')}</td>\n </tr>\n </thead>\n <tbody>\n <tr>\n <td>${I18n('os')}</td>\n <td>${userAgent.system}</td>\n <td>${userAgent.systemVersion}</td>\n </tr>\n <tr>\n <td>${I18n('browser')}</td>\n <td>${userAgent.browser}</td>\n <td>${userAgent.browserVersion}</td>\n </tr>\n <tr>\n <td>${I18n('scriptManager')}</td>\n <td>${GM_info.scriptHandler}</td>\n <td>${GM_info.version}</td>\n </tr>\n <tr>\n <td>${I18n('script')}</td>\n <td>${GM_info.script.name}</td>\n <td>${GM_info.script.version}</td>\n </tr>\n </tbody>\n </table>\n </form>\n `;
}
#initializeGlobalSettings() {
debug('开始初始化全局设置');
changeGlobalOptions('page');
whiteListOptions('page');
debug('全局设置初始化完成');
}
#setupSocialButtons() {
debug('开始设置社交媒体按钮');
this.#addSocialButton('other.twitterVerifyId', 'getTwitterUserId', 'twitterUser');
this.#addSocialButton('other.youtubeVerifyChannel', 'getYoutubeChannelId', 'youtubeChannel');
debug('社交媒体按钮设置完成');
}
#addSocialButton(inputName, buttonId, socialType) {
debug('添加社交媒体按钮', {
inputName: inputName,
buttonId: buttonId,
socialType: socialType
});
$(`input[name="${inputName}"]`).after(`<button id="${buttonId}" type="button">${I18n(`get${buttonId}`)}</button>`);
$(`#${buttonId}`).on('click', (() => this.#getId(socialType)));
debug('社交媒体按钮添加完成');
}
#setupPositionHandlers() {
debug('开始设置位置处理器');
$(this.selectors.positionInputs).on('input', (event => {
const input = $(event.target);
const type = input.attr('name')?.replace('position.', '');
if (!type) {
debug('无效的位置类型');
return;
}
debug('处理位置变化', {
type: type
});
this.#handlePositionChange(type);
}));
debug('位置处理器设置完成');
}
#handlePositionChange(type) {
debug('开始处理位置变化', {
type: type
});
const config = this.#getPositionConfig(type);
if (!config) {
debug('获取位置配置失败');
return;
}
const {distance: distance, sideX: sideX, sideY: sideY} = config;
if (!this.#isValidPosition(distance, sideX, sideY)) {
debug('无效的位置配置', {
distance: distance,
sideX: sideX,
sideY: sideY
});
return;
}
const [x, y] = distance.split(',');
const target = this.#getPositionTarget(type);
if (!target) {
debug('获取目标元素失败');
return;
}
debug('更新元素位置', {
target: target,
sideX: sideX,
sideY: sideY,
x: x,
y: y
});
this.#updateElementPosition(target, sideX, sideY, x, y);
}
#getPositionConfig(type) {
debug('获取位置配置', {
type: type
});
const baseType = type.replace(/(?:button|log|show)(?:SideX|SideY|Distance)$/, '');
const distance = $(`input[name="position.${baseType}Distance"]`).val();
const sideX = $(`input[name="position.${baseType}SideX"]`).val();
const sideY = $(`input[name="position.${baseType}SideY"]`).val();
const config = {
distance: distance,
sideX: sideX,
sideY: sideY
};
debug('位置配置', config);
return config;
}
#isValidPosition(distance, sideX, sideY) {
const isValid = VALID_SIDES_X.includes(sideX) && VALID_SIDES_Y.includes(sideY) && /^[\d]+?,[\d]+$/.test(distance);
debug('验证位置配置', {
distance: distance,
sideX: sideX,
sideY: sideY,
isValid: isValid
});
return isValid;
}
#getPositionTarget(type) {
const targetMap = {
button: this.selectors.autoTaskButtons,
showButton: this.selectors.showButtonDiv,
log: this.selectors.autoTaskInfo
};
const baseType = type.replace(/(?:SideX|SideY|Distance)$/, '');
return targetMap[baseType];
}
#updateElementPosition(selector, sideX, sideY, x, y) {
debug('更新元素位置', {
selector: selector,
sideX: sideX,
sideY: sideY,
x: x,
y: y
});
const $element = $(selector);
const oppositeX = sideX === 'right' ? 'left' : 'right';
const oppositeY = sideY === 'top' ? 'bottom' : 'top';
$element.css(sideX, `${x}px`).css(sideY, `${y}px`).css(oppositeX, '').css(oppositeY, '');
debug('元素位置更新完成');
}
#setupHotKeyHandlers() {
debug('开始设置热键处理器');
$(this.selectors.hotKeyInputs).attr('readonly', 'readonly').off('keydown').on('keydown', this.#handleHotKeyPress);
debug('热键处理器设置完成');
}
#handleHotKeyPress(event) {
debug('处理热键按下事件', {
key: event.key
});
const functionKeys = [];
if (event.altKey) {
functionKeys.push('alt');
}
if (event.ctrlKey) {
functionKeys.push('ctrl');
}
if (event.shiftKey) {
functionKeys.push('shift');
}
const key = event.key.length === 1 ? event.key.toLowerCase() : '';
const value = functionKeys.length ? `${functionKeys.join(' + ')} + ${key}` : key;
debug('设置热键值', {
functionKeys: functionKeys,
key: key,
value: value
});
$(event.target).val(value);
}
async #getId(social) {
try {
debug('开始获取社交媒体ID', {
social: social
});
const result = await Swal.fire({
title: I18n('getId', I18n(social)),
html: this.#generateIdInputHtml(social),
showCancelButton: true,
cancelButtonText: I18n('close'),
showConfirmButton: false
});
if (!result.isDismissed) {
debug('用户确认获取ID');
await this.#handleIdRetrieval(social);
} else {
debug('用户取消获取ID');
}
} catch (error) {
debug('获取社交媒体ID失败', {
error: error
});
throwError(error, 'Setting.getId');
}
}
#generateIdInputHtml(social) {
return `\n <input id="socialLink" class="swal2-input" placeholder="在此处输入链接获取id">\n <button id="link2id" data-type="${social}" class="swal2-confirm swal2-styled">获取id</button>\n `;
}
async #handleIdRetrieval(social) {
const link = $('#socialLink').val();
if (!link) {
debug('链接为空');
return;
}
debug('开始处理ID获取', {
social: social,
link: link
});
let id = '';
if (social === 'twitterUser') {
const name = link.match(/https:\/\/twitter\.com\/(.+)/)?.[1] || link;
debug('获取Twitter用户ID', {
name: name
});
id = await (new Twitter).userName2id(name) || '';
} else if (social === 'youtubeChannel') {
const name = this.#extractYoutubeUrl(link);
debug('获取YouTube频道信息', {
name: name
});
const info = await getInfo(name, 'channel');
id = info?.params?.channelId || '';
}
debug('ID获取结果', {
id: id
});
$('#socialLink').val(id);
}
#extractYoutubeUrl(link) {
debug('提取YouTube URL', {
link: link
});
if (/^https:\/\/(www\.)?google\.com.*?\/url\?.*?url=https:\/\/www.youtube.com\/.*/.test(link)) {
const extractedUrl = link.match(/url=(https:\/\/www\.youtube\.com\/.*)/)?.[1] || link;
debug('从Google搜索结果提取URL', {
extractedUrl: extractedUrl
});
return extractedUrl;
}
return link;
}
}
class History extends Keylol {
name='History';
buttons=[ 'doTask', 'undoTask', 'selectAll', 'selectNone', 'invertSelect', 'clearHistory' ];
static test() {
try {
const {host: host, pathname: pathname} = window.location;
const isMatch = [ 'auto-task.hclonely.com', 'auto-task-doc.js.org' ].includes(host) && pathname === '/history.html';
debug('检查是否为历史记录页面', {
host: host,
pathname: pathname,
isMatch: isMatch
});
return isMatch;
} catch (error) {
debug('检查历史记录页面时出错', {
error: error
});
throwError(error, 'History.test');
return false;
}
}
before() {
try {
debug('开始初始化历史记录页面');
$('body').html('<div class="container"></div>').addClass('auto-task-history');
debug('页面基础结构已创建');
const data = GM_listValues() || [];
debug('获取存储的所有值', {
count: data.length
});
const tasksHistory = data.map((value => /^[\w]+?Tasks-/.test(value) ? value : null)).filter((value => value));
debug('筛选任务历史记录', {
count: tasksHistory.length
});
tasksHistory.forEach((item => {
debug('处理任务项', {
item: item
});
this.#addItem(item);
}));
debug('历史记录页面初始化完成');
} catch (error) {
debug('初始化历史记录页面时出错', {
error: error
});
throwError(error, 'History.before');
}
}
clearHistory() {
try {
debug('开始清除历史记录');
const data = GM_listValues() || [];
debug('获取存储的所有值', {
count: data.length
});
const tasksHistory = data.map((value => /^[\w]+?Tasks-/.test(value) ? value : null)).filter((value => value));
debug('筛选要清除的任务历史记录', {
count: tasksHistory.length
});
tasksHistory.forEach((item => {
debug('删除任务项', {
item: item
});
GM_deleteValue(item);
}));
debug('历史记录清除完成');
Swal.fire({
title: I18n('clearHistoryFinished'),
icon: 'success'
});
} catch (error) {
debug('清除历史记录时出错', {
error: error
});
throwError(error, 'History.clearHistory');
}
}
#addItem(item) {
try {
debug('开始添加任务项', {
item: item
});
const tasksData = GM_getValue(item);
if (!tasksData?.tasks) {
debug('任务数据无效', {
item: item
});
return;
}
const {title: title, link: link} = this.#getTaskInfo(item);
if (!title || !link) {
debug('获取任务信息失败', {
item: item
});
return;
}
debug('生成任务HTML', {
item: item,
title: title,
link: link
});
const html = this.#generateTaskHtml(tasksData.tasks);
this.#appendTaskToContainer(item, title, link, html, tasksData.time);
this.#bindDeleteEvent();
debug('任务项添加完成', {
item: item
});
} catch (error) {
debug('添加任务项时出错', {
error: error,
item: item
});
throwError(error, 'History.addItem');
}
}
#getTaskInfo(item) {
try {
debug('开始获取任务信息', {
item: item
});
const [website, id] = item.split('-');
debug('解析任务标识符', {
website: website,
id: id
});
const taskInfoMap = {
fawTasks: {
title: `Freeanywhere[${id}]`,
link: `https://freeanywhere.net/#/giveaway/${id}`
},
gasTasks: {
title: `Giveawaysu[${id}]`,
link: `https://giveaway.su/giveaway/view/${id}`
},
gcTasks: {
title: `GiveeClub[${id}]`,
link: `https://givee.club/event/${id}`
},
gkTasks: {
title: `Givekey[${id}]`,
link: `https://givekey.ru/giveaway/${id}`
},
gleamTasks: {
title: `Gleam[${id}]`,
link: `https://gleam.io${id}`
},
khTasks: {
title: `keyhub[${id}]`,
link: `https://key-hub.eu/giveaway/${id}`
},
prysTasks: {
title: `Prys[${id}]`,
link: `https://prys.revadike.com/giveaway/?id=${id}`
}
};
const result = taskInfoMap[website] || {
title: '',
link: ''
};
debug('获取任务信息结果', {
result: result
});
return result;
} catch (error) {
debug('获取任务信息时出错', {
error: error,
item: item
});
throwError(error, 'History.getTaskInfo');
return {
title: '',
link: ''
};
}
}
#generateTaskHtml(tasks) {
try {
debug('开始生成任务HTML');
let html = '';
for (const [social, types] of Object.entries(tasks)) {
for (const [type, taskList] of Object.entries(types)) {
for (const task of taskList) {
debug('处理任务', {
social: social,
type: type,
task: task
});
const displayTask = task.length > 55 ? `${task.slice(0, 55)}...` : task;
html += `<li>\n <font class="auto-task-capitalize">${social}.${I18n(type.replace('Link', ''))}: </font>\n <a href="${task}" target="_blank">${displayTask}</a>\n </li>`;
}
}
}
debug('任务HTML生成完成');
return html;
} catch (error) {
debug('生成任务HTML时出错', {
error: error
});
throwError(error, 'History.generateTaskHtml');
return '';
}
}
#appendTaskToContainer(item, title, link, html, time) {
try {
debug('开始添加任务到容器', {
item: item,
title: title,
link: link
});
$('.container').append(`\n <div class="card" data-name="${item}">\n <div class="title">\n <a href="${link}" target="_blank">${title}</a>\n <span class="delete-task" data-name="${item}" title="${I18n('deleteTask')}">\n <svg class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2734" width="32" height="32">\n <path d="M607.897867 768.043004c-17.717453 0-31.994625-14.277171-31.994625-31.994625L575.903242 383.935495c0-17.717453 14.277171-31.994625 31.994625-31.994625s31.994625 14.277171 31.994625 31.994625l0 351.94087C639.892491 753.593818 625.61532 768.043004 607.897867 768.043004z" p-id="2735" fill="#d81e06"></path>\n <path d="M415.930119 768.043004c-17.717453 0-31.994625-14.277171-31.994625-31.994625L383.935495 383.935495c0-17.717453 14.277171-31.994625 31.994625-31.994625 17.717453 0 31.994625 14.277171 31.994625 31.994625l0 351.94087C447.924744 753.593818 433.647573 768.043004 415.930119 768.043004z" p-id="2736" fill="#d81e06"></path>\n <path d="M928.016126 223.962372l-159.973123 0L768.043004 159.973123c0-52.980346-42.659499-95.983874-95.295817-95.983874L351.94087 63.989249c-52.980346 0-95.983874 43.003528-95.983874 95.983874l0 63.989249-159.973123 0c-17.717453 0-31.994625 14.277171-31.994625 31.994625s14.277171 31.994625 31.994625 31.994625l832.032253 0c17.717453 0 31.994625-14.277171 31.994625-31.994625S945.73358 223.962372 928.016126 223.962372zM319.946246 159.973123c0-17.545439 14.449185-31.994625 31.994625-31.994625l320.806316 0c17.545439 0 31.306568 14.105157 31.306568 31.994625l0 63.989249L319.946246 223.962372 319.946246 159.973123 319.946246 159.973123z" p-id="2737" fill="#d81e06"></path>\n <path d="M736.048379 960.010751 288.123635 960.010751c-52.980346 0-95.983874-43.003528-95.983874-95.983874L192.139761 383.591466c0-17.717453 14.277171-31.994625 31.994625-31.994625s31.994625 14.277171 31.994625 31.994625l0 480.435411c0 17.717453 14.449185 31.994625 31.994625 31.994625l448.096758 0c17.717453 0 31.994625-14.277171 31.994625-31.994625L768.215018 384.795565c0-17.717453 14.277171-31.994625 31.994625-31.994625s31.994625 14.277171 31.994625 31.994625l0 479.231312C832.032253 916.835209 789.028725 960.010751 736.048379 960.010751z" p-id="2738" fill="#d81e06"></path>\n </svg>\n </span>\n </div>\n <ul>${html}</ul>\n <span class="time">${I18n('lastChangeTime')}: ${dayjs(time).format('YYYY-MM-DD HH:mm:ss')}</span>\n </div>\n `);
debug('任务已添加到容器', {
item: item
});
} catch (error) {
debug('添加任务到容器时出错', {
error: error,
item: item
});
throwError(error, 'History.appendTaskToContainer');
}
}
#bindDeleteEvent() {
try {
debug('开始绑定删除事件');
$('span.delete-task').on('click', (function() {
const itemName = $(this).attr('data-name');
debug('点击删除按钮', {
itemName: itemName
});
if (!itemName) {
debug('删除失败:未找到任务名称');
Swal.fire({
title: I18n('clearTaskFailed'),
icon: 'error'
});
return;
}
GM_deleteValue(itemName);
$(`div.card[data-name="${itemName}"]`).remove();
debug('任务删除成功', {
itemName: itemName
});
Swal.fire({
title: I18n('clearTaskFinished'),
text: itemName,
icon: 'success'
});
}));
debug('删除事件绑定完成');
} catch (error) {
debug('绑定删除事件时出错', {
error: error
});
throwError(error, 'History.bindDeleteEvent');
}
}
}
const defaultTasksTemplate$1 = {
steam: {
groupLinks: [],
wishlistLinks: [],
followLinks: [],
curatorLinks: [],
curatorLikeLinks: []
},
twitter: {
userLinks: [],
retweetLinks: []
},
twitch: {
channelLinks: []
},
discord: {
serverLinks: []
},
youtube: {
channelLinks: []
},
extra: {
giveawayHopper: []
}
};
const defaultTasks$1 = JSON.stringify(defaultTasksTemplate$1);
class GiveawayHopper extends Website {
name='GiveawayHopper';
undoneTasks=JSON.parse(defaultTasks$1);
socialTasks=JSON.parse(defaultTasks$1);
tasks=[];
buttons=[ 'doTask', 'undoTask', 'verifyTask' ];
static test() {
const {host: host} = window.location;
const isMatch = host === 'giveawayhopper.com';
debug('检查网站匹配', {
host: host,
isMatch: isMatch
});
return isMatch;
}
async after() {
try {
debug('开始执行后续操作');
if (!this.#checkLogin()) {
debug('登录检查失败');
echoLog({}).warning(I18n('checkLoginFailed'));
}
const giveawayIdResult = this.#getGiveawayId();
debug('获取抽奖ID', {
success: giveawayIdResult,
id: this.giveawayId
});
} catch (error) {
debug('后续操作失败', {
error: error
});
throwError(error, 'GiveawayHopper.after');
}
}
async init() {
try {
debug('初始化 GiveawayHopper');
const logStatus = echoLog({
text: I18n('initing')
});
const leftKeyResult = await this.#checkLeftKey();
if (!leftKeyResult) {
debug('检查剩余密钥失败');
echoLog({}).warning(I18n('checkLeftKeyFailed'));
}
this.initialized = true;
debug('初始化完成');
logStatus.success();
return true;
} catch (error) {
debug('初始化失败', {
error: error
});
throwError(error, 'GiveawayHopper.init');
return false;
}
}
async classifyTask(action) {
try {
debug('开始分类任务', {
action: action
});
if (!this.giveawayId) {
debug('未找到抽奖ID,尝试获取');
await this.#getGiveawayId();
}
const logStatus = echoLog({
text: I18n('getTasksInfo')
});
if (action === 'undo') {
debug('恢复已保存的任务信息');
this.socialTasks = GM_getValue(`giveawayHopperTasks-${this.giveawayId}`)?.tasks || JSON.parse(defaultTasks$1);
}
debug('请求任务列表');
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://giveawayhopper.com/api/v1/campaigns/${this.giveawayId}/with-auth`,
method: 'GET',
responseType: 'json',
headers: {
authorization: `Bearer ${window.sessionStorage.gw_auth}`,
'x-xsrf-token': decodeURIComponent(document.cookie.match(/XSRF-TOKEN=([^;]+)/)?.[1])
}
});
if (result !== 'Success') {
debug('请求任务列表失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200 || !data?.response?.tasks) {
debug('任务列表数据异常', {
status: data?.status,
response: data?.response
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('获取到任务列表', {
count: data.response.tasks.length
});
this.tasks = data.response.tasks;
for (const task of data.response.tasks) {
if (task.isDone) {
debug('跳过已完成任务', {
taskId: task.id,
type: task.type
});
continue;
}
debug('处理任务', {
taskId: task.id,
category: task.category,
type: task.type
});
await httpRequest({
url: `https://giveawayhopper.com/api/v1/campaigns/${this.giveawayId}/tasks/${task.id}/visited`,
method: 'GET',
responseType: 'json',
headers: {
authorization: `Bearer ${window.sessionStorage.gw_auth}`,
'x-xsrf-token': decodeURIComponent(document.cookie.match(/XSRF-TOKEN=([^;]+)/)?.[1])
}
});
if (task.category === 'Steam' && task.type === 'JoinGroup') {
debug('处理 Steam 组任务');
const steamGroupLink = await getRedirectLink(`https://steamcommunity.com/gid/${task.group_id}`);
if (!steamGroupLink) {
debug('获取 Steam 组链接失败');
continue;
}
debug('添加 Steam 组链接', {
action: action,
link: steamGroupLink
});
if (action === 'undo') {
this.socialTasks.steam.groupLinks.push(steamGroupLink);
}
if (action === 'do') {
this.undoneTasks.steam.groupLinks.push(steamGroupLink);
}
continue;
}
if (task.category === 'Discord' && task.type === 'JoinServer') {
const discordLink = `https://discord.gg/${task.invite_code}`;
debug('添加 Discord 服务器链接', {
action: action,
link: discordLink
});
if (action === 'undo') {
this.socialTasks.discord.serverLinks.push(discordLink);
}
if (action === 'do') {
this.undoneTasks.discord.serverLinks.push(discordLink);
}
continue;
}
if ([ 'TikTok', 'YouTube', 'General' ].includes(task.category)) {
debug('跳过特殊任务类型', {
category: task.category
});
continue;
}
debug('发现未知任务类型', {
category: task.category,
type: task.type
});
echoLog({}).warning(`${I18n('unKnownTaskType')}: ${task.category}-${task.type}`);
}
logStatus.success();
this.undoneTasks = this.uniqueTasks(this.undoneTasks);
this.socialTasks = this.uniqueTasks(this.socialTasks);
debug('任务分类完成', {
undoneTasks: this.undoneTasks,
socialTasks: this.socialTasks
});
GM_setValue(`giveawayHopperTasks-${this.giveawayId}`, {
tasks: this.socialTasks,
time: (new Date).getTime()
});
return true;
} catch (error) {
debug('任务分类失败', {
error: error
});
throwError(error, 'GiveawayHopper.classifyTask');
return false;
}
}
async verifyTask() {
try {
debug('开始验证任务');
for (const task of this.tasks) {
if (task.isDone) {
debug('跳过已完成任务', {
taskId: task.id
});
continue;
}
debug('验证任务', {
taskId: task.id,
name: task.displayName?.replace(':target', task.targetName) || task.name
});
const logStatus = echoLog({
text: `${I18n('verifyingTask')}[${task.displayName?.replace(':target', task.targetName) || task.name}]...`
});
if (!task.link) {
debug('获取任务链接');
task.link = this.#getTaskLink(task);
}
if (task.link) {
debug('访问任务链接', {
link: task.link
});
await this.#visitTaskLink(task);
}
await delay(1e3);
const verifyResult = await this.#verifyTask(task, logStatus);
debug('任务验证结果', {
taskId: task.id,
success: verifyResult
});
if (!verifyResult) {
continue;
}
}
debug('所有任务验证完成');
return true;
} catch (error) {
debug('任务验证失败', {
error: error
});
throwError(error, 'GiveawayHopper.verifyTask');
return false;
}
}
#getTaskLink(task) {
try {
debug('生成任务链接', {
category: task.category,
type: task.type
});
let link = '';
if (task.category === 'YouTube' && task.type === 'FollowAccount') {
link = `https://www.youtube.com/@${task.targetName}`;
} else if (task.category === 'TikTok' && task.type === 'FollowAccount') {
link = `https://www.tiktok.com/@${task.targetName}`;
} else if (task.category === 'Steam' && task.type === 'JoinGroup') {
link = '';
} else if (task.category === 'Discord' && task.type === 'JoinServer') {
link = '';
}
debug('生成的任务链接', {
link: link
});
return link;
} catch (error) {
debug('生成任务链接失败', {
error: error
});
throwError(error, 'GiveawayHopper.getTaskLink');
return '';
}
}
async #visitTaskLink(task) {
debug('访问任务链接', {
taskId: task.id,
link: task.link
});
await httpRequest({
url: `https://giveawayhopper.com/fw?url=${encodeURIComponent(task.link)}&src=campaign&src_id=${this.giveawayId}&ref=task&ref_id=${task.id}&token=${window.sessionStorage.gw_auth}`,
method: 'GET',
headers: {
authorization: `Bearer ${window.sessionStorage.gw_auth}`,
'x-xsrf-token': decodeURIComponent(document.cookie.match(/XSRF-TOKEN=([^;]+)/)?.[1])
}
});
}
async #verifyTask(task, logStatus) {
debug('验证任务', {
taskId: task.id,
category: task.category,
type: task.type
});
const postData = {
taskcategory: task.category,
taskname: task.type
};
if ([ 'YouTube', 'TikTok' ].includes(task.category)) {
postData.username = '1';
}
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: `https://giveawayhopper.com/api/v1/campaigns/${this.giveawayId}/tasks/${task.id}/complete`,
method: 'POST',
headers: {
authorization: `Bearer ${window.sessionStorage.gw_auth}`,
'x-xsrf-token': decodeURIComponent(document.cookie.match(/XSRF-TOKEN=([^;]+)/)?.[1]),
'content-type': 'application/json'
},
dataType: 'json',
data: JSON.stringify(postData)
});
if (result !== 'Success') {
debug('任务验证请求失败', {
result: result,
statusText: statusText,
status: status
});
logStatus.error(`${result}:${statusText}(${status})`);
return false;
}
if (data?.status !== 200 || !data?.response?.completed) {
debug('任务验证响应异常', {
status: data?.status,
response: data?.response
});
logStatus.error(`Error:${data?.statusText}(${data?.status})`);
return false;
}
debug('任务验证成功', {
taskId: task.id
});
logStatus.success();
return true;
}
#getGiveawayId() {
try {
debug('从URL获取抽奖ID');
const giveawayId = window.location.pathname.split('/').at(-1);
if (!giveawayId) {
debug('获取抽奖ID失败');
echoLog({
text: I18n('getFailed', 'GiveawayId')
});
return false;
}
this.giveawayId = giveawayId;
debug('获取抽奖ID成功', {
giveawayId: giveawayId
});
return true;
} catch (error) {
debug('获取抽奖ID出错', {
error: error
});
throwError(error, 'GiveawayHopper.getGiveawayId');
return false;
}
}
#checkLogin() {
try {
debug('检查登录状态');
if (!globalOptions.other.checkLogin) {
debug('跳过登录检查');
return true;
}
const needLogin = $('div.widget-connections-edit:contains("Log in")').length > 0;
if (needLogin) {
debug('未登录,自动点击登录按钮');
$('div.widget-connections-edit:contains("Log in") a')[0].click();
}
debug('登录检查完成', {
needLogin: needLogin
});
return true;
} catch (error) {
debug('登录检查失败', {
error: error
});
throwError(error, 'GiveawayHopper.checkLogin');
return false;
}
}
async #checkLeftKey() {
try {
debug('检查剩余密钥');
if (!globalOptions.other.checkLeftKey) {
debug('跳过密钥检查');
return true;
}
const keyCount = parseInt($('p.widget-single-prize span').text()?.match(/\d+/)?.[0] || '0', 10);
debug('剩余密钥数量', {
keyCount: keyCount
});
if (keyCount > 0) {
return true;
}
debug('没有剩余密钥,显示确认对话框');
const {value: value} = await Swal.fire({
icon: 'warning',
title: I18n('notice'),
text: I18n('noKeysLeft'),
confirmButtonText: I18n('confirm'),
cancelButtonText: I18n('cancel'),
showCancelButton: true
});
if (value) {
debug('用户确认关闭窗口');
window.close();
}
return true;
} catch (error) {
debug('检查剩余密钥失败', {
error: error
});
throwError(error, 'GiveawayHopper.checkLeftKey');
return false;
}
}
}
const defaultTasksTemplate = {
steam: {
groupLinks: [],
curatorLinks: [],
wishlistLinks: [],
followLinks: []
},
youtube: {
channelLinks: []
}
};
const defaultTasks = JSON.stringify(defaultTasksTemplate);
class Prys extends Website {
name='Prys';
socialTasks=JSON.parse(defaultTasks);
undoneTasks=JSON.parse(defaultTasks);
buttons=[ 'doTask', 'undoTask', 'verifyTask' ];
static test() {
const {host: host} = window.location;
const isMatch = host === 'prys.revadike.com';
debug('检查网站匹配', {
host: host,
isMatch: isMatch
});
return isMatch;
}
async after() {
try {
debug('开始执行后续操作');
if (!this.#checkLogin()) {
debug('检查登录失败');
echoLog({}).warning(I18n('checkLoginFailed'));
}
if (!await this.#checkLeftKey()) {
debug('检查剩余密钥失败');
echoLog({}).warning(I18n('checkLeftKeyFailed'));
}
} catch (error) {
debug('后续操作失败', {
error: error
});
throwError(error, 'Prys.after');
}
}
init() {
try {
debug('开始初始化');
const logStatus = echoLog({
text: I18n('initing')
});
if ($('button:contains("Sign")').length > 0) {
debug('需要登录');
logStatus.warning(I18n('needLogin'));
return false;
}
if (!this.#getGiveawayId()) {
debug('获取抽奖ID失败');
return false;
}
this.initialized = true;
debug('初始化完成');
logStatus.success();
return true;
} catch (error) {
debug('初始化失败', {
error: error
});
throwError(error, 'Prys.init');
return false;
}
}
async classifyTask(action) {
try {
debug('开始分类任务', {
action: action
});
const logStatus = echoLog({
text: I18n('getTasksInfo')
});
if (action === 'undo') {
debug('恢复已保存的任务信息');
this.socialTasks = GM_getValue(`prysTasks-${this.giveawayId}`)?.tasks || JSON.parse(defaultTasks);
}
const steps = $('#steps tbody tr');
debug('找到任务步骤', {
count: steps.length
});
for (let eq = 0; eq < steps.length; eq += 1) {
if (steps.eq(eq).find('span:contains(Success)').length === 0) {
debug('点击检查按钮', {
step: eq
});
checkClick(eq);
}
}
const pro = [];
for (const step of steps) {
const isSuccess = $(step).find('span:contains(Success)').length > 0;
if (isSuccess && action === 'do') {
debug('跳过已完成的任务');
continue;
}
const appLink = $(step).find('a[href*=\'store.steampowered.com/app/\']').attr('href');
if (appLink) {
const taskType = $(step).find('a[href*=\'store.steampowered.com/app/\']').text().includes('wishlist') ? 'wishlistLinks' : 'followLinks';
debug('添加 Steam 应用任务', {
type: taskType,
link: appLink
});
if (action === 'undo') {
this.socialTasks.steam[taskType].push(appLink);
}
if (action === 'do') {
this.undoneTasks.steam[taskType].push(appLink);
}
continue;
}
const curatorLink = $(step).find('a[href*=\'store.steampowered.com/curator/\']').attr('href');
if (curatorLink) {
debug('添加 Steam 鉴赏家任务', {
link: curatorLink
});
if (action === 'undo') {
this.socialTasks.steam.curatorLinks.push(curatorLink);
}
if (action === 'do') {
this.undoneTasks.steam.curatorLinks.push(curatorLink);
}
continue;
}
const groupLink = $(step).find('a[href*=\'steamcommunity.com/groups/\']').attr('href');
if (groupLink) {
debug('添加 Steam 组任务', {
link: groupLink
});
if (action === 'undo') {
this.socialTasks.steam.groupLinks.push(groupLink);
}
if (action === 'do') {
this.undoneTasks.steam.groupLinks.push(groupLink);
}
continue;
}
const gidLink = $(step).find('a[href*=\'steamcommunity.com/gid\']').attr('href');
if (gidLink) {
debug('处理 Steam GID 链接', {
link: gidLink
});
pro.push(getRedirectLink(gidLink).then((finalUrl => {
if (!finalUrl || !/^https?:\/\/steamcommunity\.com\/groups\//.test(finalUrl)) {
debug('无效的 Steam 组链接', {
finalUrl: finalUrl
});
return false;
}
debug('添加 Steam 组任务(从 GID)', {
link: finalUrl
});
if (action === 'undo') {
this.socialTasks.steam.groupLinks.push(finalUrl);
}
if (action === 'do') {
this.undoneTasks.steam.groupLinks.push(finalUrl);
}
})));
}
}
await Promise.allSettled(pro);
debug('任务分类完成');
logStatus.success();
this.undoneTasks = this.uniqueTasks(this.undoneTasks);
this.socialTasks = this.uniqueTasks(this.socialTasks);
if (window.DEBUG) {
console.log('%cAuto-Task[Debug]:', 'color:blue', JSON.stringify(this));
}
GM_setValue(`prysTasks-${this.giveawayId}`, {
tasks: this.socialTasks,
time: (new Date).getTime()
});
return true;
} catch (error) {
debug('任务分类失败', {
error: error
});
throwError(error, 'Prys.classifyTask');
return false;
}
}
async verifyTask() {
try {
debug('开始验证任务');
const checks = $('#steps tbody a[id^=check]');
if (checks.length === 0) {
debug('没有需要验证的任务');
echoLog({}).success(I18n('allTasksComplete'));
return;
}
const pro = [];
for (const check of checks) {
const id = $(check).attr('id')?.match(/[\d]+/)?.[0];
if (!id) {
debug('跳过无效任务ID');
continue;
}
const taskDes = $(check).parent()?.prev()?.html()?.trim();
debug('验证任务', {
id: id,
taskDes: taskDes
});
const status = echoLog({
text: `${I18n('verifyingTask')}${taskDes}...`
});
pro.push(new Promise((resolve => {
this.#checkStep(id, resolve, status);
})));
}
await Promise.all(pro);
debug('所有任务验证完成');
echoLog({}).success(I18n('allTasksComplete'));
} catch (error) {
debug('验证任务失败', {
error: error
});
throwError(error, 'Prys.verifyTask');
}
}
#getGiveawayId() {
try {
debug('开始获取抽奖ID');
const giveawayId = window.location.search.match(/id=([\d]+)/)?.[1];
if (giveawayId) {
this.giveawayId = giveawayId;
debug('获取抽奖ID成功', {
giveawayId: giveawayId
});
return true;
}
debug('获取抽奖ID失败');
echoLog({}).error(I18n('getFailed', 'GiveawayId'));
return false;
} catch (error) {
debug('获取抽奖ID出错', {
error: error
});
throwError(error, 'Prys.getGiveawayId');
return false;
}
}
async #checkLeftKey() {
try {
debug('检查剩余密钥');
if (!globalOptions.other.checkLeftKey) {
debug('跳过密钥检查');
return true;
}
const leftKey = $('#header').text().match(/([\d]+).*?prize.*?left/)?.[1];
debug('检查剩余密钥数量', {
leftKey: leftKey
});
if (leftKey !== '0') {
return true;
}
debug('没有剩余密钥,显示确认对话框');
const {value: value} = await Swal.fire({
icon: 'warning',
title: I18n('notice'),
text: I18n('noKeysLeft'),
confirmButtonText: I18n('confirm'),
cancelButtonText: I18n('cancel'),
showCancelButton: true
});
if (value) {
debug('用户确认关闭窗口');
window.close();
}
return true;
} catch (error) {
debug('检查剩余密钥失败', {
error: error
});
throwError(error, 'Prys.checkLeftKey');
return false;
}
}
#checkLogin() {
try {
debug('检查登录状态');
if (!globalOptions.other.checkLogin) {
debug('跳过登录检查');
return true;
}
if ($('button:contains("Sign")').length > 0) {
debug('未登录');
echoLog({}).warning(I18n('needLogin'));
}
debug('登录检查完成');
return true;
} catch (error) {
debug('检查登录失败', {
error: error
});
throwError(error, 'Prys.checkLogin');
return false;
}
}
#checkStep(step, resolve, status, captcha = null) {
try {
debug('开始检查步骤', {
step: step,
hasCaptcha: !!captcha
});
if (step !== 'captcha') {
debug('更新步骤状态为检查中');
$(`#check${step}`).replaceWith(`<span id="check${step}"><i class="fa fa-refresh fa-spin fa-fw"></i> Checking...</span>`);
}
debug('发送检查请求');
$.post('/api/check_step', {
step: step,
id: getURLParameter('id'),
'g-recaptcha-response': captcha
}, (json => {
resolve();
debug('收到检查响应', {
success: json.success
});
if (step !== 'captcha') {
if (json.success) {
debug('步骤检查成功');
$(`#check${step}`).replaceWith(`<span class="text-success" id="check${step}"><i class="fa fa-check"></i> Success</span>`);
status.success();
} else {
debug('步骤检查失败');
$(`#check${step}`).replaceWith(`<a id="check${step}" href="javascript:checkStep(${step})"><i class="fa fa-question"></i> Check</a>`);
status.error(json.response?.error || 'Error');
}
}
if (!json.response) {
return;
}
if (json.response.prize) {
debug('获得奖品', {
prize: json.response.prize
});
showAlert('success', `Here is your prize:<h1 role="button" align="middle" style="word-wrap: break-word;">${json.response.prize}</h2>`);
}
if (!json.response.captcha) {
return;
}
debug('需要验证码');
if (json.success) {
showAlert('info', json.response.captcha);
} else {
showAlert('warning', json.response.captcha);
}
captchaCheck();
})).fail((() => {
resolve();
debug('请求失败');
$(`#check${step}`).replaceWith(`<a id="check${step}" href="javascript:checkStep(${step})"><i class="fa fa-question"></i> Check</a>`);
status.error('Error:0');
}));
} catch (error) {
debug('检查步骤失败', {
error: error
});
throwError(error, 'prys.checkStep');
resolve(false);
}
}
}
const Websites = [ FreeAnyWhere, GiveawaySu, Indiedb, Keyhub, Givekey, GiveeClub, OpiumPulses, Keylol, Opquests, Gleam, SweepWidget, Setting, History, GiveawayHopper, Prys ];
const generateFormHtml = options => {
debug('开始生成网站选项表单HTML', {
options: options
});
const tableRows = Object.entries(options).map((([option, value]) => `\n <tr>\n <td>${option}</td>\n <td>\n <input\n class="editOption"\n type="text"\n name="${option}"\n value="${value}"\n />\n </td>\n </tr>\n `)).join('');
const formHtml = `\n <form id="websiteOptionsForm" class="auto-task-form">\n <table class="auto-task-table">\n <thead>\n <tr>\n <td>${I18n('option')}</td>\n <td>${I18n('value')}</td>\n </tr>\n </thead>\n <tbody>\n ${tableRows}\n </tbody>\n </table>\n </form>\n `;
debug('表单HTML生成完成');
return formHtml;
};
const saveOptions = (website, options, formValues) => {
debug('开始保存网站选项', {
website: website,
formValues: formValues
});
formValues.forEach((({name: name, value: value}) => {
options[name] = value;
debug('更新选项值', {
name: name,
value: value
});
}));
GM_setValue(`${website}Options`, options);
debug('选项已保存到存储', {
website: website
});
Swal.fire({
title: I18n('changeWebsiteOptionsSuccess'),
icon: 'success'
});
};
const websiteOptions = async (website, options) => {
try {
debug('开始设置网站选项', {
website: website
});
if (!website || typeof website !== 'string') {
debug('无效的网站参数', {
website: website
});
throw new Error('Invalid website parameter');
}
if (!options || typeof options !== 'object') {
debug('无效的选项参数', {
options: options
});
throw new Error('Invalid options parameter');
}
debug('显示选项编辑对话框');
const result = await Swal.fire({
title: I18n('websiteOptions'),
html: generateFormHtml(options),
showConfirmButton: true,
confirmButtonText: I18n('save'),
showCancelButton: true,
cancelButtonText: I18n('close')
});
if (result.isConfirmed) {
debug('用户确认保存选项');
const form = document.getElementById('websiteOptionsForm');
if (!form) {
debug('未找到表单元素');
throw new Error('Form element not found');
}
const formData = $('#websiteOptionsForm').serializeArray();
debug('获取表单数据', {
formData: formData
});
saveOptions(website, options, formData);
} else {
debug('用户取消保存选项');
}
} catch (error) {
debug('设置网站选项时发生错误', {
error: error
});
throwError(error, 'websiteOptions');
}
};
const UPDATE_LINKS = {
github: 'https://github.com/HCLonely/auto-task/raw/main/',
jsdelivr: 'https://cdn.jsdelivr.net/gh/HCLonely/auto-task@main/',
standby: 'https://auto-task.hclonely.com/'
};
const checkUpdate = async (updateLink, auto) => {
try {
debug('开始检查更新', {
updateLink: updateLink,
auto: auto
});
const checkUrl = `${updateLink}package.json?time=${Date.now()}`;
debug('构建检查URL', {
checkUrl: checkUrl
});
const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
url: checkUrl,
responseType: 'json',
method: 'GET',
timeout: 3e4
});
if (result === 'Success' && data?.response?.version) {
debug('成功获取更新信息', {
version: data.response.version
});
return data.response;
}
if (!auto) {
const errorMessage = data?.response?.version ? `${I18n('checkUpdateFailed')}[${data?.statusText}(${data?.status})]` : `${I18n('checkUpdateFailed')}[${result}:${statusText}(${status})]`;
debug('检查更新失败', {
errorMessage: errorMessage
});
echoLog({}).error(errorMessage);
} else {
debug('自动检查更新失败', {
result: result,
statusText: statusText,
status: status
});
}
return false;
} catch (error) {
debug('检查更新发生错误', {
error: error
});
throwError(error, 'checkUpdate');
return false;
}
};
const hasNewVersion = (currentVersion, remoteVersion) => {
try {
debug('开始比较版本号', {
currentVersion: currentVersion,
remoteVersion: remoteVersion
});
const [currentRealVersion] = currentVersion.split('-');
const [remoteRealVersion, isPreview] = remoteVersion.split('-');
if (isPreview && !globalOptions.other.receivePreview) {
debug('不接收预览版本', {
isPreview: isPreview
});
return false;
}
const currentVersionParts = currentRealVersion.split('.').map(Number);
const remoteVersionParts = remoteRealVersion.split('.').map(Number);
debug('版本号解析', {
currentVersionParts: currentVersionParts,
remoteVersionParts: remoteVersionParts
});
for (let i = 0; i < 3; i++) {
if (remoteVersionParts[i] > currentVersionParts[i]) {
debug('发现新版本', {
position: i,
current: currentVersion,
remote: remoteVersion
});
return true;
}
if (remoteVersionParts[i] < currentVersionParts[i]) {
debug('远程版本较旧', {
position: i,
current: currentVersion,
remote: remoteVersion
});
return false;
}
}
debug('版本号相同');
return false;
} catch (error) {
debug('比较版本号时发生错误', {
error: error
});
throwError(error, 'compareVersion');
return false;
}
};
const getUpdateLink = updateSource => {
debug('获取更新链接', {
updateSource: updateSource
});
const source = updateSource.toLowerCase();
const link = UPDATE_LINKS[source] || UPDATE_LINKS.github;
debug('选择的更新链接', {
source: source,
link: link
});
return link;
};
const showUpdateInfo = (packageData, currentVersion, updateLink) => {
debug('准备显示更新信息', {
currentVersion: currentVersion,
newVersion: packageData.version
});
if (hasNewVersion(currentVersion, packageData.version)) {
const scriptUrl = `${updateLink}dist/${GM_info.script.name}.user.js`;
debug('发现新版本,显示更新通知', {
scriptUrl: scriptUrl
});
echoLog({
html: `<li><font>${I18n('newVersionNotice', packageData.version, scriptUrl)}</font></li>`
});
const changeList = packageData.change?.map((change => `<li>${change}</li>`)).join('') || '';
debug('显示更新日志', {
changeListLength: packageData.change?.length
});
echoLog({
html: `<li>${I18n('updateText', packageData.version)}</li><ol class="update-text">${changeList}<li>${I18n('updateHistory')}</li></ol>`
});
} else {
debug('当前已是最新版本');
}
};
const updateChecker = async () => {
try {
debug('开始检查更新流程');
const currentVersion = GM_info.script.version;
const updateSource = globalOptions.other.autoUpdateSource;
debug('当前配置', {
currentVersion: currentVersion,
updateSource: updateSource
});
let packageData = false;
if ([ 'github', 'jsdelivr', 'standby' ].includes(updateSource.toLowerCase())) {
debug('使用指定的更新源', {
updateSource: updateSource
});
const updateLink = getUpdateLink(updateSource);
packageData = await checkUpdate(updateLink, false);
} else {
debug('按优先级尝试不同的更新源');
for (const source of [ 'github', 'jsdelivr', 'standby' ]) {
debug('尝试更新源', {
source: source
});
packageData = await checkUpdate(UPDATE_LINKS[source], true);
if (packageData) {
debug('成功获取更新信息', {
source: source
});
break;
}
}
}
if (!packageData) {
debug('所有更新源检查失败');
echoLog({}).error(I18n('checkUpdateFailed'));
return;
}
showUpdateInfo(packageData, currentVersion, getUpdateLink(updateSource));
} catch (error) {
debug('更新检查过程发生错误', {
error: error
});
throwError(error, 'updateChecker');
}
};
const byteToHex = [];
for (let i = 0; i < 256; ++i) {
byteToHex.push((i + 256).toString(16).slice(1));
}
function unsafeStringify(arr, offset = 0) {
return (byteToHex[arr[offset + 0]] + byteToHex[arr[offset + 1]] + byteToHex[arr[offset + 2]] + byteToHex[arr[offset + 3]] + '-' + byteToHex[arr[offset + 4]] + byteToHex[arr[offset + 5]] + '-' + byteToHex[arr[offset + 6]] + byteToHex[arr[offset + 7]] + '-' + byteToHex[arr[offset + 8]] + byteToHex[arr[offset + 9]] + '-' + byteToHex[arr[offset + 10]] + byteToHex[arr[offset + 11]] + byteToHex[arr[offset + 12]] + byteToHex[arr[offset + 13]] + byteToHex[arr[offset + 14]] + byteToHex[arr[offset + 15]]).toLowerCase();
}
let getRandomValues;
const rnds8 = new Uint8Array(16);
function rng() {
if (!getRandomValues) {
if (typeof crypto === 'undefined' || !crypto.getRandomValues) {
throw new Error('crypto.getRandomValues() not supported. See https://github.com/uuidjs/uuid#getrandomvalues-not-supported');
}
getRandomValues = crypto.getRandomValues.bind(crypto);
}
return getRandomValues(rnds8);
}
const randomUUID = typeof crypto !== 'undefined' && crypto.randomUUID && crypto.randomUUID.bind(crypto);
var native = {
randomUUID: randomUUID
};
function v4(options, buf, offset) {
if (native.randomUUID && true && !options) {
return native.randomUUID();
}
options = options || {};
const rnds = options.random ?? options.rng?.() ?? rng();
if (rnds.length < 16) {
throw new Error('Random bytes length must be >= 16');
}
rnds[6] = rnds[6] & 15 | 64;
rnds[8] = rnds[8] & 63 | 128;
return unsafeStringify(rnds);
}
function fawExtension() {
const hostname = window.location.hostname;
function IsJsonString(str) {
try {
JSON.parse(str);
} catch (e) {
return false;
}
return true;
}
$(window).on('load', (function() {
console.log('👌 gamesforfarm extension');
if (hostname == 'freeanywhere.net' || hostname == 'give.gamesforfarm.local' || hostname == 'gamesforfarm-testing.ru') {
const steam = $('.games_for_farm_site').data('steam');
const avatar = $('.games_for_farm_site').data('avatar');
const name = $('.games_for_farm_site').data('name');
const lang = $('.games_for_farm_site').data('lang');
let need_update = true;
GM_addValueChangeListener('FAW_STORAGE', (function(newValue, oldValue) {
if (need_update == false) {
return;
}
GM_getValue('FAW_STORAGE', (function(STORAGE) {
$.ajax({
type: 'POST',
url: '/php/extension/user_data_update.php',
data: {
extension: JSON.stringify(STORAGE)
},
success: function(data) {}
});
}));
}));
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (Object.keys(STORAGE).length === 0 || !STORAGE['tasks'] || !STORAGE['user'] || !STORAGE['games'] || !STORAGE['settings']) {
GM_deleteValue('FAW_STORAGE');
let new_storage = {};
new_storage['tasks'] = {};
new_storage['user'] = {};
new_storage['games'] = {};
new_storage['settings'] = {};
if (steam) {
new_storage['user']['steam'] = steam;
}
if (avatar) {
new_storage['user']['avatar'] = avatar;
}
if (name) {
new_storage['user']['name'] = name;
}
if (lang) {
new_storage['user']['lang'] = lang;
}
if (steam) {
$.ajax({
type: 'POST',
url: '/php/extension/user_games_get.php',
data: {
steam: steam
},
success: function(data) {
if (data != '' && IsJsonString(data)) {
const user_games = JSON.parse(data);
new_storage['games'] = user_games;
new_storage['settings']['game_update'] = parseInt(Date.now() / 1e3);
GM_setValue('FAW_STORAGE', new_storage);
}
}
});
} else {
GM_setValue('FAW_STORAGE', new_storage);
}
}
if (STORAGE['user'] && STORAGE['settings'] && STORAGE['user']['steam']) {
if (!STORAGE['settings']['game_update']) {
STORAGE['settings']['game_update'] = 0;
}
let time_now = parseInt(Date.now() / 1e3);
let time_update = parseInt(STORAGE['settings']['game_update']);
if (time_now - time_update > 60 * 60 * 24) {
$.ajax({
type: 'POST',
url: '/php/extension/user_games_get.php',
data: {
steam: STORAGE['user']['steam']
},
success: function(data) {
if (data != '' && IsJsonString(data)) {
const games = JSON.parse(data);
STORAGE['games'] = games;
STORAGE['settings']['game_update'] = parseInt(Date.now() / 1e3);
GM_setValue('FAW_STORAGE', STORAGE);
}
}
});
}
}
if (steam && STORAGE['user']) {
if (!STORAGE['user']['steam']) {
STORAGE['user']['steam'] = steam;
}
if (STORAGE['user']['steam'] != steam) {
$.ajax({
type: 'POST',
url: '/php/extension/user_data_get.php',
data: {
steam: steam
},
success: function(data) {
if (!data) {
return;
}
const db_storage = JSON.parse(data);
if (db_storage) {
GM_deleteValue('FAW_STORAGE');
need_update = false;
GM_setValue('FAW_STORAGE', db_storage);
setTimeout((function() {
need_update = true;
}), 100);
}
}
});
} else {
if (avatar) {
STORAGE['user']['avatar'] = avatar;
}
if (name) {
STORAGE['user']['name'] = name;
}
if (lang) {
STORAGE['user']['lang'] = lang;
}
GM_setValue('FAW_STORAGE', STORAGE);
}
}
if (STORAGE['tasks']) {
let update_tasks = [];
let is_update = false;
let time_now = parseInt(Date.now() / 1e3);
$.each(STORAGE['tasks'], (function(index, val) {
if (val['time'] && time_now - parseInt(val['time']) > 2 * 60 * 60) {
is_update = true;
return;
}
update_tasks.push(val);
}));
if (is_update == true) {
STORAGE['tasks'] = update_tasks;
GM_setValue('FAW_STORAGE', STORAGE);
}
}
if (STORAGE['discord']) {
if (STORAGE['discord'] && STORAGE['discord'].length > 0) {
$.ajax({
type: 'POST',
url: '/php/extension/discord_levels_update.php',
data: {
discord: JSON.stringify(STORAGE['discord'])
},
success: function(data) {
if (data.indexOf('success') != -1) {
alert('Данные discord уровней обновлены');
delete STORAGE['discord'];
GM_setValue('FAW_STORAGE', STORAGE);
} else {
alert('Возникла ошибка при обновлении discord уровней');
delete STORAGE['discord'];
GM_setValue('FAW_STORAGE', STORAGE);
}
},
error: function() {
alert('Возникла ошибка при обновлении discord уровней');
delete STORAGE['discord'];
GM_setValue('FAW_STORAGE', STORAGE);
}
});
}
}
function check_tasks_button() {
let tasks_done = true;
$('.game__content-tasks__task').each((function(index, el) {
if ($(this).hasClass('done') == false) {
tasks_done = false;
}
}));
if (tasks_done == true) {
$('.js-get-key').removeClass('inactive');
} else {
$('.js-get-key').addClass('inactive');
}
}
if ($('.games_for_farm_extension.work').length > 0) {
$('.games_for_farm_extension.not_work').remove();
$('.games_for_farm_extension.work').slideDown(200);
}
if ($('.game__content-tasks__task .task-check-extension').length > 0) {
$('.task-check-extension').removeClass('js-extentions-modal');
$('.game__content-tasks__task[data-extension=\'1\'] .task-link a').removeClass('js-extentions-modal');
$('.game__content-tasks__task .task-check-extension').on('click', (function(event) {
event.preventDefault();
let $button = $(this);
if ($button.hasClass('loading')) {
return;
}
let $parrent = $(this).parent('.game__content-tasks__task');
const type = $parrent.data('type');
const id = $parrent.data('id');
const data = $parrent.data('data');
const extension = $parrent.data('extension');
if (extension == false) {
return;
}
$button.addClass('loading');
const STORAGE = GM_getValue('FAW_STORAGE') || {};
$.ajax({
type: 'POST',
url: '/php/extension/user_data_update.php',
data: {
extension: JSON.stringify(STORAGE)
},
success: function(update) {
const getTime_start = (new Date).getTime();
$.ajax({
type: 'POST',
url: '/php/extension/user_task_update.php',
data: {
id: id,
type: type,
data: data
},
success: function(data) {
const getTime_end = (new Date).getTime();
console.log('👌 checking task in ' + (getTime_end - getTime_start) + ' ms');
if (data.indexOf('good') != -1) {
setTimeout((function() {
$parrent.addClass('done');
$parrent.removeClass('error');
$button.removeClass('loading');
check_tasks_button();
}), 1250);
} else if (data.indexOf('bad') != -1) {
setTimeout((function() {
$parrent.addClass('error');
$parrent.removeClass('done');
$button.removeClass('loading');
check_tasks_button();
}), 1250);
} else {
setTimeout((function() {
$parrent.removeClass('error');
$parrent.removeClass('done');
$button.removeClass('loading');
check_tasks_button();
}), 1250);
}
},
error: function() {
setTimeout((function() {
$parrent.removeClass('error');
$parrent.removeClass('done');
$button.removeClass('loading');
check_tasks_button();
}), 1250);
}
});
},
error: function() {
setTimeout((function() {
$parrent.removeClass('error');
$parrent.removeClass('done');
$button.removeClass('loading');
check_tasks_button();
}), 1250);
}
});
}));
}
}
function storage_tasks_update(tasks, type, value, action) {
let result = [];
let is_find = false;
$.each(tasks, (function(index, val) {
if (!val['type'] || !val['data']) {
return;
}
if (val['type'] == type && val['data'] == value) {
is_find = true;
return;
}
result.push(val);
}));
const obj = {};
switch (action) {
case 'add':
if (is_find == true) {
return;
}
const task = {};
task['type'] = type;
task['data'] = value;
task['time'] = parseInt(Date.now() / 1e3);
result.push(task);
obj['tasks'] = result;
GM_setValue('FAW_STORAGE', obj);
break;
case 'remove':
if (is_find == false) {
return;
}
obj['tasks'] = result;
GM_setValue('FAW_STORAGE', obj);
}
}
if (hostname == 'store.steampowered.com' || hostname == 'steamcommunity.com') {
if (document.querySelector('span[id^=\'CuratorUnFollowBtn_\']')) {
const curator_id = $('span[id^=\'CuratorUnFollowBtn_\']').attr('id').split('_')[1];
const follow_btn = '#CuratorFollowBtn_' + curator_id;
const unfollow_btn = '#CuratorUnFollowBtn_' + curator_id;
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
const follow = $(follow_btn).css('display');
const unfollow = $(unfollow_btn).css('display');
if (unfollow && unfollow == 'none') {
storage_tasks_update(STORAGE['tasks'], 'steam_curator_sub', curator_id, 'remove');
}
if (follow && follow == 'none') {
storage_tasks_update(STORAGE['tasks'], 'steam_curator_sub', curator_id, 'add');
}
$(follow_btn).on('click', (function(event) {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'steam_curator_sub', curator_id, 'add');
}));
$(unfollow_btn).on('click', (function(event) {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'steam_curator_sub', curator_id, 'remove');
}));
}
if (document.querySelector('.followStatsBlock')) {
const user_id = $('#HeaderUserInfoName a').attr('href').split('/').pop();
const follow_btn = '#FollowUserOptionAdd';
const unfollow_btn = '#FollowUserOptionFollowing, .followOption.remove';
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
const follow = $(follow_btn).css('visibility');
const unfollow = $(unfollow_btn).css('visibility');
if (unfollow && unfollow == 'hidden') {
storage_tasks_update(STORAGE['tasks'], 'steam_guides_sub', user_id, 'remove');
}
if (follow && follow == 'hidden') {
storage_tasks_update(STORAGE['tasks'], 'steam_guides_sub', user_id, 'add');
}
$(follow_btn).on('click', (function(event) {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'steam_guides_sub', user_id, 'add');
}));
$(unfollow_btn).on('click', (function(event) {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'steam_guides_sub', user_id, 'remove');
}));
}
if (document.querySelector('#ScrollingItemControls')) {
const manual_id = $('#PublishedFileFavorite input[name=\'id\']').val();
const follow_btn = '#FavoriteItemOptionAdd';
const unfollow_btn = '#FavoriteItemOptionFavorited, .favoriteOption.removefavorite';
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
const follow = $(follow_btn).css('visibility');
const unfollow = $(unfollow_btn).css('visibility');
if (unfollow && unfollow == 'hidden') {
storage_tasks_update(STORAGE['tasks'], 'steam_manual_favourite', manual_id, 'remove');
}
if (follow && follow == 'hidden') {
storage_tasks_update(STORAGE['tasks'], 'steam_manual_favourite', manual_id, 'add');
}
$(follow_btn).on('click', (function(event) {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'steam_manual_favourite', manual_id, 'add');
}));
$(unfollow_btn).on('click', (function(event) {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'steam_manual_favourite', manual_id, 'remove');
}));
}
if (document.querySelector('#queueBtnFollow')) {
const game_id = $('.game_page_background').data('miniprofile-appid');
const follow_btn = '#queueBtnFollow .queue_btn_inactive';
const unfollow_btn = '#queueBtnFollow .queue_btn_active';
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
const follow = $(follow_btn).css('display');
const unfollow = $(unfollow_btn).css('display');
if (unfollow && unfollow == 'none') {
storage_tasks_update(STORAGE['tasks'], 'steam_game_sub', game_id, 'remove');
}
if (follow && follow == 'none') {
storage_tasks_update(STORAGE['tasks'], 'steam_game_sub', game_id, 'add');
}
$(follow_btn).on('click', (function(event) {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'steam_game_sub', game_id, 'add');
}));
$(unfollow_btn).on('click', (function(event) {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'steam_game_sub', game_id, 'remove');
}));
}
}
if (hostname == 'www.youtube.com' || hostname == 'm.youtube.com') {
setInterval((function() {
if (document.querySelector('yt-subscribe-button-view-model')) {
const channel_id = $('meta[itemprop="identifier"]').attr('content');
const subscribe = [ 'Подписаться', 'Падпісацца', 'Підписатися', 'Abonnieren', 'Subscribe', 'Suscribirse', 'Mag-subscribe', 'S\'abonner', 'Iscriviti', 'Subskrybuj', 'Subscrever', 'Abonează-te', '订阅', 'チャンネル登録', '訂閱' ];
const subscribed = [ 'Вы подписаны', 'Вы падпісаны', 'Ви підписалися', 'Abonniert', 'Subscribed', 'Suscrito', 'Naka-subscribe', 'Abonné', 'Iscritto', 'Subskrybujesz', 'Subscrito', 'Abonat(ă)', '已订阅', '登録済み', '已訂閱' ];
const $parent = $('yt-subscribe-button-view-model');
const text = $parent.text();
if (subscribe.indexOf(text) != -1 || subscribed.indexOf(text) != -1) {
if (subscribe.indexOf(text) != -1) {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'youtube_channel_sub', channel_id, 'remove');
}
if (subscribed.indexOf(text) != -1) {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'youtube_channel_sub', channel_id, 'add');
}
} else {
if ($('.ytSubscribePlusButtonViewModelHost').length != 0) {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'youtube_channel_sub', channel_id, 'add');
} else {
const color = $parent.find('button').css('color');
if (color == '#0f0f0f' || color == 'rgb(15, 15, 15)') {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'youtube_channel_sub', channel_id, 'remove');
}
if (color == '#f1f1f1' || color == 'rgb(241, 241, 241)') {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'youtube_channel_sub', channel_id, 'add');
}
}
}
}
if (document.querySelector('.ytLikeButtonViewModelHost')) {
let video_id;
if (hostname == 'm.youtube.com') {
video_id = $('link[rel="canonical"]').attr('href').split('v=').pop();
} else {
video_id = $('meta[itemprop="identifier"]').attr('content');
}
if ($('.ytLikeButtonViewModelHost button').attr('aria-pressed') == 'false') {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'youtube_video_like', video_id, 'remove');
}
if ($('.ytLikeButtonViewModelHost button').attr('aria-pressed') == 'true') {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['tasks']) {
return;
}
storage_tasks_update(STORAGE['tasks'], 'youtube_video_like', video_id, 'add');
}
}
}), 600);
}
}));
if (hostname == 'mee6.xyz') {
var message = false;
$(document).on('keydown', '', (function(event) {
if (event.code == 'End') {
let USERS = [];
$('.md\\:block').each((function(index, el) {
const user = {};
user['name'] = $(this).find('.justify-start p').text();
user['level'] = $(this).find('.leaderboardPlayerStat .items-center').text();
USERS.push(user);
if (user['level'] == 0) {
const obj = {};
obj['discord'] = USERS;
GM_setValue('FAW_STORAGE', obj);
if (message == false) {
alert('Можно переходить на freeanywhere.net');
message = true;
}
return false;
}
}));
}
}));
}
if (hostname == 'gamesforfarm.com') {
const STORAGE = GM_getValue('FAW_STORAGE') || {};
if (!STORAGE['settings']) {
return;
}
if (!STORAGE['games']) {
return;
}
if (STORAGE['settings']['hide_games'] && STORAGE['settings']['hide_games'] == true) {
$('.product__item').each((function(index, el) {
const image = $(this).find('.product__box-image img').data('src');
const cart = $(this).find('.product__box-props a').attr('href');
if (image && image.indexOf('/apps/') != -1) {
const id = parseInt(image.split('/apps/')[1].split('/')[0]);
if (id && isNaN(id) == false) {
if (STORAGE['games'][id]) {
$(this).css('opacity', '.2');
}
}
} else if (cart && cart.indexOf('/app/') != -1) {
const id = parseInt(cart.split('/app/')[1].split('/')[0]);
if (id && isNaN(id) == false) {
if (STORAGE['games'][id]) {
$(this).css('opacity', '.2');
}
}
}
}));
}
}
}
try {
consoleLogHook();
} catch (error) {
console.error('Auto-Task[Warning]: consoleLogHook 初始化失败', error);
}
window.STYLE = GM_addStyle(style + GM_getResourceText('style'));
window.DEBUG = !!globalOptions.other?.debug;
window.TRACE = !!globalOptions.other?.debug && typeof console.trace === 'function';
const handleTwitchAuth = async () => {
debug('开始处理Twitch认证');
const authToken = Cookies.get('auth-token');
const isLogin = !!Cookies.get('login');
if (isLogin) {
const authData = {
authToken: authToken,
clientVersion: window.__twilightBuildID,
clientId: window.commonOptions?.headers?.['Client-ID'],
deviceId: window.commonOptions?.headers?.['Device-ID'],
clientSessionId: window.localStorage.local_storage_app_session_id.replace(/"/g, '')
};
GM_setValue('twitchAuth', authData);
window.close();
await Swal.fire('', I18n('closePageNotice'));
} else {
await Swal.fire('', I18n('needLogin'));
}
};
const handleRedditAuth = async () => {
debug('开始处理Reddit认证');
const betaButton = $('#redesign-beta-optin-btn');
if (betaButton.length > 0) {
betaButton[0].click();
return;
}
window.close();
await Swal.fire('', I18n('closePageNotice'));
};
const handleDiscordAuth = async () => {
debug('开始处理Discord认证');
const LocalStorage = window.localStorage;
const allLocalStorage = getAllLocalStorageAsObjects(LocalStorage);
const discordAuth = allLocalStorage.token;
if (discordAuth && discordAuth.length > 0) {
const browserInfo = await browser.getInfo();
GM_setValue('discordAuth', {
auth: discordAuth,
xSuperProperties: window.btoa(JSON.stringify({
os: browserInfo.system,
browser: browserInfo.browser,
device: '',
system_locale: browserInfo.language,
...allLocalStorage.deviceProperties || {},
browser_user_agent: navigator.userAgent,
browser_version: browserInfo.browserVersion,
os_version: browserInfo.systemVersion,
referrer: '',
referring_domain: '',
referrer_current: '',
referring_domain_current: '',
release_channel: 'stable',
client_build_number: unsafeWindow.GLOBAL_ENV.BUILD_NUMBER,
client_event_source: null,
has_client_mods: false,
client_launch_id: v4(),
client_heartbeat_session_id: allLocalStorage.LAST_CLIENT_HEARTBEAT_SESSION?.uuid,
client_app_state: 'focused'
}))
});
window.close();
Swal.fire('', I18n('closePageNotice'));
} else {
Swal.fire({
text: I18n('getDiscordAuthFailed'),
icon: 'error'
});
}
};
const handleSteamStoreAuth = async () => {
debug('开始处理Steam商店认证');
const storeSessionID = document.body.innerHTML.match(/g_sessionID = "(.+?)";/)?.[1];
if (storeSessionID) {
GM_deleteValue('ATv4_updateStoreAuth');
GM_setValue('steamStoreAuth', {
storeSessionID: storeSessionID
});
window.close();
await Swal.fire('', I18n('closePageNotice'));
} else {
await Swal.fire({
title: 'Error: Get "sessionID" failed',
icon: 'error'
});
}
};
const handleSteamCommunityAuth = async () => {
debug('开始处理Steam社区认证');
const steam64Id = document.body.innerHTML.match(/g_steamID = "(.+?)";/)?.[1];
const communitySessionID = document.body.innerHTML.match(/g_sessionID = "(.+?)";/)?.[1];
if (steam64Id && communitySessionID) {
GM_deleteValue('ATv4_updateCommunityAuth');
GM_setValue('steamCommunityAuth', {
steam64Id: steam64Id,
communitySessionID: communitySessionID
});
window.close();
await Swal.fire('', I18n('closePageNotice'));
} else {
setTimeout((async () => {
await Swal.fire({
title: 'Error: Get "sessionID" failed',
icon: 'error'
});
}), 3e3);
}
};
const initializeUI = website => {
debug('初始化UI元素', {
website: website.name
});
const $body = $('body');
$body.append(`\n <div id="auto-task-info"\n style="display:${globalOptions.other.defaultShowLog ? 'block' : 'none'};\n ${globalOptions.position.logSideX}:${globalOptions.position.logDistance.split(',')[0]}px;\n ${globalOptions.position.logSideY}:${globalOptions.position.logDistance.split(',')[1]}px;">\n </div>\n <div id="auto-task-buttons"\n style="display:${globalOptions.other.defaultShowButton ? 'block' : 'none'};\n ${globalOptions.position.buttonSideX}:${globalOptions.position.buttonDistance.split(',')[0]}px;\n ${globalOptions.position.buttonSideY}:${globalOptions.position.buttonDistance.split(',')[1]}px;">\n </div>\n <div class="show-button-div"\n style="display:${globalOptions.other.defaultShowButton ? 'none' : 'block'};\n ${globalOptions.position.showButtonSideX}:${globalOptions.position.showButtonDistance.split(',')[0]}px;\n ${globalOptions.position.showButtonSideY}:${globalOptions.position.showButtonDistance.split(',')[1]}px;">\n <a class="auto-task-website-btn"\n href="javascript:void(0);"\n target="_self"\n title="${I18n('showButton')}"> </a>\n </div>\n `);
const $autoTaskInfo = $('#auto-task-info');
const $autoTaskButtons = $('#auto-task-buttons');
const $showButtonDiv = $('div.show-button-div');
$showButtonDiv.on('click', (() => {
$autoTaskButtons.show();
$showButtonDiv.hide();
}));
if (website.buttons && $autoTaskButtons.children().length === 0) {
$autoTaskButtons.addClass(`${website.name}-buttons`);
for (const button of website.buttons) {
if (website[button]) {
const btnElement = $(`<p><a class="auto-task-website-btn ${website.name}-button" href="javascript:void(0);" target="_self">${I18n(button)}</a></p>`);
btnElement.find('a.auto-task-website-btn').on('click', (() => {
website[button]();
}));
$autoTaskButtons.append(btnElement);
}
}
}
const hideButtonElement = $(`<p><a class="auto-task-website-btn ${website.name}-button" href="javascript:void(0);" target="_self">${I18n('hideButton')}</a></p>`);
hideButtonElement.find('a.auto-task-website-btn').on('click', (() => {
$autoTaskButtons.hide();
$showButtonDiv.show();
}));
const toggleLogElement = $(`<p><a id="toggle-log" class="auto-task-website-btn ${website.name}-button" href="javascript:void(0);" target="_self" data-status="${globalOptions.other.defaultShowLog ? 'show' : 'hide'}">${globalOptions.other.defaultShowLog ? I18n('hideLog') : I18n('showLog')}</a></p>`);
const toggleLog = () => {
const $toggleLog = $('#toggle-log');
const status = $toggleLog.attr('data-status');
if (status === 'show') {
$autoTaskInfo.hide();
$toggleLog.attr('data-status', 'hide').text(I18n('showLog'));
} else {
$autoTaskInfo.show();
$toggleLog.attr('data-status', 'show').text(I18n('hideLog'));
}
};
toggleLogElement.find('a.auto-task-website-btn').on('click', toggleLog);
$autoTaskButtons.append(hideButtonElement).append(toggleLogElement);
if (website.options) {
GM_registerMenuCommand(I18n('changeWebsiteOptions'), (() => {
websiteOptions(website.name, website.options);
}));
}
};
const initializeHotkeys = website => {
debug('初始化热键', {
website: website.name
});
keyboardJS.bind(globalOptions.hotKey.doTaskKey, (() => {
if (website.doTask) {
website.doTask();
}
}));
keyboardJS.bind(globalOptions.hotKey.undoTaskKey, (() => {
if (website.undoTask) {
website.undoTask();
}
}));
keyboardJS.bind(globalOptions.hotKey.toggleLogKey, (() => {
const $toggleLog = $('#toggle-log');
const status = $toggleLog.attr('data-status');
const $autoTaskInfo = $('#auto-task-info');
if (status === 'show') {
$autoTaskInfo.hide();
$toggleLog.attr('data-status', 'hide').text(I18n('showLog'));
} else {
$autoTaskInfo.show();
$toggleLog.attr('data-status', 'show').text(I18n('hideLog'));
}
}));
};
const checkSteamASFStatus = async () => {
debug('检查Steam ASF状态');
if (!globalOptions.ASF.AsfEnabled || !globalOptions.ASF.AsfIpcUrl || !globalOptions.ASF.AsfIpcPassword) {
return;
}
const stopPlayTime = GM_getValue('stopPlayTime', 0) || 0;
if (stopPlayTime === 0 || stopPlayTime >= Date.now()) {
return;
}
const stopPlayTimeMinutes = Math.floor((Date.now() - stopPlayTime) / 6e4);
await Swal.fire({
title: I18n('stopPlayTimeTitle'),
text: I18n('stopPlayTimeText', stopPlayTimeMinutes.toString()),
icon: 'warning',
confirmButtonText: I18n('confirm')
});
let steamASF = new SteamASF(globalOptions.ASF);
try {
const isInitialized = await steamASF.init();
if (!isInitialized) {
return;
}
const isGamesStopped = await steamASF.stopPlayGames();
if (!isGamesStopped) {
return;
}
const taskLink = GM_getValue('taskLink', []) || [];
for (const link of taskLink) {
GM_openInTab(link, {
active: true
});
}
GM_setValue('stopPlayTime', 0);
GM_setValue('playedGames', []);
GM_setValue('taskLink', []);
} catch (error) {
console.error('SteamASF operation failed:', error);
} finally {
steamASF = null;
}
};
const checkVersionAndNotice = () => {
debug('检查版本和通知');
const {scriptHandler: scriptHandler} = GM_info;
if (scriptHandler !== 'Tampermonkey') {
debug('未知脚本管理器', {
scriptHandler: scriptHandler
});
echoLog({}).warning(I18n('unknownScriptHandler'));
return;
}
const [v1, v2] = GM_info.version?.split('.') || [];
if (!(parseInt(v1, 10) >= 5 && parseInt(v2, 10) >= 2)) {
echoLog({}).error(I18n('versionNotMatched'));
}
if (!GM_getValue('notice')) {
Swal.fire({
title: I18n('swalNotice'),
icon: 'warning'
}).then((() => {
GM_openInTab(I18n('noticeLink'), {
active: true
});
GM_setValue('notice', (new Date).getTime());
}));
echoLog({
html: `<li><font class="warning">${I18n('echoNotice', I18n('noticeLink'))}</font></li>`
}).font?.find('a').on('click', (() => {
GM_setValue('notice', (new Date).getTime());
}));
}
};
const loadScript = async () => {
debug('主程序入口 loadScript 开始');
if (window.name === 'ATv4_twitchAuth' && window.location.hostname === 'www.twitch.tv') {
debug('检测到Twitch认证窗口');
await handleTwitchAuth();
return;
}
if (window.name === 'ATv4_redditAuth' && window.location.hostname === 'www.reddit.com') {
debug('检测到Reddit认证窗口');
await handleRedditAuth();
return;
}
let website;
for (const Website of Websites) {
if (Website.test()) {
debug('识别到支持的网站', {
website: Website.name
});
website = new Website;
break;
}
}
if (!website) {
debug('未识别到支持的网站,脚本停止加载');
console.log('%c%s', 'color:#ff0000', 'Auto-Task[Warning]: 脚本停止加载,当前网站不支持!');
return;
}
if (website.before) {
debug('执行网站 before 钩子');
await website.before();
}
initializeUI(website);
initializeHotkeys(website);
if (website.after) {
debug('执行网站 after 钩子');
await website.after();
}
if (website.name !== 'Setting') {
debug('注册全局菜单命令');
GM_registerMenuCommand(I18n('changeGlobalOptions'), (() => {
changeGlobalOptions('swal');
}));
GM_registerMenuCommand(I18n('settingPage'), (() => {
GM_openInTab('https://auto-task.hclonely.com/setting.html', {
active: true
});
}));
}
debug('脚本加载完成');
console.log('%c%s', 'color:#1bbe1a', 'Auto-Task[Load]: 脚本加载完成');
if (window.DEBUG) {
echoLog({}).warning(I18n('debugModeNotice'));
}
await checkSteamASFStatus();
checkVersionAndNotice();
updateChecker();
};
try {
debug('主程序入口开始', {
hostname: window.location.hostname,
windowName: window.name
});
if ([ 'freeanywhere.net', 'give.gamesforfarm.local', 'gamesforfarm-testing.ru', 'store.steampowered.com', 'steamcommunity.com', 'www.youtube.com', 'm.youtube.com', 'mee6.xyz', 'gamesforfarm.com' ].includes(window.location.hostname) && $('.task-check-extension').length > 0) {
debug('检测到freeanywhere.com,加载扩展');
fawExtension();
}
if (window.location.hostname === 'discord.com') {
if (window.name === 'ATv4_discordAuth') {
debug('检测到Discord认证窗口');
handleDiscordAuth();
} else {
debug('检测到Discord主站');
const discordAuth = window.localStorage?.getItem('token')?.replace(/^"|"$/g, '');
if (discordAuth && discordAuth.length > 0) {
debug('获取到Discord认证token');
GM_setValue('discordAuth', {
auth: discordAuth
});
}
}
} else if (window.location.hostname === 'opquests.com') {
debug('检测到opquests.com,加载主脚本');
loadScript();
} else if ((window.name === 'ATv4_updateStoreAuth' || GM_getValue('ATv4_updateStoreAuth')) && window.location.host === 'store.steampowered.com') {
debug('检测到Steam商店认证窗口');
$((() => {
if ($('[data-miniprofile]').length === 0) {
return;
}
handleSteamStoreAuth();
}));
} else if ((window.name === 'ATv4_updateCommunityAuth' || GM_getValue('ATv4_updateCommunityAuth')) && window.location.host === 'steamcommunity.com') {
debug('检测到Steam社区认证窗口');
$((() => {
handleSteamCommunityAuth();
}));
} else {
if (window.location.hostname === 'key-hub.eu') {
debug('检测到key-hub.eu,设置全局变量');
unsafeWindow.keyhubtracker = 1;
unsafeWindow.gaData = {};
}
debug('加载主脚本');
$(loadScript);
}
} catch (error) {
debug('主程序入口发生异常', {
error: error
});
}
})(Swal, Cookies, browser, util, dayjs, keyboardJS);