feat: phase 4 - 积分竞猜和游戏黑名单 v0.3.0

竞猜功能:发起竞猜、下注、关闭、开奖、奖池分配
黑名单功能:标记游戏、按原因/严重程度筛选、详情展开
修复:双重结算、TOCTOU竞态、订阅泄漏、选项选择兼容性

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
congsh
2026-04-19 00:21:43 +08:00
parent 2d56df940d
commit 60ad9a04cd
24 changed files with 4047 additions and 14 deletions
@@ -0,0 +1,114 @@
/// <reference path="../pb_data/types.d.ts" />
migrate((db) => {
const collection = new Collection({
"id": "gblacklist_col",
"created": "2026-04-18 21:00:01.000Z",
"updated": "2026-04-18 21:00:01.000Z",
"name": "game_blacklist",
"type": "base",
"system": false,
"schema": [
{
"system": false,
"id": "gb_group",
"name": "group",
"type": "relation",
"required": true,
"options": {
"collectionId": "es63bkyiblpnxdf",
"cascadeDelete": true,
"minSelect": null,
"maxSelect": 1,
"displayFields": null
}
},
{
"system": false,
"id": "gb_reporter",
"name": "reporter",
"type": "relation",
"required": true,
"options": {
"collectionId": "_pb_users_auth_",
"cascadeDelete": true,
"minSelect": null,
"maxSelect": 1,
"displayFields": null
}
},
{
"system": false,
"id": "gb_game",
"name": "game",
"type": "relation",
"required": false,
"options": {
"collectionId": "x5adjlc0txf16r8",
"cascadeDelete": false,
"minSelect": null,
"maxSelect": 1,
"displayFields": null
}
},
{
"system": false,
"id": "gb_gamename",
"name": "gameName",
"type": "text",
"required": true,
"options": {
"min": 1,
"max": 200,
"pattern": ""
}
},
{
"system": false,
"id": "gb_reason",
"name": "reason",
"type": "select",
"required": true,
"options": {
"maxSelect": 1,
"values": ["behavior", "cheating", "abandonment", "toxic", "other"]
}
},
{
"system": false,
"id": "gb_desc",
"name": "description",
"type": "text",
"required": true,
"options": {
"min": 1,
"max": 500,
"pattern": ""
}
},
{
"system": false,
"id": "gb_severity",
"name": "severity",
"type": "select",
"required": true,
"options": {
"maxSelect": 1,
"values": ["mild", "medium", "severe"]
}
}
],
"indexes": [],
"listRule": "@request.auth.id != \"\" && group.members ~ @request.auth.id",
"viewRule": "@request.auth.id != \"\" && group.members ~ @request.auth.id",
"createRule": "@request.auth.id != \"\" && group.members ~ @request.auth.id",
"updateRule": null,
"deleteRule": "reporter = @request.auth.id || group.owner = @request.auth.id",
"options": {}
});
return Dao(db).saveCollection(collection);
}, (db) => {
const dao = new Dao(db);
const collection = dao.findCollectionByNameOrId("gblacklist_col");
return dao.deleteCollection(collection);
})
@@ -0,0 +1,149 @@
/// <reference path="../pb_data/types.d.ts" />
migrate((db) => {
const collection = new Collection({
"id": "bets_col",
"created": "2026-04-18 21:00:02.000Z",
"updated": "2026-04-18 21:00:02.000Z",
"name": "bets",
"type": "base",
"system": false,
"schema": [
{
"system": false,
"id": "bt_group",
"name": "group",
"type": "relation",
"required": true,
"options": {
"collectionId": "es63bkyiblpnxdf",
"cascadeDelete": true,
"minSelect": null,
"maxSelect": 1,
"displayFields": null
}
},
{
"system": false,
"id": "bt_creator",
"name": "creator",
"type": "relation",
"required": true,
"options": {
"collectionId": "_pb_users_auth_",
"cascadeDelete": true,
"minSelect": null,
"maxSelect": 1,
"displayFields": null
}
},
{
"system": false,
"id": "bt_title",
"name": "title",
"type": "text",
"required": true,
"options": {
"min": 1,
"max": 200,
"pattern": ""
}
},
{
"system": false,
"id": "bt_desc",
"name": "description",
"type": "text",
"required": false,
"options": {
"min": null,
"max": 1000,
"pattern": ""
}
},
{
"system": false,
"id": "bt_minstake",
"name": "minStake",
"type": "number",
"required": true,
"options": {
"min": 1,
"max": null,
"noDecimal": true
}
},
{
"system": false,
"id": "bt_maxstake",
"name": "maxStake",
"type": "number",
"required": true,
"options": {
"min": 1,
"max": null,
"noDecimal": true
}
},
{
"system": false,
"id": "bt_status",
"name": "status",
"type": "select",
"required": true,
"options": {
"maxSelect": 1,
"values": ["open", "closed", "settled"]
}
},
{
"system": false,
"id": "bt_result",
"name": "resultOption",
"type": "relation",
"required": false,
"options": {
"collectionId": "betopts_col",
"cascadeDelete": false,
"minSelect": null,
"maxSelect": 1,
"displayFields": null
}
},
{
"system": false,
"id": "bt_deadline",
"name": "deadline",
"type": "date",
"required": true,
"options": {
"min": "",
"max": ""
}
},
{
"system": false,
"id": "bt_settledat",
"name": "settledAt",
"type": "date",
"required": false,
"options": {
"min": "",
"max": ""
}
}
],
"indexes": [],
"listRule": "@request.auth.id != \"\" && group.members ~ @request.auth.id",
"viewRule": "@request.auth.id != \"\" && group.members ~ @request.auth.id",
"createRule": "@request.auth.id != \"\" && group.members ~ @request.auth.id",
"updateRule": "creator = @request.auth.id",
"deleteRule": "creator = @request.auth.id || group.owner = @request.auth.id",
"options": {}
});
return Dao(db).saveCollection(collection);
}, (db) => {
const dao = new Dao(db);
const collection = dao.findCollectionByNameOrId("bets_col");
return dao.deleteCollection(collection);
})
@@ -0,0 +1,64 @@
/// <reference path="../pb_data/types.d.ts" />
migrate((db) => {
const collection = new Collection({
"id": "betopts_col",
"created": "2026-04-18 21:00:03.000Z",
"updated": "2026-04-18 21:00:03.000Z",
"name": "bet_options",
"type": "base",
"system": false,
"schema": [
{
"system": false,
"id": "bo_bet",
"name": "bet",
"type": "relation",
"required": true,
"options": {
"collectionId": "bets_col",
"cascadeDelete": true,
"minSelect": null,
"maxSelect": 1,
"displayFields": null
}
},
{
"system": false,
"id": "bo_content",
"name": "content",
"type": "text",
"required": true,
"options": {
"min": 1,
"max": 200,
"pattern": ""
}
},
{
"system": false,
"id": "bo_order",
"name": "order",
"type": "number",
"required": true,
"options": {
"min": 1,
"max": null,
"noDecimal": true
}
}
],
"indexes": [],
"listRule": "@request.auth.id != \"\" && bet.group.members ~ @request.auth.id",
"viewRule": "@request.auth.id != \"\" && bet.group.members ~ @request.auth.id",
"createRule": "@request.auth.id != \"\" && bet.group.members ~ @request.auth.id",
"updateRule": "bet.creator = @request.auth.id",
"deleteRule": "bet.creator = @request.auth.id",
"options": {}
});
return Dao(db).saveCollection(collection);
}, (db) => {
const dao = new Dao(db);
const collection = dao.findCollectionByNameOrId("betopts_col");
return dao.deleteCollection(collection);
})
@@ -0,0 +1,88 @@
/// <reference path="../pb_data/types.d.ts" />
migrate((db) => {
const collection = new Collection({
"id": "betentries_col",
"created": "2026-04-18 21:00:04.000Z",
"updated": "2026-04-18 21:00:04.000Z",
"name": "bet_entries",
"type": "base",
"system": false,
"schema": [
{
"system": false,
"id": "be_bet",
"name": "bet",
"type": "relation",
"required": true,
"options": {
"collectionId": "bets_col",
"cascadeDelete": true,
"minSelect": null,
"maxSelect": 1,
"displayFields": null
}
},
{
"system": false,
"id": "be_user",
"name": "user",
"type": "relation",
"required": true,
"options": {
"collectionId": "_pb_users_auth_",
"cascadeDelete": false,
"minSelect": null,
"maxSelect": 1,
"displayFields": null
}
},
{
"system": false,
"id": "be_option",
"name": "option",
"type": "relation",
"required": true,
"options": {
"collectionId": "betopts_col",
"cascadeDelete": false,
"minSelect": null,
"maxSelect": 1,
"displayFields": null
}
},
{
"system": false,
"id": "be_stake",
"name": "stake",
"type": "number",
"required": true,
"options": {
"min": 1,
"max": null,
"noDecimal": true
}
},
{
"system": false,
"id": "be_won",
"name": "won",
"type": "bool",
"required": false,
"options": {}
}
],
"indexes": [],
"listRule": "@request.auth.id != \"\" && bet.group.members ~ @request.auth.id",
"viewRule": "@request.auth.id != \"\" && bet.group.members ~ @request.auth.id",
"createRule": "@request.auth.id != \"\" && user = @request.auth.id && bet.group.members ~ @request.auth.id",
"updateRule": "bet.creator = @request.auth.id",
"deleteRule": "user = @request.auth.id && bet.status = \"open\"",
"options": {}
});
return Dao(db).saveCollection(collection);
}, (db) => {
const dao = new Dao(db);
const collection = dao.findCollectionByNameOrId("betentries_col");
return dao.deleteCollection(collection);
})
@@ -0,0 +1,116 @@
/// <reference path="../pb_data/types.d.ts" />
migrate((db) => {
const dao = new Dao(db)
const collection = dao.findCollectionByNameOrId("h5adxdw9gm0aw8s")
// update
collection.schema.addField(new SchemaField({
"system": false,
"id": "sd27cbh8",
"name": "action",
"type": "select",
"required": true,
"presentable": false,
"unique": false,
"options": {
"maxSelect": 1,
"values": [
"vote",
"team",
"memory",
"bet",
"settle"
]
}
}))
// update
collection.schema.addField(new SchemaField({
"system": false,
"id": "iszqa13h",
"name": "points",
"type": "number",
"required": true,
"presentable": false,
"unique": false,
"options": {
"min": null,
"max": null,
"noDecimal": false
}
}))
// update
collection.schema.addField(new SchemaField({
"system": false,
"id": "pnipfzbd",
"name": "relatedId",
"type": "text",
"required": false,
"presentable": false,
"unique": false,
"options": {
"min": null,
"max": null,
"pattern": ""
}
}))
return dao.saveCollection(collection)
}, (db) => {
const dao = new Dao(db)
const collection = dao.findCollectionByNameOrId("h5adxdw9gm0aw8s")
// update
collection.schema.addField(new SchemaField({
"system": false,
"id": "sd27cbh8",
"name": "action",
"type": "select",
"required": true,
"presentable": false,
"unique": false,
"options": {
"maxSelect": 1,
"values": [
"vote",
"team",
"memory"
]
}
}))
// update
collection.schema.addField(new SchemaField({
"system": false,
"id": "iszqa13h",
"name": "points",
"type": "number",
"required": true,
"presentable": false,
"unique": false,
"options": {
"min": 1,
"max": null,
"noDecimal": false
}
}))
// update
collection.schema.addField(new SchemaField({
"system": false,
"id": "pnipfzbd",
"name": "relatedId",
"type": "text",
"required": true,
"presentable": false,
"unique": false,
"options": {
"min": null,
"max": null,
"pattern": ""
}
}))
return dao.saveCollection(collection)
})