├── voice_worker.js
├── LICENSE
├── .github
└── workflows
│ ├── sync.yml
│ └── deploy.yml
├── api_worker.js
├── README.md
└── free_worker.js
/voice_worker.js:
--------------------------------------------------------------------------------
1 | export default {
2 | async fetch(request, env) {
3 | const url = new URL(request.url);
4 | url.host = 'voice.oaifree.com';
5 | return fetch(new Request(url, request));
6 | }
7 | }
8 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2024 YX Jian
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/.github/workflows/sync.yml:
--------------------------------------------------------------------------------
1 | name: Upstream Sync
2 |
3 | permissions:
4 | contents: write
5 |
6 | on:
7 | schedule:
8 | - cron: "0 0 * * *" # every day
9 | workflow_dispatch:
10 |
11 | jobs:
12 | sync_latest_from_upstream:
13 | name: Sync latest commits from upstream repo
14 | runs-on: ubuntu-latest
15 | if: ${{ github.event.repository.fork }}
16 |
17 | steps:
18 | # Step 1: run a standard checkout action
19 | - name: Checkout target repo
20 | uses: actions/checkout@v3
21 |
22 | # Step 2: run the sync action
23 | - name: Sync upstream changes
24 | id: sync
25 | uses: aormsby/Fork-Sync-With-Upstream-action@v3.4
26 | with:
27 | upstream_sync_repo: jyx04/oaifree-helper
28 | upstream_sync_branch: main
29 | target_sync_branch: main
30 | target_repo_token: ${{ secrets.GITHUB_TOKEN }} # automatically generated, no need to set
31 |
32 | # Set test_mode true to run tests instead of the true action!!
33 | test_mode: false
34 |
35 | - name: Sync check
36 | if: failure()
37 | run: |
38 | echo "[Error] 由于上游仓库的 workflow 文件变更,导致 GitHub 自动暂停了本次自动更新,你需要手动 Sync Fork 一次"
39 | echo "[Error] Due to a change in the workflow file of the upstream repository, GitHub has automatically suspended the scheduled automatic update. You need to manually sync your fork."
40 | exit 1
41 |
--------------------------------------------------------------------------------
/api_worker.js:
--------------------------------------------------------------------------------
1 | addEventListener('fetch', event => {
2 | const url = new URL(event.request.url);
3 | //const pathname =url.pathname;
4 | event.respondWith(handleRequest(event.request,url.pathname));
5 |
6 | })
7 |
8 | // @ts-ignore
9 | const KV = oai_global_variables;
10 |
11 | function parseJwt(token) {
12 | const base64Url = token.split('.')[1];// 获取载荷部分
13 | const base64 = base64Url.replace(/-/g, '+').replace(/_/g, '/');
14 | const jsonPayload = decodeURIComponent(atob(base64).split('').map(function (c) {
15 | return '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2);
16 | }).join(''));
17 | return JSON.parse(jsonPayload);// 返回载荷解析后的 JSON 对象
18 | }
19 |
20 |
21 | async function refreshAT(tochecktoken,an) {
22 | const accessTokenKey = `at_${an}`;
23 | const token = tochecktoken || await KV.get(accessTokenKey) ||'';
24 | if (token && token !== "Bad_RT" && token !== "Old_AT")
25 | {
26 | const payload = parseJwt(token);
27 | const currentTime = Math.floor(Date.now() / 1000);
28 | if (payload.exp > currentTime ){
29 | return token
30 | }
31 | }
32 | const refreshTokenKey = `rt_${an}`;
33 | const url = 'https://token.oaifree.com/api/auth/refresh';
34 | const refreshToken = await KV.get(refreshTokenKey);
35 | if (refreshToken) {
36 | // 发送 POST 请求
37 | const response = await fetch(url, {
38 | method: 'POST',
39 | headers: {
40 | 'Content-Type': 'application/x-www-form-urlencoded;charset=UTF-8'
41 | },
42 | body: `refresh_token=${refreshToken}`
43 | });
44 |
45 | // 检查响应状态
46 | if (response.ok) {
47 | const data = await response.json();
48 | const newAccessToken = data.access_token;
49 | await KV.put(accessTokenKey, newAccessToken);
50 | return newAccessToken;
51 | } else {
52 | await KV.put(accessTokenKey, "Bad_RT");
53 | return '';
54 | }
55 | }
56 | else {
57 | await KV.put(accessTokenKey, "Old_AT");
58 | return '';
59 | }
60 |
61 | }
62 |
63 |
64 |
65 |
66 |
67 | async function handleRequest(request,pathname) {
68 | // 检查Authorization头是否包含正确的秘钥
69 | const auth = request.headers.get('Authorization');
70 | const adminKeys = await KV.get('Admin');
71 | const adminKeyList = adminKeys.split(',');
72 |
73 | if (!auth || !adminKeyList.includes(auth.replace('Bearer ', ''))) {
74 | return new Response('Succeed', { status: 200 });
75 | }
76 |
77 | // 从请求中获取用户的数据
78 | const requestData = await request.json();
79 |
80 | // 获取 aliveaccount 的值并解析
81 | const aliveAccount = await KV.get('PlusAliveAccounts');
82 | let aliveAccountList = aliveAccount.split(',');
83 |
84 | if (aliveAccountList.length > 0) {
85 | // 从 aliveAccountList 中随机选一个
86 | const accountNumber = aliveAccountList[Math.floor(Math.random() * aliveAccountList.length)];
87 | const newaccesstoken = await refreshAT('',accountNumber);
88 | //console.log(`Selected account number: ${accountNumber}, Access token: ${newaccesstoken}`);
89 |
90 | // 构建API请求
91 | const apiRequest = new Request(`https://api.oaifree.com${pathname}`, {
92 | method: 'POST',
93 | headers: {
94 | 'Content-Type': 'application/json',
95 | 'Authorization': `Bearer ${newaccesstoken}`
96 | },
97 | body: JSON.stringify(requestData)
98 | });
99 |
100 | // 发送API请求并获取响应
101 | const apiResponse = await fetch(apiRequest);
102 | const responseBody = await apiResponse.text();
103 |
104 |
105 | // 记录响应状态和响应体
106 | //console.log(`Request: ${JSON.stringify(requestData)}`);
107 | //console.log(`Response status: ${apiResponse.status}, Response body: ${responseBody}`);
108 |
109 | if (apiResponse.status === 401) {
110 | // 如果状态码是401,从 aliveAccountList 中删除对应的序号
111 | aliveAccountList = aliveAccountList.filter(account => account !== accountNumber.toString());
112 | await KV.put('PlusAliveAccounts', aliveAccountList.join(','));
113 | // console.log(`Removed account number: ${accountNumber} from aliveAccountList`);
114 | await deletelog('API', accountNumber,'Plus')
115 | }
116 |
117 | // 返回API响应给用户
118 | return new Response(responseBody, {
119 | status: apiResponse.status,
120 | headers: apiResponse.headers
121 | });
122 | }
123 | }
124 |
125 | async function deletelog(userName, accountNumber,antype) {
126 | const currentTime = new Date().toLocaleString("zh-CN", { timeZone: "Asia/Shanghai" });
127 | const logEntry = {
128 | user: userName,
129 | time: currentTime,
130 | accountNumber: accountNumber
131 | };
132 | // Retrieve the existing log array or create a new one if it doesn't exist
133 | // @ts-ignore
134 | const lastDeleteLogs = await KV.get(`${antype}DeleteLogs`);
135 | let logArray = [];
136 | if (lastDeleteLogs) {
137 | logArray = JSON.parse(lastDeleteLogs);
138 | }
139 | logArray.push(logEntry);
140 | // @ts-ignore
141 | await KV.put(`${antype}DeleteLogs`, JSON.stringify(logArray));
142 | }
143 |
144 |
145 |
146 |
147 |
--------------------------------------------------------------------------------
/.github/workflows/deploy.yml:
--------------------------------------------------------------------------------
1 | name: Deploy to Cloudflare Workers
2 |
3 | on:
4 | workflow_dispatch:
5 | inputs:
6 | environment:
7 | description: 'wrangler environment to deploy to'
8 | required: true
9 | default: 'dev'
10 | type: choice
11 | options:
12 | - dev
13 | - prod
14 | commit:
15 | description: 'Git commit to deploy'
16 | default: 'main'
17 | required: true
18 |
19 | push:
20 | branches:
21 | - "main"
22 | repository_dispatch:
23 |
24 | env:
25 | GIT_REF: ${{ github.event.inputs.commit || github.ref }}
26 | WORKERS_ENV: ${{ github.event.inputs.environment || 'dev' }}
27 |
28 | jobs:
29 | deploy:
30 | name: Deploy workers
31 | runs-on: ubuntu-latest
32 | steps:
33 | - name: Checkout
34 | uses: actions/checkout@v2
35 | with:
36 | ref: ${{ env.GIT_REF }}
37 |
38 | - name: Install Wrangler
39 | run: npm install -g wrangler
40 |
41 | - name: Set Cloudflare Environment Variables
42 | run: |
43 | echo "CLOUDFLARE_ACCOUNT_ID=${{ secrets.CLOUDFLARE_ACCOUNT_ID }}" >> $GITHUB_ENV
44 | echo "CLOUDFLARE_API_TOKEN=${{ secrets.CLOUDFLARE_API_TOKEN }}" >> $GITHUB_ENV
45 |
46 | - name: Validate Cloudflare Credentials
47 | run: |
48 | curl -X GET "https://api.cloudflare.com/client/v4/accounts" \
49 | -H "Authorization: Bearer ${{ secrets.CLOUDFLARE_API_TOKEN }}" \
50 | -H "Content-Type: application/json"
51 |
52 | - name: Create or Use Existing KV Namespace
53 | id: create_or_find_kv
54 | run: |
55 | echo "Listing KV namespaces"
56 | namespace_output=$(wrangler kv namespace list)
57 | echo "Namespace list output: $namespace_output"
58 | existing_namespace_id=$(echo "$namespace_output" | jq -r '.[] | select(.title == "oai_global_variables" or .title == "worker-oai_global_variables" or .title == "oaifreehelper-oai_global_variables" or .title == "oaifreehelper-oai_global_variables_preview" or .title == "worker-oai_global_variables_preview") | .id' | head -n 1)
59 | if [ -z "$existing_namespace_id" ]; then
60 | echo "No existing KV namespace found, creating a new one."
61 | namespace_creation_output=$(wrangler kv namespace create "oai_global_variables") || exit 1
62 | echo "Namespace creation output: $namespace_creation_output"
63 | namespace_id=$(echo "$namespace_creation_output" | grep -oP '(?<=id = ")[^"]+')
64 | echo "CF_KV_NAMESPACE_ID=$namespace_id" >> $GITHUB_ENV
65 | else
66 | echo "Found existing KV namespace with ID: $existing_namespace_id"
67 | echo "CF_KV_NAMESPACE_ID=$existing_namespace_id" >> $GITHUB_ENV
68 | fi
69 |
70 |
71 | - name: Generate wrangler.toml for main worker
72 | run: |
73 | echo "name = \"oaifreehelper\"" > wrangler.toml
74 | echo "workers_dev = true" >> wrangler.toml
75 | echo "main = \"_worker.js\"" >> wrangler.toml
76 | echo "compatibility_date = \"2024-06-01\"" >> wrangler.toml
77 | echo "[[kv_namespaces]]" >> wrangler.toml
78 | echo "binding = \"oai_global_variables\"" >> wrangler.toml
79 | echo "id = \"$CF_KV_NAMESPACE_ID\"" >> wrangler.toml
80 |
81 | - name: Publish main worker to Cloudflare Workers
82 | env:
83 | CLOUDFLARE_API_TOKEN: ${{ secrets.CLOUDFLARE_API_TOKEN }}
84 | run: wrangler deploy
85 |
86 | - name: Generate wrangler.toml for api worker
87 | run: |
88 | echo "name = \"api\"" > wrangler.toml
89 | echo "workers_dev = true" >> wrangler.toml
90 | echo "main = \"api_worker.js\"" >> wrangler.toml
91 | echo "compatibility_date = \"2024-06-01\"" >> wrangler.toml
92 | echo "[[kv_namespaces]]" >> wrangler.toml
93 | echo "binding = \"oai_global_variables\"" >> wrangler.toml
94 | echo "id = \"$CF_KV_NAMESPACE_ID\"" >> wrangler.toml
95 |
96 | - name: Publish api worker to Cloudflare Workers
97 | env:
98 | CLOUDFLARE_API_TOKEN: ${{ secrets.CLOUDFLARE_API_TOKEN }}
99 | run: wrangler deploy
100 |
101 | - name: Generate wrangler.toml for free worker
102 | run: |
103 | echo "name = \"free\"" > wrangler.toml
104 | echo "workers_dev = true" >> wrangler.toml
105 | echo "main = \"free_worker.js\"" >> wrangler.toml
106 | echo "compatibility_date = \"2024-06-01\"" >> wrangler.toml
107 | echo "[[kv_namespaces]]" >> wrangler.toml
108 | echo "binding = \"oai_global_variables\"" >> wrangler.toml
109 | echo "id = \"$CF_KV_NAMESPACE_ID\"" >> wrangler.toml
110 |
111 | - name: Publish free worker to Cloudflare Workers
112 | env:
113 | CLOUDFLARE_API_TOKEN: ${{ secrets.CLOUDFLARE_API_TOKEN }}
114 | run: wrangler deploy
115 |
116 | - name: Generate wrangler.toml for voice worker
117 | run: |
118 | echo "name = \"voice\"" > wrangler.toml
119 | echo "workers_dev = true" >> wrangler.toml
120 | echo "main = \"voice_worker.js\"" >> wrangler.toml
121 | echo "compatibility_date = \"2024-06-01\"" >> wrangler.toml
122 |
123 | - name: Publish api worker to Cloudflare Workers
124 | run: wrangler deploy
125 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | 本项目完全依赖Linuxdo始皇大神的服务,致力于更优雅个性化访问new.oaifree站。如您无Linuxdo等级,不必浪费时间部署,本服务对您大概率无用或不好用!!!
2 | # oaifree-helper
3 | ### 本项目基于始皇的new站服务。利用单个Worker&Pages优雅访问始皇镜像站,组建合租共享车队。包含直链登陆、前端登陆页、用户管理、token池管理、车队管理、用户注册、用量查询等等功能。全程无需服务器和域名,无需改代码。
4 |
5 | # 首先,致敬始皇,致敬所有热佬,没有他们的项目和服务就没有这个项目。
6 | ### [体验站](https://oaifreehelper.haibara-ai.workers.dev) 密码linux.do,无功能,请勿填写敏感信息。
7 | ### 主要功能
8 | - 原理是储存`refreshtoken`和`accesstoken`,并调用始皇的各项接口获取`sharetoken`一键直达始皇的new.oaifree.com镜像站
9 | - 用户使用唯一用户名登陆即可后台自动分配`sharetoken`,自带始皇的聊天隔离功能。包含简易的用户体系,储存各类用户,设置各类用户的限额和限制
10 | - 支持使用/?un=xxx的直链登陆,分享更省心。
11 | - 自带注册功能,分享激活码给朋友,不用总手动录入用户
12 | - 支持组建token池,可前端面板储存token,支持自动判断rt/at,自动解析json
13 | - 支持token自动刷新,若遇at过期,自动调用始皇接口刷新at
14 | - 包含多种选车模式,可手动/指定用户专车/顺序轮询/随机选车
15 | - 支持禁用失效车次
16 | - 自动检测官方服务状态,如遇官方故障自动禁止用户登陆,甩锅官方
17 | - 支持人机验证
18 | - 点击登录页Logo跳转管理面板,包含用户管理、token池管理、用量查询、token批量导出
19 | - 支持替换Chat页面显示的头像/用户名/邮箱【新】
20 | - 支持道德审查接口【新】
21 |
22 |
23 |
24 |
25 |
26 |
27 | # Worker 部署(一键直达)
28 | [](https://deploy.workers.cloudflare.com/?url=https://github.com/jyx04/oaifree_helper)
29 | - 一键为全家桶,包含主服务/选车面板服务/API服务/反代voice服务,且无需手动关联KV,即点即用
30 | - 配置完成后,如需添加人机验证器防爆破,请按照下方Turnstile人机验证服务教程,获得`站点密钥`和`密钥`
31 | - 访问部署域名,在初始界面一键保存各项变量,完成部署!
32 | - 添加token:在登陆页点击logo,选择Token Management进入token管理面板添加
33 | # Worker 部署(手动部署)
34 | ### 1. 配置Turnstile人机验证服务(不建议跳过)
35 | - 如需跳过,后期将`RemoveTurnstile`参数设置为1即可
36 | - 注册/登陆你的cloudflare,右上角可设置语言为中文。
37 | - 左侧找到`Turnstile`,选择`添加站点`
38 | - `站点名称`随意,`域`为:`workers.dev`或你自己的域名
39 | - 创建,记录好`站点密钥`和`密钥`,备用
40 |
41 | ### 2. 部署 Cloudflare Worker:
42 | - 在左侧列表找到`Worker和Pages`
43 | - 选择`KV`,创建一个名为`oai_global_variables`的KV备用
44 | - 选择`概述`-`创建应用程序`-`Worker`,为项目命名,并创建worker
45 | - 进入`worker`-`设置`-`变量`,在`KV 命名空间绑定`添加绑定KV,变量名`oai_global_variables`
46 | - 【可选】在worker的`设置`-`触发器`-`添加自定义域`绑定自己的域名
47 | - 回到本GitHub项目,复制`_worker.js`中的全部内容,在worker配置页面点击 `编辑代码`,清空原有内容粘贴后点右上角`部署`
48 | - 大功告成!
49 | - 访问`自定义的域名`,或点击`部署`-`查看版本`,在初始面板一键保存各项环境变量(初次保存后后该页面自动禁用,若需更改请至KV中调整)
50 |
51 | ### 3. 环境变量
52 | - 以下是所有变量,全部无需手动填写,部署完项目后直接第一次进入可前端面板一键保存。
53 | - 如需进行修改,位置在KV,而非worker的环境变量!
54 | ```
55 | Admin //管理员,用于管理面板的验证使用,且可看所有聊天记录【必填】
56 | TurnstileKeys //turnsile的密钥【必填】
57 | TurnstileSiteKey //turnsile的密钥【必填】
58 | RemoveTurnstile//跳过turnsile人机验证。设置跳过,以上两参数随便填
59 | WebName //站点名称
60 | WorkerURL //站点域名,无需https://若无自己的域名,则为worker默认域名:[worker名].[用户名].workers.dev
61 | LogoURL //图片地址,需https://,若无图床可填图片本地base64编码,不宜过大
62 | ChatLogoURL //chat界面显示的用户头像地址,需https://,若无图床可填图片本地base64编码,不宜过大
63 | ChatUserName //chat界面显示的用户名
64 | ChatMail //chat界面显示的用户邮箱
65 |
66 | Users //默认用户,以aaa,bbb,ccc形式填写,能访问所有车牌
67 | VIPUsers //vip用户,即私车用户,无速率和时间限制
68 | FreeUsers //限制用户,有速率和时间限制
69 |
70 | ForceAN //强制选车,若设置为1,用户名为xxx_n的私车用户用登陆强制进入n号车,忽略登陆所选车号
71 | SetAN //选车模式。如只有一辆车则填1。如多辆车用户手动选则留空。如需开启随机或顺序轮询,填True,并用下面两个变量控制
72 | PlusMode //plus号随机的轮询方式,Order或者Random
73 | FreeMode //普号随机的轮询方式,Order/Random或Plus(使用plus号池和配置)
74 |
75 | CDKEY //注册可用的激活码,以aaa,bbb,ccc格式
76 | AutoDeleteCDK //设置为1则激活码只可用一次
77 | FKDomain //把sharetoken当at用时,走的默认域名
78 | Status //服务状态,若为非空,无视openai官方故障通告,始终允许登陆
79 | TemporaryAN //强制启用临时聊天的车牌,以1,2,3格式
80 |
81 | //以下在管理员面板添加更方便
82 | PlusAliveAccounts //plus号池存活序号,以1,2,3格式
83 | FreeAliveAccounts //普号存活序号,以1,2,3格式
84 | rt_1
85 | rt_2
86 | at_1//(若已有rt,at可不填)
87 | at_2
88 | ……
89 | ```
90 |
91 | ### 4. 选车面板(可选)
92 | - 通过文件`free_worker.js`部署worker,即可配置基于普号号池的选车上车界面。(一键部署已包含)
93 | - 大部分变量同上,可以额外配置以下变量
94 | ```
95 | FreeURL //单独的URL
96 | FreeWebName //选车上车页的站点名
97 | FreeWebIntro //选车上车页的简介,可用html代码插入文本、超链接等
98 | ```
99 |
100 |
101 | ### 5. API接口(可选)
102 | - 通过文件`api_worker.js`部署worker,即可配置基于plus号池的api服务。(一键部署已包含)
103 |
104 | ### 6. 反代始皇的Voice服务(新增)
105 | - 通过文件`voice_worker.js`部署voice服务的反代worker。(一键部署已包含)
106 | - 需在系统KV配置变量`VoiceURL`为此worker的链接(无需https://)
107 | - 点击镜像页中间的logo,优雅访问voice服务
108 |
109 | # 使用教程
110 | ### 1. 管理面板
111 | - 配置完成后,点击登录页面的logo,可进入管理面板
112 |
113 | ### 2. Token管理
114 | - 见管理员面板的Token Management功能
115 | - 获取token:可通过[始皇的服务](https://token.oaifree.com/auth)获取普号或Plus号的rt/at(需linux.do高级用户)。也可自行通过网页获取at(自行查询教程)
116 | - 添加token:可批量输入 rt/at,以','分割,支持自动识别token类型。也可粘贴单个token的完整json,自动提取添加。添加的token将自动识别普号/plus,将序号加入对应的AliveAccountLists索引。随token添加的user为跟车用户,自动绑定车号,若设置ForceAN则强制以该车号登录。
117 | - 更新token:若存有rt,at过期将自动刷新。若无rt,将在登陆页提醒at过期
118 | - 禁用token:AliveAccountLists存有所有有效token的序号。通过账号登录页面可报告账号问题,删除序号。也可通过API调用,自动删除失效token的序号。
119 |
120 | ### 3. 批量导出号池token功能(新增)
121 | - 见管理员面板的ExportTokens功能
122 | - 可选导出Plus/Free号池
123 | - 可选生成导出链接or直接下载txt文件
124 | - txt文件格式为每行一个token,便于挪至其他服务使用
125 |
126 | ### 4. 用户和车次管理
127 | - 见管理员面板的User Management功能
128 | - 用户添加:VIPUser的有效期最长,无用量限制;User为普通用户;FreeUser为受限用户。
129 | - 用户车号选择:`SetAN`不填则用户手动选,序号则所有用户以该序号登录,True则由系统选。
130 | - 系统车号选择:当`SetAN`为True,可在`PlusMode`填入Random或Order,应用VIPUSer和User的自动选车模式。在`FreeMode`填入Random/Order或Plus,应用FreeUser的自动选车模式。Random为随机,Order为顺序,Plus为使用Plus号池和模式。
131 |
132 | ### 5. 用户注册
133 | - `CDKEY`内存有效激活码,`AutoDeleteCDK`非空则激活码只能使用一次,否则用后自动删除
134 |
135 | ### 6. 用量查询
136 | - 见管理员面板的Query User Usage功能
137 | - 若输入管理员账号,可分别查询用户和免费用户的用量,可储存和为用户名打码
138 | - 若输入非管理员账号,则查询当前用户用量
139 |
140 | ### 7. API接口(同样基于始皇的转API服务)
141 | - 本接口同样采用始皇的服务,使用plus号池内账号的token,随机调取,失效自动禁用
142 | - api的BaseURL为api worker的地址,`apikey`为admin密码,支持的服务请参考始皇的服务文档
143 | - OneAPI/NewAPI示例:
144 | -
145 |
146 |
147 | ### 8. 接入道德审查功能(新增)
148 | - 此功能代码完全来自[Linux.do-Lvguanjun](https://linux.do/t/topic/99742),感谢大佬!
149 | - 如需启用此功能,需在KV中新增变量`ModerationApiKey`,填入始皇oaipro的apikey
150 |
151 | ### 9. 个性化和杂项
152 | - 参考以下环境变量
153 | ```
154 | WebName //站点名称
155 | WorkerURL //站点域名,无需https://若无自己的域名,贼为worker默认域名:[worker名].[用户名].workers.dev【必填】
156 | LogoURL //图片地址,需https://,若无图床可填图片本地base64编码,不宜过大
157 | ChatLogoURL //chat界面显示的用户头像地址,需https://,若无图床可填图片本地base64编码,不宜过大
158 | ChatUesrName //chat界面显示的用户名
159 | ChatMail //chat界面显示的用户邮箱
160 | FreeWebName //选车上车页的站点名
161 | FreeWebIntro //选车上车页的简介,可用html代码插入文本、超链接等
162 | ```
163 | - 如需修改用户的默认用量限制等,请修改worker的`getShareToken`函数,内有详细注释
164 |
165 | ### 10. 常见问题
166 | - 一些功能(如自定义头像/道德审查等),如无需使用,可在fork的项目代码内注释掉,避免聊天过程中频繁请求KV造成额度不足。
167 | - 若用量较大导致KV额度不足,建议将以上个性化参数都定义进代码,避免网页请求都频繁调用KV
168 |
169 |
170 | ### 11. About
171 | - 本服务实质是Token储存和分发,所有功能和实际服务都基于始皇大神的付出。再次向大神致敬!
172 | - Bug反馈和功能建议请在Github提交issues。故障反馈需包含log
173 | - 本项目不会收费也不会引流,个人代码能力有限,会尽力维护
174 |
175 | # 日志
176 | - 建立GitHub项目
177 | - 创建一键部署,新增选车界面和api服务
178 | - 优化用量查询功能
179 | - 新增token导出功能,可导出所选号池的rt/at为txt文件
180 | - 支持反代始皇新彩蛋:voice服务
181 | - 支持替换Chat页面显示的头像/用户名/邮箱
182 | - 新增Chat页面用户名显示车次的功能
183 | - 优化token批量导出功能和转api功能,确保at自动刷新
184 | - 支持接入道德审查接口
185 |
--------------------------------------------------------------------------------
/free_worker.js:
--------------------------------------------------------------------------------
1 | addEventListener('fetch', event => {
2 | event.respondWith(handleRequest(event.request));
3 | });
4 |
5 | // @ts-ignore
6 | const KV = oai_global_variables;
7 | const logo = '';
8 |
9 |
10 | function parseJwt(token) {
11 | const base64Url = token.split('.')[1];// 获取载荷部分
12 | const base64 = base64Url.replace(/-/g, '+').replace(/_/g, '/');
13 | const jsonPayload = decodeURIComponent(atob(base64).split('').map(function (c) {
14 | return '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2);
15 | }).join(''));
16 | return JSON.parse(jsonPayload);// 返回载荷解析后的 JSON 对象
17 | }
18 |
19 | // 使用哈希函数加密username
20 | function generatePassword(token) {
21 | let hash = 7
22 | for (let i = 0; i < token.length; i++) {
23 | const char = token.charCodeAt(i)
24 | hash = (hash << 5) - hash + char
25 | hash = hash & hash // Convert to 32bit integer
26 | }
27 | // 将哈希值转换为正数,并转换为字符串
28 | let hashStr = Math.abs(hash).toString()
29 | // 如果 hashStr 长度不足 10 位,用 '7' 填充
30 | while (hashStr.length < 15) {
31 | hashStr = '7' + hashStr
32 | }
33 | // 截取前10位作为密码
34 | return hashStr.substring(0, 15)
35 | }
36 |
37 | function isTokenExpired(token) {
38 | // 检查 token 是否存在,如果不存在或为空字符串,直接返回 true
39 | if (!token || token === "Bad_RT" ||token === "Bad_AT" ) {
40 | return true;
41 | }
42 | const payload = parseJwt(token);
43 | const currentTime = Math.floor(Date.now() / 1000);// 获取当前时间戳(秒)
44 | return payload.exp < currentTime;// 检查 token 是否过期
45 | }
46 |
47 | async function getOAuthLink(shareToken, proxiedDomain) {
48 | //const url = `https://${proxiedDomain}/api/auth/oauth_token`;
49 | // 不知道为什么,好像没法直接通过反代的服务器获取oauth link
50 | const url = `https://new.oaifree.com/api/auth/oauth_token`;
51 |
52 | const response = await fetch(url, {
53 | method: 'POST',
54 | headers: {
55 | 'Origin': `https://${proxiedDomain}`,
56 | 'Content-Type': 'application/json',
57 | },
58 | body: JSON.stringify({
59 | share_token: shareToken
60 | })
61 | })
62 | const data = await response.json();
63 | return data.login_url;
64 | }
65 |
66 | async function usermatch(userName, usertype) {
67 | const typeUsers = await KV.get(usertype) ||'';
68 | const typeUsersArray = typeUsers.split(","); // 将返回的用户类型字符串分割成数组
69 | return typeUsersArray.includes(userName); // 检查用户名是否在类型用户数组中
70 | }
71 |
72 | async function getShareToken(userName, accessToken,accountNumber) {
73 | const url = 'https://chat.oaifree.com/token/register';
74 | /*
75 | const tokenPrefix = await KV.get('token_prefix');
76 | const baseUserName = tokenPrefix + userName.replace(/_\d+$/, ''); // 移除用户名后的编号
77 | */
78 | const passwd = await generatePassword(userName);
79 | const isAdmin = await usermatch(userName, 'Admin') || userName=='atdirect';
80 | const isVIP = await usermatch(userName, 'VIPUsers') || await usermatch(userName, 'Admin') ;
81 | const isFreeUsers = await usermatch(userName, 'FreeUsers');
82 | const isTemporary = await usermatch(accountNumber, 'TemporaryAN') && !isAdmin;
83 |
84 |
85 | //console.log(`getShareToken - userName: ${userName}, accountNumber: ${accountNumber}, showConversations: ${isAdmin}, isVIP: ${isVIP}, isTemporary: ${isTemporary}, accessToken: ${accessToken}`);
86 | const body = new URLSearchParams({
87 | access_token: accessToken, // 使用从全局变量中获取的 accessToken
88 | unique_name: passwd, //前缀+无后缀用户名
89 | site_limit: '', // 限制的网站
90 | expires_in: isVIP ? '0' : '0', // token有效期(单位为秒),填 0 则永久有效
91 | gpt35_limit: '-1', // gpt3.5 对话限制
92 | gpt4_limit: isFreeUsers ? '-1' : '-1', // gpt4 对话限制,-1为不限制
93 | show_conversations: isAdmin ? 'true' : 'false', // 是否显示所有人的会话
94 | temporary_chat: isTemporary ? 'true' : 'false', //默认启用临时聊天
95 | show_userinfo: 'false', // 是否显示用户信息
96 | reset_limit: 'false' // 是否重置对话限制
97 | }).toString();
98 | const apiResponse = await fetch(url, {
99 | method: 'POST',
100 | headers: {
101 | 'Content-Type': 'application/x-www-form-urlencoded'
102 | },
103 | body: body
104 | });
105 | const responseText = await apiResponse.text();
106 | const tokenKeyMatch = /"token_key":"([^"]+)"/.exec(responseText);
107 | const tokenKey = tokenKeyMatch ? tokenKeyMatch[1] : 'Can not get share token.';
108 | return tokenKey;
109 | }
110 |
111 |
112 | async function handleRequest(request) {
113 | const url = new URL(request.url);
114 | const voiceURL = await KV.get('VoiceURL');
115 | const chatlogourl = await KV.get('ChatLogoURL') || await KV.get('LogoURL') || logo;
116 | const chatusername = await KV.get('ChatUserName') || 'Haibara AI';
117 | const chatmail = await KV.get('ChatMail') || 'Power by Pandora';
118 | const cookies = request.headers.get('Cookie');
119 | let aian = '';
120 | if (cookies) {
121 | const cookiesArray = cookies.split(';');
122 | for (const cookie of cookiesArray) {
123 | const [name, value] = cookie.trim().split('=');
124 | if (name === 'aian') {
125 | aian = value;
126 | }
127 | }
128 | }
129 |
130 |
131 | // Specific handling for /auth/login_auth0
132 | if (url.pathname === '/auth/login_auth0') {
133 | const html = await getHTMLSelectionPage();
134 | return new Response(html, { headers: { 'Content-Type': 'text/html' } });
135 |
136 | }
137 | if (url.pathname === '/auth/login_auth0/login') {
138 | const params = new URLSearchParams(url.search);
139 | const encryptedAN = params.get('an');
140 | if (request.method === 'GET') {
141 | if (encryptedAN) {
142 | if (encryptedAN == 'Random'){
143 | const accountNumber = 'Random';
144 | const html = await getHTMLLoginPage(accountNumber);
145 | return new Response(html, { headers: { 'Content-Type': 'text/html' } });
146 | }
147 | else {try {
148 | const accountNumber = await decrypt(encryptedAN, 'eGJjZWFsZGVuYmVqZGZzY3VhbmZpZGZz'); //必须是32位
149 | const html = await getHTMLLoginPage(accountNumber);
150 | return new Response(html, { headers: { 'Content-Type': 'text/html' } });
151 | } catch (e) {
152 | return new Response('Invalid account number', { status: 400 });
153 | }}
154 | } else {
155 | const html = await getHTMLSelectionPage();
156 | return new Response(html, { headers: { 'Content-Type': 'text/html' } });
157 | }}
158 | if (request.method === 'POST') {
159 | return handlePostRequest(request);
160 | }
161 | }
162 |
163 | if (url.pathname === '/auth/login') {
164 | /* const token = url.searchParams.get('token');
165 | if (!token) {
166 | if (request.method === 'GET') {
167 | return handleLoginGetRequest(request);
168 | } else if (request.method === 'POST') {
169 | return handleLoginPostRequest(request);
170 | } else {
171 | return new Response('Method not allowed', { status: 200 });
172 | }
173 | } */
174 | url.host = 'new.oaifree.com';
175 | url.protocol = 'https';
176 | return fetch(new Request(url, request));
177 | }
178 | //Voice地址和其他
179 | url.host = 'new.oaifree.com';
180 | url.protocol = 'https';
181 | const modifiedRequest = new Request(url, request);
182 | if(voiceURL){
183 | modifiedRequest.headers.set('X-Voice-Base', `https://${voiceURL}`);
184 | }
185 | const response = await fetch(modifiedRequest);
186 |
187 | //去掉小锁
188 | if (url.pathname === '/backend-api/conversations') {
189 | const data = await response.json();
190 | data.items = data.items.filter(item => item.title !== "🔒");
191 | return new Response(JSON.stringify(data), {
192 | status: response.status,
193 | headers: response.headers
194 | });
195 | }
196 |
197 | //修改用户信息
198 | if (url.pathname === '/backend-api/me') {
199 | const data = await response.json();
200 | data.picture = `${chatlogourl}`;
201 | data.email = `${chatmail}`;
202 | data.name = `${chatusername} [${aian}]`;
203 | return new Response(JSON.stringify(data), {
204 | status: response.status,
205 | headers: response.headers
206 | });
207 | }
208 | if (url.pathname === '/backend-api/accounts/check') {
209 | const data = await response.json();
210 | for (const accountId in data.accounts) {
211 | if (data.accounts[accountId].account) {
212 | data.accounts[accountId].account.name = `${chatusername} [${aian}]`;
213 | }
214 | }
215 | return new Response(JSON.stringify(data), {
216 | status: response.status,
217 | headers: response.headers
218 | });
219 | }
220 | if (url.pathname === '/backend-api/gizmo_creator_profile') {
221 | const data = await response.json();
222 | data.name = `${chatusername} [${aian}]`;
223 | data.display_name = `${chatusername} [${aian}]`;
224 | return new Response(JSON.stringify(data), {
225 | status: response.status,
226 | headers: response.headers
227 | });
228 | }
229 | return response;
230 | }
231 |
232 |
233 |
234 | async function handlePostRequest(request) {
235 | const formData = await request.formData();
236 | const userName = formData.get('un');
237 | const accountNumber = formData.get('accountNumber');
238 | const turnstileResponse = formData.get('cf-turnstile-response');
239 | const anissues = formData.get('anissues') === 'on';
240 |
241 | return await handleLogin(userName, accountNumber, turnstileResponse, anissues);
242 | }
243 |
244 | async function handleLogin(userName, initialaccountNumber, turnstileResponse, anissues) {
245 | //Turnsile认证
246 | if (turnstileResponse !== 'do not need Turnstle' && (!turnstileResponse || !await verifyTurnstile(turnstileResponse))) {
247 | return generateErrorResponse('Turnstile verification failed',initialaccountNumber);
248 | }
249 | const GPTState = await getGPTStatus();
250 | const status = await KV.get('Status');
251 | if ((GPTState == 'major_performance')&&(!status)){
252 | await loginlog(userName, 'Bad_OAIStatus','Error');
253 | return generateErrorResponse(`OpenAI service is under maintenance.
Official status: ${GPTState}
More details: https://status.openai.com`);
254 | }
255 |
256 |
257 |
258 | const proxiedDomain = await KV.get('FreeURL') || await KV.get('WorkerURL');//修改为反代new站的地址
259 | /*
260 | try {
261 | const tokenData = JSON.parse(userName);
262 | if (tokenData.accessToken) {
263 | const jsonAccessToken = tokenData.accessToken;
264 | const shareToken = await getShareToken('atdirect', jsonAccessToken, '0');
265 | if (shareToken === 'Can not get share token.') {
266 | return generateErrorResponse(`Error fetching share token.`,accountNumber);
267 | }
268 |
269 | return Response.redirect(await getOAuthLink(shareToken, proxiedDomain), 302);
270 | }
271 | } catch (e) {
272 | // 输入不是 JSON 格式
273 | }
274 |
275 |
276 | // 如果输入用户名长度大于50,直接视作accessToken
277 | if (userName.length > 50) {
278 | const shareToken = await getShareToken('atdirect', userName, '0');
279 |
280 | if (shareToken === 'Can not get share token.') {
281 | return generateErrorResponse(`Error fetching share token.`,accountNumber);
282 | }
283 |
284 | return Response.redirect(await getOAuthLink(shareToken, proxiedDomain), 302);
285 | }
286 |
287 |
288 | // 如果输入用户名fk开头,直接视作sharetoken
289 | if (userName.startsWith('fk-')) {
290 | const shareToken = userName;
291 | return Response.redirect(await getOAuthLink(shareToken, proxiedDomain), 302);
292 | }
293 |
294 | */
295 |
296 | const userRegex = new RegExp(`^${userName}_(\\d+)$`);
297 | let fullUserName = userName;
298 | let foundSuffix = false;
299 | let accountNumber = '';
300 | // let suffix = '';
301 |
302 |
303 | //const freeforcecar = await KV.get("FreeForceAN");
304 | const defaultusers = await KV.get("Users");
305 | const vipusers = await KV.get("VIPUsers");
306 | const freeusers = await KV.get("FreeUsers");
307 | const admin = await KV.get("Admin");
308 |
309 | // 合并所有用户
310 | const users = `${defaultusers},${vipusers},${freeusers},${admin}`;
311 |
312 |
313 | // 自动查找匹配的用户名格式abc_xxx,并添加后缀
314 | users.split(",").forEach(user => {
315 | const match = user.match(userRegex);
316 | if (match) {
317 | foundSuffix = true;
318 | // suffix = match[1]; // 更新后缀为实际的账号编号
319 | fullUserName = user; // 更新为完整的用户名
320 | }
321 | });
322 |
323 |
324 | if (!foundSuffix && !users.split(",").includes(userName)) {
325 | await loginlog(userName, '404','Error');
326 | return generateErrorResponse('Unauthorized access.',initialaccountNumber);
327 | }
328 |
329 | //禁止免费用户使用其他车
330 | const notfreeaccount = !(await usermatch(initialaccountNumber, 'FreeAliveAccounts')) && (await usermatch(fullUserName, 'Freeusers'));
331 | if (notfreeaccount) {
332 | return generateErrorResponse('Unauthorized access, please switch accounts.',initialaccountNumber);
333 | }
334 | if (initialaccountNumber =='Random'){
335 | accountNumber = await getAccountNumber(fullUserName,'', 'Free', 'Random',anissues);
336 | }
337 | else{
338 | accountNumber = await getAccountNumber(fullUserName,initialaccountNumber, 'Free', 'Check',anissues);
339 | }
340 | const refreshTokenKey = `rt_${accountNumber}`;
341 | const accessTokenKey = `at_${accountNumber}`;
342 | const accessToken = await KV.get(accessTokenKey);
343 |
344 | //使用佬友的sharetoken
345 | if (accessToken){
346 | if (accessToken.startsWith('fk-')) {
347 | const fkDomain = await KV.get('FKDomain') ||proxiedDomain;
348 | //return Response.redirect(await getOAuthLink(accessToken, fkDomain), 302);
349 | return Response.redirect(`https://${fkDomain}/auth/login_share?token=${accessToken}`)
350 | }
351 | }
352 |
353 | if (isTokenExpired(accessToken)) {
354 | // 给没有refresh token的萌新用(比如我),取消下面这行注释即可享用
355 | // return generateErrorResponse('The current access token has not been updated.',accountNumber);
356 | // 如果 Token 过期,执行获取新 Token 的逻辑
357 | const url = 'https://token.oaifree.com/api/auth/refresh';
358 | const refreshToken = await KV.get(refreshTokenKey);
359 | if (refreshToken) {
360 |
361 | // 发送 POST 请求
362 | const response = await fetch(url, {
363 | method: 'POST',
364 | headers: {
365 | 'Content-Type': 'application/x-www-form-urlencoded;charset=UTF-8'
366 | },
367 | body: `refresh_token=${refreshToken}`
368 | });
369 |
370 | // 检查响应状态
371 | if (response.ok) {
372 | const data = await response.json();
373 | const newAccessToken = data.access_token;
374 | await KV.put(accessTokenKey, newAccessToken);
375 | } else {
376 | await KV.put(accessTokenKey, "Bad_RT");
377 | return generateErrorResponse(`Error fetching access token.`,accountNumber);
378 | }
379 | }
380 | else {
381 | return generateErrorResponse('The current access token has not been updated.',accountNumber);
382 | }
383 | }
384 | const finalaccessToken = await KV.get(accessTokenKey);
385 | const shareToken = await getShareToken(fullUserName, finalaccessToken,accountNumber);
386 |
387 | if (shareToken === 'Can not get share token.') {
388 | //await KV.put(accessTokenKey, "Bad_AT");
389 | return generateErrorResponse(`Error fetching share token.`,accountNumber);
390 | }
391 |
392 |
393 | // Log the successful login
394 | await loginlog(userName, accountNumber,'Free');
395 | const oauthLink = await getOAuthLink(shareToken, proxiedDomain);
396 | const headers = new Headers();
397 | headers.append('Location', oauthLink);
398 | headers.append('Set-Cookie', `aian=${accountNumber}; Path=/`);
399 |
400 |
401 | const response = new Response(null, {
402 | status: 302,
403 | headers: headers
404 | });
405 | return response;
406 |
407 | }
408 |
409 |
410 | async function verifyTurnstile(responseToken) {
411 | const removeTurnstile = await KV.get('RemoveTurnstile')||'';
412 | if (removeTurnstile){return 'true'}
413 | const verifyUrl = 'https://challenges.cloudflare.com/turnstile/v0/siteverify';
414 | const secretKey = await KV.get('TurnstileKeys');
415 | const response = await fetch(verifyUrl, {
416 | method: 'POST',
417 | headers: { 'Content-Type': 'application/json' },
418 | body: JSON.stringify({
419 | secret: secretKey,
420 | response: responseToken
421 | })
422 | });
423 | const data = await response.json();
424 | return data.success;
425 | }
426 |
427 |
428 | async function loginlog(userName, accountNumber, antype) {
429 | const currentTime = new Date().toLocaleString("zh-CN", { timeZone: "Asia/Shanghai" });
430 | const timestamp = Date.now();
431 | const logEntry = {
432 | user: userName,
433 | accountNumber: accountNumber,
434 | time: currentTime,
435 | timestamp: timestamp
436 | };
437 | // Retrieve the existing log array or create a new one if it doesn't exist
438 | const lastLoginLogs = await KV.get(`${antype}LoginLogs`);
439 | let logArray = [];
440 | if (lastLoginLogs) {
441 | logArray = JSON.parse(lastLoginLogs);
442 | }
443 | logArray.push(logEntry);
444 | await KV.put(`${antype}LoginLogs`, JSON.stringify(logArray));
445 | }
446 |
447 | async function deletelog(userName, accountNumber,antype) {
448 | const currentTime = new Date().toLocaleString("zh-CN", { timeZone: "Asia/Shanghai" });
449 | const logEntry = {
450 | user: userName,
451 | time: currentTime,
452 | accountNumber: accountNumber
453 | };
454 | // Retrieve the existing log array or create a new one if it doesn't exist
455 | const lastDeleteLogs = await KV.get(`${antype}DeleteLogs`);
456 | let logArray = [];
457 | if (lastDeleteLogs) {
458 | logArray = JSON.parse(lastDeleteLogs);
459 | }
460 | logArray.push(logEntry);
461 | await KV.put(`${antype}DeleteLogs`, JSON.stringify(logArray));
462 | }
463 |
464 | //AN获取和删除
465 | async function getAccountNumber(userName, initialaccountNumber, antype, mode, anissues) {
466 | const currentTime = Date.now()
467 | const Milliseconds = 3 * 60 * 1000;
468 |
469 | const checkAndRemoveIssueAccount = async (accountNumber) => {
470 | // Retrieve the login logs
471 | const lastLoginLogs = await KV.get(`${antype}LoginLogs`);
472 | if (lastLoginLogs) {
473 | const logArray = JSON.parse(lastLoginLogs);
474 | const userLogs = logArray.filter(log => log.user === userName && log.accountNumber === accountNumber);
475 | if (userLogs.length > 0) {
476 | const recentLogins = userLogs.filter(log => {
477 | const logTime = log.timestamp;
478 | return currentTime - logTime <= Milliseconds;
479 | });
480 | if (recentLogins.length >= 1 && anissues) {
481 | // 删除问题账号
482 | const aliveAccount = await KV.get(`${antype}AliveAccounts`);
483 | let aliveAccountList = aliveAccount.split(',');
484 | aliveAccountList = aliveAccountList.filter(acc => acc !== accountNumber.toString());
485 | await KV.put(`${antype}AliveAccounts`, aliveAccountList.join(','));
486 | await deletelog(userName, accountNumber,antype);
487 | return true;
488 | }
489 | }
490 | }
491 | return false;
492 | };
493 |
494 | // 顺序读取
495 | if (mode == 'Order') {
496 | const aliveAccountString = await KV.get(`${antype}AliveAccounts`) || '';
497 | let aliveAccounts = aliveAccountString
498 | .split(',')
499 | .map(num => parseInt(num, 10))
500 | .filter(num => !isNaN(num));
501 |
502 | if (aliveAccounts.length > 0) {
503 | let minAccount = Math.min(...aliveAccounts);
504 | if (await checkAndRemoveIssueAccount(minAccount)) {
505 | aliveAccounts = aliveAccounts.filter(acc => acc !== minAccount);
506 | minAccount = aliveAccounts.length > 0 ? Math.min(...aliveAccounts) : 1;
507 | }
508 | return minAccount;
509 | }
510 | return 1;
511 | }
512 |
513 | // 检测和删除问题账号
514 | if (mode == 'Check') {
515 | await checkAndRemoveIssueAccount(initialaccountNumber);
516 | return initialaccountNumber;
517 | }
518 |
519 | // 随机读取
520 | if (mode == 'Random') {
521 | // Retrieve the last login logs
522 | const lastLoginLogs = await KV.get(`${antype}LoginLogs`);
523 | if (lastLoginLogs) {
524 | const logArray = JSON.parse(lastLoginLogs);
525 | const userLogs = logArray.filter(log => log.user === userName);
526 | const recentLogins = userLogs.filter(log => {
527 | const logTime = log.timestamp;
528 | return currentTime - logTime <= Milliseconds;
529 | });
530 |
531 | if (recentLogins.length > 0) {
532 | const lastAccount = recentLogins[recentLogins.length - 1].accountNumber;
533 | if (await checkAndRemoveIssueAccount(lastAccount)) {
534 | const aliveAccountString = await KV.get(`${antype}AliveAccounts`) || '';
535 | const aliveAccounts = aliveAccountString
536 | .split(',')
537 | .map(num => parseInt(num, 10))
538 | .filter(num => !isNaN(num));
539 |
540 | if (aliveAccounts.length > 0) {
541 | const randomAccount = aliveAccounts[Math.floor(Math.random() * aliveAccounts.length)];
542 | return randomAccount;
543 | }
544 | return 0;
545 | }
546 | return lastAccount;
547 | }
548 | }
549 |
550 |
551 | const aliveAccountString = await KV.get(`${antype}AliveAccounts`) || '';
552 | let aliveAccounts = aliveAccountString
553 | .split(',')
554 | .map(num => parseInt(num, 10))
555 | .filter(num => !isNaN(num));
556 |
557 | if (aliveAccounts.length > 0) {
558 | let randomAccount = aliveAccounts[Math.floor(Math.random() * aliveAccounts.length)];
559 | if (await checkAndRemoveIssueAccount(randomAccount)) {
560 | aliveAccounts = aliveAccounts.filter(acc => acc !== randomAccount);
561 | if (aliveAccounts.length > 0) {
562 | randomAccount = aliveAccounts[Math.floor(Math.random() * aliveAccounts.length)];
563 | return randomAccount;
564 | }
565 | return 0;
566 | }
567 | return randomAccount;
568 | }
569 | return 0;
570 | }
571 |
572 | return initialaccountNumber;
573 | }
574 |
575 |
576 |
577 | async function generateErrorResponse(message,accountNumber) {
578 |
579 | const errorHtml = `
580 |
Don't have an account? Sign Up
1318 |