@windrun-huaiin/backend-core 27.0.0 → 28.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (147) hide show
  1. package/dist/app/api/ai/route.d.ts +2 -0
  2. package/dist/app/api/ai/route.d.ts.map +1 -0
  3. package/dist/app/api/ai/route.js +10 -0
  4. package/dist/app/api/ai/route.mjs +1 -0
  5. package/dist/app/api/stripe/checkout/route.js +1 -1
  6. package/dist/app/api/stripe/checkout/route.mjs +1 -1
  7. package/dist/app/api/stripe/customer-portal/route.js +1 -1
  8. package/dist/app/api/stripe/customer-portal/route.mjs +1 -1
  9. package/dist/app/api/user/credit-overview/route.d.ts +13 -0
  10. package/dist/app/api/user/credit-overview/route.d.ts.map +1 -0
  11. package/dist/app/api/user/credit-overview/route.js +32 -0
  12. package/dist/app/api/user/credit-overview/route.mjs +30 -0
  13. package/dist/app/api/user/pricing-context/route.d.ts +3 -0
  14. package/dist/app/api/user/pricing-context/route.d.ts.map +1 -0
  15. package/dist/app/api/user/pricing-context/route.js +19 -0
  16. package/dist/app/api/user/pricing-context/route.mjs +17 -0
  17. package/dist/app/api/webhook/clerk/user/route.js +1 -1
  18. package/dist/app/api/webhook/clerk/user/route.mjs +1 -1
  19. package/dist/auth/auth-utils.js +1 -1
  20. package/dist/auth/auth-utils.mjs +1 -1
  21. package/dist/core-prisma/browser.d.ts +40 -0
  22. package/dist/core-prisma/browser.d.ts.map +1 -0
  23. package/dist/core-prisma/client.d.ts +57 -0
  24. package/dist/core-prisma/client.d.ts.map +1 -0
  25. package/dist/core-prisma/client.js +55 -0
  26. package/dist/core-prisma/client.mjs +33 -0
  27. package/dist/core-prisma/commonInputTypes.d.ts +495 -0
  28. package/dist/core-prisma/commonInputTypes.d.ts.map +1 -0
  29. package/dist/core-prisma/enums.d.ts +2 -0
  30. package/dist/core-prisma/enums.d.ts.map +1 -0
  31. package/dist/core-prisma/internal/class.d.ts +193 -0
  32. package/dist/core-prisma/internal/class.d.ts.map +1 -0
  33. package/dist/core-prisma/internal/class.js +108 -0
  34. package/dist/core-prisma/internal/class.mjs +87 -0
  35. package/dist/core-prisma/internal/prismaNamespace.d.ts +1180 -0
  36. package/dist/core-prisma/internal/prismaNamespace.d.ts.map +1 -0
  37. package/dist/core-prisma/internal/prismaNamespace.js +278 -0
  38. package/dist/core-prisma/internal/prismaNamespace.mjs +226 -0
  39. package/dist/core-prisma/internal/prismaNamespaceBrowser.d.ts +233 -0
  40. package/dist/core-prisma/internal/prismaNamespaceBrowser.d.ts.map +1 -0
  41. package/dist/core-prisma/models/Apilog.d.ts +1088 -0
  42. package/dist/core-prisma/models/Apilog.d.ts.map +1 -0
  43. package/dist/core-prisma/models/Credit.d.ts +1554 -0
  44. package/dist/core-prisma/models/Credit.d.ts.map +1 -0
  45. package/dist/core-prisma/models/CreditAuditLog.d.ts +1362 -0
  46. package/dist/core-prisma/models/CreditAuditLog.d.ts.map +1 -0
  47. package/dist/core-prisma/models/Subscription.d.ts +1441 -0
  48. package/dist/core-prisma/models/Subscription.d.ts.map +1 -0
  49. package/dist/core-prisma/models/Transaction.d.ts +2222 -0
  50. package/dist/core-prisma/models/Transaction.d.ts.map +1 -0
  51. package/dist/core-prisma/models/User.d.ts +1775 -0
  52. package/dist/core-prisma/models/User.d.ts.map +1 -0
  53. package/dist/core-prisma/models/UserBackup.d.ts +1228 -0
  54. package/dist/core-prisma/models/UserBackup.d.ts.map +1 -0
  55. package/dist/core-prisma/models.d.ts +9 -0
  56. package/dist/core-prisma/models.d.ts.map +1 -0
  57. package/dist/credit/server.d.ts +1 -0
  58. package/dist/credit/server.d.ts.map +1 -1
  59. package/dist/credit/server.js +2 -0
  60. package/dist/credit/server.mjs +1 -0
  61. package/dist/index.js +7 -0
  62. package/dist/index.mjs +5 -2
  63. package/dist/lib/credit-overview.d.ts +17 -0
  64. package/dist/lib/credit-overview.d.ts.map +1 -0
  65. package/dist/lib/credit-overview.js +117 -0
  66. package/dist/lib/credit-overview.mjs +115 -0
  67. package/dist/lib/money-price-helper.d.ts.map +1 -1
  68. package/dist/lib/money-price-helper.js +1 -0
  69. package/dist/lib/money-price-helper.mjs +2 -1
  70. package/dist/lib/stripe-config.js +1 -1
  71. package/dist/lib/stripe-config.mjs +1 -1
  72. package/dist/prisma/index.d.ts +4 -1
  73. package/dist/prisma/index.d.ts.map +1 -1
  74. package/dist/prisma/index.js +6 -0
  75. package/dist/prisma/index.mjs +4 -1
  76. package/dist/prisma/prisma-transaction-util.d.ts +1 -1
  77. package/dist/prisma/prisma-transaction-util.d.ts.map +1 -1
  78. package/dist/prisma/prisma-transaction-util.js +1 -1
  79. package/dist/prisma/prisma-transaction-util.mjs +2 -2
  80. package/dist/prisma/prisma.d.ts +22 -3
  81. package/dist/prisma/prisma.d.ts.map +1 -1
  82. package/dist/prisma/prisma.js +83 -20
  83. package/dist/prisma/prisma.mjs +81 -21
  84. package/dist/services/aggregate/anonymous.aggregate.service.js +3 -2
  85. package/dist/services/aggregate/anonymous.aggregate.service.mjs +3 -2
  86. package/dist/services/aggregate/billing.aggregate.service.js +1 -1
  87. package/dist/services/aggregate/billing.aggregate.service.mjs +1 -1
  88. package/dist/services/aggregate/user.aggregate.service.d.ts.map +1 -1
  89. package/dist/services/aggregate/user.aggregate.service.js +3 -3
  90. package/dist/services/aggregate/user.aggregate.service.mjs +3 -3
  91. package/dist/services/context/index.js +1 -0
  92. package/dist/services/context/index.mjs +1 -1
  93. package/dist/services/context/user-context-finalizer.d.ts +1 -0
  94. package/dist/services/context/user-context-finalizer.d.ts.map +1 -1
  95. package/dist/services/context/user-context-finalizer.js +17 -0
  96. package/dist/services/context/user-context-finalizer.mjs +17 -1
  97. package/dist/services/context/user-context-service.js +1 -1
  98. package/dist/services/context/user-context-service.mjs +1 -1
  99. package/dist/services/database/apilog.service.js +1 -0
  100. package/dist/services/database/apilog.service.mjs +1 -0
  101. package/dist/services/database/credit.service.js +5 -4
  102. package/dist/services/database/credit.service.mjs +3 -2
  103. package/dist/services/database/creditAuditLog.service.js +5 -4
  104. package/dist/services/database/creditAuditLog.service.mjs +3 -2
  105. package/dist/services/database/prisma-model-type.d.ts +2 -2
  106. package/dist/services/database/prisma-model-type.d.ts.map +1 -1
  107. package/dist/services/database/subscription.service.js +1 -0
  108. package/dist/services/database/subscription.service.mjs +1 -0
  109. package/dist/services/database/transaction.service.js +5 -4
  110. package/dist/services/database/transaction.service.mjs +3 -2
  111. package/dist/services/database/user.service.js +1 -0
  112. package/dist/services/database/user.service.mjs +1 -0
  113. package/dist/services/database/userBackup.service.js +1 -0
  114. package/dist/services/database/userBackup.service.mjs +1 -0
  115. package/dist/services/stripe/webhook-handler.js +1 -1
  116. package/dist/services/stripe/webhook-handler.mjs +1 -1
  117. package/package.json +25 -8
  118. package/prisma/schema.prisma +3 -1
  119. package/src/app/api/ai/route.ts +6 -0
  120. package/src/app/api/user/credit-overview/route.ts +43 -0
  121. package/src/app/api/user/pricing-context/route.ts +14 -0
  122. package/src/core-prisma/browser.ts +54 -0
  123. package/src/core-prisma/client.ts +78 -0
  124. package/src/core-prisma/commonInputTypes.ts +565 -0
  125. package/src/core-prisma/enums.ts +15 -0
  126. package/src/core-prisma/internal/class.ts +293 -0
  127. package/src/core-prisma/internal/prismaNamespace.ts +1419 -0
  128. package/src/core-prisma/internal/prismaNamespaceBrowser.ts +259 -0
  129. package/src/core-prisma/models/Apilog.ts +1244 -0
  130. package/src/core-prisma/models/Credit.ts +1722 -0
  131. package/src/core-prisma/models/CreditAuditLog.ts +1536 -0
  132. package/src/core-prisma/models/Subscription.ts +1610 -0
  133. package/src/core-prisma/models/Transaction.ts +2398 -0
  134. package/src/core-prisma/models/User.ts +1982 -0
  135. package/src/core-prisma/models/UserBackup.ts +1384 -0
  136. package/src/core-prisma/models.ts +18 -0
  137. package/src/credit/server.ts +1 -0
  138. package/src/lib/credit-overview.ts +153 -0
  139. package/src/lib/money-price-helper.ts +3 -0
  140. package/src/prisma/index.ts +18 -1
  141. package/src/prisma/prisma-transaction-util.ts +3 -3
  142. package/src/prisma/prisma.ts +130 -21
  143. package/src/services/aggregate/user.aggregate.service.ts +2 -1
  144. package/src/services/context/user-context-finalizer.ts +16 -0
  145. package/src/services/database/credit.service.ts +2 -2
  146. package/src/services/database/prisma-model-type.ts +2 -2
  147. package/src/services/database/userBackup.service.ts +2 -2
@@ -1,20 +1,21 @@
1
1
  'use strict';
2
2
 
3
- var client = require('@prisma/client');
3
+ var adapterPg = require('@prisma/adapter-pg');
4
+ var client = require('../core-prisma/client.js');
4
5
 
5
- var _a;
6
6
  const globalForPrisma = globalThis;
7
7
  // ==================== 日志配置 ====================
8
8
  const getLogConfig = () => {
9
+ if (process.env.PRISMA_DEBUG === 'true') {
10
+ return [
11
+ { emit: 'event', level: 'query' },
12
+ { emit: 'stdout', level: 'info' },
13
+ { emit: 'stdout', level: 'warn' },
14
+ { emit: 'stdout', level: 'error' },
15
+ ];
16
+ }
9
17
  const env = process.env.NODE_ENV || 'development';
10
18
  switch (env) {
11
- case 'development':
12
- return [
13
- { emit: 'event', level: 'query' },
14
- { emit: 'stdout', level: 'info' },
15
- { emit: 'stdout', level: 'warn' },
16
- { emit: 'stdout', level: 'error' },
17
- ];
18
19
  case 'test':
19
20
  return [
20
21
  { emit: 'stdout', level: 'warn' },
@@ -25,22 +26,80 @@ const getLogConfig = () => {
25
26
  }
26
27
  };
27
28
  const logConfig = getLogConfig();
28
- // ==================== 创建 Prisma 全局单例 ====================
29
- const prisma = (_a = globalForPrisma.prisma) !== null && _a !== void 0 ? _a : new client.PrismaClient({
30
- log: logConfig,
31
- });
32
- if (process.env.NODE_ENV !== 'production') {
33
- globalForPrisma.prisma = prisma;
29
+ function isPrismaDebugEnabled() {
30
+ return process.env.PRISMA_DEBUG === 'true';
31
+ }
32
+ function createPrismaInstanceId(prefix = 'core-prisma') {
33
+ return `${prefix}_${Date.now()}_${Math.random().toString(36).slice(2, 7)}`;
34
+ }
35
+ function createPrismaPgConfig(databaseUrl) {
36
+ const ca = process.env.SUPABASE_DB_CA_CERT;
37
+ if (!ca && !globalForPrisma.__prisma_ssl_warning_logged) {
38
+ console.warn('SUPABASE_DB_CA_CERT is not configured. Prisma will request TLS without certificate verification unless DATABASE_URL SSL parameters override this behavior. Configure SUPABASE_DB_CA_CERT for certificate verification.');
39
+ globalForPrisma.__prisma_ssl_warning_logged = true;
40
+ }
41
+ return {
42
+ connectionString: databaseUrl,
43
+ ssl: ca
44
+ ? {
45
+ ca,
46
+ rejectUnauthorized: true,
47
+ }
48
+ : {
49
+ rejectUnauthorized: false,
50
+ },
51
+ };
52
+ }
53
+ function createPrismaClient(databaseUrl = process.env.DATABASE_URL) {
54
+ if (!databaseUrl) {
55
+ throw new Error('DATABASE_URL is required to create PrismaClient');
56
+ }
57
+ const adapter = new adapterPg.PrismaPg(createPrismaPgConfig(databaseUrl));
58
+ const instanceId = createPrismaInstanceId();
59
+ if (isPrismaDebugEnabled()) {
60
+ console.log(`Prisma Client Created | ID: ${instanceId}`);
61
+ }
62
+ const client$1 = new client.PrismaClient({
63
+ adapter,
64
+ log: logConfig,
65
+ });
66
+ return configureBackendCorePrisma(client$1, instanceId);
67
+ }
68
+ function configureBackendCorePrisma(prismaClient, instanceId) {
69
+ var _a;
70
+ if (instanceId === void 0) { instanceId = (_a = globalForPrisma.__prisma_instance_id) !== null && _a !== void 0 ? _a : createPrismaInstanceId(); }
71
+ globalForPrisma.prisma = prismaClient;
72
+ globalForPrisma.__prisma_instance_id = instanceId;
73
+ registerDevelopmentQueryLogger(prismaClient, instanceId);
74
+ return prismaClient;
34
75
  }
35
- if (process.env.NODE_ENV === 'development') {
76
+ function getBackendCorePrisma() {
77
+ if (!globalForPrisma.prisma) {
78
+ configureBackendCorePrisma(createPrismaClient());
79
+ }
80
+ return globalForPrisma.prisma;
81
+ }
82
+ // Backward-compatible lazy export. Accessing a property creates the client,
83
+ // importing this module does not.
84
+ const prisma = new Proxy({}, {
85
+ get(_target, property, receiver) {
86
+ return Reflect.get(getBackendCorePrisma(), property, receiver);
87
+ },
88
+ });
89
+ function registerDevelopmentQueryLogger(prismaClient, instanceId) {
90
+ var _a;
91
+ if (!isPrismaDebugEnabled()) {
92
+ return;
93
+ }
36
94
  const REGISTERED_KEY = '__prisma_query_logger_registered';
37
95
  const ID_KEY = '__prisma_query_logger_id';
38
96
  if (globalForPrisma[REGISTERED_KEY]) {
39
- console.log(`Prisma Query Logger Already Registered | ID: ${globalForPrisma[ID_KEY]}`);
97
+ console.log(`Prisma Query Logger Already Registered | Listener ID: ${globalForPrisma[ID_KEY]} | Instance ID: ${globalForPrisma.__prisma_instance_id}`);
40
98
  }
41
99
  else {
42
100
  const listenerId = `listener_${Date.now()}_${Math.random().toString(36).substr(2, 5)}`;
43
101
  globalForPrisma[ID_KEY] = listenerId;
102
+ console.log(`Prisma Query Logger Registered | Listener ID: ${listenerId} | Instance ID: ${instanceId}`);
44
103
  // --- 自定义SQL拼接 ---
45
104
  const interpolate = (query, params) => {
46
105
  // 1. 【核心修改】:安全检查和参数解析
@@ -84,7 +143,7 @@ if (process.env.NODE_ENV === 'development') {
84
143
  return sql;
85
144
  };
86
145
  const wrappedHandler = (event) => {
87
- const ms = event.duration;
146
+ const ms = Math.round(event.duration);
88
147
  const slow = ms >= 200 ? '🐌 SLOW SQL ' : '🚀 SQL';
89
148
  const interpolatedSql = interpolate(event.query, event.params);
90
149
  const clean = interpolatedSql
@@ -92,18 +151,22 @@ if (process.env.NODE_ENV === 'development') {
92
151
  .replace(/= '([^']+)'/g, `= '$1'`) // 已经替换成单引号,此处可以优化
93
152
  .replace(/"/g, ''); // 彻底灭双引号
94
153
  console.log('─'.repeat(60));
154
+ console.log(`Prisma Instance ID: ${instanceId} | Listener ID: ${listenerId}`);
95
155
  console.log(`${clean};`);
96
156
  console.log(`⏰ 耗时: ${ms}ms, ${slow}`);
97
157
  };
98
158
  // 注册包装后的 handler
99
- prisma.$on('query', wrappedHandler);
159
+ (_a = prismaClient.$on) === null || _a === void 0 ? void 0 : _a.call(prismaClient, 'query', wrappedHandler);
100
160
  globalForPrisma[REGISTERED_KEY] = true;
101
161
  }
102
162
  }
103
163
  // ==================== 便捷方法, 入参事务客户端不存在或者不传, 就返回全局非事务客户端 ====================
104
164
  function checkAndFallbackWithNonTCClient(tx) {
105
- return tx !== null && tx !== void 0 ? tx : prisma;
165
+ return tx !== null && tx !== void 0 ? tx : getBackendCorePrisma();
106
166
  }
107
167
 
108
168
  exports.checkAndFallbackWithNonTCClient = checkAndFallbackWithNonTCClient;
169
+ exports.configureBackendCorePrisma = configureBackendCorePrisma;
170
+ exports.createPrismaClient = createPrismaClient;
171
+ exports.getBackendCorePrisma = getBackendCorePrisma;
109
172
  exports.prisma = prisma;
@@ -1,18 +1,19 @@
1
- import { PrismaClient } from '@prisma/client';
1
+ import { PrismaPg } from '@prisma/adapter-pg';
2
+ import { PrismaClient } from '../core-prisma/client.mjs';
2
3
 
3
- var _a;
4
4
  const globalForPrisma = globalThis;
5
5
  // ==================== 日志配置 ====================
6
6
  const getLogConfig = () => {
7
+ if (process.env.PRISMA_DEBUG === 'true') {
8
+ return [
9
+ { emit: 'event', level: 'query' },
10
+ { emit: 'stdout', level: 'info' },
11
+ { emit: 'stdout', level: 'warn' },
12
+ { emit: 'stdout', level: 'error' },
13
+ ];
14
+ }
7
15
  const env = process.env.NODE_ENV || 'development';
8
16
  switch (env) {
9
- case 'development':
10
- return [
11
- { emit: 'event', level: 'query' },
12
- { emit: 'stdout', level: 'info' },
13
- { emit: 'stdout', level: 'warn' },
14
- { emit: 'stdout', level: 'error' },
15
- ];
16
17
  case 'test':
17
18
  return [
18
19
  { emit: 'stdout', level: 'warn' },
@@ -23,22 +24,80 @@ const getLogConfig = () => {
23
24
  }
24
25
  };
25
26
  const logConfig = getLogConfig();
26
- // ==================== 创建 Prisma 全局单例 ====================
27
- const prisma = (_a = globalForPrisma.prisma) !== null && _a !== void 0 ? _a : new PrismaClient({
28
- log: logConfig,
29
- });
30
- if (process.env.NODE_ENV !== 'production') {
31
- globalForPrisma.prisma = prisma;
27
+ function isPrismaDebugEnabled() {
28
+ return process.env.PRISMA_DEBUG === 'true';
29
+ }
30
+ function createPrismaInstanceId(prefix = 'core-prisma') {
31
+ return `${prefix}_${Date.now()}_${Math.random().toString(36).slice(2, 7)}`;
32
+ }
33
+ function createPrismaPgConfig(databaseUrl) {
34
+ const ca = process.env.SUPABASE_DB_CA_CERT;
35
+ if (!ca && !globalForPrisma.__prisma_ssl_warning_logged) {
36
+ console.warn('SUPABASE_DB_CA_CERT is not configured. Prisma will request TLS without certificate verification unless DATABASE_URL SSL parameters override this behavior. Configure SUPABASE_DB_CA_CERT for certificate verification.');
37
+ globalForPrisma.__prisma_ssl_warning_logged = true;
38
+ }
39
+ return {
40
+ connectionString: databaseUrl,
41
+ ssl: ca
42
+ ? {
43
+ ca,
44
+ rejectUnauthorized: true,
45
+ }
46
+ : {
47
+ rejectUnauthorized: false,
48
+ },
49
+ };
50
+ }
51
+ function createPrismaClient(databaseUrl = process.env.DATABASE_URL) {
52
+ if (!databaseUrl) {
53
+ throw new Error('DATABASE_URL is required to create PrismaClient');
54
+ }
55
+ const adapter = new PrismaPg(createPrismaPgConfig(databaseUrl));
56
+ const instanceId = createPrismaInstanceId();
57
+ if (isPrismaDebugEnabled()) {
58
+ console.log(`Prisma Client Created | ID: ${instanceId}`);
59
+ }
60
+ const client = new PrismaClient({
61
+ adapter,
62
+ log: logConfig,
63
+ });
64
+ return configureBackendCorePrisma(client, instanceId);
65
+ }
66
+ function configureBackendCorePrisma(prismaClient, instanceId) {
67
+ var _a;
68
+ if (instanceId === void 0) { instanceId = (_a = globalForPrisma.__prisma_instance_id) !== null && _a !== void 0 ? _a : createPrismaInstanceId(); }
69
+ globalForPrisma.prisma = prismaClient;
70
+ globalForPrisma.__prisma_instance_id = instanceId;
71
+ registerDevelopmentQueryLogger(prismaClient, instanceId);
72
+ return prismaClient;
32
73
  }
33
- if (process.env.NODE_ENV === 'development') {
74
+ function getBackendCorePrisma() {
75
+ if (!globalForPrisma.prisma) {
76
+ configureBackendCorePrisma(createPrismaClient());
77
+ }
78
+ return globalForPrisma.prisma;
79
+ }
80
+ // Backward-compatible lazy export. Accessing a property creates the client,
81
+ // importing this module does not.
82
+ const prisma = new Proxy({}, {
83
+ get(_target, property, receiver) {
84
+ return Reflect.get(getBackendCorePrisma(), property, receiver);
85
+ },
86
+ });
87
+ function registerDevelopmentQueryLogger(prismaClient, instanceId) {
88
+ var _a;
89
+ if (!isPrismaDebugEnabled()) {
90
+ return;
91
+ }
34
92
  const REGISTERED_KEY = '__prisma_query_logger_registered';
35
93
  const ID_KEY = '__prisma_query_logger_id';
36
94
  if (globalForPrisma[REGISTERED_KEY]) {
37
- console.log(`Prisma Query Logger Already Registered | ID: ${globalForPrisma[ID_KEY]}`);
95
+ console.log(`Prisma Query Logger Already Registered | Listener ID: ${globalForPrisma[ID_KEY]} | Instance ID: ${globalForPrisma.__prisma_instance_id}`);
38
96
  }
39
97
  else {
40
98
  const listenerId = `listener_${Date.now()}_${Math.random().toString(36).substr(2, 5)}`;
41
99
  globalForPrisma[ID_KEY] = listenerId;
100
+ console.log(`Prisma Query Logger Registered | Listener ID: ${listenerId} | Instance ID: ${instanceId}`);
42
101
  // --- 自定义SQL拼接 ---
43
102
  const interpolate = (query, params) => {
44
103
  // 1. 【核心修改】:安全检查和参数解析
@@ -82,7 +141,7 @@ if (process.env.NODE_ENV === 'development') {
82
141
  return sql;
83
142
  };
84
143
  const wrappedHandler = (event) => {
85
- const ms = event.duration;
144
+ const ms = Math.round(event.duration);
86
145
  const slow = ms >= 200 ? '🐌 SLOW SQL ' : '🚀 SQL';
87
146
  const interpolatedSql = interpolate(event.query, event.params);
88
147
  const clean = interpolatedSql
@@ -90,17 +149,18 @@ if (process.env.NODE_ENV === 'development') {
90
149
  .replace(/= '([^']+)'/g, `= '$1'`) // 已经替换成单引号,此处可以优化
91
150
  .replace(/"/g, ''); // 彻底灭双引号
92
151
  console.log('─'.repeat(60));
152
+ console.log(`Prisma Instance ID: ${instanceId} | Listener ID: ${listenerId}`);
93
153
  console.log(`${clean};`);
94
154
  console.log(`⏰ 耗时: ${ms}ms, ${slow}`);
95
155
  };
96
156
  // 注册包装后的 handler
97
- prisma.$on('query', wrappedHandler);
157
+ (_a = prismaClient.$on) === null || _a === void 0 ? void 0 : _a.call(prismaClient, 'query', wrappedHandler);
98
158
  globalForPrisma[REGISTERED_KEY] = true;
99
159
  }
100
160
  }
101
161
  // ==================== 便捷方法, 入参事务客户端不存在或者不传, 就返回全局非事务客户端 ====================
102
162
  function checkAndFallbackWithNonTCClient(tx) {
103
- return tx !== null && tx !== void 0 ? tx : prisma;
163
+ return tx !== null && tx !== void 0 ? tx : getBackendCorePrisma();
104
164
  }
105
165
 
106
- export { checkAndFallbackWithNonTCClient, prisma };
166
+ export { checkAndFallbackWithNonTCClient, configureBackendCorePrisma, createPrismaClient, getBackendCorePrisma, prisma };
@@ -4,11 +4,12 @@ var tslib = require('tslib');
4
4
  var user_service = require('../database/user.service.js');
5
5
  var subscription_service = require('../database/subscription.service.js');
6
6
  var credit_service = require('../database/credit.service.js');
7
- var client = require('@prisma/client');
7
+ require('../../core-prisma/client.js');
8
8
  var constants = require('../database/constants.js');
9
9
  require('../../prisma/prisma.js');
10
10
  var prismaTransactionUtil = require('../../prisma/prisma-transaction-util.js');
11
11
  var creditInit = require('../../lib/credit-init.js');
12
+ var prismaNamespace = require('../../core-prisma/internal/prismaNamespace.js');
12
13
 
13
14
  const ANONYMOUS_INIT_LOCK_NAMESPACE = 92831;
14
15
  class AnonymousAggregateService {
@@ -16,7 +17,7 @@ class AnonymousAggregateService {
16
17
  return tslib.__awaiter(this, void 0, void 0, function* () {
17
18
  yield tx.$executeRaw `
18
19
  SELECT pg_advisory_xact_lock(
19
- ${client.Prisma.raw(String(ANONYMOUS_INIT_LOCK_NAMESPACE))},
20
+ ${prismaNamespace.raw(String(ANONYMOUS_INIT_LOCK_NAMESPACE))},
20
21
  hashtext(${fingerprintId})
21
22
  )
22
23
  `;
@@ -2,11 +2,12 @@ import { __awaiter } from 'tslib';
2
2
  import { userService } from '../database/user.service.mjs';
3
3
  import { subscriptionService } from '../database/subscription.service.mjs';
4
4
  import { creditService } from '../database/credit.service.mjs';
5
- import { Prisma } from '@prisma/client';
5
+ import '../../core-prisma/client.mjs';
6
6
  import { UserStatus, OperationType, CreditType } from '../database/constants.mjs';
7
7
  import '../../prisma/prisma.mjs';
8
8
  import { runInTransaction } from '../../prisma/prisma-transaction-util.mjs';
9
9
  import { freeAmount } from '../../lib/credit-init.mjs';
10
+ import { raw } from '../../core-prisma/internal/prismaNamespace.mjs';
10
11
 
11
12
  const ANONYMOUS_INIT_LOCK_NAMESPACE = 92831;
12
13
  class AnonymousAggregateService {
@@ -14,7 +15,7 @@ class AnonymousAggregateService {
14
15
  return __awaiter(this, void 0, void 0, function* () {
15
16
  yield tx.$executeRaw `
16
17
  SELECT pg_advisory_xact_lock(
17
- ${Prisma.raw(String(ANONYMOUS_INIT_LOCK_NAMESPACE))},
18
+ ${raw(String(ANONYMOUS_INIT_LOCK_NAMESPACE))},
18
19
  hashtext(${fingerprintId})
19
20
  )
20
21
  `;
@@ -4,10 +4,10 @@ var tslib = require('tslib');
4
4
  var constants = require('../database/constants.js');
5
5
  require('../../prisma/prisma.js');
6
6
  var prismaTransactionUtil = require('../../prisma/prisma-transaction-util.js');
7
+ require('../../core-prisma/client.js');
7
8
  var subscription_service = require('../database/subscription.service.js');
8
9
  var credit_service = require('../database/credit.service.js');
9
10
  var transaction_service = require('../database/transaction.service.js');
10
- require('@prisma/client');
11
11
 
12
12
  class BillingAggregateService {
13
13
  recordSubscriptionInitPayment(context) {
@@ -2,10 +2,10 @@ import { __awaiter } from 'tslib';
2
2
  import { SubscriptionStatus, PaymentStatus, OrderStatus, TransactionType, PaySupplier, OperationType, CreditType } from '../database/constants.mjs';
3
3
  import '../../prisma/prisma.mjs';
4
4
  import { runInTransaction } from '../../prisma/prisma-transaction-util.mjs';
5
+ import '../../core-prisma/client.mjs';
5
6
  import { subscriptionService } from '../database/subscription.service.mjs';
6
7
  import { creditService } from '../database/credit.service.mjs';
7
8
  import { transactionService } from '../database/transaction.service.mjs';
8
- import '@prisma/client';
9
9
 
10
10
  class BillingAggregateService {
11
11
  recordSubscriptionInitPayment(context) {
@@ -1 +1 @@
1
- {"version":3,"file":"user.aggregate.service.d.ts","sourceRoot":"","sources":["../../../src/services/aggregate/user.aggregate.service.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,4BAA4B,CAAC;AAKvE,qBAAa,oBAAoB;IAEzB,iBAAiB,CACrB,aAAa,EAAE,MAAM,EACrB,OAAO,CAAC,EAAE;QAAE,SAAS,CAAC,EAAE,MAAM,CAAC,cAAc,CAAC;KAAE,GAC/C,OAAO,CAAC;QAAE,OAAO,EAAE,IAAI,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;KAAE,CAAC;IA6B9C;;;;;;;;;;;;OAYG;IACG,uBAAuB,CAC3B,WAAW,EAAE,MAAM,EACnB,KAAK,CAAC,EAAE,MAAM,EACd,aAAa,CAAC,EAAE,MAAM,EACtB,QAAQ,CAAC,EAAE,MAAM,EACjB,SAAS,CAAC,EAAE,MAAM,CAAC,cAAc,GAChC,OAAO,CAAC;QAAE,OAAO,EAAE,IAAI,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;KAAE,CAAC;IA+BxC,mBAAmB,CACvB,MAAM,EAAE,MAAM,EACd,KAAK,EAAE,MAAM,EACb,WAAW,EAAE,MAAM,EACnB,QAAQ,CAAC,EAAE,MAAM,GAChB,OAAO,CAAC;QAAE,UAAU,EAAE,IAAI,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;KAAE,CAAC;IA+B3C,oBAAoB,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC;YAwBzD,qBAAqB;CAcpC;AAED,eAAO,MAAM,oBAAoB,sBAA6B,CAAC"}
1
+ {"version":3,"file":"user.aggregate.service.d.ts","sourceRoot":"","sources":["../../../src/services/aggregate/user.aggregate.service.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,4BAA4B,CAAC;AAKvE,qBAAa,oBAAoB;IAEzB,iBAAiB,CACrB,aAAa,EAAE,MAAM,EACrB,OAAO,CAAC,EAAE;QAAE,SAAS,CAAC,EAAE,MAAM,CAAC,cAAc,CAAC;KAAE,GAC/C,OAAO,CAAC;QAAE,OAAO,EAAE,IAAI,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;KAAE,CAAC;IA6B9C;;;;;;;;;;;;OAYG;IACG,uBAAuB,CAC3B,WAAW,EAAE,MAAM,EACnB,KAAK,CAAC,EAAE,MAAM,EACd,aAAa,CAAC,EAAE,MAAM,EACtB,QAAQ,CAAC,EAAE,MAAM,EACjB,SAAS,CAAC,EAAE,MAAM,CAAC,cAAc,GAChC,OAAO,CAAC;QAAE,OAAO,EAAE,IAAI,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;KAAE,CAAC;IAgCxC,mBAAmB,CACvB,MAAM,EAAE,MAAM,EACd,KAAK,EAAE,MAAM,EACb,WAAW,EAAE,MAAM,EACnB,QAAQ,CAAC,EAAE,MAAM,GAChB,OAAO,CAAC;QAAE,UAAU,EAAE,IAAI,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;KAAE,CAAC;IA+B3C,oBAAoB,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC;YAwBzD,qBAAqB;CAcpC;AAED,eAAO,MAAM,oBAAoB,sBAA6B,CAAC"}
@@ -5,7 +5,7 @@ var constants = require('../database/constants.js');
5
5
  var user_service = require('../database/user.service.js');
6
6
  var subscription_service = require('../database/subscription.service.js');
7
7
  var credit_service = require('../database/credit.service.js');
8
- require('@prisma/client');
8
+ require('../../core-prisma/client.js');
9
9
  require('../../prisma/prisma.js');
10
10
  var prismaTransactionUtil = require('../../prisma/prisma-transaction-util.js');
11
11
  var creditInit = require('../../lib/credit-init.js');
@@ -63,7 +63,7 @@ class UserAggregateService {
63
63
  operationType: constants.OperationType.SYS_GIFT,
64
64
  operationReferId: newUser.userId,
65
65
  creditsChange: creditInit.freeRegisterAmount,
66
- });
66
+ }, tx);
67
67
  yield subscription_service.subscriptionService.initializeSubscription(newUser.userId, tx);
68
68
  return { newUser, credit };
69
69
  }));
@@ -104,7 +104,7 @@ class UserAggregateService {
104
104
  }
105
105
  const userId = user.userId;
106
106
  // 更改用户状态,保留user信息尤其是FingerprintId,防止反复注册薅羊毛
107
- yield user_service.userService.unregister(user.userId);
107
+ yield user_service.userService.unregister(user.userId, tx);
108
108
  // 清空积分
109
109
  yield credit_service.creditService.purgeCredit(userId, 'soft_delete_user', userId, tx);
110
110
  const subscription = yield subscription_service.subscriptionService.getActiveSubscription(userId, tx);
@@ -3,7 +3,7 @@ import { UserStatus, OperationType, CreditType } from '../database/constants.mjs
3
3
  import { userService } from '../database/user.service.mjs';
4
4
  import { subscriptionService } from '../database/subscription.service.mjs';
5
5
  import { creditService } from '../database/credit.service.mjs';
6
- import '@prisma/client';
6
+ import '../../core-prisma/client.mjs';
7
7
  import '../../prisma/prisma.mjs';
8
8
  import { runInTransaction } from '../../prisma/prisma-transaction-util.mjs';
9
9
  import { freeAmount, freeRegisterAmount } from '../../lib/credit-init.mjs';
@@ -61,7 +61,7 @@ class UserAggregateService {
61
61
  operationType: OperationType.SYS_GIFT,
62
62
  operationReferId: newUser.userId,
63
63
  creditsChange: freeRegisterAmount,
64
- });
64
+ }, tx);
65
65
  yield subscriptionService.initializeSubscription(newUser.userId, tx);
66
66
  return { newUser, credit };
67
67
  }));
@@ -102,7 +102,7 @@ class UserAggregateService {
102
102
  }
103
103
  const userId = user.userId;
104
104
  // 更改用户状态,保留user信息尤其是FingerprintId,防止反复注册薅羊毛
105
- yield userService.unregister(user.userId);
105
+ yield userService.unregister(user.userId, tx);
106
106
  // 清空积分
107
107
  yield creditService.purgeCredit(userId, 'soft_delete_user', userId, tx);
108
108
  const subscription = yield subscriptionService.getActiveSubscription(userId, tx);
@@ -11,4 +11,5 @@ exports.fetchUserContextByClerkUserId = userContextService.fetchUserContextByCle
11
11
  exports.mapCreditToXCredit = userContextService.mapCreditToXCredit;
12
12
  exports.mapSubscriptionToXSubscription = userContextService.mapSubscriptionToXSubscription;
13
13
  exports.mapUserToXUser = userContextService.mapUserToXUser;
14
+ exports.applyMoneyPriceMockUserDelay = userContextFinalizer.applyMoneyPriceMockUserDelay;
14
15
  exports.finalizeUserContext = userContextFinalizer.finalizeUserContext;
@@ -1,2 +1,2 @@
1
1
  export { buildInitUserContextFromEntities, fetchLatestUserContextByFingerprintId, fetchUserContextByClerkUserId, mapCreditToXCredit, mapSubscriptionToXSubscription, mapUserToXUser } from './user-context-service.mjs';
2
- export { finalizeUserContext } from './user-context-finalizer.mjs';
2
+ export { applyMoneyPriceMockUserDelay, finalizeUserContext } from './user-context-finalizer.mjs';
@@ -3,6 +3,7 @@ type FinalizableUserContext = {
3
3
  xUser: XUser | null;
4
4
  xSubscription: XSubscription | null;
5
5
  };
6
+ export declare function applyMoneyPriceMockUserDelay(): Promise<void>;
6
7
  /**
7
8
  * Output finalizer for user-context payloads.
8
9
  * Real data assembly should stay in user-context-service; any optional test-only
@@ -1 +1 @@
1
- {"version":3,"file":"user-context-finalizer.d.ts","sourceRoot":"","sources":["../../../src/services/context/user-context-finalizer.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,aAAa,EAAE,KAAK,EAAE,MAAM,sCAAsC,CAAC;AAEjF,KAAK,sBAAsB,GAAG;IAC5B,KAAK,EAAE,KAAK,GAAG,IAAI,CAAC;IACpB,aAAa,EAAE,aAAa,GAAG,IAAI,CAAC;CACrC,CAAC;AAEF;;;;GAIG;AACH,wBAAgB,mBAAmB,CAAC,CAAC,SAAS,sBAAsB,EAAE,OAAO,EAAE,CAAC,GAAG,CAAC,CAsEnF"}
1
+ {"version":3,"file":"user-context-finalizer.d.ts","sourceRoot":"","sources":["../../../src/services/context/user-context-finalizer.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,aAAa,EAAE,KAAK,EAAE,MAAM,sCAAsC,CAAC;AAEjF,KAAK,sBAAsB,GAAG;IAC5B,KAAK,EAAE,KAAK,GAAG,IAAI,CAAC;IACpB,aAAa,EAAE,aAAa,GAAG,IAAI,CAAC;CACrC,CAAC;AAIF,wBAAsB,4BAA4B,IAAI,OAAO,CAAC,IAAI,CAAC,CAYlE;AAED;;;;GAIG;AACH,wBAAgB,mBAAmB,CAAC,CAAC,SAAS,sBAAsB,EAAE,OAAO,EAAE,CAAC,GAAG,CAAC,CAsEnF"}
@@ -1,7 +1,23 @@
1
1
  'use strict';
2
2
 
3
+ var tslib = require('tslib');
3
4
  var utils = require('@windrun-huaiin/lib/utils');
4
5
 
6
+ const MAX_MONEY_PRICE_MOCK_DELAY_MS = 10000;
7
+ function applyMoneyPriceMockUserDelay() {
8
+ return tslib.__awaiter(this, void 0, void 0, function* () {
9
+ var _a;
10
+ if (process.env.MONEY_PRICE_MOCK_USER_ENABLED !== 'true') {
11
+ return;
12
+ }
13
+ const delaySeconds = Number((_a = process.env.MONEY_PRICE_MOCK_USER_DELAY_SECONDS) !== null && _a !== void 0 ? _a : 0);
14
+ if (!Number.isFinite(delaySeconds) || delaySeconds <= 0) {
15
+ return;
16
+ }
17
+ const delayMs = Math.min(Math.floor(delaySeconds * 1000), MAX_MONEY_PRICE_MOCK_DELAY_MS);
18
+ yield new Promise((resolve) => setTimeout(resolve, delayMs));
19
+ });
20
+ }
5
21
  /**
6
22
  * Output finalizer for user-context payloads.
7
23
  * Real data assembly should stay in user-context-service; any optional test-only
@@ -71,4 +87,5 @@ function finalizeUserContext(context) {
71
87
  return context;
72
88
  }
73
89
 
90
+ exports.applyMoneyPriceMockUserDelay = applyMoneyPriceMockUserDelay;
74
91
  exports.finalizeUserContext = finalizeUserContext;
@@ -1,5 +1,21 @@
1
+ import { __awaiter } from 'tslib';
1
2
  import { viewLocalTime } from '@windrun-huaiin/lib/utils';
2
3
 
4
+ const MAX_MONEY_PRICE_MOCK_DELAY_MS = 10000;
5
+ function applyMoneyPriceMockUserDelay() {
6
+ return __awaiter(this, void 0, void 0, function* () {
7
+ var _a;
8
+ if (process.env.MONEY_PRICE_MOCK_USER_ENABLED !== 'true') {
9
+ return;
10
+ }
11
+ const delaySeconds = Number((_a = process.env.MONEY_PRICE_MOCK_USER_DELAY_SECONDS) !== null && _a !== void 0 ? _a : 0);
12
+ if (!Number.isFinite(delaySeconds) || delaySeconds <= 0) {
13
+ return;
14
+ }
15
+ const delayMs = Math.min(Math.floor(delaySeconds * 1000), MAX_MONEY_PRICE_MOCK_DELAY_MS);
16
+ yield new Promise((resolve) => setTimeout(resolve, delayMs));
17
+ });
18
+ }
3
19
  /**
4
20
  * Output finalizer for user-context payloads.
5
21
  * Real data assembly should stay in user-context-service; any optional test-only
@@ -69,4 +85,4 @@ function finalizeUserContext(context) {
69
85
  return context;
70
86
  }
71
87
 
72
- export { finalizeUserContext };
88
+ export { applyMoneyPriceMockUserDelay, finalizeUserContext };
@@ -4,7 +4,7 @@ var tslib = require('tslib');
4
4
  var user_service = require('../database/user.service.js');
5
5
  var subscription_service = require('../database/subscription.service.js');
6
6
  var credit_service = require('../database/credit.service.js');
7
- require('@prisma/client');
7
+ require('../../core-prisma/client.js');
8
8
  require('../../prisma/prisma.js');
9
9
  var utils = require('@windrun-huaiin/lib/utils');
10
10
 
@@ -2,7 +2,7 @@ import { __awaiter } from 'tslib';
2
2
  import { userService } from '../database/user.service.mjs';
3
3
  import { subscriptionService } from '../database/subscription.service.mjs';
4
4
  import { creditService } from '../database/credit.service.mjs';
5
- import '@prisma/client';
5
+ import '../../core-prisma/client.mjs';
6
6
  import '../../prisma/prisma.mjs';
7
7
  import { viewLocalTime } from '@windrun-huaiin/lib/utils';
8
8
 
@@ -2,6 +2,7 @@
2
2
 
3
3
  var tslib = require('tslib');
4
4
  var prisma = require('../../prisma/prisma.js');
5
+ require('../../core-prisma/client.js');
5
6
 
6
7
  /* eslint-disable @typescript-eslint/no-explicit-any */
7
8
  class ApilogService {
@@ -1,5 +1,6 @@
1
1
  import { __awaiter } from 'tslib';
2
2
  import { checkAndFallbackWithNonTCClient } from '../../prisma/prisma.mjs';
3
+ import '../../core-prisma/client.mjs';
3
4
 
4
5
  /* eslint-disable @typescript-eslint/no-explicit-any */
5
6
  class ApilogService {
@@ -1,11 +1,12 @@
1
1
  'use strict';
2
2
 
3
3
  var tslib = require('tslib');
4
- var client = require('@prisma/client');
4
+ require('../../core-prisma/client.js');
5
5
  var constants = require('./constants.js');
6
6
  var creditInit = require('../../lib/credit-init.js');
7
7
  var prisma = require('../../prisma/prisma.js');
8
8
  var creditAuditLog_service = require('./creditAuditLog.service.js');
9
+ var prismaNamespace = require('../../core-prisma/internal/prismaNamespace.js');
9
10
 
10
11
  const CREDIT_PURGE_CONFIG = {
11
12
  [constants.CreditType.FREE]: {
@@ -454,13 +455,13 @@ class CreditService {
454
455
  // Get Users with Low Credit Balance
455
456
  getLowBalanceUsers() {
456
457
  return tslib.__awaiter(this, arguments, void 0, function* (threshold = 10, tx) {
457
- const client$1 = prisma.checkAndFallbackWithNonTCClient(tx);
458
- const query = client.Prisma.sql `
458
+ const client = prisma.checkAndFallbackWithNonTCClient(tx);
459
+ const query = prismaNamespace.sql `
459
460
  SELECT * FROM credits
460
461
  WHERE (balance_free + balance_paid + balance_onetime_paid) < ${threshold}
461
462
  ORDER BY (balance_free + balance_paid + balance_onetime_paid) ASC
462
463
  `;
463
- return yield client$1.$queryRaw(query);
464
+ return yield client.$queryRaw(query);
464
465
  });
465
466
  }
466
467
  // Get Credit Statistics
@@ -1,9 +1,10 @@
1
1
  import { __awaiter } from 'tslib';
2
- import { Prisma } from '@prisma/client';
2
+ import '../../core-prisma/client.mjs';
3
3
  import { CreditType, OperationType } from './constants.mjs';
4
4
  import { freeExpiredDays } from '../../lib/credit-init.mjs';
5
5
  import { checkAndFallbackWithNonTCClient } from '../../prisma/prisma.mjs';
6
6
  import { creditAuditLogService } from './creditAuditLog.service.mjs';
7
+ import { sql } from '../../core-prisma/internal/prismaNamespace.mjs';
7
8
 
8
9
  const CREDIT_PURGE_CONFIG = {
9
10
  [CreditType.FREE]: {
@@ -453,7 +454,7 @@ class CreditService {
453
454
  getLowBalanceUsers() {
454
455
  return __awaiter(this, arguments, void 0, function* (threshold = 10, tx) {
455
456
  const client = checkAndFallbackWithNonTCClient(tx);
456
- const query = Prisma.sql `
457
+ const query = sql `
457
458
  SELECT * FROM credits
458
459
  WHERE (balance_free + balance_paid + balance_onetime_paid) < ${threshold}
459
460
  ORDER BY (balance_free + balance_paid + balance_onetime_paid) ASC
@@ -1,9 +1,10 @@
1
1
  'use strict';
2
2
 
3
3
  var tslib = require('tslib');
4
- var client = require('@prisma/client');
4
+ require('../../core-prisma/client.js');
5
5
  var constants = require('./constants.js');
6
6
  var prisma = require('../../prisma/prisma.js');
7
+ var prismaNamespace = require('../../core-prisma/internal/prismaNamespace.js');
7
8
 
8
9
  /* eslint-disable @typescript-eslint/no-explicit-any */
9
10
  class CreditAuditLogService {
@@ -188,9 +189,9 @@ class CreditAuditLogService {
188
189
  return tslib.__awaiter(this, arguments, void 0, function* (days = 30, userId, tx) {
189
190
  const startDate = new Date();
190
191
  startDate.setDate(startDate.getDate() - days);
191
- const client$1 = prisma.checkAndFallbackWithNonTCClient(tx);
192
- const userFilter = userId ? client.Prisma.sql `AND user_id = ${userId}` : client.Prisma.sql ``;
193
- const result = yield client$1.$queryRaw `
192
+ const client = prisma.checkAndFallbackWithNonTCClient(tx);
193
+ const userFilter = userId ? prismaNamespace.sql `AND user_id = ${userId}` : prismaNamespace.sql ``;
194
+ const result = yield client.$queryRaw `
194
195
  SELECT
195
196
  DATE(created_at) as date,
196
197
  SUM(CASE WHEN operation_type = 'consume' THEN credits_used ELSE 0 END) as consumed,