world_cup_activity.sql 44 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820
  1. -- World Cup activity tables for SQL Server.
  2. -- Amount fields use integer cents (NumConfig::NUM_VALUE = 100).
  3. IF OBJECT_ID('agent.dbo.world_cup_user_state', 'U') IS NULL
  4. BEGIN
  5. -- 字段说明:
  6. -- user_id: QPAccountsDB.dbo.AccountsInfo.UserID,活动用户主键。
  7. -- first_bet_used: 是否已使用世界杯首注奖励,1=已使用,0=未使用。
  8. -- invite_code: 用户自己的邀请码,前端展示并用于手动补绑。
  9. -- referred_by_user_id: 邀请人的 UserID,未绑定时为空。
  10. -- referral_bind_at: 邀请关系绑定时间。
  11. -- referral_bind_type: 绑定方式,link=分享链接自动绑定,manual=24h 内手动输入邀请码。
  12. -- device_fp: 设备指纹,用于风控识别同设备自邀。
  13. -- pay_account_hash: 支付账号哈希,用于风控识别同支付账号。
  14. -- signup_ip: 注册 IP,用于风控识别同 IP/同段聚集。
  15. -- created_at: 记录创建时间。
  16. -- updated_at: 记录更新时间。
  17. CREATE TABLE agent.dbo.world_cup_user_state (
  18. user_id INT NOT NULL PRIMARY KEY,
  19. first_bet_used BIT NOT NULL DEFAULT 0,
  20. invite_code VARCHAR(32) NULL,
  21. referred_by_user_id INT NULL,
  22. referral_bind_at DATETIME NULL,
  23. referral_bind_type VARCHAR(16) NULL,
  24. device_fp VARCHAR(128) NULL,
  25. pay_account_hash VARCHAR(128) NULL,
  26. signup_ip VARCHAR(64) NULL,
  27. created_at DATETIME NOT NULL DEFAULT GETDATE(),
  28. updated_at DATETIME NOT NULL DEFAULT GETDATE()
  29. );
  30. CREATE UNIQUE INDEX idx_world_cup_user_state_invite_code
  31. ON agent.dbo.world_cup_user_state(invite_code)
  32. WHERE invite_code IS NOT NULL;
  33. END;
  34. IF OBJECT_ID('agent.dbo.world_cup_matches', 'U') IS NULL
  35. BEGIN
  36. -- 字段说明:
  37. -- match_id: 比赛 ID,单场 1X2 盘口主键。
  38. -- match_no: FIFA 官方赛程 Match 编号。
  39. -- competition: 赛事名称,例如 World Cup。
  40. -- stage: 赛事阶段,group=小组赛,round_32=32 强,round_16=16 强,quarter_final=1/4 决赛,semi_final=半决赛,final=决赛。
  41. -- group_name: 小组名称,小组赛为 A-L,淘汰赛可为空。
  42. -- home_team: 主队名称。
  43. -- away_team: 客队名称。
  44. -- venue: 比赛场馆名称。
  45. -- kickoff_at: 开赛时间,UTC 时间戳;开赛前 1 小时停止投注。
  46. -- status: 比赛状态,scheduled=可维护/待开赛,closed=关闭投注,finished=已完赛。
  47. -- result: 比赛赛果,home/draw/away;未结算时为空。
  48. -- created_at: 记录创建时间。
  49. -- updated_at: 记录更新时间。
  50. CREATE TABLE agent.dbo.world_cup_matches (
  51. match_id INT IDENTITY(1,1) NOT NULL PRIMARY KEY,
  52. match_no INT NULL,
  53. competition VARCHAR(64) NOT NULL,
  54. stage VARCHAR(32) NOT NULL DEFAULT 'group',
  55. group_name VARCHAR(8) NULL,
  56. home_team VARCHAR(64) NOT NULL,
  57. away_team VARCHAR(64) NOT NULL,
  58. venue VARCHAR(128) NULL,
  59. kickoff_at DATETIME NOT NULL,
  60. status VARCHAR(16) NOT NULL DEFAULT 'scheduled',
  61. result VARCHAR(16) NULL,
  62. created_at DATETIME NOT NULL DEFAULT GETDATE(),
  63. updated_at DATETIME NOT NULL DEFAULT GETDATE()
  64. );
  65. CREATE INDEX idx_world_cup_matches_open
  66. ON agent.dbo.world_cup_matches(status, kickoff_at);
  67. END;
  68. IF OBJECT_ID('agent.dbo.world_cup_matches', 'U') IS NOT NULL
  69. BEGIN
  70. IF COL_LENGTH('agent.dbo.world_cup_matches', 'match_no') IS NULL
  71. ALTER TABLE agent.dbo.world_cup_matches ADD match_no INT NULL;
  72. IF COL_LENGTH('agent.dbo.world_cup_matches', 'stage') IS NULL
  73. ALTER TABLE agent.dbo.world_cup_matches ADD stage VARCHAR(32) NOT NULL DEFAULT 'group';
  74. IF COL_LENGTH('agent.dbo.world_cup_matches', 'group_name') IS NULL
  75. ALTER TABLE agent.dbo.world_cup_matches ADD group_name VARCHAR(8) NULL;
  76. IF COL_LENGTH('agent.dbo.world_cup_matches', 'venue') IS NULL
  77. ALTER TABLE agent.dbo.world_cup_matches ADD venue VARCHAR(128) NULL;
  78. END;
  79. GO
  80. IF OBJECT_ID('agent.dbo.world_cup_odds', 'U') IS NULL
  81. BEGIN
  82. -- 字段说明:
  83. -- odds_id: 赔率记录 ID。
  84. -- match_id: 比赛 ID;Winner 夺冠盘为空,1X2 单场盘必填。
  85. -- market: 盘口类型,winner=夺冠盘,1x2=单场胜平负。
  86. -- selection: 投注选项;Winner 为球队名,1X2 为 home/draw/away。
  87. -- decimal_odds: 当前十进制赔率。
  88. -- previous_odds: 上一次赔率,用于前端展示涨跌箭头。
  89. -- is_active: 是否展示/可投注,1=启用,0=隐藏。
  90. -- locked_weight: 后台排序权重,数值越大越靠前。
  91. -- updated_at: 赔率更新时间。
  92. CREATE TABLE agent.dbo.world_cup_odds (
  93. odds_id INT IDENTITY(1,1) NOT NULL PRIMARY KEY,
  94. match_id INT NULL,
  95. market VARCHAR(16) NOT NULL,
  96. selection VARCHAR(64) NOT NULL,
  97. decimal_odds DECIMAL(10, 2) NOT NULL,
  98. previous_odds DECIMAL(10, 2) NULL,
  99. is_active BIT NOT NULL DEFAULT 1,
  100. locked_weight INT NOT NULL DEFAULT 0,
  101. updated_at DATETIME NOT NULL DEFAULT GETDATE()
  102. );
  103. CREATE INDEX idx_world_cup_odds_market
  104. ON agent.dbo.world_cup_odds(market, match_id, is_active);
  105. END;
  106. IF OBJECT_ID('agent.dbo.world_cup_match_favorites', 'U') IS NULL
  107. BEGIN
  108. -- 字段说明:
  109. -- id: 收藏记录 ID。
  110. -- user_id: 收藏用户 UserID。
  111. -- match_id: 被收藏的单场比赛 ID,仅支持 1X2 比赛,不支持 Winner 盘。
  112. -- created_at: 收藏创建时间。
  113. CREATE TABLE agent.dbo.world_cup_match_favorites (
  114. id INT IDENTITY(1,1) NOT NULL PRIMARY KEY,
  115. user_id INT NOT NULL,
  116. match_id INT NOT NULL,
  117. created_at DATETIME NOT NULL DEFAULT GETDATE()
  118. );
  119. CREATE UNIQUE INDEX idx_world_cup_match_favorites_user_match
  120. ON agent.dbo.world_cup_match_favorites(user_id, match_id);
  121. END;
  122. IF OBJECT_ID('agent.dbo.world_cup_bets', 'U') IS NULL
  123. BEGIN
  124. -- 字段说明:
  125. -- bet_id: 世界杯注单 ID。
  126. -- user_id: 下注用户 UserID。
  127. -- game_id: 前端展示的 8 位 GameID,冗余保存便于查询。
  128. -- idempotency_key: 客户端幂等键,同一用户同一 key 不重复下单。
  129. -- market: 盘口类型,winner=夺冠盘,1x2=单场胜平负。
  130. -- match_id: 比赛 ID;Winner 夺冠盘为空。
  131. -- selection: 投注选项,球队名或 home/draw/away。
  132. -- stake: 下注额,整数分。
  133. -- odds: 下单时锁定的十进制赔率。
  134. -- is_first_bet: 是否为用户世界杯首注,1=首注。
  135. -- bonus_amount: 首注 +50% 额外奖励金额,整数分。
  136. -- potential_payout: 潜在派彩金额,整数分,stake * odds + bonus_amount。
  137. -- status: 注单状态,pending=待结算,won=赢,lost=输。
  138. -- settled_at: 结算时间。
  139. -- created_at: 下单时间。
  140. -- updated_at: 注单更新时间。
  141. CREATE TABLE agent.dbo.world_cup_bets (
  142. bet_id BIGINT IDENTITY(1,1) NOT NULL PRIMARY KEY,
  143. user_id INT NOT NULL,
  144. game_id INT NOT NULL,
  145. idempotency_key VARCHAR(64) NOT NULL,
  146. market VARCHAR(16) NOT NULL,
  147. match_id INT NULL,
  148. selection VARCHAR(64) NOT NULL,
  149. stake INT NOT NULL,
  150. odds DECIMAL(10, 2) NOT NULL,
  151. is_first_bet BIT NOT NULL DEFAULT 0,
  152. bonus_amount INT NOT NULL DEFAULT 0,
  153. potential_payout INT NOT NULL DEFAULT 0,
  154. status VARCHAR(16) NOT NULL DEFAULT 'pending',
  155. settled_at DATETIME NULL,
  156. created_at DATETIME NOT NULL DEFAULT GETDATE(),
  157. updated_at DATETIME NOT NULL DEFAULT GETDATE()
  158. );
  159. CREATE UNIQUE INDEX idx_world_cup_bets_user_idempotency
  160. ON agent.dbo.world_cup_bets(user_id, idempotency_key);
  161. CREATE INDEX idx_world_cup_bets_match_status
  162. ON agent.dbo.world_cup_bets(match_id, status);
  163. END;
  164. IF OBJECT_ID('agent.dbo.world_cup_referrals', 'U') IS NULL
  165. BEGIN
  166. -- 字段说明:
  167. -- id: 邀请绑定记录 ID。
  168. -- referrer_id: 邀请人 UserID。
  169. -- invitee_id: 被邀请人 UserID;唯一,一个被邀请人只能绑定一次。
  170. -- bind_type: 绑定方式,link=分享链接自动绑定,manual=手动补填邀请码。
  171. -- bind_at: 绑定发生时间。
  172. -- created_at: 记录创建时间。
  173. CREATE TABLE agent.dbo.world_cup_referrals (
  174. id BIGINT IDENTITY(1,1) NOT NULL PRIMARY KEY,
  175. referrer_id INT NOT NULL,
  176. invitee_id INT NOT NULL,
  177. bind_type VARCHAR(16) NOT NULL,
  178. bind_at DATETIME NOT NULL DEFAULT GETDATE(),
  179. created_at DATETIME NOT NULL DEFAULT GETDATE()
  180. );
  181. CREATE UNIQUE INDEX idx_world_cup_referrals_invitee
  182. ON agent.dbo.world_cup_referrals(invitee_id);
  183. END;
  184. IF OBJECT_ID('agent.dbo.world_cup_referral_rewards', 'U') IS NULL
  185. BEGIN
  186. -- 字段说明:
  187. -- reward_id: 邀请奖励单 ID,后台审核对象。
  188. -- referrer_id: 邀请人 UserID。
  189. -- invitee_id: 被邀请人 UserID;唯一,同一被邀请人首充只生成一次奖励单。
  190. -- first_deposit_order_sn: 触发奖励的首充订单号,用于事件监听幂等。
  191. -- first_deposit_amt: 触发奖励时的首充实际入账金额,整数分。
  192. -- reward_each: 双方各得奖励金额,整数分;规则为 min(first_deposit_amt * 50%, 10000)。
  193. -- total_liability: 本奖励单总赔付,整数分,reward_each * 2,最高 20000。
  194. -- risk_score: 风控评分,0-100。
  195. -- risk_level: 风险等级,low/medium/high。
  196. -- signals: 命中的风控信号 JSON 文本。
  197. -- status: 审核状态,reviewing/approved/rejected/on_hold/clawed_back。
  198. -- reason_code: 驳回或追回原因代码。
  199. -- review_by: 审核员标识。
  200. -- reviewed_at: 审核时间。
  201. -- submitted_at: 奖励单提交审核时间,用于计算 24h SLA。
  202. -- created_at: 记录创建时间。
  203. -- updated_at: 记录更新时间。
  204. CREATE TABLE agent.dbo.world_cup_referral_rewards (
  205. reward_id BIGINT IDENTITY(1,1) NOT NULL PRIMARY KEY,
  206. referrer_id INT NOT NULL,
  207. invitee_id INT NOT NULL,
  208. first_deposit_order_sn VARCHAR(64) NULL,
  209. first_deposit_amt INT NOT NULL,
  210. reward_each INT NOT NULL,
  211. total_liability INT NOT NULL,
  212. risk_score INT NOT NULL DEFAULT 0,
  213. risk_level VARCHAR(16) NOT NULL DEFAULT 'low',
  214. signals NVARCHAR(MAX) NULL,
  215. status VARCHAR(16) NOT NULL DEFAULT 'reviewing',
  216. reason_code VARCHAR(32) NULL,
  217. review_by VARCHAR(64) NULL,
  218. reviewed_at DATETIME NULL,
  219. submitted_at DATETIME NOT NULL DEFAULT GETDATE(),
  220. created_at DATETIME NOT NULL DEFAULT GETDATE(),
  221. updated_at DATETIME NOT NULL DEFAULT GETDATE()
  222. );
  223. CREATE UNIQUE INDEX idx_world_cup_referral_rewards_invitee
  224. ON agent.dbo.world_cup_referral_rewards(invitee_id);
  225. CREATE INDEX idx_world_cup_referral_rewards_queue
  226. ON agent.dbo.world_cup_referral_rewards(status, risk_level, submitted_at);
  227. END;
  228. IF OBJECT_ID('agent.dbo.world_cup_referral_rewards', 'U') IS NOT NULL
  229. BEGIN
  230. IF COL_LENGTH('agent.dbo.world_cup_referral_rewards', 'first_deposit_order_sn') IS NULL
  231. ALTER TABLE agent.dbo.world_cup_referral_rewards ADD first_deposit_order_sn VARCHAR(64) NULL;
  232. END;
  233. GO
  234. IF OBJECT_ID('agent.dbo.world_cup_referral_rewards', 'U') IS NOT NULL
  235. AND NOT EXISTS (
  236. SELECT 1
  237. FROM sys.indexes
  238. WHERE name = 'idx_world_cup_referral_rewards_order_sn'
  239. AND object_id = OBJECT_ID('agent.dbo.world_cup_referral_rewards')
  240. )
  241. BEGIN
  242. CREATE UNIQUE INDEX idx_world_cup_referral_rewards_order_sn
  243. ON agent.dbo.world_cup_referral_rewards(first_deposit_order_sn)
  244. WHERE first_deposit_order_sn IS NOT NULL;
  245. END;
  246. IF OBJECT_ID('agent.dbo.world_cup_odds', 'U') IS NOT NULL
  247. AND NOT EXISTS (
  248. SELECT 1
  249. FROM sys.indexes
  250. WHERE name = 'idx_world_cup_odds_unique_market'
  251. AND object_id = OBJECT_ID('agent.dbo.world_cup_odds')
  252. )
  253. BEGIN
  254. CREATE UNIQUE INDEX idx_world_cup_odds_unique_market
  255. ON agent.dbo.world_cup_odds(market, match_id, selection);
  256. END;
  257. IF OBJECT_ID('agent.dbo.world_cup_risk_signals', 'U') IS NULL
  258. BEGIN
  259. -- 字段说明:
  260. -- code: 风控信号代码。
  261. -- label: 后台展示名称。
  262. -- weight: 风控权重,命中后累加到 risk_score。
  263. -- is_active: 是否启用,1=启用,0=停用。
  264. -- updated_at: 配置更新时间。
  265. CREATE TABLE agent.dbo.world_cup_risk_signals (
  266. code VARCHAR(32) NOT NULL PRIMARY KEY,
  267. label VARCHAR(128) NOT NULL,
  268. weight INT NOT NULL,
  269. is_active BIT NOT NULL DEFAULT 1,
  270. updated_at DATETIME NOT NULL DEFAULT GETDATE()
  271. );
  272. INSERT INTO agent.dbo.world_cup_risk_signals (code, label, weight)
  273. VALUES
  274. ('same_device', 'Same device', 40),
  275. ('same_payment', 'Same payment account', 40),
  276. ('same_ip', 'Same IP/subnet', 20),
  277. ('reg_velocity', 'Registration velocity', 25),
  278. ('fast_deposit', 'Fast first deposit', 15),
  279. ('deposit_reversed', 'Deposit reversed', 50),
  280. ('multi_account_ring', 'Multi-account ring', 40),
  281. ('threshold_clustering', 'Threshold clustering', 20);
  282. END;
  283. IF OBJECT_ID('agent.dbo.world_cup_audit_log', 'U') IS NULL
  284. BEGIN
  285. -- 字段说明:
  286. -- id: 审计日志 ID。
  287. -- reward_id: 关联的邀请奖励单 ID,可为空。
  288. -- actor: 操作人,后台审核员或系统任务。
  289. -- action: 操作动作,例如 approve/reject/hold/clawback/auto_score。
  290. -- reason_code: 操作原因代码。
  291. -- before_status: 操作前状态。
  292. -- after_status: 操作后状态。
  293. -- payload: 操作上下文 JSON 文本。
  294. -- created_at: 日志创建时间。
  295. CREATE TABLE agent.dbo.world_cup_audit_log (
  296. id BIGINT IDENTITY(1,1) NOT NULL PRIMARY KEY,
  297. reward_id BIGINT NULL,
  298. actor VARCHAR(64) NOT NULL,
  299. action VARCHAR(32) NOT NULL,
  300. reason_code VARCHAR(32) NULL,
  301. before_status VARCHAR(16) NULL,
  302. after_status VARCHAR(16) NULL,
  303. payload NVARCHAR(MAX) NULL,
  304. created_at DATETIME NOT NULL DEFAULT GETDATE()
  305. );
  306. CREATE INDEX idx_world_cup_audit_log_reward
  307. ON agent.dbo.world_cup_audit_log(reward_id, created_at);
  308. END;
  309. IF OBJECT_ID('agent.dbo.world_cup_matches', 'U') IS NOT NULL
  310. BEGIN
  311. SET IDENTITY_INSERT agent.dbo.world_cup_matches ON;
  312. MERGE agent.dbo.world_cup_matches AS target
  313. USING (VALUES
  314. (1, 1, 'World Cup', 'group', 'A', 'Mexico', 'South Africa', 'Mexico City', '2026-06-11 19:00:00', 'scheduled'),
  315. (2, 2, 'World Cup', 'group', 'A', 'South Korea', 'Czechia', 'Guadalajara', '2026-06-12 02:00:00', 'scheduled'),
  316. (3, 3, 'World Cup', 'group', 'B', 'Canada', 'Bosnia & Herzegovina', 'Toronto', '2026-06-12 19:00:00', 'scheduled'),
  317. (4, 4, 'World Cup', 'group', 'B', 'Qatar', 'Switzerland', 'Santa Clara', '2026-06-13 19:00:00', 'scheduled'),
  318. (5, 5, 'World Cup', 'group', 'C', 'Brazil', 'Morocco', 'East Rutherford', '2026-06-13 22:00:00', 'scheduled'),
  319. (6, 6, 'World Cup', 'group', 'C', 'Haiti', 'Scotland', 'Foxborough', '2026-06-14 01:00:00', 'scheduled'),
  320. (7, 7, 'World Cup', 'group', 'D', 'United States', 'Paraguay', 'Inglewood', '2026-06-13 01:00:00', 'scheduled'),
  321. (8, 8, 'World Cup', 'group', 'D', 'Australia', 'Turkiye', 'Vancouver', '2026-06-14 16:00:00', 'scheduled'),
  322. (9, 9, 'World Cup', 'group', 'E', 'Germany', 'Curacao', 'Houston', '2026-06-14 17:00:00', 'scheduled'),
  323. (10, 10, 'World Cup', 'group', 'E', 'Ivory Coast', 'Ecuador', 'Philadelphia', '2026-06-14 23:00:00', 'scheduled'),
  324. (11, 11, 'World Cup', 'group', 'F', 'Netherlands', 'Japan', 'Arlington', '2026-06-14 20:00:00', 'scheduled'),
  325. (12, 12, 'World Cup', 'group', 'F', 'Sweden', 'Tunisia', 'Monterrey', '2026-06-15 02:00:00', 'scheduled'),
  326. (13, 13, 'World Cup', 'group', 'G', 'Belgium', 'Egypt', 'Seattle', '2026-06-15 19:00:00', 'scheduled'),
  327. (14, 14, 'World Cup', 'group', 'G', 'Iran', 'New Zealand', 'Inglewood', '2026-06-16 01:00:00', 'scheduled'),
  328. (15, 15, 'World Cup', 'group', 'H', 'Spain', 'Cape Verde', 'Atlanta', '2026-06-15 16:00:00', 'scheduled'),
  329. (16, 16, 'World Cup', 'group', 'H', 'Saudi Arabia', 'Uruguay', 'Miami Gardens', '2026-06-15 22:00:00', 'scheduled'),
  330. (17, 17, 'World Cup', 'group', 'I', 'France', 'Senegal', 'East Rutherford', '2026-06-16 19:00:00', 'scheduled'),
  331. (18, 18, 'World Cup', 'group', 'I', 'Iraq', 'Norway', 'Foxborough', '2026-06-16 22:00:00', 'scheduled'),
  332. (19, 19, 'World Cup', 'group', 'J', 'Argentina', 'Algeria', 'Kansas City', '2026-06-17 01:00:00', 'scheduled'),
  333. (20, 20, 'World Cup', 'group', 'J', 'Austria', 'Jordan', 'Santa Clara', '2026-06-17 04:00:00', 'scheduled'),
  334. (21, 21, 'World Cup', 'group', 'K', 'Portugal', 'DR Congo', 'Houston', '2026-06-17 17:00:00', 'scheduled'),
  335. (22, 22, 'World Cup', 'group', 'K', 'Uzbekistan', 'Colombia', 'Mexico City', '2026-06-18 02:00:00', 'scheduled'),
  336. (23, 23, 'World Cup', 'group', 'L', 'England', 'Croatia', 'Arlington', '2026-06-17 20:00:00', 'scheduled'),
  337. (24, 24, 'World Cup', 'group', 'L', 'Ghana', 'Panama', 'Toronto', '2026-06-17 23:00:00', 'scheduled'),
  338. (25, 25, 'World Cup', 'group', 'A', 'Czechia', 'South Africa', 'Atlanta', '2026-06-18 16:00:00', 'scheduled'),
  339. (26, 26, 'World Cup', 'group', 'A', 'Mexico', 'South Korea', 'Guadalajara', '2026-06-19 01:00:00', 'scheduled'),
  340. (27, 27, 'World Cup', 'group', 'B', 'Switzerland', 'Bosnia & Herzegovina', 'Inglewood', '2026-06-18 19:00:00', 'scheduled'),
  341. (28, 28, 'World Cup', 'group', 'B', 'Canada', 'Qatar', 'Vancouver', '2026-06-18 22:00:00', 'scheduled'),
  342. (29, 29, 'World Cup', 'group', 'C', 'Scotland', 'Morocco', 'Foxborough', '2026-06-19 22:00:00', 'scheduled'),
  343. (30, 30, 'World Cup', 'group', 'C', 'Brazil', 'Haiti', 'Philadelphia', '2026-06-20 00:30:00', 'scheduled'),
  344. (31, 31, 'World Cup', 'group', 'D', 'United States', 'Australia', 'Seattle', '2026-06-19 19:00:00', 'scheduled'),
  345. (32, 32, 'World Cup', 'group', 'D', 'Turkiye', 'Paraguay', 'Santa Clara', '2026-06-20 03:00:00', 'scheduled'),
  346. (33, 33, 'World Cup', 'group', 'E', 'Germany', 'Ivory Coast', 'Toronto', '2026-06-20 20:00:00', 'scheduled'),
  347. (34, 34, 'World Cup', 'group', 'E', 'Ecuador', 'Curacao', 'Kansas City', '2026-06-21 00:00:00', 'scheduled'),
  348. (35, 35, 'World Cup', 'group', 'F', 'Netherlands', 'Sweden', 'Houston', '2026-06-20 17:00:00', 'scheduled'),
  349. (36, 36, 'World Cup', 'group', 'F', 'Tunisia', 'Japan', 'Monterrey', '2026-06-21 04:00:00', 'scheduled'),
  350. (37, 37, 'World Cup', 'group', 'G', 'Belgium', 'Iran', 'Inglewood', '2026-06-21 19:00:00', 'scheduled'),
  351. (38, 38, 'World Cup', 'group', 'G', 'New Zealand', 'Egypt', 'Vancouver', '2026-06-22 01:00:00', 'scheduled'),
  352. (39, 39, 'World Cup', 'group', 'H', 'Spain', 'Saudi Arabia', 'Atlanta', '2026-06-21 16:00:00', 'scheduled'),
  353. (40, 40, 'World Cup', 'group', 'H', 'Uruguay', 'Cape Verde', 'Miami Gardens', '2026-06-21 22:00:00', 'scheduled'),
  354. (41, 41, 'World Cup', 'group', 'I', 'France', 'Iraq', 'Philadelphia', '2026-06-22 21:00:00', 'scheduled'),
  355. (42, 42, 'World Cup', 'group', 'I', 'Norway', 'Senegal', 'East Rutherford', '2026-06-23 00:00:00', 'scheduled'),
  356. (43, 43, 'World Cup', 'group', 'J', 'Argentina', 'Austria', 'Arlington', '2026-06-22 17:00:00', 'scheduled'),
  357. (44, 44, 'World Cup', 'group', 'J', 'Jordan', 'Algeria', 'Santa Clara', '2026-06-23 03:00:00', 'scheduled'),
  358. (45, 45, 'World Cup', 'group', 'K', 'Portugal', 'Uzbekistan', 'Houston', '2026-06-23 17:00:00', 'scheduled'),
  359. (46, 46, 'World Cup', 'group', 'K', 'Colombia', 'DR Congo', 'Guadalajara', '2026-06-24 02:00:00', 'scheduled'),
  360. (47, 47, 'World Cup', 'group', 'L', 'England', 'Ghana', 'Foxborough', '2026-06-23 20:00:00', 'scheduled'),
  361. (48, 48, 'World Cup', 'group', 'L', 'Panama', 'Croatia', 'Toronto', '2026-06-23 23:00:00', 'scheduled'),
  362. (49, 49, 'World Cup', 'group', 'A', 'Czechia', 'Mexico', 'Mexico City', '2026-06-25 01:00:00', 'scheduled'),
  363. (50, 50, 'World Cup', 'group', 'A', 'South Africa', 'South Korea', 'Monterrey', '2026-06-25 01:00:00', 'scheduled'),
  364. (51, 51, 'World Cup', 'group', 'B', 'Switzerland', 'Canada', 'Vancouver', '2026-06-24 19:00:00', 'scheduled'),
  365. (52, 52, 'World Cup', 'group', 'B', 'Bosnia & Herzegovina', 'Qatar', 'Seattle', '2026-06-24 19:00:00', 'scheduled'),
  366. (53, 53, 'World Cup', 'group', 'C', 'Scotland', 'Brazil', 'Miami Gardens', '2026-06-24 22:00:00', 'scheduled'),
  367. (54, 54, 'World Cup', 'group', 'C', 'Morocco', 'Haiti', 'Atlanta', '2026-06-24 22:00:00', 'scheduled'),
  368. (55, 55, 'World Cup', 'group', 'D', 'Turkiye', 'United States', 'Inglewood', '2026-06-26 02:00:00', 'scheduled'),
  369. (56, 56, 'World Cup', 'group', 'D', 'Paraguay', 'Australia', 'Santa Clara', '2026-06-26 02:00:00', 'scheduled'),
  370. (57, 57, 'World Cup', 'group', 'E', 'Curacao', 'Ivory Coast', 'Philadelphia', '2026-06-25 20:00:00', 'scheduled'),
  371. (58, 58, 'World Cup', 'group', 'E', 'Ecuador', 'Germany', 'East Rutherford', '2026-06-25 20:00:00', 'scheduled'),
  372. (59, 59, 'World Cup', 'group', 'F', 'Japan', 'Sweden', 'Arlington', '2026-06-25 23:00:00', 'scheduled'),
  373. (60, 60, 'World Cup', 'group', 'F', 'Tunisia', 'Netherlands', 'Kansas City', '2026-06-25 23:00:00', 'scheduled'),
  374. (61, 61, 'World Cup', 'group', 'G', 'Egypt', 'Iran', 'Seattle', '2026-06-27 03:00:00', 'scheduled'),
  375. (62, 62, 'World Cup', 'group', 'G', 'New Zealand', 'Belgium', 'Vancouver', '2026-06-27 03:00:00', 'scheduled'),
  376. (63, 63, 'World Cup', 'group', 'H', 'Cape Verde', 'Saudi Arabia', 'Houston', '2026-06-27 00:00:00', 'scheduled'),
  377. (64, 64, 'World Cup', 'group', 'H', 'Uruguay', 'Spain', 'Guadalajara', '2026-06-27 00:00:00', 'scheduled'),
  378. (65, 65, 'World Cup', 'group', 'I', 'Norway', 'France', 'Foxborough', '2026-06-26 19:00:00', 'scheduled'),
  379. (66, 66, 'World Cup', 'group', 'I', 'Senegal', 'Iraq', 'Toronto', '2026-06-26 19:00:00', 'scheduled'),
  380. (67, 67, 'World Cup', 'group', 'J', 'Algeria', 'Austria', 'Kansas City', '2026-06-28 02:00:00', 'scheduled'),
  381. (68, 68, 'World Cup', 'group', 'J', 'Jordan', 'Argentina', 'Arlington', '2026-06-28 02:00:00', 'scheduled'),
  382. (69, 69, 'World Cup', 'group', 'K', 'Colombia', 'Portugal', 'Miami Gardens', '2026-06-27 23:30:00', 'scheduled'),
  383. (70, 70, 'World Cup', 'group', 'K', 'DR Congo', 'Uzbekistan', 'Atlanta', '2026-06-27 23:30:00', 'scheduled'),
  384. (71, 71, 'World Cup', 'group', 'L', 'Panama', 'England', 'East Rutherford', '2026-06-27 21:00:00', 'scheduled'),
  385. (72, 72, 'World Cup', 'group', 'L', 'Croatia', 'Ghana', 'Philadelphia', '2026-06-27 21:00:00', 'scheduled'),
  386. (73, 73, 'World Cup', 'round_32', NULL, 'Winner 73 A', 'Winner 73 B', 'Inglewood', '2026-06-28 19:00:00', 'closed'),
  387. (74, 74, 'World Cup', 'round_32', NULL, 'Winner 74 A', 'Winner 74 B', 'Houston', '2026-06-29 17:00:00', 'closed'),
  388. (75, 75, 'World Cup', 'round_32', NULL, 'Winner 75 A', 'Winner 75 B', 'Foxborough', '2026-06-29 20:30:00', 'closed'),
  389. (76, 76, 'World Cup', 'round_32', NULL, 'Winner 76 A', 'Winner 76 B', 'Monterrey', '2026-06-30 01:00:00', 'closed'),
  390. (77, 77, 'World Cup', 'round_32', NULL, 'Winner 77 A', 'Winner 77 B', 'Arlington', '2026-06-30 17:00:00', 'closed'),
  391. (78, 78, 'World Cup', 'round_32', NULL, 'Winner 78 A', 'Winner 78 B', 'East Rutherford', '2026-06-30 21:00:00', 'closed'),
  392. (79, 79, 'World Cup', 'round_32', NULL, 'Winner 79 A', 'Winner 79 B', 'Mexico City', '2026-07-01 01:00:00', 'closed'),
  393. (80, 80, 'World Cup', 'round_32', NULL, 'Winner 80 A', 'Winner 80 B', 'Atlanta', '2026-07-01 16:00:00', 'closed'),
  394. (81, 81, 'World Cup', 'round_32', NULL, 'Winner 81 A', 'Winner 81 B', 'Seattle', '2026-07-01 20:00:00', 'closed'),
  395. (82, 82, 'World Cup', 'round_32', NULL, 'Winner 82 A', 'Winner 82 B', 'Santa Clara', '2026-07-02 00:00:00', 'closed'),
  396. (83, 83, 'World Cup', 'round_32', NULL, 'Winner 83 A', 'Winner 83 B', 'Inglewood', '2026-07-02 19:00:00', 'closed'),
  397. (84, 84, 'World Cup', 'round_32', NULL, 'Winner 84 A', 'Winner 84 B', 'Toronto', '2026-07-02 23:00:00', 'closed'),
  398. (85, 85, 'World Cup', 'round_32', NULL, 'Winner 85 A', 'Winner 85 B', 'Vancouver', '2026-07-03 03:00:00', 'closed'),
  399. (86, 86, 'World Cup', 'round_32', NULL, 'Winner 86 A', 'Winner 86 B', 'Arlington', '2026-07-03 18:00:00', 'closed'),
  400. (87, 87, 'World Cup', 'round_32', NULL, 'Winner 87 A', 'Winner 87 B', 'Miami Gardens', '2026-07-03 22:00:00', 'closed'),
  401. (88, 88, 'World Cup', 'round_32', NULL, 'Winner 88 A', 'Winner 88 B', 'Kansas City', '2026-07-04 01:30:00', 'closed'),
  402. (89, 89, 'World Cup', 'round_16', NULL, 'Winner 89 A', 'Winner 89 B', 'Houston', '2026-07-04 17:00:00', 'closed'),
  403. (90, 90, 'World Cup', 'round_16', NULL, 'Winner 90 A', 'Winner 90 B', 'Philadelphia', '2026-07-04 21:00:00', 'closed'),
  404. (91, 91, 'World Cup', 'round_16', NULL, 'Winner 91 A', 'Winner 91 B', 'East Rutherford', '2026-07-05 20:00:00', 'closed'),
  405. (92, 92, 'World Cup', 'round_16', NULL, 'Winner 92 A', 'Winner 92 B', 'Mexico City', '2026-07-06 00:00:00', 'closed'),
  406. (93, 93, 'World Cup', 'round_16', NULL, 'Winner 93 A', 'Winner 93 B', 'Arlington', '2026-07-06 19:00:00', 'closed'),
  407. (94, 94, 'World Cup', 'round_16', NULL, 'Winner 94 A', 'Winner 94 B', 'Seattle', '2026-07-07 00:00:00', 'closed'),
  408. (95, 95, 'World Cup', 'round_16', NULL, 'Winner 95 A', 'Winner 95 B', 'Atlanta', '2026-07-07 16:00:00', 'closed'),
  409. (96, 96, 'World Cup', 'round_16', NULL, 'Winner 96 A', 'Winner 96 B', 'Vancouver', '2026-07-07 20:00:00', 'closed'),
  410. (97, 97, 'World Cup', 'quarter_final', NULL, 'Winner 97 A', 'Winner 97 B', 'Foxborough', '2026-07-09 20:00:00', 'closed'),
  411. (98, 98, 'World Cup', 'quarter_final', NULL, 'Winner 98 A', 'Winner 98 B', 'Inglewood', '2026-07-10 19:00:00', 'closed'),
  412. (99, 99, 'World Cup', 'quarter_final', NULL, 'Winner 99 A', 'Winner 99 B', 'Miami Gardens', '2026-07-11 21:00:00', 'closed'),
  413. (100, 100, 'World Cup', 'quarter_final', NULL, 'Winner 100 A', 'Winner 100 B', 'Kansas City', '2026-07-12 01:00:00', 'closed'),
  414. (101, 101, 'World Cup', 'semi_final', NULL, 'Winner 101 A', 'Winner 101 B', 'Arlington', '2026-07-14 19:00:00', 'closed'),
  415. (102, 102, 'World Cup', 'semi_final', NULL, 'Winner 102 A', 'Winner 102 B', 'Atlanta', '2026-07-15 19:00:00', 'closed'),
  416. (103, 103, 'World Cup', 'third_place', NULL, 'Winner 103 A', 'Winner 103 B', 'Miami Gardens', '2026-07-18 21:00:00', 'closed'),
  417. (104, 104, 'World Cup', 'final', NULL, 'Winner 104 A', 'Winner 104 B', 'East Rutherford', '2026-07-19 19:00:00', 'closed')
  418. ) AS source (
  419. match_id,
  420. match_no,
  421. competition,
  422. stage,
  423. group_name,
  424. home_team,
  425. away_team,
  426. venue,
  427. kickoff_at,
  428. status
  429. )
  430. ON target.match_id = source.match_id
  431. WHEN MATCHED THEN
  432. UPDATE SET
  433. match_no = source.match_no,
  434. competition = source.competition,
  435. stage = source.stage,
  436. group_name = source.group_name,
  437. home_team = source.home_team,
  438. away_team = source.away_team,
  439. venue = source.venue,
  440. kickoff_at = source.kickoff_at,
  441. status = source.status,
  442. updated_at = GETDATE()
  443. WHEN NOT MATCHED THEN
  444. INSERT (match_id, match_no, competition, stage, group_name, home_team, away_team, venue, kickoff_at, status)
  445. VALUES (
  446. source.match_id,
  447. source.match_no,
  448. source.competition,
  449. source.stage,
  450. source.group_name,
  451. source.home_team,
  452. source.away_team,
  453. source.venue,
  454. source.kickoff_at,
  455. source.status
  456. );
  457. SET IDENTITY_INSERT agent.dbo.world_cup_matches OFF;
  458. END;
  459. IF OBJECT_ID('agent.dbo.world_cup_odds', 'U') IS NOT NULL
  460. BEGIN
  461. MERGE agent.dbo.world_cup_odds AS target
  462. USING (VALUES
  463. (1, '1x2', 'home', 1.40, 0),
  464. (1, '1x2', 'draw', 4.60, 0),
  465. (1, '1x2', 'away', 8.00, 0),
  466. (2, '1x2', 'home', 2.62, 0),
  467. (2, '1x2', 'draw', 3.10, 0),
  468. (2, '1x2', 'away', 2.75, 0),
  469. (7, '1x2', 'home', 2.05, 0),
  470. (7, '1x2', 'draw', 3.30, 0),
  471. (7, '1x2', 'away', 3.90, 0),
  472. (5, '1x2', 'home', 1.61, 0),
  473. (5, '1x2', 'draw', 3.90, 0),
  474. (5, '1x2', 'away', 5.50, 0),
  475. (6, '1x2', 'home', 6.25, 0),
  476. (6, '1x2', 'draw', 4.33, 0),
  477. (6, '1x2', 'away', 1.50, 0),
  478. (8, '1x2', 'home', 4.40, 0),
  479. (8, '1x2', 'draw', 3.50, 0),
  480. (8, '1x2', 'away', 1.80, 0),
  481. (9, '1x2', 'home', 1.04, 0),
  482. (9, '1x2', 'draw', 16.00, 0),
  483. (9, '1x2', 'away', 46.00, 0),
  484. (11, '1x2', 'home', 2.00, 0),
  485. (11, '1x2', 'draw', 3.50, 0),
  486. (11, '1x2', 'away', 3.50, 0),
  487. (10, '1x2', 'home', 3.50, 0),
  488. (10, '1x2', 'draw', 2.88, 0),
  489. (10, '1x2', 'away', 2.30, 0),
  490. (12, '1x2', 'home', 1.91, 0),
  491. (12, '1x2', 'draw', 3.30, 0),
  492. (12, '1x2', 'away', 4.00, 0),
  493. (15, '1x2', 'home', 1.11, 0),
  494. (15, '1x2', 'draw', 10.00, 0),
  495. (15, '1x2', 'away', 23.00, 0),
  496. (NULL, 'winner', 'Spain', 5.50, 999),
  497. (NULL, 'winner', 'France', 5.80, 998),
  498. (NULL, 'winner', 'England', 7.50, 997),
  499. (NULL, 'winner', 'Brazil', 10.00, 996),
  500. (NULL, 'winner', 'Portugal', 9.50, 995),
  501. (NULL, 'winner', 'Argentina', 10.50, 994),
  502. (NULL, 'winner', 'Germany', 14.00, 993),
  503. (NULL, 'winner', 'Netherlands', 18.00, 992),
  504. (NULL, 'winner', 'Belgium', 23.00, 991),
  505. (NULL, 'winner', 'Norway', 34.00, 990),
  506. (NULL, 'winner', 'Colombia', 41.00, 989),
  507. (NULL, 'winner', 'Japan', 46.00, 988),
  508. (NULL, 'winner', 'Morocco', 56.00, 987),
  509. (NULL, 'winner', 'United States', 56.00, 986),
  510. (NULL, 'winner', 'Uruguay', 61.00, 985),
  511. (NULL, 'winner', 'Mexico', 61.00, 984),
  512. (NULL, 'winner', 'Switzerland', 76.00, 983),
  513. (NULL, 'winner', 'Croatia', 81.00, 982),
  514. (NULL, 'winner', 'Ecuador', 101.00, 981),
  515. (NULL, 'winner', 'Austria', 101.00, 980),
  516. (NULL, 'winner', 'Turkiye', 81.00, 979),
  517. (NULL, 'winner', 'Senegal', 151.00, 978),
  518. (NULL, 'winner', 'Ivory Coast', 151.00, 977),
  519. (NULL, 'winner', 'Sweden', 176.00, 976),
  520. (NULL, 'winner', 'Canada', 201.00, 975),
  521. (NULL, 'winner', 'Paraguay', 226.00, 974),
  522. (NULL, 'winner', 'Bosnia & Herzegovina', 226.00, 973),
  523. (NULL, 'winner', 'Algeria', 251.00, 972),
  524. (NULL, 'winner', 'Scotland', 301.00, 971),
  525. (NULL, 'winner', 'Egypt', 351.00, 970),
  526. (NULL, 'winner', 'South Korea', 401.00, 969),
  527. (NULL, 'winner', 'Ghana', 401.00, 968),
  528. (NULL, 'winner', 'Czechia', 401.00, 967),
  529. (NULL, 'winner', 'Iran', 751.00, 966),
  530. (NULL, 'winner', 'Tunisia', 2001.00, 965),
  531. (NULL, 'winner', 'Australia', 2501.00, 964),
  532. (NULL, 'winner', 'DR Congo', 2501.00, 963),
  533. (NULL, 'winner', 'Cape Verde', 2501.00, 962),
  534. (NULL, 'winner', 'Uzbekistan', 2501.00, 961),
  535. (NULL, 'winner', 'Panama', 2501.00, 960),
  536. (NULL, 'winner', 'Curacao', 2501.00, 959),
  537. (NULL, 'winner', 'Haiti', 1501.00, 958),
  538. (NULL, 'winner', 'Qatar', 2501.00, 957),
  539. (NULL, 'winner', 'Saudi Arabia', 2501.00, 956),
  540. (NULL, 'winner', 'Iraq', 2501.00, 955),
  541. (NULL, 'winner', 'New Zealand', 2501.00, 954),
  542. (NULL, 'winner', 'Jordan', 2501.00, 953),
  543. (NULL, 'winner', 'South Africa', 2501.00, 952)
  544. ) AS source (match_id, market, selection, decimal_odds, locked_weight)
  545. ON target.market = source.market
  546. AND (
  547. target.match_id = source.match_id
  548. OR (target.match_id IS NULL AND source.match_id IS NULL)
  549. )
  550. AND target.selection = source.selection
  551. WHEN MATCHED THEN
  552. UPDATE SET
  553. decimal_odds = source.decimal_odds,
  554. previous_odds = target.decimal_odds,
  555. is_active = 1,
  556. locked_weight = source.locked_weight,
  557. updated_at = GETDATE()
  558. WHEN NOT MATCHED THEN
  559. INSERT (match_id, market, selection, decimal_odds, previous_odds, is_active, locked_weight, updated_at)
  560. VALUES (
  561. source.match_id,
  562. source.market,
  563. source.selection,
  564. source.decimal_odds,
  565. NULL,
  566. 1,
  567. source.locked_weight,
  568. GETDATE()
  569. );
  570. END;
  571. USE [agent]
  572. GO
  573. DECLARE @WorldCupDescriptions TABLE (
  574. table_name SYSNAME NOT NULL,
  575. column_name SYSNAME NULL,
  576. description NVARCHAR(4000) NOT NULL
  577. );
  578. INSERT INTO @WorldCupDescriptions (table_name, column_name, description)
  579. VALUES
  580. (N'world_cup_user_state', NULL, N'世界杯活动用户状态表'),
  581. (N'world_cup_user_state', N'user_id', N'活动用户 UserID,对应 QPAccountsDB.dbo.AccountsInfo.UserID'),
  582. (N'world_cup_user_state', N'first_bet_used', N'是否已使用世界杯首注奖励,1=已使用,0=未使用'),
  583. (N'world_cup_user_state', N'invite_code', N'用户自己的邀请码,前端展示并用于手动绑定'),
  584. (N'world_cup_user_state', N'referred_by_user_id', N'邀请人的 UserID,未绑定时为空'),
  585. (N'world_cup_user_state', N'referral_bind_at', N'邀请关系绑定时间'),
  586. (N'world_cup_user_state', N'referral_bind_type', N'绑定方式,link=分享链接自动绑定,manual=24h 内手动输入邀请码'),
  587. (N'world_cup_user_state', N'device_fp', N'设备指纹,用于风控识别同设备自邀'),
  588. (N'world_cup_user_state', N'pay_account_hash', N'支付账号哈希,用于风控识别同支付账号'),
  589. (N'world_cup_user_state', N'signup_ip', N'注册 IP,用于风控识别同 IP 或同网段聚集'),
  590. (N'world_cup_user_state', N'created_at', N'记录创建时间'),
  591. (N'world_cup_user_state', N'updated_at', N'记录更新时间'),
  592. (N'world_cup_matches', NULL, N'世界杯单场比赛表'),
  593. (N'world_cup_matches', N'match_id', N'比赛 ID,单场 1X2 盘口主键'),
  594. (N'world_cup_matches', N'match_no', N'FIFA 官方赛程 Match 编号'),
  595. (N'world_cup_matches', N'competition', N'赛事名称,例如 World Cup'),
  596. (N'world_cup_matches', N'stage', N'赛事阶段,group=小组赛,round_32=32 强,round_16=16 强,quarter_final=1/4 决赛,semi_final=半决赛,third_place=季军赛,final=决赛'),
  597. (N'world_cup_matches', N'group_name', N'小组名称,小组赛为 A-L,淘汰赛为空'),
  598. (N'world_cup_matches', N'home_team', N'主队名称'),
  599. (N'world_cup_matches', N'away_team', N'客队名称'),
  600. (N'world_cup_matches', N'venue', N'比赛城市或场馆名称'),
  601. (N'world_cup_matches', N'kickoff_at', N'开赛时间;开赛前 1 小时停止投注'),
  602. (N'world_cup_matches', N'status', N'比赛状态,scheduled=待开赛可投注,closed=关闭投注,finished=已完赛'),
  603. (N'world_cup_matches', N'result', N'比赛结果,home/draw/away;未结算时为空'),
  604. (N'world_cup_matches', N'created_at', N'记录创建时间'),
  605. (N'world_cup_matches', N'updated_at', N'记录更新时间'),
  606. (N'world_cup_odds', NULL, N'世界杯盘口赔率表'),
  607. (N'world_cup_odds', N'odds_id', N'赔率记录 ID'),
  608. (N'world_cup_odds', N'match_id', N'比赛 ID;Winner 夺冠盘为空,1X2 单场盘必填'),
  609. (N'world_cup_odds', N'market', N'盘口类型,winner=夺冠盘,1x2=单场胜平负'),
  610. (N'world_cup_odds', N'selection', N'投注选项;Winner 为球队名,1X2 为 home/draw/away'),
  611. (N'world_cup_odds', N'decimal_odds', N'当前十进制赔率'),
  612. (N'world_cup_odds', N'previous_odds', N'上一版赔率,用于前端展示涨跌箭头'),
  613. (N'world_cup_odds', N'is_active', N'是否展示并允许投注,1=启用,0=隐藏'),
  614. (N'world_cup_odds', N'locked_weight', N'后台排序权重,数值越大越靠前'),
  615. (N'world_cup_odds', N'updated_at', N'赔率更新时间'),
  616. (N'world_cup_match_favorites', NULL, N'世界杯比赛收藏表'),
  617. (N'world_cup_match_favorites', N'id', N'收藏记录 ID'),
  618. (N'world_cup_match_favorites', N'user_id', N'收藏用户 UserID'),
  619. (N'world_cup_match_favorites', N'match_id', N'被收藏的单场比赛 ID,仅支持 1X2 比赛,不支持 Winner 盘'),
  620. (N'world_cup_match_favorites', N'created_at', N'收藏创建时间'),
  621. (N'world_cup_bets', NULL, N'世界杯投注订单表'),
  622. (N'world_cup_bets', N'bet_id', N'世界杯注单 ID'),
  623. (N'world_cup_bets', N'user_id', N'下注用户 UserID'),
  624. (N'world_cup_bets', N'game_id', N'前端展示的 8 位 GameID,冗余保存便于查询'),
  625. (N'world_cup_bets', N'idempotency_key', N'客户端幂等键,同一用户同一 key 不重复下单'),
  626. (N'world_cup_bets', N'market', N'盘口类型,winner=夺冠盘,1x2=单场胜平负'),
  627. (N'world_cup_bets', N'match_id', N'比赛 ID;Winner 夺冠盘为空'),
  628. (N'world_cup_bets', N'selection', N'投注选项,球队名或 home/draw/away'),
  629. (N'world_cup_bets', N'stake', N'下注金额,整数分'),
  630. (N'world_cup_bets', N'odds', N'下单时锁定的十进制赔率'),
  631. (N'world_cup_bets', N'is_first_bet', N'是否为用户世界杯首注,1=首注'),
  632. (N'world_cup_bets', N'bonus_amount', N'首注 +50% 额外奖励金额,整数分'),
  633. (N'world_cup_bets', N'potential_payout', N'潜在派彩金额,整数分,stake * odds + bonus_amount'),
  634. (N'world_cup_bets', N'status', N'注单状态,pending=待结算,won=赢,lost=输'),
  635. (N'world_cup_bets', N'settled_at', N'结算时间'),
  636. (N'world_cup_bets', N'created_at', N'下单时间'),
  637. (N'world_cup_bets', N'updated_at', N'注单更新时间'),
  638. (N'world_cup_referrals', NULL, N'世界杯邀请绑定关系表'),
  639. (N'world_cup_referrals', N'id', N'邀请绑定记录 ID'),
  640. (N'world_cup_referrals', N'referrer_id', N'邀请人 UserID'),
  641. (N'world_cup_referrals', N'invitee_id', N'被邀请人 UserID;唯一,一个被邀请人只能绑定一次'),
  642. (N'world_cup_referrals', N'bind_type', N'绑定方式,link=分享链接自动绑定,manual=手动补填邀请码'),
  643. (N'world_cup_referrals', N'bind_at', N'绑定发生时间'),
  644. (N'world_cup_referrals', N'created_at', N'记录创建时间'),
  645. (N'world_cup_referral_rewards', NULL, N'世界杯邀请奖励审核表'),
  646. (N'world_cup_referral_rewards', N'reward_id', N'邀请奖励单 ID,后台审核对象'),
  647. (N'world_cup_referral_rewards', N'referrer_id', N'邀请人 UserID'),
  648. (N'world_cup_referral_rewards', N'invitee_id', N'被邀请人 UserID;同一被邀请人首充只生成一次奖励单'),
  649. (N'world_cup_referral_rewards', N'first_deposit_order_sn', N'触发邀请奖励的首充订单号,用于 OrderPaid 事件监听幂等'),
  650. (N'world_cup_referral_rewards', N'first_deposit_amt', N'触发奖励时的首充实际入账金额,整数分'),
  651. (N'world_cup_referral_rewards', N'reward_each', N'双方各得奖励金额,整数分;规则为 min(first_deposit_amt * 50%, 10000)'),
  652. (N'world_cup_referral_rewards', N'total_liability', N'本奖励单总赔付,整数分,reward_each * 2,最高 20000'),
  653. (N'world_cup_referral_rewards', N'risk_score', N'风控评分,0-100'),
  654. (N'world_cup_referral_rewards', N'risk_level', N'风险等级,low/medium/high'),
  655. (N'world_cup_referral_rewards', N'signals', N'命中的风控信号 JSON 文本'),
  656. (N'world_cup_referral_rewards', N'status', N'审核状态,reviewing/approved/rejected/on_hold/clawed_back'),
  657. (N'world_cup_referral_rewards', N'reason_code', N'驳回或追回原因代码'),
  658. (N'world_cup_referral_rewards', N'review_by', N'审核员标识'),
  659. (N'world_cup_referral_rewards', N'reviewed_at', N'审核时间'),
  660. (N'world_cup_referral_rewards', N'submitted_at', N'奖励单提交审核时间,用于计算 24h SLA'),
  661. (N'world_cup_referral_rewards', N'created_at', N'记录创建时间'),
  662. (N'world_cup_referral_rewards', N'updated_at', N'记录更新时间'),
  663. (N'world_cup_risk_signals', NULL, N'世界杯风控信号配置表'),
  664. (N'world_cup_risk_signals', N'code', N'风控信号代码'),
  665. (N'world_cup_risk_signals', N'label', N'后台展示名称'),
  666. (N'world_cup_risk_signals', N'weight', N'风控权重,命中后累加到 risk_score'),
  667. (N'world_cup_risk_signals', N'is_active', N'是否启用,1=启用,0=停用'),
  668. (N'world_cup_risk_signals', N'updated_at', N'配置更新时间'),
  669. (N'world_cup_audit_log', NULL, N'世界杯审核操作日志表'),
  670. (N'world_cup_audit_log', N'id', N'审核日志 ID'),
  671. (N'world_cup_audit_log', N'reward_id', N'关联的邀请奖励单 ID,可为空'),
  672. (N'world_cup_audit_log', N'actor', N'操作人,后台审核员或系统任务'),
  673. (N'world_cup_audit_log', N'action', N'操作动作,例如 approve/reject/hold/clawback/auto_score'),
  674. (N'world_cup_audit_log', N'reason_code', N'操作原因代码'),
  675. (N'world_cup_audit_log', N'before_status', N'操作前状态'),
  676. (N'world_cup_audit_log', N'after_status', N'操作后状态'),
  677. (N'world_cup_audit_log', N'payload', N'操作上下文 JSON 文本'),
  678. (N'world_cup_audit_log', N'created_at', N'日志创建时间');
  679. DECLARE
  680. @tableName SYSNAME,
  681. @columnName SYSNAME,
  682. @description NVARCHAR(4000),
  683. @objectName NVARCHAR(512);
  684. DECLARE world_cup_description_cursor CURSOR LOCAL FAST_FORWARD FOR
  685. SELECT table_name, column_name, description
  686. FROM @WorldCupDescriptions;
  687. OPEN world_cup_description_cursor;
  688. FETCH NEXT FROM world_cup_description_cursor INTO @tableName, @columnName, @description;
  689. WHILE @@FETCH_STATUS = 0
  690. BEGIN
  691. SET @objectName = N'dbo.' + @tableName;
  692. IF OBJECT_ID(@objectName, 'U') IS NOT NULL
  693. AND (@columnName IS NULL OR COL_LENGTH(@objectName, @columnName) IS NOT NULL)
  694. BEGIN
  695. IF @columnName IS NULL
  696. BEGIN
  697. IF EXISTS (
  698. SELECT 1
  699. FROM sys.extended_properties ep
  700. INNER JOIN sys.tables t ON ep.major_id = t.object_id
  701. INNER JOIN sys.schemas s ON t.schema_id = s.schema_id
  702. WHERE ep.name = N'MS_Description'
  703. AND ep.minor_id = 0
  704. AND s.name = N'dbo'
  705. AND t.name = @tableName
  706. )
  707. BEGIN
  708. EXEC sys.sp_updateextendedproperty
  709. @name = N'MS_Description',
  710. @value = @description,
  711. @level0type = N'SCHEMA',
  712. @level0name = N'dbo',
  713. @level1type = N'TABLE',
  714. @level1name = @tableName;
  715. END
  716. ELSE
  717. BEGIN
  718. EXEC sys.sp_addextendedproperty
  719. @name = N'MS_Description',
  720. @value = @description,
  721. @level0type = N'SCHEMA',
  722. @level0name = N'dbo',
  723. @level1type = N'TABLE',
  724. @level1name = @tableName;
  725. END;
  726. END
  727. ELSE
  728. BEGIN
  729. IF EXISTS (
  730. SELECT 1
  731. FROM sys.extended_properties ep
  732. INNER JOIN sys.tables t ON ep.major_id = t.object_id
  733. INNER JOIN sys.schemas s ON t.schema_id = s.schema_id
  734. INNER JOIN sys.columns c ON ep.major_id = c.object_id
  735. AND ep.minor_id = c.column_id
  736. WHERE ep.name = N'MS_Description'
  737. AND s.name = N'dbo'
  738. AND t.name = @tableName
  739. AND c.name = @columnName
  740. )
  741. BEGIN
  742. EXEC sys.sp_updateextendedproperty
  743. @name = N'MS_Description',
  744. @value = @description,
  745. @level0type = N'SCHEMA',
  746. @level0name = N'dbo',
  747. @level1type = N'TABLE',
  748. @level1name = @tableName,
  749. @level2type = N'COLUMN',
  750. @level2name = @columnName;
  751. END
  752. ELSE
  753. BEGIN
  754. EXEC sys.sp_addextendedproperty
  755. @name = N'MS_Description',
  756. @value = @description,
  757. @level0type = N'SCHEMA',
  758. @level0name = N'dbo',
  759. @level1type = N'TABLE',
  760. @level1name = @tableName,
  761. @level2type = N'COLUMN',
  762. @level2name = @columnName;
  763. END;
  764. END;
  765. END;
  766. FETCH NEXT FROM world_cup_description_cursor INTO @tableName, @columnName, @description;
  767. END;
  768. CLOSE world_cup_description_cursor;
  769. DEALLOCATE world_cup_description_cursor;
  770. GO