├── 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 | image 22 | image 23 | image 24 | image 25 | 26 | 27 | # Worker 部署(一键直达) 28 | [![Deploy to Cloudflare Workers](https://deploy.workers.cloudflare.com/button)](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 | image 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 | - image 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 |
581 |
582 | 583 | ${message} 584 |
585 |
586 | `; 587 | const html = await getHTMLLoginPage(accountNumber); 588 | const responseHtml = html.replace( 589 | '', 590 | errorHtml + '' 591 | ); 592 | return new Response(responseHtml, { headers: { 'Content-Type': 'text/html' } }); 593 | } 594 | 595 | 596 | // 将密钥转换为 CryptoKey 对象 597 | async function getCryptoKey(secret) { 598 | return crypto.subtle.importKey( 599 | 'raw', 600 | new TextEncoder().encode(secret), 601 | 'AES-GCM', 602 | false, 603 | ['encrypt', 'decrypt'] 604 | ); 605 | } 606 | 607 | // 加密函数 608 | async function encrypt(text, secret) { 609 | const iv = crypto.getRandomValues(new Uint8Array(12)); // 初始化向量 610 | const key = await getCryptoKey(secret); 611 | const encrypted = await crypto.subtle.encrypt( 612 | { 613 | name: 'AES-GCM', 614 | iv: iv 615 | }, 616 | key, 617 | new TextEncoder().encode(text) 618 | ); 619 | 620 | // 返回 base64 编码的 iv 和加密结果 621 | return btoa(String.fromCharCode(...iv)) + ':' + btoa(String.fromCharCode(...new Uint8Array(encrypted))); 622 | } 623 | 624 | // 解密函数 625 | async function decrypt(encryptedText, secret) { 626 | const [iv, data] = encryptedText.split(':').map(part => Uint8Array.from(atob(part), c => c.charCodeAt(0))); 627 | const key = await getCryptoKey(secret); 628 | const decrypted = await crypto.subtle.decrypt( 629 | { 630 | name: 'AES-GCM', 631 | iv: iv 632 | }, 633 | key, 634 | data 635 | ); 636 | 637 | return new TextDecoder().decode(decrypted); 638 | } 639 | 640 | async function getGPTStatus(){ 641 | const url = 'https://status.openai.com/api/v2/summary.json'; 642 | 643 | 644 | // 发送 POST 请求 645 | const response = await fetch(url, { 646 | method: 'GET', 647 | headers: { 648 | 'Content-Type': 'application/x-www-form-urlencoded;charset=UTF-8' 649 | }, 650 | }); 651 | // 检查响应状态 652 | if (response.ok) { 653 | const data = await response.json(); 654 | const status = data.components.find((component) => component.name === 'ChatGPT'); 655 | //return JSON.stringify(status); 656 | return status.status; 657 | } 658 | else {return 'operational';} 659 | } 660 | 661 | async function getHTMLSelectionPage() { 662 | const aliveAccounts = await KV.get('FreeAliveAccounts') || ''; 663 | const aliveAccountsArray = aliveAccounts.split(',').map(num => num.trim()).filter(num => num !== ''); 664 | const websiteName = await KV.get('FreeWebName') || await KV.get('WebName') || 'Haibara AI'; 665 | const introContent = await KV.get('FreeWebIntro') || ''; 666 | 667 | const options = await Promise.all(aliveAccountsArray.map(async (accountNumber) => { 668 | const encryptedAN = await encrypt(accountNumber, 'eGJjZWFsZGVuYmVqZGZzY3VhbmZpZGZz'); // 必须是32位 669 | return ` 670 |
671 | 672 | 673 |
674 | `; 675 | })); 676 | 677 | return ` 678 | 679 | 680 | 681 | 682 | 683 | ${websiteName} 684 | 685 | 686 | 687 | 854 | 861 | 869 | 870 | 871 | 872 |
873 |

${websiteName}

874 |
875 |
876 | ${introContent ? ` 877 |
878 | ${introContent} 879 |
` : ''} 880 | 881 |
882 | 883 | 884 | 887 |
888 |
889 | 890 | 893 | 894 | 895 | `; 896 | } 897 | 898 | 899 | async function getHTMLLoginPage(accountNumber) { 900 | const logourl = await KV.get('LogoURL') || logo; 901 | const WorkerURL=await KV.get('WorkerURL'); 902 | const turnstileSiteKey=await KV.get('TurnstileSiteKey'); 903 | const websiteName = await KV.get('FreeWebName') || await KV.get('WebName') || 'Haibara AI'; 904 | const removeTurnstile = await KV.get('RemoveTurnstile')||''; 905 | return ` 906 | 907 | 908 | 909 | 910 | 911 | 912 | 913 | 914 | Free - ${websiteName} 915 | 1277 | 1278 | 1279 | 1280 |
1281 |
1282 |
1283 | 1284 | Logo 1285 | 1286 |
1287 |
1288 |
1289 |

${websiteName}

1290 | 1319 |
1320 |
1321 |
1322 |
1323 | 1358 | 1359 | 1360 | 1361 | `; 1362 | } 1363 | 1364 | --------------------------------------------------------------------------------