mvc-common-toolkit 1.43.10 → 1.44.0

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 (210) hide show
  1. package/dist/src/constants.d.ts +59 -0
  2. package/dist/src/constants.js +69 -0
  3. package/dist/src/constants.js.map +1 -0
  4. package/dist/src/gateways/alibaba-cloud-gateway.d.ts +30 -0
  5. package/dist/src/gateways/alibaba-cloud-gateway.js +120 -0
  6. package/dist/src/gateways/alibaba-cloud-gateway.js.map +1 -0
  7. package/dist/src/gateways/http-audit-gateway.d.ts +20 -0
  8. package/dist/src/gateways/http-audit-gateway.js +76 -0
  9. package/dist/src/gateways/http-audit-gateway.js.map +1 -0
  10. package/dist/src/gateways/index.d.ts +5 -0
  11. package/dist/src/gateways/index.js +22 -0
  12. package/dist/src/gateways/index.js.map +1 -0
  13. package/dist/src/gateways/internal-auth-gateway.d.ts +8 -0
  14. package/dist/src/gateways/internal-auth-gateway.js +40 -0
  15. package/dist/src/gateways/internal-auth-gateway.js.map +1 -0
  16. package/dist/src/gateways/stdout-audit-gateway.d.ts +7 -0
  17. package/dist/src/gateways/stdout-audit-gateway.js +25 -0
  18. package/dist/src/gateways/stdout-audit-gateway.js.map +1 -0
  19. package/dist/src/gateways/webhook-audit-gateway.d.ts +14 -0
  20. package/dist/src/gateways/webhook-audit-gateway.js +27 -0
  21. package/dist/src/gateways/webhook-audit-gateway.js.map +1 -0
  22. package/dist/src/interfaces.d.ts +218 -0
  23. package/dist/src/interfaces.js +3 -0
  24. package/dist/src/interfaces.js.map +1 -0
  25. package/dist/src/models/audit-log.d.ts +77 -0
  26. package/dist/src/models/audit-log.js +92 -0
  27. package/dist/src/models/audit-log.js.map +1 -0
  28. package/dist/src/models/index.d.ts +1 -0
  29. package/dist/src/models/index.js +18 -0
  30. package/dist/src/models/index.js.map +1 -0
  31. package/dist/src/pkg/array-helper.d.ts +1 -0
  32. package/dist/src/pkg/array-helper.js +12 -0
  33. package/dist/src/pkg/array-helper.js.map +1 -0
  34. package/dist/src/pkg/bcrypt-helper.d.ts +2 -0
  35. package/dist/src/pkg/bcrypt-helper.js +36 -0
  36. package/dist/src/pkg/bcrypt-helper.js.map +1 -0
  37. package/dist/src/pkg/crypto-helper.d.ts +2 -0
  38. package/dist/src/pkg/crypto-helper.js +16 -0
  39. package/dist/src/pkg/crypto-helper.js.map +1 -0
  40. package/dist/src/pkg/encryption-helper.d.ts +18 -0
  41. package/dist/src/pkg/encryption-helper.js +89 -0
  42. package/dist/src/pkg/encryption-helper.js.map +1 -0
  43. package/dist/src/pkg/encryption-helper.spec.d.ts +1 -0
  44. package/dist/src/pkg/encryption-helper.spec.js +238 -0
  45. package/dist/src/pkg/encryption-helper.spec.js.map +1 -0
  46. package/dist/src/pkg/filter-helper.d.ts +2 -0
  47. package/dist/src/pkg/filter-helper.js +102 -0
  48. package/dist/src/pkg/filter-helper.js.map +1 -0
  49. package/dist/src/pkg/filter-helper.spec.d.ts +1 -0
  50. package/dist/src/pkg/filter-helper.spec.js +94 -0
  51. package/dist/src/pkg/filter-helper.spec.js.map +1 -0
  52. package/dist/src/pkg/geoip-helper.d.ts +2 -0
  53. package/dist/src/pkg/geoip-helper.js +32 -0
  54. package/dist/src/pkg/geoip-helper.js.map +1 -0
  55. package/dist/src/pkg/hash-helper.d.ts +1 -0
  56. package/dist/src/pkg/hash-helper.js +37 -0
  57. package/dist/src/pkg/hash-helper.js.map +1 -0
  58. package/dist/src/pkg/http-request-utils.d.ts +4 -0
  59. package/dist/src/pkg/http-request-utils.js +55 -0
  60. package/dist/src/pkg/http-request-utils.js.map +1 -0
  61. package/dist/src/pkg/index.d.ts +19 -0
  62. package/dist/src/pkg/index.js +46 -0
  63. package/dist/src/pkg/index.js.map +1 -0
  64. package/dist/src/pkg/key-helper.d.ts +2 -0
  65. package/dist/src/pkg/key-helper.js +20 -0
  66. package/dist/src/pkg/key-helper.js.map +1 -0
  67. package/dist/src/pkg/logger.d.ts +9 -0
  68. package/dist/src/pkg/logger.js +23 -0
  69. package/dist/src/pkg/logger.js.map +1 -0
  70. package/dist/src/pkg/object-helper.d.ts +2 -0
  71. package/dist/src/pkg/object-helper.js +37 -0
  72. package/dist/src/pkg/object-helper.js.map +1 -0
  73. package/dist/src/pkg/paginated-cache-registry.d.ts +8 -0
  74. package/dist/src/pkg/paginated-cache-registry.js +23 -0
  75. package/dist/src/pkg/paginated-cache-registry.js.map +1 -0
  76. package/dist/src/pkg/query-helper.d.ts +3 -0
  77. package/dist/src/pkg/query-helper.js +60 -0
  78. package/dist/src/pkg/query-helper.js.map +1 -0
  79. package/dist/src/pkg/referral-tree-utils.d.ts +33 -0
  80. package/dist/src/pkg/referral-tree-utils.js +71 -0
  81. package/dist/src/pkg/referral-tree-utils.js.map +1 -0
  82. package/dist/src/pkg/scripts/index.d.ts +1 -0
  83. package/dist/src/pkg/scripts/index.js +28 -0
  84. package/dist/src/pkg/scripts/index.js.map +1 -0
  85. package/dist/src/pkg/scripts/lua.d.ts +10 -0
  86. package/dist/src/pkg/scripts/lua.js +109 -0
  87. package/dist/src/pkg/scripts/lua.js.map +1 -0
  88. package/dist/src/pkg/sort-helper.d.ts +3 -0
  89. package/dist/src/pkg/sort-helper.js +18 -0
  90. package/dist/src/pkg/sort-helper.js.map +1 -0
  91. package/dist/src/pkg/string-utils.d.ts +10 -0
  92. package/dist/src/pkg/string-utils.js +79 -0
  93. package/dist/src/pkg/string-utils.js.map +1 -0
  94. package/dist/src/pkg/task-helper.d.ts +2 -0
  95. package/dist/src/pkg/task-helper.js +30 -0
  96. package/dist/src/pkg/task-helper.js.map +1 -0
  97. package/dist/src/pkg/trading-pair-helper.d.ts +9 -0
  98. package/dist/src/pkg/trading-pair-helper.js +44 -0
  99. package/dist/src/pkg/trading-pair-helper.js.map +1 -0
  100. package/dist/src/pkg/trading-pair-helper.spec.d.ts +1 -0
  101. package/dist/src/pkg/trading-pair-helper.spec.js +132 -0
  102. package/dist/src/pkg/trading-pair-helper.spec.js.map +1 -0
  103. package/dist/src/pkg/workflow/delayed-task-registry.d.ts +10 -0
  104. package/dist/src/pkg/workflow/delayed-task-registry.js +67 -0
  105. package/dist/src/pkg/workflow/delayed-task-registry.js.map +1 -0
  106. package/dist/src/pkg/workflow/delayed-task.d.ts +18 -0
  107. package/dist/src/pkg/workflow/delayed-task.js +95 -0
  108. package/dist/src/pkg/workflow/delayed-task.js.map +1 -0
  109. package/dist/src/pkg/workflow/index.d.ts +5 -0
  110. package/dist/src/pkg/workflow/index.js +22 -0
  111. package/dist/src/pkg/workflow/index.js.map +1 -0
  112. package/dist/src/pkg/workflow/processing-milestone.d.ts +18 -0
  113. package/dist/src/pkg/workflow/processing-milestone.js +39 -0
  114. package/dist/src/pkg/workflow/processing-milestone.js.map +1 -0
  115. package/dist/src/pkg/workflow/retry-task.d.ts +24 -0
  116. package/dist/src/pkg/workflow/retry-task.js +89 -0
  117. package/dist/src/pkg/workflow/retry-task.js.map +1 -0
  118. package/dist/src/pkg/workflow/retry-task.spec.d.ts +1 -0
  119. package/dist/src/pkg/workflow/retry-task.spec.js +145 -0
  120. package/dist/src/pkg/workflow/retry-task.spec.js.map +1 -0
  121. package/dist/src/pkg/workflow/sync-taskqueue.d.ts +32 -0
  122. package/dist/src/pkg/workflow/sync-taskqueue.js +108 -0
  123. package/dist/src/pkg/workflow/sync-taskqueue.js.map +1 -0
  124. package/dist/src/pkg/worksheet.utils.d.ts +27 -0
  125. package/dist/src/pkg/worksheet.utils.js +116 -0
  126. package/dist/src/pkg/worksheet.utils.js.map +1 -0
  127. package/dist/src/services/audit-service.d.ts +7 -0
  128. package/dist/src/services/audit-service.js +32 -0
  129. package/dist/src/services/audit-service.js.map +1 -0
  130. package/dist/src/services/excel.service.d.ts +25 -0
  131. package/dist/src/services/excel.service.js +95 -0
  132. package/dist/src/services/excel.service.js.map +1 -0
  133. package/dist/src/services/http-service.d.ts +7 -0
  134. package/dist/src/services/http-service.js +67 -0
  135. package/dist/src/services/http-service.js.map +1 -0
  136. package/dist/src/services/index.d.ts +8 -0
  137. package/dist/src/services/index.js +25 -0
  138. package/dist/src/services/index.js.map +1 -0
  139. package/dist/src/services/kafka-service.d.ts +15 -0
  140. package/dist/src/services/kafka-service.js +68 -0
  141. package/dist/src/services/kafka-service.js.map +1 -0
  142. package/dist/src/services/mailer-service.d.ts +15 -0
  143. package/dist/src/services/mailer-service.js +44 -0
  144. package/dist/src/services/mailer-service.js.map +1 -0
  145. package/dist/src/services/paginated-cache.d.ts +16 -0
  146. package/dist/src/services/paginated-cache.js +115 -0
  147. package/dist/src/services/paginated-cache.js.map +1 -0
  148. package/dist/src/services/paginated-cache.spec.d.ts +1 -0
  149. package/dist/src/services/paginated-cache.spec.js +284 -0
  150. package/dist/src/services/paginated-cache.spec.js.map +1 -0
  151. package/dist/src/services/redis-service.d.ts +33 -0
  152. package/dist/src/services/redis-service.js +230 -0
  153. package/dist/src/services/redis-service.js.map +1 -0
  154. package/dist/src/services/security-service.d.ts +11 -0
  155. package/dist/src/services/security-service.js +68 -0
  156. package/dist/src/services/security-service.js.map +1 -0
  157. package/dist/tsconfig.tsbuildinfo +1 -1
  158. package/package.json +1 -1
  159. package/src/constants.ts +66 -0
  160. package/src/gateways/alibaba-cloud-gateway.ts +127 -0
  161. package/src/gateways/http-audit-gateway.ts +104 -0
  162. package/src/gateways/index.ts +5 -0
  163. package/src/gateways/internal-auth-gateway.ts +42 -0
  164. package/src/gateways/stdout-audit-gateway.ts +23 -0
  165. package/src/gateways/webhook-audit-gateway.ts +33 -0
  166. package/src/interfaces.ts +304 -0
  167. package/src/models/audit-log.ts +126 -0
  168. package/src/models/index.ts +1 -0
  169. package/src/pkg/array-helper.ts +7 -0
  170. package/src/pkg/bcrypt-helper.ts +9 -0
  171. package/src/pkg/crypto-helper.ts +18 -0
  172. package/src/pkg/encryption-helper.spec.ts +423 -0
  173. package/src/pkg/encryption-helper.ts +155 -0
  174. package/src/pkg/filter-helper.spec.ts +105 -0
  175. package/src/pkg/filter-helper.ts +139 -0
  176. package/src/pkg/geoip-helper.ts +5 -0
  177. package/src/pkg/hash-helper.ts +12 -0
  178. package/src/pkg/http-request-utils.ts +75 -0
  179. package/src/pkg/index.ts +19 -0
  180. package/src/pkg/key-helper.ts +20 -0
  181. package/src/pkg/logger.ts +23 -0
  182. package/src/pkg/object-helper.ts +42 -0
  183. package/src/pkg/paginated-cache-registry.ts +25 -0
  184. package/src/pkg/query-helper.ts +79 -0
  185. package/src/pkg/referral-tree-utils.ts +165 -0
  186. package/src/pkg/scripts/index.ts +1 -0
  187. package/src/pkg/scripts/lua.ts +112 -0
  188. package/src/pkg/sort-helper.ts +19 -0
  189. package/src/pkg/string-utils.ts +104 -0
  190. package/src/pkg/task-helper.ts +25 -0
  191. package/src/pkg/trading-pair-helper.spec.ts +146 -0
  192. package/src/pkg/trading-pair-helper.ts +78 -0
  193. package/src/pkg/workflow/delayed-task-registry.ts +54 -0
  194. package/src/pkg/workflow/delayed-task.ts +106 -0
  195. package/src/pkg/workflow/index.ts +5 -0
  196. package/src/pkg/workflow/processing-milestone.ts +54 -0
  197. package/src/pkg/workflow/retry-task.spec.ts +194 -0
  198. package/src/pkg/workflow/retry-task.ts +119 -0
  199. package/src/pkg/workflow/sync-taskqueue.ts +118 -0
  200. package/src/pkg/worksheet.utils.ts +178 -0
  201. package/src/services/audit-service.ts +22 -0
  202. package/src/services/excel.service.ts +103 -0
  203. package/src/services/http-service.ts +71 -0
  204. package/src/services/index.ts +8 -0
  205. package/src/services/kafka-service.ts +81 -0
  206. package/src/services/mailer-service.ts +43 -0
  207. package/src/services/paginated-cache.spec.ts +519 -0
  208. package/src/services/paginated-cache.ts +122 -0
  209. package/src/services/redis-service.ts +238 -0
  210. package/src/services/security-service.ts +80 -0
@@ -0,0 +1,79 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.maskFn = exports.validatePasswordStrengthWithMessage = exports.generatePassword = exports.generateRandomId = void 0;
4
+ const cuid2_1 = require("@paralleldrive/cuid2");
5
+ function generateRandomId() {
6
+ return (0, cuid2_1.createId)();
7
+ }
8
+ exports.generateRandomId = generateRandomId;
9
+ function generatePassword(length = 16) {
10
+ const chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
11
+ const numbers = "0123456789";
12
+ const symbols = "!@#$%^&*(){}|+-";
13
+ const all = chars + numbers + symbols;
14
+ let retVal = "";
15
+ retVal += chars.charAt(Math.floor(Math.random() * chars.length));
16
+ retVal += numbers.charAt(Math.floor(Math.random() * numbers.length));
17
+ retVal += symbols.charAt(Math.floor(Math.random() * symbols.length));
18
+ for (let i = 0; i < length - 3; ++i) {
19
+ retVal += all.charAt(Math.floor(Math.random() * all.length));
20
+ }
21
+ return retVal;
22
+ }
23
+ exports.generatePassword = generatePassword;
24
+ function escapeRegexCharacters(str) {
25
+ return str.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
26
+ }
27
+ function validatePasswordStrengthWithMessage(password) {
28
+ const escapedPassword = escapeRegexCharacters(password);
29
+ if (escapedPassword.length < 8) {
30
+ return "Password must be at least 8 characters long";
31
+ }
32
+ if (!escapedPassword.match(/[a-z]/)) {
33
+ return "Password must contain at least one lowercase letter";
34
+ }
35
+ if (!escapedPassword.match(/[A-Z]/)) {
36
+ return "Password must contain at least one uppercase letter";
37
+ }
38
+ if (!escapedPassword.match(/[0-9]/)) {
39
+ return "Password must contain at least one number";
40
+ }
41
+ if (!escapedPassword.match(/[^a-zA-Z0-9]/)) {
42
+ return "Password must contain at least one special character";
43
+ }
44
+ return "";
45
+ }
46
+ exports.validatePasswordStrengthWithMessage = validatePasswordStrengthWithMessage;
47
+ const standardlize = (f1) => f1.replace(new RegExp("_", "g"), "").toLowerCase();
48
+ const maskRules = [
49
+ {
50
+ keys: [
51
+ "password",
52
+ "Authorization",
53
+ "access_token",
54
+ "refresh_token",
55
+ "signature",
56
+ ],
57
+ pattern: /^(.*)$/,
58
+ replacer: () => "***masked***",
59
+ },
60
+ {
61
+ keys: ["txid", "txnid"],
62
+ pattern: /^(.{10})(.*)(.{36})$/,
63
+ replacer: (_match, prefix, middle, suffix) => `${prefix}${"*".repeat(middle.length)}${suffix}`,
64
+ },
65
+ ];
66
+ const maskFn = (key, value, additionalMaskRules = []) => {
67
+ if (typeof value !== "string") {
68
+ return value;
69
+ }
70
+ const normalizedKey = standardlize(key);
71
+ const allRules = [...maskRules, ...additionalMaskRules];
72
+ const rule = allRules.find(({ keys }) => keys.some((k) => standardlize(k) === normalizedKey));
73
+ if (!rule) {
74
+ return value;
75
+ }
76
+ return value.replace(rule.pattern, rule.replacer);
77
+ };
78
+ exports.maskFn = maskFn;
79
+ //# sourceMappingURL=string-utils.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"string-utils.js","sourceRoot":"","sources":["../../../src/pkg/string-utils.ts"],"names":[],"mappings":";;;AAAA,gDAAgD;AAEhD,SAAgB,gBAAgB;IAC9B,OAAO,IAAA,gBAAQ,GAAE,CAAC;AACpB,CAAC;AAFD,4CAEC;AAED,SAAgB,gBAAgB,CAAC,MAAM,GAAG,EAAE;IAC1C,MAAM,KAAK,GAAG,sDAAsD,CAAC;IACrE,MAAM,OAAO,GAAG,YAAY,CAAC;IAC7B,MAAM,OAAO,GAAG,iBAAiB,CAAC;IAClC,MAAM,GAAG,GAAG,KAAK,GAAG,OAAO,GAAG,OAAO,CAAC;IAEtC,IAAI,MAAM,GAAG,EAAE,CAAC;IAChB,MAAM,IAAI,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC;IACjE,MAAM,IAAI,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC;IACrE,MAAM,IAAI,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC;IAErE,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,MAAM,GAAG,CAAC,EAAE,EAAE,CAAC,EAAE;QACnC,MAAM,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC;KAC9D;IACD,OAAO,MAAM,CAAC;AAChB,CAAC;AAfD,4CAeC;AAED,SAAS,qBAAqB,CAAC,GAAW;IACxC,OAAO,GAAG,CAAC,OAAO,CAAC,qBAAqB,EAAE,MAAM,CAAC,CAAC;AACpD,CAAC;AAED,SAAgB,mCAAmC,CAAC,QAAgB;IAClE,MAAM,eAAe,GAAG,qBAAqB,CAAC,QAAQ,CAAC,CAAC;IAExD,IAAI,eAAe,CAAC,MAAM,GAAG,CAAC,EAAE;QAC9B,OAAO,6CAA6C,CAAC;KACtD;IACD,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE;QACnC,OAAO,qDAAqD,CAAC;KAC9D;IACD,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE;QACnC,OAAO,qDAAqD,CAAC;KAC9D;IACD,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE;QACnC,OAAO,2CAA2C,CAAC;KACpD;IACD,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,cAAc,CAAC,EAAE;QAC1C,OAAO,sDAAsD,CAAC;KAC/D;IACD,OAAO,EAAE,CAAC;AACZ,CAAC;AAnBD,kFAmBC;AAED,MAAM,YAAY,GAAG,CAAC,EAAU,EAAE,EAAE,CAClC,EAAE,CAAC,OAAO,CAAC,IAAI,MAAM,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,EAAE,CAAC,CAAC,WAAW,EAAE,CAAC;AAQrD,MAAM,SAAS,GAAe;IAC5B;QACE,IAAI,EAAE;YACJ,UAAU;YACV,eAAe;YACf,cAAc;YACd,eAAe;YACf,WAAW;SACZ;QACD,OAAO,EAAE,QAAQ;QACjB,QAAQ,EAAE,GAAG,EAAE,CAAC,cAAc;KAC/B;IACD;QACE,IAAI,EAAE,CAAC,MAAM,EAAE,OAAO,CAAC;QACvB,OAAO,EAAE,sBAAsB;QAC/B,QAAQ,EAAE,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,EAAE,CAC3C,GAAG,MAAM,GAAG,GAAG,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,GAAG,MAAM,EAAE;KACnD;CACF,CAAC;AAEK,MAAM,MAAM,GAAG,CACpB,GAAW,EACX,KAAc,EACd,sBAAkC,EAAE,EAC3B,EAAE;IAEX,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE;QAC7B,OAAO,KAAK,CAAC;KACd;IAED,MAAM,aAAa,GAAG,YAAY,CAAC,GAAG,CAAC,CAAC;IAExC,MAAM,QAAQ,GAAG,CAAC,GAAG,SAAS,EAAE,GAAG,mBAAmB,CAAC,CAAC;IAGxD,MAAM,IAAI,GAAG,QAAQ,CAAC,IAAI,CAAC,CAAC,EAAE,IAAI,EAAE,EAAE,EAAE,CACtC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,YAAY,CAAC,CAAC,CAAC,KAAK,aAAa,CAAC,CACpD,CAAC;IAGF,IAAI,CAAC,IAAI,EAAE;QACT,OAAO,KAAK,CAAC;KACd;IAGD,OAAO,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;AACpD,CAAC,CAAC;AA1BW,QAAA,MAAM,UA0BjB"}
@@ -0,0 +1,2 @@
1
+ import { TaskFn } from "../interfaces";
2
+ export declare function runWithTimeout<T = any>(handler: TaskFn, timeoutInMs: number): Promise<T>;
@@ -0,0 +1,30 @@
1
+ "use strict";
2
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
3
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
4
+ return new (P || (P = Promise))(function (resolve, reject) {
5
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
6
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
7
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
8
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
9
+ });
10
+ };
11
+ Object.defineProperty(exports, "__esModule", { value: true });
12
+ exports.runWithTimeout = void 0;
13
+ function runWithTimeout(handler, timeoutInMs) {
14
+ return __awaiter(this, void 0, void 0, function* () {
15
+ return new Promise((resolve, reject) => {
16
+ const timeout = setTimeout(() => reject(new Error("task timed out")), timeoutInMs);
17
+ handler()
18
+ .then((result) => {
19
+ clearTimeout(timeout);
20
+ resolve(result);
21
+ })
22
+ .catch((error) => {
23
+ clearTimeout(timeout);
24
+ reject(error);
25
+ });
26
+ });
27
+ });
28
+ }
29
+ exports.runWithTimeout = runWithTimeout;
30
+ //# sourceMappingURL=task-helper.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"task-helper.js","sourceRoot":"","sources":["../../../src/pkg/task-helper.ts"],"names":[],"mappings":";;;;;;;;;;;;AAEA,SAAsB,cAAc,CAClC,OAAe,EACf,WAAmB;;QAEnB,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,MAAM,OAAO,GAAG,UAAU,CACxB,GAAG,EAAE,CAAC,MAAM,CAAC,IAAI,KAAK,CAAC,gBAAgB,CAAC,CAAC,EACzC,WAAW,CACZ,CAAC;YAEF,OAAO,EAAE;iBACN,IAAI,CAAC,CAAC,MAAS,EAAE,EAAE;gBAClB,YAAY,CAAC,OAAO,CAAC,CAAC;gBAEtB,OAAO,CAAC,MAAM,CAAC,CAAC;YAClB,CAAC,CAAC;iBACD,KAAK,CAAC,CAAC,KAAY,EAAE,EAAE;gBACtB,YAAY,CAAC,OAAO,CAAC,CAAC;gBAEtB,MAAM,CAAC,KAAK,CAAC,CAAC;YAChB,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;IACL,CAAC;CAAA;AAtBD,wCAsBC"}
@@ -0,0 +1,9 @@
1
+ import { CRYPTO_TOKEN, STABLE_COIN } from "../constants";
2
+ export type PairNameFormater = (data: string[], separator: string) => string;
3
+ export type NormalizePairNameOptions = {
4
+ sortOrder?: "asc" | "desc";
5
+ stableCoinRule?: "first" | "last" | "default";
6
+ separator?: string;
7
+ outputFormater?: PairNameFormater;
8
+ };
9
+ export declare const normalizePairName: (stableCoinList?: typeof STABLE_COIN, cryptoList?: typeof CRYPTO_TOKEN) => (pairName: string, options?: NormalizePairNameOptions) => string;
@@ -0,0 +1,44 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.normalizePairName = void 0;
4
+ const constants_1 = require("../constants");
5
+ const esc = (s) => s.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
6
+ const defaultFormatter = (data, separator) => data.join(separator);
7
+ const normalizePairName = (stableCoinList = constants_1.STABLE_COIN, cryptoList = constants_1.CRYPTO_TOKEN) => {
8
+ const stableTokensLower = Object.values(stableCoinList).map((i) => i.toLowerCase());
9
+ const stableRe = new RegExp(`(${stableTokensLower.map(esc).join("|")})`, "i");
10
+ const cryptoTokensLower = Object.values(cryptoList)
11
+ .map((i) => i.toLowerCase())
12
+ .sort((a, b) => b.length - a.length);
13
+ const cryptoTokensEscaped = cryptoTokensLower.map(esc);
14
+ return (pairName, options) => {
15
+ const { outputFormater = defaultFormatter, sortOrder = "asc", stableCoinRule = "last", separator = "/", } = options || {};
16
+ const normalized = pairName.replace(/[|/]/g, "").toLowerCase();
17
+ const parts = [];
18
+ const hasStable = stableTokensLower.some((t) => normalized.includes(t));
19
+ if (stableCoinRule !== "default" && hasStable) {
20
+ const other = normalized.replace(stableRe, "");
21
+ const stable = normalized.replace(other, "");
22
+ const ordered = stableCoinRule === "first" ? [stable, other] : [other, stable];
23
+ return outputFormater(ordered, separator);
24
+ }
25
+ for (let i = 0; i < cryptoTokensLower.length; i++) {
26
+ const raw = cryptoTokensLower[i];
27
+ if (normalized.includes(raw)) {
28
+ const pat = new RegExp(cryptoTokensEscaped[i], "i");
29
+ const remainder = normalized.replace(pat, "");
30
+ parts.push(raw, remainder);
31
+ break;
32
+ }
33
+ }
34
+ if (parts.length === 0) {
35
+ throw new Error("pairname does not contain crypto from list");
36
+ }
37
+ const ordered = sortOrder === "asc"
38
+ ? parts.sort((a, b) => a.localeCompare(b))
39
+ : parts.sort((a, b) => b.localeCompare(a));
40
+ return outputFormater(ordered, separator);
41
+ };
42
+ };
43
+ exports.normalizePairName = normalizePairName;
44
+ //# sourceMappingURL=trading-pair-helper.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"trading-pair-helper.js","sourceRoot":"","sources":["../../../src/pkg/trading-pair-helper.ts"],"names":[],"mappings":";;;AAAA,4CAAyD;AAEzD,MAAM,GAAG,GAAG,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,qBAAqB,EAAE,MAAM,CAAC,CAAC;AAIpE,MAAM,gBAAgB,GAAqB,CAAC,IAAI,EAAE,SAAS,EAAE,EAAE,CAC7D,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;AAShB,MAAM,iBAAiB,GAAG,CAC/B,cAAc,GAAG,uBAAW,EAC5B,UAAU,GAAG,wBAAY,EACzB,EAAE;IACF,MAAM,iBAAiB,GAAG,MAAM,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAChE,CAAC,CAAC,WAAW,EAAE,CAChB,CAAC;IACF,MAAM,QAAQ,GAAG,IAAI,MAAM,CAAC,IAAI,iBAAiB,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;IAG9E,MAAM,iBAAiB,GAAG,MAAM,CAAC,MAAM,CAAC,UAAU,CAAC;SAChD,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC;SAC3B,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,GAAG,CAAC,CAAC,MAAM,CAAC,CAAC;IAEvC,MAAM,mBAAmB,GAAG,iBAAiB,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;IAEvD,OAAO,CAAC,QAAgB,EAAE,OAAkC,EAAU,EAAE;QACtE,MAAM,EACJ,cAAc,GAAG,gBAAgB,EACjC,SAAS,GAAG,KAAK,EACjB,cAAc,GAAG,MAAM,EACvB,SAAS,GAAG,GAAG,GAChB,GAAG,OAAO,IAAI,EAAE,CAAC;QAGlB,MAAM,UAAU,GAAG,QAAQ,CAAC,OAAO,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,WAAW,EAAE,CAAC;QAC/D,MAAM,KAAK,GAAa,EAAE,CAAC;QAG3B,MAAM,SAAS,GAAG,iBAAiB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC;QACxE,IAAI,cAAc,KAAK,SAAS,IAAI,SAAS,EAAE;YAC7C,MAAM,KAAK,GAAG,UAAU,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;YAC/C,MAAM,MAAM,GAAG,UAAU,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;YAC7C,MAAM,OAAO,GACX,cAAc,KAAK,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;YAEjE,OAAO,cAAc,CAAC,OAAO,EAAE,SAAS,CAAC,CAAC;SAC3C;QAGD,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,iBAAiB,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;YACjD,MAAM,GAAG,GAAG,iBAAiB,CAAC,CAAC,CAAC,CAAC;YACjC,IAAI,UAAU,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE;gBAC5B,MAAM,GAAG,GAAG,IAAI,MAAM,CAAC,mBAAmB,CAAC,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;gBACpD,MAAM,SAAS,GAAG,UAAU,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;gBAC9C,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,SAAS,CAAC,CAAC;gBAC3B,MAAM;aACP;SACF;QAED,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE;YACtB,MAAM,IAAI,KAAK,CAAC,4CAA4C,CAAC,CAAC;SAC/D;QAED,MAAM,OAAO,GACX,SAAS,KAAK,KAAK;YACjB,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC;YAC1C,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;QAE/C,OAAO,cAAc,CAAC,OAAO,EAAE,SAAS,CAAC,CAAC;IAC5C,CAAC,CAAC;AACJ,CAAC,CAAC;AA7DW,QAAA,iBAAiB,qBA6D5B"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,132 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const mocha_1 = require("mocha");
4
+ const chai_1 = require("chai");
5
+ const trading_pair_helper_1 = require("./trading-pair-helper");
6
+ const constants_1 = require("../constants");
7
+ (0, mocha_1.describe)("normalizePairName (functional factory)", () => {
8
+ const normalize = (0, trading_pair_helper_1.normalizePairName)(constants_1.STABLE_COIN, constants_1.CRYPTO_TOKEN);
9
+ (0, mocha_1.it)("places stable coin last when stableCoinRule = 'last' (order-agnostic input)", () => {
10
+ const a = normalize("USDTWVPC", {
11
+ stableCoinRule: "last",
12
+ sortOrder: "asc",
13
+ separator: "/",
14
+ outputFormater: undefined,
15
+ });
16
+ const b = normalize("WVPCUSDT", {
17
+ stableCoinRule: "last",
18
+ sortOrder: "asc",
19
+ separator: "/",
20
+ outputFormater: undefined,
21
+ });
22
+ (0, chai_1.expect)(a).to.equal("wvpc/usdt");
23
+ (0, chai_1.expect)(b).to.equal("wvpc/usdt");
24
+ });
25
+ (0, mocha_1.it)("places stable coin first when stableCoinRule = 'first'", () => {
26
+ const out = normalize("WVPCUSDC", {
27
+ stableCoinRule: "first",
28
+ sortOrder: "asc",
29
+ separator: "/",
30
+ outputFormater: undefined,
31
+ });
32
+ (0, chai_1.expect)(out).to.equal("usdc/wvpc");
33
+ });
34
+ (0, mocha_1.it)("with stableCoinRule = 'default', uses general sort (no special placement)", () => {
35
+ const outAsc = normalize("WVPCUSDT", {
36
+ stableCoinRule: "default",
37
+ sortOrder: "asc",
38
+ separator: "/",
39
+ outputFormater: undefined,
40
+ });
41
+ const outDesc = normalize("WVPCUSDT", {
42
+ stableCoinRule: "default",
43
+ sortOrder: "desc",
44
+ separator: "/",
45
+ outputFormater: undefined,
46
+ });
47
+ (0, chai_1.expect)(outAsc).to.equal("usdt/wvpc");
48
+ (0, chai_1.expect)(outDesc).to.equal("wvpc/usdt");
49
+ });
50
+ (0, mocha_1.it)("strips separators and is case-insensitive", () => {
51
+ const a = normalize("ETH/USDT", {
52
+ stableCoinRule: "last",
53
+ sortOrder: "asc",
54
+ separator: "/",
55
+ outputFormater: undefined,
56
+ });
57
+ const b = normalize("ETH|USDT", {
58
+ stableCoinRule: "last",
59
+ sortOrder: "asc",
60
+ separator: "/",
61
+ outputFormater: undefined,
62
+ });
63
+ const c = normalize("wVpCusdT", {
64
+ stableCoinRule: "last",
65
+ sortOrder: "asc",
66
+ separator: "/",
67
+ outputFormater: undefined,
68
+ });
69
+ (0, chai_1.expect)(a).to.equal("eth/usdt");
70
+ (0, chai_1.expect)(b).to.equal("eth/usdt");
71
+ (0, chai_1.expect)(c).to.equal("wvpc/usdt");
72
+ });
73
+ (0, mocha_1.it)("handles overlapping tokens by preferring longest match (e.g., WETH vs ETH)", () => {
74
+ const outAsc = normalize("WETHETH", {
75
+ stableCoinRule: "default",
76
+ sortOrder: "asc",
77
+ separator: "/",
78
+ outputFormater: undefined,
79
+ });
80
+ const outDesc = normalize("WETHETH", {
81
+ stableCoinRule: "default",
82
+ sortOrder: "desc",
83
+ separator: "/",
84
+ outputFormater: undefined,
85
+ });
86
+ (0, chai_1.expect)(outAsc).to.equal("eth/weth");
87
+ (0, chai_1.expect)(outDesc).to.equal("weth/eth");
88
+ });
89
+ (0, mocha_1.it)("sorts non-stable pairs lexicographically based on sortOrder", () => {
90
+ const outAsc = normalize("BTCETH", {
91
+ stableCoinRule: "default",
92
+ sortOrder: "asc",
93
+ separator: "/",
94
+ outputFormater: undefined,
95
+ });
96
+ const outDesc = normalize("BTCETH", {
97
+ stableCoinRule: "default",
98
+ sortOrder: "desc",
99
+ separator: "/",
100
+ outputFormater: undefined,
101
+ });
102
+ (0, chai_1.expect)(outAsc).to.equal("btc/eth");
103
+ (0, chai_1.expect)(outDesc).to.equal("eth/btc");
104
+ });
105
+ (0, mocha_1.it)("passes the separator into the output formatter", () => {
106
+ const out = normalize("WVPCUSDT", {
107
+ stableCoinRule: "last",
108
+ sortOrder: "asc",
109
+ separator: "-",
110
+ outputFormater: (parts, sep) => parts.join(sep),
111
+ });
112
+ (0, chai_1.expect)(out).to.equal("wvpc-usdt");
113
+ });
114
+ (0, mocha_1.it)("supports custom formatter (e.g., uppercase concatenation)", () => {
115
+ const out = normalize("WVPCUSDT", {
116
+ stableCoinRule: "last",
117
+ sortOrder: "asc",
118
+ separator: "/",
119
+ outputFormater: (parts) => parts.map((p) => p.toUpperCase()).join(""),
120
+ });
121
+ (0, chai_1.expect)(out).to.equal("WVPCUSDT");
122
+ });
123
+ (0, mocha_1.it)("throws when no known tokens found", () => {
124
+ (0, chai_1.expect)(() => normalize("FOOBAR", {
125
+ stableCoinRule: "default",
126
+ sortOrder: "asc",
127
+ separator: "/",
128
+ outputFormater: undefined,
129
+ })).to.throw(/does not contain crypto from list/i);
130
+ });
131
+ });
132
+ //# sourceMappingURL=trading-pair-helper.spec.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"trading-pair-helper.spec.js","sourceRoot":"","sources":["../../../src/pkg/trading-pair-helper.spec.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,10 @@
1
+ import { DelayedTaskOps, ITask, TaskRegistry } from "../../interfaces";
2
+ export declare class DelayedTaskRegistry implements TaskRegistry {
3
+ protected taskRegistry: Map<string, ITask>;
4
+ register(options: DelayedTaskOps): Promise<ITask>;
5
+ count(): Promise<number>;
6
+ getTasks(): Promise<ITask[]>;
7
+ getTaskById(id: string): Promise<ITask>;
8
+ cancelTaskById(id: string): Promise<void>;
9
+ startTaskById(id: string): Promise<void>;
10
+ }
@@ -0,0 +1,67 @@
1
+ "use strict";
2
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
3
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
4
+ return new (P || (P = Promise))(function (resolve, reject) {
5
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
6
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
7
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
8
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
9
+ });
10
+ };
11
+ Object.defineProperty(exports, "__esModule", { value: true });
12
+ exports.DelayedTaskRegistry = void 0;
13
+ const delayed_task_1 = require("./delayed-task");
14
+ class DelayedTaskRegistry {
15
+ constructor() {
16
+ this.taskRegistry = new Map();
17
+ }
18
+ register(options) {
19
+ return __awaiter(this, void 0, void 0, function* () {
20
+ const delayedTask = new delayed_task_1.DelayedTask(options);
21
+ this.taskRegistry.set(delayedTask.id, delayedTask);
22
+ return delayedTask;
23
+ });
24
+ }
25
+ count() {
26
+ return __awaiter(this, void 0, void 0, function* () {
27
+ return this.taskRegistry.size;
28
+ });
29
+ }
30
+ getTasks() {
31
+ return __awaiter(this, void 0, void 0, function* () {
32
+ return Object.values(this.taskRegistry);
33
+ });
34
+ }
35
+ getTaskById(id) {
36
+ return __awaiter(this, void 0, void 0, function* () {
37
+ return this.taskRegistry.get(id);
38
+ });
39
+ }
40
+ cancelTaskById(id) {
41
+ return __awaiter(this, void 0, void 0, function* () {
42
+ const task = this.taskRegistry.get(id);
43
+ if (!task) {
44
+ throw new Error(`task with id ${id} not exist`);
45
+ }
46
+ if (!task.isRunning) {
47
+ return;
48
+ }
49
+ yield task.cancel();
50
+ this.taskRegistry.delete(id);
51
+ });
52
+ }
53
+ startTaskById(id) {
54
+ return __awaiter(this, void 0, void 0, function* () {
55
+ const task = this.taskRegistry.get(id);
56
+ if (!task) {
57
+ throw new Error(`task with id ${id} not exist`);
58
+ }
59
+ if (task.isRunning) {
60
+ return;
61
+ }
62
+ return task.start();
63
+ });
64
+ }
65
+ }
66
+ exports.DelayedTaskRegistry = DelayedTaskRegistry;
67
+ //# sourceMappingURL=delayed-task-registry.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"delayed-task-registry.js","sourceRoot":"","sources":["../../../../src/pkg/workflow/delayed-task-registry.ts"],"names":[],"mappings":";;;;;;;;;;;;AACA,iDAA6C;AAE7C,MAAa,mBAAmB;IAAhC;QACY,iBAAY,GAAG,IAAI,GAAG,EAAiB,CAAC;IAiDpD,CAAC;IA/Cc,QAAQ,CAAC,OAAuB;;YAC3C,MAAM,WAAW,GAAG,IAAI,0BAAW,CAAC,OAAO,CAAC,CAAC;YAE7C,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,WAAW,CAAC,EAAE,EAAE,WAAW,CAAC,CAAC;YAEnD,OAAO,WAAW,CAAC;QACrB,CAAC;KAAA;IAEY,KAAK;;YAChB,OAAO,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC;QAChC,CAAC;KAAA;IAEY,QAAQ;;YACnB,OAAO,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QAC1C,CAAC;KAAA;IAEY,WAAW,CAAC,EAAU;;YACjC,OAAO,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QACnC,CAAC;KAAA;IAEY,cAAc,CAAC,EAAU;;YACpC,MAAM,IAAI,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YACvC,IAAI,CAAC,IAAI,EAAE;gBACT,MAAM,IAAI,KAAK,CAAC,gBAAgB,EAAE,YAAY,CAAC,CAAC;aACjD;YAED,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE;gBACnB,OAAO;aACR;YAED,MAAM,IAAI,CAAC,MAAM,EAAE,CAAC;YAEpB,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;QAC/B,CAAC;KAAA;IAEY,aAAa,CAAC,EAAU;;YACnC,MAAM,IAAI,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YACvC,IAAI,CAAC,IAAI,EAAE;gBACT,MAAM,IAAI,KAAK,CAAC,gBAAgB,EAAE,YAAY,CAAC,CAAC;aACjD;YAED,IAAI,IAAI,CAAC,SAAS,EAAE;gBAClB,OAAO;aACR;YAED,OAAO,IAAI,CAAC,KAAK,EAAE,CAAC;QACtB,CAAC;KAAA;CACF;AAlDD,kDAkDC"}
@@ -0,0 +1,18 @@
1
+ import { DelayFn, DelayedTaskOps, ITask, Subscription, TaskRescheduleOps } from "../../interfaces";
2
+ export declare class DelayedTask implements ITask {
3
+ protected options: DelayedTaskOps;
4
+ protected delayFn: DelayFn;
5
+ id: string;
6
+ isCron: boolean;
7
+ isRunning: boolean;
8
+ isCancelled: boolean;
9
+ protected _lastRun: Date;
10
+ protected subscription: Subscription;
11
+ constructor(options: DelayedTaskOps, delayFn?: DelayFn);
12
+ protected _start(): void;
13
+ cancel(): Promise<void>;
14
+ reschedule(options: TaskRescheduleOps): Promise<void>;
15
+ start(): Promise<void>;
16
+ lastRun(): Promise<Date>;
17
+ nextRun(): Promise<Date>;
18
+ }
@@ -0,0 +1,95 @@
1
+ "use strict";
2
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
3
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
4
+ return new (P || (P = Promise))(function (resolve, reject) {
5
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
6
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
7
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
8
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
9
+ });
10
+ };
11
+ var __importDefault = (this && this.__importDefault) || function (mod) {
12
+ return (mod && mod.__esModule) ? mod : { "default": mod };
13
+ };
14
+ Object.defineProperty(exports, "__esModule", { value: true });
15
+ exports.DelayedTask = void 0;
16
+ const cuid2_1 = require("@paralleldrive/cuid2");
17
+ const dayjs_1 = __importDefault(require("dayjs"));
18
+ const nativeTimeout = (callback, timeout) => {
19
+ const timeoutId = setTimeout(callback, timeout);
20
+ return {
21
+ unsubscribe() {
22
+ return __awaiter(this, void 0, void 0, function* () {
23
+ clearTimeout(timeoutId);
24
+ });
25
+ },
26
+ };
27
+ };
28
+ class DelayedTask {
29
+ constructor(options, delayFn = nativeTimeout) {
30
+ this.options = options;
31
+ this.delayFn = delayFn;
32
+ this.isCron = false;
33
+ this.isRunning = false;
34
+ this.isCancelled = false;
35
+ this.id = (0, cuid2_1.createId)();
36
+ if (options === null || options === void 0 ? void 0 : options.startOnCreate) {
37
+ this._start();
38
+ }
39
+ }
40
+ _start() {
41
+ this.subscription = this.delayFn(() => {
42
+ this.options.callback();
43
+ this._lastRun = new Date();
44
+ }, this.options.timeout);
45
+ this.isRunning = true;
46
+ }
47
+ cancel() {
48
+ this.isCancelled = true;
49
+ this.isRunning = false;
50
+ return this.subscription.unsubscribe();
51
+ }
52
+ reschedule(options) {
53
+ return __awaiter(this, void 0, void 0, function* () {
54
+ if (!(options === null || options === void 0 ? void 0 : options.msFromNow) && !(options === null || options === void 0 ? void 0 : options.runTime)) {
55
+ throw new Error("must pass either msFromNow or runTime");
56
+ }
57
+ yield this.cancel();
58
+ if (options.msFromNow && options.runTime) {
59
+ throw new Error("only accept either msFromNow or runTime.");
60
+ }
61
+ if (options.msFromNow) {
62
+ this.options.timeout = options.msFromNow;
63
+ return this._start();
64
+ }
65
+ if (options.runTime) {
66
+ const timeDiff = (0, dayjs_1.default)(options.runTime).diff(new Date());
67
+ if (timeDiff <= 0) {
68
+ throw new Error("new run time must be after now");
69
+ }
70
+ this.options.timeout = timeDiff;
71
+ return this._start();
72
+ }
73
+ });
74
+ }
75
+ start() {
76
+ return __awaiter(this, void 0, void 0, function* () {
77
+ this._start();
78
+ });
79
+ }
80
+ lastRun() {
81
+ return __awaiter(this, void 0, void 0, function* () {
82
+ return this._lastRun;
83
+ });
84
+ }
85
+ nextRun() {
86
+ return __awaiter(this, void 0, void 0, function* () {
87
+ if (!this.isRunning) {
88
+ return null;
89
+ }
90
+ return (0, dayjs_1.default)(this._lastRun).add(this.options.timeout).toDate();
91
+ });
92
+ }
93
+ }
94
+ exports.DelayedTask = DelayedTask;
95
+ //# sourceMappingURL=delayed-task.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"delayed-task.js","sourceRoot":"","sources":["../../../../src/pkg/workflow/delayed-task.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ export * from "./delayed-task-registry";
2
+ export * from "./delayed-task";
3
+ export * from "./sync-taskqueue";
4
+ export * from "./retry-task";
5
+ export * from "./processing-milestone";
@@ -0,0 +1,22 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./delayed-task-registry"), exports);
18
+ __exportStar(require("./delayed-task"), exports);
19
+ __exportStar(require("./sync-taskqueue"), exports);
20
+ __exportStar(require("./retry-task"), exports);
21
+ __exportStar(require("./processing-milestone"), exports);
22
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/pkg/workflow/index.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;AAAA,0DAAwC;AACxC,iDAA+B;AAC/B,mDAAiC;AACjC,+CAA6B;AAC7B,yDAAuC"}
@@ -0,0 +1,18 @@
1
+ export interface Step {
2
+ name?: string;
3
+ onFailure?: OnFailure;
4
+ }
5
+ export interface OnFailure {
6
+ reason: string;
7
+ }
8
+ export declare class SingleProcessingMilestone {
9
+ protected pipelineName?: string;
10
+ protected logger?: (msg: string) => void;
11
+ protected _steps: Step[];
12
+ protected _isFinished: boolean;
13
+ constructor(pipelineName?: string, logger?: (msg: string) => void);
14
+ protected throwIfEnded(): void;
15
+ get totalSteps(): number;
16
+ addStep(name?: string, onFailure?: OnFailure): void;
17
+ result(): Step;
18
+ }
@@ -0,0 +1,39 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.SingleProcessingMilestone = void 0;
4
+ class SingleProcessingMilestone {
5
+ constructor(pipelineName, logger) {
6
+ this.pipelineName = pipelineName;
7
+ this.logger = logger;
8
+ this._isFinished = false;
9
+ }
10
+ throwIfEnded() {
11
+ if (this._isFinished) {
12
+ throw new Error("pipeline has already ended");
13
+ }
14
+ }
15
+ get totalSteps() {
16
+ this.throwIfEnded();
17
+ return this._steps.length;
18
+ }
19
+ addStep(name, onFailure) {
20
+ this.throwIfEnded();
21
+ const stepNo = this.totalSteps + 1;
22
+ this._steps.push({
23
+ name: name || `step_${stepNo}`,
24
+ onFailure: onFailure || {
25
+ reason: "unknown",
26
+ },
27
+ });
28
+ if (this.pipelineName && this.logger) {
29
+ this.logger(`step ${stepNo} of pipeline ${this.pipelineName} reached!`);
30
+ }
31
+ }
32
+ result() {
33
+ this.throwIfEnded();
34
+ this._isFinished = true;
35
+ return this._steps.pop();
36
+ }
37
+ }
38
+ exports.SingleProcessingMilestone = SingleProcessingMilestone;
39
+ //# sourceMappingURL=processing-milestone.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"processing-milestone.js","sourceRoot":"","sources":["../../../../src/pkg/workflow/processing-milestone.ts"],"names":[],"mappings":";;;AASA,MAAa,yBAAyB;IAIpC,YACY,YAAqB,EACrB,MAA8B;QAD9B,iBAAY,GAAZ,YAAY,CAAS;QACrB,WAAM,GAAN,MAAM,CAAwB;QAJhC,gBAAW,GAAG,KAAK,CAAC;IAK3B,CAAC;IAEM,YAAY;QACpB,IAAI,IAAI,CAAC,WAAW,EAAE;YACpB,MAAM,IAAI,KAAK,CAAC,4BAA4B,CAAC,CAAC;SAC/C;IACH,CAAC;IAED,IAAW,UAAU;QACnB,IAAI,CAAC,YAAY,EAAE,CAAC;QAEpB,OAAO,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC;IAC5B,CAAC;IAEM,OAAO,CAAC,IAAa,EAAE,SAAqB;QACjD,IAAI,CAAC,YAAY,EAAE,CAAC;QACpB,MAAM,MAAM,GAAG,IAAI,CAAC,UAAU,GAAG,CAAC,CAAC;QAEnC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC;YACf,IAAI,EAAE,IAAI,IAAI,QAAQ,MAAM,EAAE;YAC9B,SAAS,EAAE,SAAS,IAAI;gBACtB,MAAM,EAAE,SAAS;aAClB;SACF,CAAC,CAAC;QAEH,IAAI,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,MAAM,EAAE;YACpC,IAAI,CAAC,MAAM,CAAC,QAAQ,MAAM,gBAAgB,IAAI,CAAC,YAAY,WAAW,CAAC,CAAC;SACzE;IACH,CAAC;IAEM,MAAM;QACX,IAAI,CAAC,YAAY,EAAE,CAAC;QAEpB,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;QAExB,OAAO,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC;IAC3B,CAAC;CACF;AA5CD,8DA4CC"}