@adtrackify/at-service-common 4.0.3 → 4.0.4

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 (753) hide show
  1. package/dist/cjs/__tests__/clients/acuity-client.spec.d.ts +1 -1
  2. package/dist/cjs/__tests__/clients/acuity-client.spec.js +43 -43
  3. package/dist/cjs/__tests__/clients/cross-platform-compression.spec.d.ts +1 -1
  4. package/dist/cjs/__tests__/clients/cross-platform-compression.spec.js +354 -354
  5. package/dist/cjs/__tests__/clients/dynamodb-client.spec.d.ts +1 -1
  6. package/dist/cjs/__tests__/clients/dynamodb-client.spec.js +194 -194
  7. package/dist/cjs/__tests__/clients/sqs-bundled-client.spec.d.ts +1 -1
  8. package/dist/cjs/__tests__/clients/sqs-bundled-client.spec.js +931 -931
  9. package/dist/cjs/__tests__/clients/sqs-bundling-contracts.spec.d.ts +1 -1
  10. package/dist/cjs/__tests__/clients/sqs-bundling-contracts.spec.js +563 -563
  11. package/dist/cjs/__tests__/clients/sqs-client.spec.d.ts +1 -1
  12. package/dist/cjs/__tests__/clients/sqs-client.spec.js +191 -191
  13. package/dist/cjs/__tests__/clients/sqs-unbundle.spec.d.ts +1 -1
  14. package/dist/cjs/__tests__/clients/sqs-unbundle.spec.js +1357 -1357
  15. package/dist/cjs/__tests__/db/contact-enrichments-db-service.spec.d.ts +1 -1
  16. package/dist/cjs/__tests__/db/contact-enrichments-db-service.spec.js +68 -68
  17. package/dist/cjs/__tests__/db/destinations-db-service.spec.d.ts +1 -1
  18. package/dist/cjs/__tests__/db/destinations-db-service.spec.js +125 -125
  19. package/dist/cjs/__tests__/db/products-db-service.spec.d.ts +1 -0
  20. package/dist/cjs/__tests__/db/products-db-service.spec.js +90 -0
  21. package/dist/cjs/__tests__/db/products-db-service.spec.js.map +1 -0
  22. package/dist/cjs/__tests__/db/shared-read-db-services.spec.d.ts +1 -1
  23. package/dist/cjs/__tests__/db/shared-read-db-services.spec.js +89 -89
  24. package/dist/cjs/__tests__/db/shopify-app-installs-db-service.spec.d.ts +1 -1
  25. package/dist/cjs/__tests__/db/shopify-app-installs-db-service.spec.js +104 -104
  26. package/dist/cjs/__tests__/db/subscriptions-db-service.spec.d.ts +1 -1
  27. package/dist/cjs/__tests__/db/subscriptions-db-service.spec.js +95 -95
  28. package/dist/cjs/__tests__/db/user-accounts-db-service.spec.d.ts +1 -1
  29. package/dist/cjs/__tests__/db/user-accounts-db-service.spec.js +76 -76
  30. package/dist/cjs/__tests__/helpers/account-users-helper.spec.d.ts +1 -1
  31. package/dist/cjs/__tests__/helpers/account-users-helper.spec.js +220 -220
  32. package/dist/cjs/__tests__/helpers/acuity-helper.spec.d.ts +1 -1
  33. package/dist/cjs/__tests__/helpers/acuity-helper.spec.js +69 -69
  34. package/dist/cjs/__tests__/helpers/api-key-auth-helper.spec.d.ts +1 -1
  35. package/dist/cjs/__tests__/helpers/api-key-auth-helper.spec.js +82 -82
  36. package/dist/cjs/__tests__/identity-cache/identity-cache-db-service.spec.d.ts +1 -1
  37. package/dist/cjs/__tests__/identity-cache/identity-cache-db-service.spec.js +676 -676
  38. package/dist/cjs/__tests__/identity-cache/identity-cache-dynamodb-service.spec.d.ts +1 -1
  39. package/dist/cjs/__tests__/identity-cache/identity-cache-dynamodb-service.spec.js +1140 -1140
  40. package/dist/cjs/__tests__/identity-cache/identity-cache-tier-routing.spec.d.ts +1 -1
  41. package/dist/cjs/__tests__/identity-cache/identity-cache-tier-routing.spec.js +851 -851
  42. package/dist/cjs/__tests__/identity-cache/trait-merging-and-staleness.spec.d.ts +1 -1
  43. package/dist/cjs/__tests__/identity-cache/trait-merging-and-staleness.spec.js +1060 -1060
  44. package/dist/cjs/__tests__/identity-cache/volatile-traits-optimization.spec.d.ts +1 -1
  45. package/dist/cjs/__tests__/identity-cache/volatile-traits-optimization.spec.js +818 -818
  46. package/dist/cjs/__tests__/integration/sqs-bundling-roundtrip.spec.d.ts +1 -1
  47. package/dist/cjs/__tests__/integration/sqs-bundling-roundtrip.spec.js +584 -584
  48. package/dist/cjs/__tests__/libs/compress-decompress.spec.d.ts +1 -1
  49. package/dist/cjs/__tests__/libs/compress-decompress.spec.js +16 -16
  50. package/dist/cjs/__tests__/libs/contacts.spec.d.ts +1 -1
  51. package/dist/cjs/__tests__/libs/contacts.spec.js +294 -294
  52. package/dist/cjs/__tests__/libs/currency.spec.d.ts +1 -1
  53. package/dist/cjs/__tests__/libs/currency.spec.js +220 -220
  54. package/dist/cjs/__tests__/libs/dates.spec.d.ts +1 -1
  55. package/dist/cjs/__tests__/libs/dates.spec.js +130 -130
  56. package/dist/cjs/__tests__/libs/domain.spec.d.ts +1 -1
  57. package/dist/cjs/__tests__/libs/domain.spec.js +107 -107
  58. package/dist/cjs/__tests__/libs/numbers.spec.d.ts +1 -1
  59. package/dist/cjs/__tests__/libs/numbers.spec.js +261 -261
  60. package/dist/cjs/__tests__/s3-client/s3-client.spec.d.ts +1 -1
  61. package/dist/cjs/__tests__/s3-client/s3-client.spec.js +33 -33
  62. package/dist/cjs/__tests__/services/acuity-api-service.spec.d.ts +1 -1
  63. package/dist/cjs/__tests__/services/acuity-api-service.spec.js +71 -71
  64. package/dist/cjs/__tests__/services/cost/cost-calculation-types.spec.d.ts +1 -0
  65. package/dist/cjs/__tests__/services/cost/cost-calculation-types.spec.js +24 -0
  66. package/dist/cjs/__tests__/services/cost/cost-calculation-types.spec.js.map +1 -0
  67. package/dist/cjs/__tests__/services/cost/cost-calculator-service.spec.d.ts +1 -0
  68. package/dist/cjs/__tests__/services/cost/cost-calculator-service.spec.js +3320 -0
  69. package/dist/cjs/__tests__/services/cost/cost-calculator-service.spec.js.map +1 -0
  70. package/dist/cjs/__tests__/services/cost/cost-currency-service.spec.d.ts +1 -0
  71. package/dist/cjs/__tests__/services/cost/cost-currency-service.spec.js +115 -0
  72. package/dist/cjs/__tests__/services/cost/cost-currency-service.spec.js.map +1 -0
  73. package/dist/cjs/__tests__/services/cost/cost-filter-service.spec.d.ts +1 -0
  74. package/dist/cjs/__tests__/services/cost/cost-filter-service.spec.js +469 -0
  75. package/dist/cjs/__tests__/services/cost/cost-filter-service.spec.js.map +1 -0
  76. package/dist/cjs/__tests__/services/cost/order-cost/order-cost-resolution-service.spec.d.ts +1 -0
  77. package/dist/cjs/__tests__/services/cost/order-cost/order-cost-resolution-service.spec.js +207 -0
  78. package/dist/cjs/__tests__/services/cost/order-cost/order-cost-resolution-service.spec.js.map +1 -0
  79. package/dist/cjs/__tests__/services/currency-exchange-rate-lookup-service.spec.d.ts +1 -0
  80. package/dist/cjs/__tests__/services/currency-exchange-rate-lookup-service.spec.js +35 -0
  81. package/dist/cjs/__tests__/services/currency-exchange-rate-lookup-service.spec.js.map +1 -0
  82. package/dist/cjs/__tests__/services/email-verification/contact-email-verification-service.spec.d.ts +1 -1
  83. package/dist/cjs/__tests__/services/email-verification/contact-email-verification-service.spec.js +93 -93
  84. package/dist/cjs/__tests__/services/email-verification/email-verification-service.spec.d.ts +1 -1
  85. package/dist/cjs/__tests__/services/email-verification/email-verification-service.spec.js +57 -57
  86. package/dist/cjs/__tests__/shopify/shopify-graphql-transformer.spec.d.ts +1 -1
  87. package/dist/cjs/__tests__/shopify/shopify-graphql-transformer.spec.js +35 -35
  88. package/dist/cjs/__tests__/unit/libs/api-router/public-api-router.spec.d.ts +1 -1
  89. package/dist/cjs/__tests__/unit/libs/api-router/public-api-router.spec.js +181 -181
  90. package/dist/cjs/__tests__/unit/libs/api-router/route-matcher.spec.d.ts +1 -1
  91. package/dist/cjs/__tests__/unit/libs/api-router/route-matcher.spec.js +69 -69
  92. package/dist/cjs/__tests__/utils/custom-measure-formula-utils.spec.d.ts +1 -1
  93. package/dist/cjs/__tests__/utils/custom-measure-formula-utils.spec.js +139 -139
  94. package/dist/cjs/clients/generic/cognito-client.d.ts +23 -23
  95. package/dist/cjs/clients/generic/cognito-client.js +209 -209
  96. package/dist/cjs/clients/generic/dynamodb-client.d.ts +20 -20
  97. package/dist/cjs/clients/generic/dynamodb-client.js +235 -235
  98. package/dist/cjs/clients/generic/eventbridge-client.d.ts +14 -14
  99. package/dist/cjs/clients/generic/eventbridge-client.js +51 -51
  100. package/dist/cjs/clients/generic/http-client.d.ts +14 -14
  101. package/dist/cjs/clients/generic/http-client.js +61 -61
  102. package/dist/cjs/clients/generic/index.d.ts +13 -13
  103. package/dist/cjs/clients/generic/index.js +29 -29
  104. package/dist/cjs/clients/generic/lambda-invoke-client.d.ts +10 -10
  105. package/dist/cjs/clients/generic/lambda-invoke-client.js +39 -39
  106. package/dist/cjs/clients/generic/location-client.d.ts +8 -8
  107. package/dist/cjs/clients/generic/location-client.js +31 -31
  108. package/dist/cjs/clients/generic/redis-client.d.ts +33 -33
  109. package/dist/cjs/clients/generic/redis-client.js +191 -191
  110. package/dist/cjs/clients/generic/s3-client.d.ts +23 -23
  111. package/dist/cjs/clients/generic/s3-client.js +216 -216
  112. package/dist/cjs/clients/generic/singlestore-db-client.d.ts +14 -14
  113. package/dist/cjs/clients/generic/singlestore-db-client.js +67 -67
  114. package/dist/cjs/clients/generic/sqs-bundled-client.d.ts +15 -15
  115. package/dist/cjs/clients/generic/sqs-bundled-client.js +311 -311
  116. package/dist/cjs/clients/generic/sqs-bundled-client.types.d.ts +53 -53
  117. package/dist/cjs/clients/generic/sqs-bundled-client.types.js +17 -17
  118. package/dist/cjs/clients/generic/sqs-client.d.ts +53 -53
  119. package/dist/cjs/clients/generic/sqs-client.js +285 -285
  120. package/dist/cjs/clients/generic/sqs-unbundle.d.ts +32 -32
  121. package/dist/cjs/clients/generic/sqs-unbundle.js +144 -144
  122. package/dist/cjs/clients/index.d.ts +3 -3
  123. package/dist/cjs/clients/index.js +19 -19
  124. package/dist/cjs/clients/internal-api/accounts-client.d.ts +91 -91
  125. package/dist/cjs/clients/internal-api/accounts-client.js +129 -129
  126. package/dist/cjs/clients/internal-api/cache-lambda-client.d.ts +26 -26
  127. package/dist/cjs/clients/internal-api/cache-lambda-client.js +89 -89
  128. package/dist/cjs/clients/internal-api/db-management-client.d.ts +18 -18
  129. package/dist/cjs/clients/internal-api/db-management-client.js +36 -36
  130. package/dist/cjs/clients/internal-api/destinations-client.d.ts +34 -34
  131. package/dist/cjs/clients/internal-api/destinations-client.js +79 -79
  132. package/dist/cjs/clients/internal-api/event-collector-client.d.ts +20 -20
  133. package/dist/cjs/clients/internal-api/event-collector-client.js +36 -36
  134. package/dist/cjs/clients/internal-api/identity-client.d.ts +31 -31
  135. package/dist/cjs/clients/internal-api/identity-client.js +91 -91
  136. package/dist/cjs/clients/internal-api/index.d.ts +9 -9
  137. package/dist/cjs/clients/internal-api/index.js +25 -25
  138. package/dist/cjs/clients/internal-api/shopify-app-install-client.d.ts +37 -37
  139. package/dist/cjs/clients/internal-api/shopify-app-install-client.js +81 -81
  140. package/dist/cjs/clients/internal-api/subscriptions-client.d.ts +26 -26
  141. package/dist/cjs/clients/internal-api/subscriptions-client.js +77 -77
  142. package/dist/cjs/clients/internal-api/users-auth-client.d.ts +35 -35
  143. package/dist/cjs/clients/internal-api/users-auth-client.js +110 -110
  144. package/dist/cjs/clients/third-party/acuity-client.d.ts +10 -10
  145. package/dist/cjs/clients/third-party/acuity-client.js +40 -40
  146. package/dist/cjs/clients/third-party/emailable-client.d.ts +7 -7
  147. package/dist/cjs/clients/third-party/emailable-client.js +25 -25
  148. package/dist/cjs/clients/third-party/exchange-rate-api-client.d.ts +17 -17
  149. package/dist/cjs/clients/third-party/exchange-rate-api-client.js +19 -19
  150. package/dist/cjs/clients/third-party/index.d.ts +5 -5
  151. package/dist/cjs/clients/third-party/index.js +21 -21
  152. package/dist/cjs/clients/third-party/loops-client.d.ts +10 -10
  153. package/dist/cjs/clients/third-party/loops-client.js +30 -30
  154. package/dist/cjs/clients/third-party/shopify/graphql-order-queries.d.ts +25 -25
  155. package/dist/cjs/clients/third-party/shopify/graphql-order-queries.js +4 -4
  156. package/dist/cjs/clients/third-party/shopify/graphql-product-queries.d.ts +2 -2
  157. package/dist/cjs/clients/third-party/shopify/graphql-product-queries.js +5 -5
  158. package/dist/cjs/clients/third-party/shopify/shopify-graphql-client.d.ts +10 -10
  159. package/dist/cjs/clients/third-party/shopify/shopify-graphql-client.js +161 -161
  160. package/dist/cjs/clients/third-party/shopify-client.d.ts +29 -29
  161. package/dist/cjs/clients/third-party/shopify-client.js +146 -146
  162. package/dist/cjs/constants/index.d.ts +1 -1
  163. package/dist/cjs/constants/index.js +17 -17
  164. package/dist/cjs/constants/sqs.d.ts +20 -20
  165. package/dist/cjs/constants/sqs.js +26 -26
  166. package/dist/cjs/helpers/account-users-helper.d.ts +2 -2
  167. package/dist/cjs/helpers/account-users-helper.js +22 -22
  168. package/dist/cjs/helpers/acuity-helper.d.ts +4 -4
  169. package/dist/cjs/helpers/acuity-helper.js +56 -56
  170. package/dist/cjs/helpers/api-key-auth-helper.d.ts +9 -9
  171. package/dist/cjs/helpers/api-key-auth-helper.js +40 -40
  172. package/dist/cjs/helpers/api-key-authorizer-helper.d.ts +36 -36
  173. package/dist/cjs/helpers/api-key-authorizer-helper.js +87 -87
  174. package/dist/cjs/helpers/identity-cache-helper.d.ts +30 -30
  175. package/dist/cjs/helpers/identity-cache-helper.js +253 -253
  176. package/dist/cjs/helpers/index.d.ts +10 -10
  177. package/dist/cjs/helpers/index.js +26 -26
  178. package/dist/cjs/helpers/input-validation-helper.d.ts +3 -3
  179. package/dist/cjs/helpers/input-validation-helper.js +22 -22
  180. package/dist/cjs/helpers/logging-helper.d.ts +16 -16
  181. package/dist/cjs/helpers/logging-helper.js +84 -84
  182. package/dist/cjs/helpers/response-helper.d.ts +18 -18
  183. package/dist/cjs/helpers/response-helper.js +43 -43
  184. package/dist/cjs/helpers/shopify-helper.d.ts +9 -9
  185. package/dist/cjs/helpers/shopify-helper.js +26 -26
  186. package/dist/cjs/helpers/sqs-utils.d.ts +6 -6
  187. package/dist/cjs/helpers/sqs-utils.js +14 -14
  188. package/dist/cjs/index.d.ts +7 -7
  189. package/dist/cjs/index.js +23 -23
  190. package/dist/cjs/libs/api-router/index.d.ts +2 -2
  191. package/dist/cjs/libs/api-router/index.js +18 -18
  192. package/dist/cjs/libs/api-router/public-api-router.d.ts +3 -3
  193. package/dist/cjs/libs/api-router/public-api-router.js +36 -36
  194. package/dist/cjs/libs/api-router/route-matcher.d.ts +21 -21
  195. package/dist/cjs/libs/api-router/route-matcher.js +36 -36
  196. package/dist/cjs/libs/click-id-parser.d.ts +23 -23
  197. package/dist/cjs/libs/click-id-parser.js +49 -49
  198. package/dist/cjs/libs/compression.d.ts +2 -2
  199. package/dist/cjs/libs/compression.js +33 -33
  200. package/dist/cjs/libs/contacts.d.ts +7 -7
  201. package/dist/cjs/libs/contacts.js +152 -152
  202. package/dist/cjs/libs/cookie.d.ts +17 -17
  203. package/dist/cjs/libs/cookie.js +76 -76
  204. package/dist/cjs/libs/crypto.d.ts +4 -4
  205. package/dist/cjs/libs/crypto.js +25 -25
  206. package/dist/cjs/libs/csv.d.ts +2 -2
  207. package/dist/cjs/libs/csv.js +35 -35
  208. package/dist/cjs/libs/currency.d.ts +1 -1
  209. package/dist/cjs/libs/currency.js +29 -29
  210. package/dist/cjs/libs/dates.d.ts +12 -12
  211. package/dist/cjs/libs/dates.js +96 -96
  212. package/dist/cjs/libs/domain.d.ts +2 -2
  213. package/dist/cjs/libs/domain.js +38 -38
  214. package/dist/cjs/libs/emails.d.ts +8 -8
  215. package/dist/cjs/libs/emails.js +154 -154
  216. package/dist/cjs/libs/http-error.d.ts +21 -21
  217. package/dist/cjs/libs/http-error.js +63 -63
  218. package/dist/cjs/libs/http-status-codes.d.ts +58 -58
  219. package/dist/cjs/libs/http-status-codes.js +62 -62
  220. package/dist/cjs/libs/index.d.ts +19 -19
  221. package/dist/cjs/libs/index.js +35 -35
  222. package/dist/cjs/libs/numbers.d.ts +1 -1
  223. package/dist/cjs/libs/numbers.js +15 -15
  224. package/dist/cjs/libs/referrer-parser/index.d.ts +2 -2
  225. package/dist/cjs/libs/referrer-parser/index.js +18 -18
  226. package/dist/cjs/libs/referrer-parser/referrer-data.d.ts +9 -9
  227. package/dist/cjs/libs/referrer-parser/referrer-data.js +3307 -3307
  228. package/dist/cjs/libs/referrer-parser/referrer-parser-util.d.ts +20 -20
  229. package/dist/cjs/libs/referrer-parser/referrer-parser-util.js +131 -131
  230. package/dist/cjs/libs/strings.d.ts +3 -3
  231. package/dist/cjs/libs/strings.js +46 -46
  232. package/dist/cjs/libs/traits.d.ts +6 -6
  233. package/dist/cjs/libs/traits.js +65 -65
  234. package/dist/cjs/libs/url.d.ts +1 -1
  235. package/dist/cjs/libs/url.js +13 -13
  236. package/dist/cjs/services/acuity-api-service.d.ts +9 -9
  237. package/dist/cjs/services/acuity-api-service.js +73 -73
  238. package/dist/cjs/services/cache/generic-cached-object.d.ts +5 -5
  239. package/dist/cjs/services/cache/generic-cached-object.js +2 -2
  240. package/dist/cjs/services/cache/index.d.ts +1 -1
  241. package/dist/cjs/services/cache/index.js +17 -17
  242. package/dist/cjs/services/cache/product-cache-service.d.ts +21 -21
  243. package/dist/cjs/services/cache/product-cache-service.js +76 -76
  244. package/dist/cjs/services/cost/cost-calculation-types.d.ts +69 -0
  245. package/dist/cjs/services/cost/cost-calculation-types.js +20 -0
  246. package/dist/cjs/services/cost/cost-calculation-types.js.map +1 -0
  247. package/dist/cjs/services/cost/cost-calculator-service.d.ts +24 -0
  248. package/dist/cjs/services/cost/cost-calculator-service.js +457 -0
  249. package/dist/cjs/services/cost/cost-calculator-service.js.map +1 -0
  250. package/dist/cjs/services/cost/cost-currency-service.d.ts +6 -0
  251. package/dist/cjs/services/cost/cost-currency-service.js +88 -0
  252. package/dist/cjs/services/cost/cost-currency-service.js.map +1 -0
  253. package/dist/cjs/services/cost/cost-filter-service.d.ts +10 -0
  254. package/dist/cjs/services/cost/cost-filter-service.js +122 -0
  255. package/dist/cjs/services/cost/cost-filter-service.js.map +1 -0
  256. package/dist/cjs/services/cost/index.d.ts +5 -0
  257. package/dist/cjs/services/cost/index.js +22 -0
  258. package/dist/cjs/services/cost/index.js.map +1 -0
  259. package/dist/cjs/services/cost/order-cost/index.d.ts +2 -0
  260. package/dist/cjs/services/cost/order-cost/index.js +19 -0
  261. package/dist/cjs/services/cost/order-cost/index.js.map +1 -0
  262. package/dist/cjs/services/cost/order-cost/order-cost-resolution-service.d.ts +23 -0
  263. package/dist/cjs/services/cost/order-cost/order-cost-resolution-service.js +362 -0
  264. package/dist/cjs/services/cost/order-cost/order-cost-resolution-service.js.map +1 -0
  265. package/dist/cjs/services/cost/order-cost/order-cost-resolution-types.d.ts +37 -0
  266. package/dist/cjs/services/cost/order-cost/order-cost-resolution-types.js +3 -0
  267. package/dist/cjs/services/cost/order-cost/order-cost-resolution-types.js.map +1 -0
  268. package/dist/cjs/services/currency-exchange-rate-lookup-service.d.ts +12 -11
  269. package/dist/cjs/services/currency-exchange-rate-lookup-service.js +94 -66
  270. package/dist/cjs/services/currency-exchange-rate-lookup-service.js.map +1 -1
  271. package/dist/cjs/services/db/accounts-db-service.d.ts +9 -9
  272. package/dist/cjs/services/db/accounts-db-service.js +33 -33
  273. package/dist/cjs/services/db/api-keys-db-service.d.ts +10 -10
  274. package/dist/cjs/services/db/api-keys-db-service.js +36 -36
  275. package/dist/cjs/services/db/contact-enrichments-db-service.d.ts +15 -15
  276. package/dist/cjs/services/db/contact-enrichments-db-service.js +94 -94
  277. package/dist/cjs/services/db/currency-exchange-rates-db-service.d.ts +21 -21
  278. package/dist/cjs/services/db/currency-exchange-rates-db-service.js +39 -39
  279. package/dist/cjs/services/db/custom-measures-db-service.d.ts +14 -14
  280. package/dist/cjs/services/db/custom-measures-db-service.js +48 -48
  281. package/dist/cjs/services/db/destinations-db-service.d.ts +13 -13
  282. package/dist/cjs/services/db/destinations-db-service.js +74 -74
  283. package/dist/cjs/services/db/identity-cache-db-service.d.ts +28 -28
  284. package/dist/cjs/services/db/identity-cache-db-service.js +320 -320
  285. package/dist/cjs/services/db/identity-cache-dynamodb-service.d.ts +44 -44
  286. package/dist/cjs/services/db/identity-cache-dynamodb-service.js +734 -734
  287. package/dist/cjs/services/db/index.d.ts +19 -17
  288. package/dist/cjs/services/db/index.js +35 -33
  289. package/dist/cjs/services/db/index.js.map +1 -1
  290. package/dist/cjs/services/db/log-events-db-service.d.ts +11 -11
  291. package/dist/cjs/services/db/log-events-db-service.js +181 -181
  292. package/dist/cjs/services/db/pixels-db-service.d.ts +8 -8
  293. package/dist/cjs/services/db/pixels-db-service.js +35 -35
  294. package/dist/cjs/services/db/products-db-service-types.d.ts +10 -0
  295. package/dist/cjs/services/db/products-db-service-types.js +3 -0
  296. package/dist/cjs/services/db/products-db-service-types.js.map +1 -0
  297. package/dist/cjs/services/db/products-db-service.d.ts +19 -0
  298. package/dist/cjs/services/db/products-db-service.js +282 -0
  299. package/dist/cjs/services/db/products-db-service.js.map +1 -0
  300. package/dist/cjs/services/db/purchasable-contacts-db-service.d.ts +9 -9
  301. package/dist/cjs/services/db/purchasable-contacts-db-service.js +43 -43
  302. package/dist/cjs/services/db/purchased-contacts/index.d.ts +2 -2
  303. package/dist/cjs/services/db/purchased-contacts/index.js +18 -18
  304. package/dist/cjs/services/db/purchased-contacts/purchased-contacts-db-service.d.ts +18 -18
  305. package/dist/cjs/services/db/purchased-contacts/purchased-contacts-db-service.js +152 -152
  306. package/dist/cjs/services/db/purchased-contacts/types.d.ts +11 -11
  307. package/dist/cjs/services/db/purchased-contacts/types.js +2 -2
  308. package/dist/cjs/services/db/shopify-app-installs-db-service.d.ts +10 -10
  309. package/dist/cjs/services/db/shopify-app-installs-db-service.js +52 -52
  310. package/dist/cjs/services/db/shopify-products-cache-db-service.d.ts +16 -16
  311. package/dist/cjs/services/db/shopify-products-cache-db-service.js +73 -73
  312. package/dist/cjs/services/db/subscriptions-db-service.d.ts +11 -11
  313. package/dist/cjs/services/db/subscriptions-db-service.js +38 -38
  314. package/dist/cjs/services/db/tracking-events-db-service.d.ts +21 -21
  315. package/dist/cjs/services/db/tracking-events-db-service.js +188 -188
  316. package/dist/cjs/services/db/user-accounts-db-service.d.ts +7 -7
  317. package/dist/cjs/services/db/user-accounts-db-service.js +17 -17
  318. package/dist/cjs/services/email-verification/contact-email-verification-service.d.ts +7 -7
  319. package/dist/cjs/services/email-verification/contact-email-verification-service.js +101 -101
  320. package/dist/cjs/services/email-verification/email-verification-service.d.ts +19 -19
  321. package/dist/cjs/services/email-verification/email-verification-service.js +131 -131
  322. package/dist/cjs/services/email-verification/index.d.ts +2 -2
  323. package/dist/cjs/services/email-verification/index.js +18 -18
  324. package/dist/cjs/services/eventbridge-integration-service.d.ts +9 -9
  325. package/dist/cjs/services/eventbridge-integration-service.js +28 -28
  326. package/dist/cjs/services/events/index.d.ts +3 -3
  327. package/dist/cjs/services/events/index.js +19 -19
  328. package/dist/cjs/services/events/log-event-service.d.ts +19 -19
  329. package/dist/cjs/services/events/log-event-service.js +77 -77
  330. package/dist/cjs/services/events/metric-event-service.d.ts +9 -9
  331. package/dist/cjs/services/events/metric-event-service.js +49 -49
  332. package/dist/cjs/services/events/tracking-event-sqs-service.d.ts +8 -8
  333. package/dist/cjs/services/events/tracking-event-sqs-service.js +34 -34
  334. package/dist/cjs/services/generic-cache-service.d.ts +7 -7
  335. package/dist/cjs/services/generic-cache-service.js +33 -33
  336. package/dist/cjs/services/index.d.ts +11 -10
  337. package/dist/cjs/services/index.js +27 -26
  338. package/dist/cjs/services/index.js.map +1 -1
  339. package/dist/cjs/services/ipdata-lookup-service.d.ts +20 -20
  340. package/dist/cjs/services/ipdata-lookup-service.js +112 -112
  341. package/dist/cjs/services/shopify/index.d.ts +2 -2
  342. package/dist/cjs/services/shopify/index.js +18 -18
  343. package/dist/cjs/services/shopify/products/index.d.ts +1 -1
  344. package/dist/cjs/services/shopify/products/index.js +17 -17
  345. package/dist/cjs/services/shopify/products/shopify-products-serviceV2.d.ts +17 -17
  346. package/dist/cjs/services/shopify/products/shopify-products-serviceV2.js +112 -112
  347. package/dist/cjs/services/shopify/shopify-graphql-transformer.d.ts +8 -8
  348. package/dist/cjs/services/shopify/shopify-graphql-transformer.js +141 -141
  349. package/dist/cjs/types/acuity-types.d.ts +74 -74
  350. package/dist/cjs/types/acuity-types.js +2 -2
  351. package/dist/cjs/types/api-response.d.ts +6 -6
  352. package/dist/cjs/types/api-response.js +2 -2
  353. package/dist/cjs/types/index.d.ts +4 -4
  354. package/dist/cjs/types/index.js +33 -33
  355. package/dist/cjs/types/internal-events/event-detail-types.d.ts +20 -20
  356. package/dist/cjs/types/internal-events/event-detail-types.js +27 -27
  357. package/dist/cjs/types/internal-events/index.d.ts +1 -1
  358. package/dist/cjs/types/internal-events/index.js +17 -17
  359. package/dist/cjs/types/shopify-graphql-types/admin.generated.d.ts +123 -123
  360. package/dist/cjs/types/shopify-graphql-types/admin.generated.js +2 -2
  361. package/dist/cjs/types/shopify-graphql-types/admin.types.d.ts +26289 -26289
  362. package/dist/cjs/types/shopify-graphql-types/admin.types.js +5311 -5311
  363. package/dist/cjs/types/shopify-graphql-types/index.d.ts +2 -2
  364. package/dist/cjs/types/shopify-graphql-types/index.js +18 -18
  365. package/dist/cjs/types/shopify-rest-types.d.ts +767 -767
  366. package/dist/cjs/types/shopify-rest-types.js +2 -2
  367. package/dist/cjs/utils/compression.d.ts +36 -36
  368. package/dist/cjs/utils/compression.js +198 -198
  369. package/dist/cjs/utils/custom-measure-formula-utils.d.ts +6 -6
  370. package/dist/cjs/utils/custom-measure-formula-utils.js +209 -209
  371. package/dist/cjs/utils/index.d.ts +4 -4
  372. package/dist/cjs/utils/index.js +20 -20
  373. package/dist/cjs/utils/retry-envelope.d.ts +12 -12
  374. package/dist/cjs/utils/retry-envelope.js +28 -28
  375. package/dist/cjs/utils/size.d.ts +2 -2
  376. package/dist/cjs/utils/size.js +49 -49
  377. package/dist/esm/__tests__/clients/acuity-client.spec.d.ts +1 -1
  378. package/dist/esm/__tests__/clients/acuity-client.spec.js +41 -41
  379. package/dist/esm/__tests__/clients/cross-platform-compression.spec.d.ts +1 -1
  380. package/dist/esm/__tests__/clients/cross-platform-compression.spec.js +329 -329
  381. package/dist/esm/__tests__/clients/dynamodb-client.spec.d.ts +1 -1
  382. package/dist/esm/__tests__/clients/dynamodb-client.spec.js +192 -192
  383. package/dist/esm/__tests__/clients/sqs-bundled-client.spec.d.ts +1 -1
  384. package/dist/esm/__tests__/clients/sqs-bundled-client.spec.js +906 -906
  385. package/dist/esm/__tests__/clients/sqs-bundling-contracts.spec.d.ts +1 -1
  386. package/dist/esm/__tests__/clients/sqs-bundling-contracts.spec.js +538 -538
  387. package/dist/esm/__tests__/clients/sqs-client.spec.d.ts +1 -1
  388. package/dist/esm/__tests__/clients/sqs-client.spec.js +189 -189
  389. package/dist/esm/__tests__/clients/sqs-unbundle.spec.d.ts +1 -1
  390. package/dist/esm/__tests__/clients/sqs-unbundle.spec.js +1355 -1355
  391. package/dist/esm/__tests__/db/contact-enrichments-db-service.spec.d.ts +1 -1
  392. package/dist/esm/__tests__/db/contact-enrichments-db-service.spec.js +66 -66
  393. package/dist/esm/__tests__/db/destinations-db-service.spec.d.ts +1 -1
  394. package/dist/esm/__tests__/db/destinations-db-service.spec.js +123 -123
  395. package/dist/esm/__tests__/db/products-db-service.spec.d.ts +1 -0
  396. package/dist/esm/__tests__/db/products-db-service.spec.js +88 -0
  397. package/dist/esm/__tests__/db/products-db-service.spec.js.map +1 -0
  398. package/dist/esm/__tests__/db/shared-read-db-services.spec.d.ts +1 -1
  399. package/dist/esm/__tests__/db/shared-read-db-services.spec.js +87 -87
  400. package/dist/esm/__tests__/db/shopify-app-installs-db-service.spec.d.ts +1 -1
  401. package/dist/esm/__tests__/db/shopify-app-installs-db-service.spec.js +102 -102
  402. package/dist/esm/__tests__/db/subscriptions-db-service.spec.d.ts +1 -1
  403. package/dist/esm/__tests__/db/subscriptions-db-service.spec.js +93 -93
  404. package/dist/esm/__tests__/db/user-accounts-db-service.spec.d.ts +1 -1
  405. package/dist/esm/__tests__/db/user-accounts-db-service.spec.js +74 -74
  406. package/dist/esm/__tests__/helpers/account-users-helper.spec.d.ts +1 -1
  407. package/dist/esm/__tests__/helpers/account-users-helper.spec.js +218 -218
  408. package/dist/esm/__tests__/helpers/acuity-helper.spec.d.ts +1 -1
  409. package/dist/esm/__tests__/helpers/acuity-helper.spec.js +67 -67
  410. package/dist/esm/__tests__/helpers/api-key-auth-helper.spec.d.ts +1 -1
  411. package/dist/esm/__tests__/helpers/api-key-auth-helper.spec.js +80 -80
  412. package/dist/esm/__tests__/identity-cache/identity-cache-db-service.spec.d.ts +1 -1
  413. package/dist/esm/__tests__/identity-cache/identity-cache-db-service.spec.js +674 -674
  414. package/dist/esm/__tests__/identity-cache/identity-cache-dynamodb-service.spec.d.ts +1 -1
  415. package/dist/esm/__tests__/identity-cache/identity-cache-dynamodb-service.spec.js +1138 -1138
  416. package/dist/esm/__tests__/identity-cache/identity-cache-tier-routing.spec.d.ts +1 -1
  417. package/dist/esm/__tests__/identity-cache/identity-cache-tier-routing.spec.js +849 -849
  418. package/dist/esm/__tests__/identity-cache/trait-merging-and-staleness.spec.d.ts +1 -1
  419. package/dist/esm/__tests__/identity-cache/trait-merging-and-staleness.spec.js +1058 -1058
  420. package/dist/esm/__tests__/identity-cache/volatile-traits-optimization.spec.d.ts +1 -1
  421. package/dist/esm/__tests__/identity-cache/volatile-traits-optimization.spec.js +816 -816
  422. package/dist/esm/__tests__/integration/sqs-bundling-roundtrip.spec.d.ts +1 -1
  423. package/dist/esm/__tests__/integration/sqs-bundling-roundtrip.spec.js +582 -582
  424. package/dist/esm/__tests__/libs/compress-decompress.spec.d.ts +1 -1
  425. package/dist/esm/__tests__/libs/compress-decompress.spec.js +14 -14
  426. package/dist/esm/__tests__/libs/contacts.spec.d.ts +1 -1
  427. package/dist/esm/__tests__/libs/contacts.spec.js +292 -292
  428. package/dist/esm/__tests__/libs/currency.spec.d.ts +1 -1
  429. package/dist/esm/__tests__/libs/currency.spec.js +218 -218
  430. package/dist/esm/__tests__/libs/dates.spec.d.ts +1 -1
  431. package/dist/esm/__tests__/libs/dates.spec.js +128 -128
  432. package/dist/esm/__tests__/libs/domain.spec.d.ts +1 -1
  433. package/dist/esm/__tests__/libs/domain.spec.js +105 -105
  434. package/dist/esm/__tests__/libs/numbers.spec.d.ts +1 -1
  435. package/dist/esm/__tests__/libs/numbers.spec.js +259 -259
  436. package/dist/esm/__tests__/s3-client/s3-client.spec.d.ts +1 -1
  437. package/dist/esm/__tests__/s3-client/s3-client.spec.js +31 -31
  438. package/dist/esm/__tests__/services/acuity-api-service.spec.d.ts +1 -1
  439. package/dist/esm/__tests__/services/acuity-api-service.spec.js +69 -69
  440. package/dist/esm/__tests__/services/cost/cost-calculation-types.spec.d.ts +1 -0
  441. package/dist/esm/__tests__/services/cost/cost-calculation-types.spec.js +22 -0
  442. package/dist/esm/__tests__/services/cost/cost-calculation-types.spec.js.map +1 -0
  443. package/dist/esm/__tests__/services/cost/cost-calculator-service.spec.d.ts +1 -0
  444. package/dist/esm/__tests__/services/cost/cost-calculator-service.spec.js +3318 -0
  445. package/dist/esm/__tests__/services/cost/cost-calculator-service.spec.js.map +1 -0
  446. package/dist/esm/__tests__/services/cost/cost-currency-service.spec.d.ts +1 -0
  447. package/dist/esm/__tests__/services/cost/cost-currency-service.spec.js +113 -0
  448. package/dist/esm/__tests__/services/cost/cost-currency-service.spec.js.map +1 -0
  449. package/dist/esm/__tests__/services/cost/cost-filter-service.spec.d.ts +1 -0
  450. package/dist/esm/__tests__/services/cost/cost-filter-service.spec.js +467 -0
  451. package/dist/esm/__tests__/services/cost/cost-filter-service.spec.js.map +1 -0
  452. package/dist/esm/__tests__/services/cost/order-cost/order-cost-resolution-service.spec.d.ts +1 -0
  453. package/dist/esm/__tests__/services/cost/order-cost/order-cost-resolution-service.spec.js +205 -0
  454. package/dist/esm/__tests__/services/cost/order-cost/order-cost-resolution-service.spec.js.map +1 -0
  455. package/dist/esm/__tests__/services/currency-exchange-rate-lookup-service.spec.d.ts +1 -0
  456. package/dist/esm/__tests__/services/currency-exchange-rate-lookup-service.spec.js +33 -0
  457. package/dist/esm/__tests__/services/currency-exchange-rate-lookup-service.spec.js.map +1 -0
  458. package/dist/esm/__tests__/services/email-verification/contact-email-verification-service.spec.d.ts +1 -1
  459. package/dist/esm/__tests__/services/email-verification/contact-email-verification-service.spec.js +91 -91
  460. package/dist/esm/__tests__/services/email-verification/email-verification-service.spec.d.ts +1 -1
  461. package/dist/esm/__tests__/services/email-verification/email-verification-service.spec.js +55 -55
  462. package/dist/esm/__tests__/shopify/shopify-graphql-transformer.spec.d.ts +1 -1
  463. package/dist/esm/__tests__/shopify/shopify-graphql-transformer.spec.js +33 -33
  464. package/dist/esm/__tests__/unit/libs/api-router/public-api-router.spec.d.ts +1 -1
  465. package/dist/esm/__tests__/unit/libs/api-router/public-api-router.spec.js +156 -156
  466. package/dist/esm/__tests__/unit/libs/api-router/route-matcher.spec.d.ts +1 -1
  467. package/dist/esm/__tests__/unit/libs/api-router/route-matcher.spec.js +67 -67
  468. package/dist/esm/__tests__/utils/custom-measure-formula-utils.spec.d.ts +1 -1
  469. package/dist/esm/__tests__/utils/custom-measure-formula-utils.spec.js +137 -137
  470. package/dist/esm/clients/generic/cognito-client.d.ts +23 -23
  471. package/dist/esm/clients/generic/cognito-client.js +204 -204
  472. package/dist/esm/clients/generic/dynamodb-client.d.ts +20 -20
  473. package/dist/esm/clients/generic/dynamodb-client.js +231 -231
  474. package/dist/esm/clients/generic/eventbridge-client.d.ts +14 -14
  475. package/dist/esm/clients/generic/eventbridge-client.js +47 -47
  476. package/dist/esm/clients/generic/http-client.d.ts +14 -14
  477. package/dist/esm/clients/generic/http-client.js +53 -53
  478. package/dist/esm/clients/generic/index.d.ts +13 -13
  479. package/dist/esm/clients/generic/index.js +13 -13
  480. package/dist/esm/clients/generic/lambda-invoke-client.d.ts +10 -10
  481. package/dist/esm/clients/generic/lambda-invoke-client.js +35 -35
  482. package/dist/esm/clients/generic/location-client.d.ts +8 -8
  483. package/dist/esm/clients/generic/location-client.js +27 -27
  484. package/dist/esm/clients/generic/redis-client.d.ts +33 -33
  485. package/dist/esm/clients/generic/redis-client.js +184 -184
  486. package/dist/esm/clients/generic/s3-client.d.ts +23 -23
  487. package/dist/esm/clients/generic/s3-client.js +209 -209
  488. package/dist/esm/clients/generic/singlestore-db-client.d.ts +14 -14
  489. package/dist/esm/clients/generic/singlestore-db-client.js +40 -40
  490. package/dist/esm/clients/generic/sqs-bundled-client.d.ts +15 -15
  491. package/dist/esm/clients/generic/sqs-bundled-client.js +307 -307
  492. package/dist/esm/clients/generic/sqs-bundled-client.types.d.ts +53 -53
  493. package/dist/esm/clients/generic/sqs-bundled-client.types.js +14 -14
  494. package/dist/esm/clients/generic/sqs-client.d.ts +53 -53
  495. package/dist/esm/clients/generic/sqs-client.js +281 -281
  496. package/dist/esm/clients/generic/sqs-unbundle.d.ts +32 -32
  497. package/dist/esm/clients/generic/sqs-unbundle.js +137 -137
  498. package/dist/esm/clients/index.d.ts +3 -3
  499. package/dist/esm/clients/index.js +3 -3
  500. package/dist/esm/clients/internal-api/accounts-client.d.ts +91 -91
  501. package/dist/esm/clients/internal-api/accounts-client.js +125 -125
  502. package/dist/esm/clients/internal-api/cache-lambda-client.d.ts +26 -26
  503. package/dist/esm/clients/internal-api/cache-lambda-client.js +85 -85
  504. package/dist/esm/clients/internal-api/db-management-client.d.ts +18 -18
  505. package/dist/esm/clients/internal-api/db-management-client.js +32 -32
  506. package/dist/esm/clients/internal-api/destinations-client.d.ts +34 -34
  507. package/dist/esm/clients/internal-api/destinations-client.js +75 -75
  508. package/dist/esm/clients/internal-api/event-collector-client.d.ts +20 -20
  509. package/dist/esm/clients/internal-api/event-collector-client.js +32 -32
  510. package/dist/esm/clients/internal-api/identity-client.d.ts +31 -31
  511. package/dist/esm/clients/internal-api/identity-client.js +87 -87
  512. package/dist/esm/clients/internal-api/index.d.ts +9 -9
  513. package/dist/esm/clients/internal-api/index.js +9 -9
  514. package/dist/esm/clients/internal-api/shopify-app-install-client.d.ts +37 -37
  515. package/dist/esm/clients/internal-api/shopify-app-install-client.js +77 -77
  516. package/dist/esm/clients/internal-api/subscriptions-client.d.ts +26 -26
  517. package/dist/esm/clients/internal-api/subscriptions-client.js +73 -73
  518. package/dist/esm/clients/internal-api/users-auth-client.d.ts +35 -35
  519. package/dist/esm/clients/internal-api/users-auth-client.js +106 -106
  520. package/dist/esm/clients/third-party/acuity-client.d.ts +10 -10
  521. package/dist/esm/clients/third-party/acuity-client.js +36 -36
  522. package/dist/esm/clients/third-party/emailable-client.d.ts +7 -7
  523. package/dist/esm/clients/third-party/emailable-client.js +21 -21
  524. package/dist/esm/clients/third-party/exchange-rate-api-client.d.ts +17 -17
  525. package/dist/esm/clients/third-party/exchange-rate-api-client.js +15 -15
  526. package/dist/esm/clients/third-party/index.d.ts +5 -5
  527. package/dist/esm/clients/third-party/index.js +5 -5
  528. package/dist/esm/clients/third-party/loops-client.d.ts +10 -10
  529. package/dist/esm/clients/third-party/loops-client.js +26 -26
  530. package/dist/esm/clients/third-party/shopify/graphql-order-queries.d.ts +25 -25
  531. package/dist/esm/clients/third-party/shopify/graphql-order-queries.js +1 -1
  532. package/dist/esm/clients/third-party/shopify/graphql-product-queries.d.ts +2 -2
  533. package/dist/esm/clients/third-party/shopify/graphql-product-queries.js +2 -2
  534. package/dist/esm/clients/third-party/shopify/shopify-graphql-client.d.ts +10 -10
  535. package/dist/esm/clients/third-party/shopify/shopify-graphql-client.js +157 -157
  536. package/dist/esm/clients/third-party/shopify-client.d.ts +29 -29
  537. package/dist/esm/clients/third-party/shopify-client.js +142 -142
  538. package/dist/esm/constants/index.d.ts +1 -1
  539. package/dist/esm/constants/index.js +1 -1
  540. package/dist/esm/constants/sqs.d.ts +20 -20
  541. package/dist/esm/constants/sqs.js +22 -22
  542. package/dist/esm/helpers/account-users-helper.d.ts +2 -2
  543. package/dist/esm/helpers/account-users-helper.js +18 -18
  544. package/dist/esm/helpers/acuity-helper.d.ts +4 -4
  545. package/dist/esm/helpers/acuity-helper.js +51 -51
  546. package/dist/esm/helpers/api-key-auth-helper.d.ts +9 -9
  547. package/dist/esm/helpers/api-key-auth-helper.js +35 -35
  548. package/dist/esm/helpers/api-key-authorizer-helper.d.ts +36 -36
  549. package/dist/esm/helpers/api-key-authorizer-helper.js +83 -83
  550. package/dist/esm/helpers/identity-cache-helper.d.ts +30 -30
  551. package/dist/esm/helpers/identity-cache-helper.js +248 -248
  552. package/dist/esm/helpers/index.d.ts +10 -10
  553. package/dist/esm/helpers/index.js +10 -10
  554. package/dist/esm/helpers/input-validation-helper.d.ts +3 -3
  555. package/dist/esm/helpers/input-validation-helper.js +18 -18
  556. package/dist/esm/helpers/logging-helper.d.ts +16 -16
  557. package/dist/esm/helpers/logging-helper.js +56 -56
  558. package/dist/esm/helpers/response-helper.d.ts +18 -18
  559. package/dist/esm/helpers/response-helper.js +37 -37
  560. package/dist/esm/helpers/shopify-helper.d.ts +9 -9
  561. package/dist/esm/helpers/shopify-helper.js +21 -21
  562. package/dist/esm/helpers/sqs-utils.d.ts +6 -6
  563. package/dist/esm/helpers/sqs-utils.js +9 -9
  564. package/dist/esm/index.d.ts +7 -7
  565. package/dist/esm/index.js +7 -7
  566. package/dist/esm/libs/api-router/index.d.ts +2 -2
  567. package/dist/esm/libs/api-router/index.js +2 -2
  568. package/dist/esm/libs/api-router/public-api-router.d.ts +3 -3
  569. package/dist/esm/libs/api-router/public-api-router.js +32 -32
  570. package/dist/esm/libs/api-router/route-matcher.d.ts +21 -21
  571. package/dist/esm/libs/api-router/route-matcher.js +30 -30
  572. package/dist/esm/libs/click-id-parser.d.ts +23 -23
  573. package/dist/esm/libs/click-id-parser.js +45 -45
  574. package/dist/esm/libs/compression.d.ts +2 -2
  575. package/dist/esm/libs/compression.js +25 -25
  576. package/dist/esm/libs/contacts.d.ts +7 -7
  577. package/dist/esm/libs/contacts.js +143 -143
  578. package/dist/esm/libs/cookie.d.ts +17 -17
  579. package/dist/esm/libs/cookie.js +70 -70
  580. package/dist/esm/libs/crypto.d.ts +4 -4
  581. package/dist/esm/libs/crypto.js +15 -15
  582. package/dist/esm/libs/csv.d.ts +2 -2
  583. package/dist/esm/libs/csv.js +30 -30
  584. package/dist/esm/libs/currency.d.ts +1 -1
  585. package/dist/esm/libs/currency.js +22 -22
  586. package/dist/esm/libs/dates.d.ts +12 -12
  587. package/dist/esm/libs/dates.js +83 -83
  588. package/dist/esm/libs/domain.d.ts +2 -2
  589. package/dist/esm/libs/domain.js +33 -33
  590. package/dist/esm/libs/emails.d.ts +8 -8
  591. package/dist/esm/libs/emails.js +146 -146
  592. package/dist/esm/libs/http-error.d.ts +21 -21
  593. package/dist/esm/libs/http-error.js +59 -59
  594. package/dist/esm/libs/http-status-codes.d.ts +58 -58
  595. package/dist/esm/libs/http-status-codes.js +59 -59
  596. package/dist/esm/libs/index.d.ts +19 -19
  597. package/dist/esm/libs/index.js +19 -19
  598. package/dist/esm/libs/numbers.d.ts +1 -1
  599. package/dist/esm/libs/numbers.js +11 -11
  600. package/dist/esm/libs/referrer-parser/index.d.ts +2 -2
  601. package/dist/esm/libs/referrer-parser/index.js +2 -2
  602. package/dist/esm/libs/referrer-parser/referrer-data.d.ts +9 -9
  603. package/dist/esm/libs/referrer-parser/referrer-data.js +3304 -3304
  604. package/dist/esm/libs/referrer-parser/referrer-parser-util.d.ts +20 -20
  605. package/dist/esm/libs/referrer-parser/referrer-parser-util.js +124 -124
  606. package/dist/esm/libs/strings.d.ts +3 -3
  607. package/dist/esm/libs/strings.js +40 -40
  608. package/dist/esm/libs/traits.d.ts +6 -6
  609. package/dist/esm/libs/traits.js +54 -54
  610. package/dist/esm/libs/url.d.ts +1 -1
  611. package/dist/esm/libs/url.js +9 -9
  612. package/dist/esm/services/acuity-api-service.d.ts +9 -9
  613. package/dist/esm/services/acuity-api-service.js +69 -69
  614. package/dist/esm/services/cache/generic-cached-object.d.ts +5 -5
  615. package/dist/esm/services/cache/generic-cached-object.js +1 -1
  616. package/dist/esm/services/cache/index.d.ts +1 -1
  617. package/dist/esm/services/cache/index.js +1 -1
  618. package/dist/esm/services/cache/product-cache-service.d.ts +21 -21
  619. package/dist/esm/services/cache/product-cache-service.js +68 -68
  620. package/dist/esm/services/cost/cost-calculation-types.d.ts +69 -0
  621. package/dist/esm/services/cost/cost-calculation-types.js +16 -0
  622. package/dist/esm/services/cost/cost-calculation-types.js.map +1 -0
  623. package/dist/esm/services/cost/cost-calculator-service.d.ts +24 -0
  624. package/dist/esm/services/cost/cost-calculator-service.js +451 -0
  625. package/dist/esm/services/cost/cost-calculator-service.js.map +1 -0
  626. package/dist/esm/services/cost/cost-currency-service.d.ts +6 -0
  627. package/dist/esm/services/cost/cost-currency-service.js +85 -0
  628. package/dist/esm/services/cost/cost-currency-service.js.map +1 -0
  629. package/dist/esm/services/cost/cost-filter-service.d.ts +10 -0
  630. package/dist/esm/services/cost/cost-filter-service.js +119 -0
  631. package/dist/esm/services/cost/cost-filter-service.js.map +1 -0
  632. package/dist/esm/services/cost/index.d.ts +5 -0
  633. package/dist/esm/services/cost/index.js +6 -0
  634. package/dist/esm/services/cost/index.js.map +1 -0
  635. package/dist/esm/services/cost/order-cost/index.d.ts +2 -0
  636. package/dist/esm/services/cost/order-cost/index.js +3 -0
  637. package/dist/esm/services/cost/order-cost/index.js.map +1 -0
  638. package/dist/esm/services/cost/order-cost/order-cost-resolution-service.d.ts +23 -0
  639. package/dist/esm/services/cost/order-cost/order-cost-resolution-service.js +356 -0
  640. package/dist/esm/services/cost/order-cost/order-cost-resolution-service.js.map +1 -0
  641. package/dist/esm/services/cost/order-cost/order-cost-resolution-types.d.ts +37 -0
  642. package/dist/esm/services/cost/order-cost/order-cost-resolution-types.js +2 -0
  643. package/dist/esm/services/cost/order-cost/order-cost-resolution-types.js.map +1 -0
  644. package/dist/esm/services/currency-exchange-rate-lookup-service.d.ts +12 -11
  645. package/dist/esm/services/currency-exchange-rate-lookup-service.js +90 -62
  646. package/dist/esm/services/currency-exchange-rate-lookup-service.js.map +1 -1
  647. package/dist/esm/services/db/accounts-db-service.d.ts +9 -9
  648. package/dist/esm/services/db/accounts-db-service.js +29 -29
  649. package/dist/esm/services/db/api-keys-db-service.d.ts +10 -10
  650. package/dist/esm/services/db/api-keys-db-service.js +32 -32
  651. package/dist/esm/services/db/contact-enrichments-db-service.d.ts +15 -15
  652. package/dist/esm/services/db/contact-enrichments-db-service.js +90 -90
  653. package/dist/esm/services/db/currency-exchange-rates-db-service.d.ts +21 -21
  654. package/dist/esm/services/db/currency-exchange-rates-db-service.js +35 -35
  655. package/dist/esm/services/db/custom-measures-db-service.d.ts +14 -14
  656. package/dist/esm/services/db/custom-measures-db-service.js +44 -44
  657. package/dist/esm/services/db/destinations-db-service.d.ts +13 -13
  658. package/dist/esm/services/db/destinations-db-service.js +70 -70
  659. package/dist/esm/services/db/identity-cache-db-service.d.ts +28 -28
  660. package/dist/esm/services/db/identity-cache-db-service.js +313 -313
  661. package/dist/esm/services/db/identity-cache-dynamodb-service.d.ts +44 -44
  662. package/dist/esm/services/db/identity-cache-dynamodb-service.js +727 -727
  663. package/dist/esm/services/db/index.d.ts +19 -17
  664. package/dist/esm/services/db/index.js +19 -17
  665. package/dist/esm/services/db/index.js.map +1 -1
  666. package/dist/esm/services/db/log-events-db-service.d.ts +11 -11
  667. package/dist/esm/services/db/log-events-db-service.js +177 -177
  668. package/dist/esm/services/db/pixels-db-service.d.ts +8 -8
  669. package/dist/esm/services/db/pixels-db-service.js +31 -31
  670. package/dist/esm/services/db/products-db-service-types.d.ts +10 -0
  671. package/dist/esm/services/db/products-db-service-types.js +2 -0
  672. package/dist/esm/services/db/products-db-service-types.js.map +1 -0
  673. package/dist/esm/services/db/products-db-service.d.ts +19 -0
  674. package/dist/esm/services/db/products-db-service.js +278 -0
  675. package/dist/esm/services/db/products-db-service.js.map +1 -0
  676. package/dist/esm/services/db/purchasable-contacts-db-service.d.ts +9 -9
  677. package/dist/esm/services/db/purchasable-contacts-db-service.js +39 -39
  678. package/dist/esm/services/db/purchased-contacts/index.d.ts +2 -2
  679. package/dist/esm/services/db/purchased-contacts/index.js +2 -2
  680. package/dist/esm/services/db/purchased-contacts/purchased-contacts-db-service.d.ts +18 -18
  681. package/dist/esm/services/db/purchased-contacts/purchased-contacts-db-service.js +148 -148
  682. package/dist/esm/services/db/purchased-contacts/types.d.ts +11 -11
  683. package/dist/esm/services/db/purchased-contacts/types.js +1 -1
  684. package/dist/esm/services/db/shopify-app-installs-db-service.d.ts +10 -10
  685. package/dist/esm/services/db/shopify-app-installs-db-service.js +48 -48
  686. package/dist/esm/services/db/shopify-products-cache-db-service.d.ts +16 -16
  687. package/dist/esm/services/db/shopify-products-cache-db-service.js +66 -66
  688. package/dist/esm/services/db/subscriptions-db-service.d.ts +11 -11
  689. package/dist/esm/services/db/subscriptions-db-service.js +34 -34
  690. package/dist/esm/services/db/tracking-events-db-service.d.ts +21 -21
  691. package/dist/esm/services/db/tracking-events-db-service.js +184 -184
  692. package/dist/esm/services/db/user-accounts-db-service.d.ts +7 -7
  693. package/dist/esm/services/db/user-accounts-db-service.js +13 -13
  694. package/dist/esm/services/email-verification/contact-email-verification-service.d.ts +7 -7
  695. package/dist/esm/services/email-verification/contact-email-verification-service.js +97 -97
  696. package/dist/esm/services/email-verification/email-verification-service.d.ts +19 -19
  697. package/dist/esm/services/email-verification/email-verification-service.js +127 -127
  698. package/dist/esm/services/email-verification/index.d.ts +2 -2
  699. package/dist/esm/services/email-verification/index.js +2 -2
  700. package/dist/esm/services/eventbridge-integration-service.d.ts +9 -9
  701. package/dist/esm/services/eventbridge-integration-service.js +24 -24
  702. package/dist/esm/services/events/index.d.ts +3 -3
  703. package/dist/esm/services/events/index.js +3 -3
  704. package/dist/esm/services/events/log-event-service.d.ts +19 -19
  705. package/dist/esm/services/events/log-event-service.js +73 -73
  706. package/dist/esm/services/events/metric-event-service.d.ts +9 -9
  707. package/dist/esm/services/events/metric-event-service.js +45 -45
  708. package/dist/esm/services/events/tracking-event-sqs-service.d.ts +8 -8
  709. package/dist/esm/services/events/tracking-event-sqs-service.js +30 -30
  710. package/dist/esm/services/generic-cache-service.d.ts +7 -7
  711. package/dist/esm/services/generic-cache-service.js +29 -29
  712. package/dist/esm/services/index.d.ts +11 -10
  713. package/dist/esm/services/index.js +11 -10
  714. package/dist/esm/services/index.js.map +1 -1
  715. package/dist/esm/services/ipdata-lookup-service.d.ts +20 -20
  716. package/dist/esm/services/ipdata-lookup-service.js +108 -108
  717. package/dist/esm/services/shopify/index.d.ts +2 -2
  718. package/dist/esm/services/shopify/index.js +2 -2
  719. package/dist/esm/services/shopify/products/index.d.ts +1 -1
  720. package/dist/esm/services/shopify/products/index.js +1 -1
  721. package/dist/esm/services/shopify/products/shopify-products-serviceV2.d.ts +17 -17
  722. package/dist/esm/services/shopify/products/shopify-products-serviceV2.js +108 -108
  723. package/dist/esm/services/shopify/shopify-graphql-transformer.d.ts +8 -8
  724. package/dist/esm/services/shopify/shopify-graphql-transformer.js +138 -138
  725. package/dist/esm/types/acuity-types.d.ts +74 -74
  726. package/dist/esm/types/acuity-types.js +1 -1
  727. package/dist/esm/types/api-response.d.ts +6 -6
  728. package/dist/esm/types/api-response.js +1 -1
  729. package/dist/esm/types/index.d.ts +4 -4
  730. package/dist/esm/types/index.js +4 -4
  731. package/dist/esm/types/internal-events/event-detail-types.d.ts +20 -20
  732. package/dist/esm/types/internal-events/event-detail-types.js +24 -24
  733. package/dist/esm/types/internal-events/index.d.ts +1 -1
  734. package/dist/esm/types/internal-events/index.js +1 -1
  735. package/dist/esm/types/shopify-graphql-types/admin.generated.d.ts +123 -123
  736. package/dist/esm/types/shopify-graphql-types/admin.generated.js +1 -1
  737. package/dist/esm/types/shopify-graphql-types/admin.types.d.ts +26289 -26289
  738. package/dist/esm/types/shopify-graphql-types/admin.types.js +5299 -5299
  739. package/dist/esm/types/shopify-graphql-types/index.d.ts +2 -2
  740. package/dist/esm/types/shopify-graphql-types/index.js +2 -2
  741. package/dist/esm/types/shopify-rest-types.d.ts +767 -767
  742. package/dist/esm/types/shopify-rest-types.js +1 -1
  743. package/dist/esm/utils/compression.d.ts +36 -36
  744. package/dist/esm/utils/compression.js +187 -187
  745. package/dist/esm/utils/custom-measure-formula-utils.d.ts +6 -6
  746. package/dist/esm/utils/custom-measure-formula-utils.js +201 -201
  747. package/dist/esm/utils/index.d.ts +4 -4
  748. package/dist/esm/utils/index.js +4 -4
  749. package/dist/esm/utils/retry-envelope.d.ts +12 -12
  750. package/dist/esm/utils/retry-envelope.js +22 -22
  751. package/dist/esm/utils/size.d.ts +2 -2
  752. package/dist/esm/utils/size.js +44 -44
  753. package/package.json +134 -134
@@ -1,852 +1,852 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- const at_tracking_event_types_1 = require("@adtrackify/at-tracking-event-types");
4
- const identity_cache_helper_1 = require("../../helpers/identity-cache-helper");
5
- const identity_cache_dynamodb_service_1 = require("../../services/db/identity-cache-dynamodb-service");
6
- const mockInvokeFunction = jest.fn();
7
- jest.mock('../../clients/index.js', () => ({
8
- DynamoDbClient: {
9
- safeGet: jest.fn(),
10
- safePut: jest.fn(),
11
- safeDelete: jest.fn(),
12
- safeBatchGet: jest.fn(),
13
- safeBatchWrite: jest.fn(),
14
- safeQueryByGSI: jest.fn(),
15
- batchWrite: jest.fn(),
16
- },
17
- LambdaInvokeClient: jest.fn().mockImplementation(() => ({
18
- invokeFunction: mockInvokeFunction,
19
- })),
20
- }));
21
- jest.mock('../../helpers/index.js', () => ({
22
- Logger: {
23
- debug: jest.fn(),
24
- info: jest.fn(),
25
- warn: jest.fn(),
26
- error: jest.fn(),
27
- },
28
- }));
29
- const identity_cache_dynamodb_service_js_1 = require("../../services/db/identity-cache-dynamodb-service.js");
30
- const index_js_1 = require("../../clients/index.js");
31
- const index_js_2 = require("../../helpers/index.js");
32
- const mockedDynamoDbClient = index_js_1.DynamoDbClient;
33
- const mockedLogger = index_js_2.Logger;
34
- describe('Identity Cache Three-Tier Routing', () => {
35
- const pixelId = 'pixel123';
36
- const lambdaArn = 'arn:aws:lambda:us-east-1:123456789:function:identity-private';
37
- beforeEach(() => {
38
- jest.clearAllMocks();
39
- mockedDynamoDbClient.safeGet.mockReset();
40
- mockedDynamoDbClient.safePut.mockReset();
41
- mockedDynamoDbClient.safeDelete.mockReset();
42
- mockedDynamoDbClient.safeBatchGet.mockReset();
43
- mockedDynamoDbClient.safeBatchWrite.mockReset();
44
- mockedDynamoDbClient.safeQueryByGSI.mockReset();
45
- mockedDynamoDbClient.batchWrite.mockReset();
46
- mockInvokeFunction.mockReset();
47
- identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.lambdaInvokeClient = undefined;
48
- });
49
- describe('TIER: VOLATILE_ONLY — Returns merged identity, NO Neptune, NO cache write', () => {
50
- it('should return merged identity without Neptune call when only ipAddress is stale', async () => {
51
- const incomingIdentity = {
52
- identityId: 'identity456',
53
- traits: {
54
- emails: ['test@email.com'],
55
- ipAddress: '10.0.0.1',
56
- },
57
- };
58
- const cachedResponse = {
59
- pk: `identity#${pixelId}#identity456`,
60
- pixelId,
61
- identityId: 'identity456',
62
- response: {
63
- identityId: 'identity456',
64
- traits: {
65
- emails: ['test@email.com'],
66
- ipAddress: '192.168.1.1',
67
- },
68
- },
69
- updatedAt: new Date().toISOString(),
70
- };
71
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
72
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
73
- expect(result).toBeDefined();
74
- expect(result?.identityId).toBe('identity456');
75
- expect(mockInvokeFunction).not.toHaveBeenCalled();
76
- expect(mockedDynamoDbClient.safeBatchWrite).not.toHaveBeenCalled();
77
- });
78
- it('should return merged identity without Neptune call when only userAgent is stale', async () => {
79
- const incomingIdentity = {
80
- identityId: 'identity456',
81
- traits: {
82
- emails: ['test@email.com'],
83
- userAgent: 'Chrome/120.0',
84
- },
85
- };
86
- const cachedResponse = {
87
- pk: `identity#${pixelId}#identity456`,
88
- pixelId,
89
- identityId: 'identity456',
90
- response: {
91
- identityId: 'identity456',
92
- traits: {
93
- emails: ['test@email.com'],
94
- userAgent: 'Mozilla/5.0',
95
- },
96
- },
97
- updatedAt: new Date().toISOString(),
98
- };
99
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
100
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
101
- expect(result).toBeDefined();
102
- expect(mockInvokeFunction).not.toHaveBeenCalled();
103
- expect(mockedDynamoDbClient.safeBatchWrite).not.toHaveBeenCalled();
104
- });
105
- it('should return merged identity without Neptune call when only IP-sourced addresses are stale', async () => {
106
- const incomingIdentity = {
107
- identityId: 'identity456',
108
- traits: {
109
- emails: ['test@email.com'],
110
- addresses: [{ city: 'New York', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.IP_ADDRESS }],
111
- },
112
- };
113
- const cachedResponse = {
114
- pk: `identity#${pixelId}#identity456`,
115
- pixelId,
116
- identityId: 'identity456',
117
- response: {
118
- identityId: 'identity456',
119
- traits: {
120
- emails: ['test@email.com'],
121
- addresses: [{ city: 'Chicago', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.IP_ADDRESS }],
122
- },
123
- },
124
- updatedAt: new Date().toISOString(),
125
- };
126
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
127
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
128
- expect(result).toBeDefined();
129
- expect(mockInvokeFunction).not.toHaveBeenCalled();
130
- expect(mockedDynamoDbClient.safeBatchWrite).not.toHaveBeenCalled();
131
- });
132
- });
133
- describe('TIER: ADDRESS_UPDATE — Writes to cache, NO Neptune call', () => {
134
- it('should write to cache but skip Neptune when new non-IP address is added', async () => {
135
- const incomingIdentity = {
136
- identityId: 'identity456',
137
- traits: {
138
- emails: ['test@email.com'],
139
- addresses: [{ city: 'Boston', firstName: 'John', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT }],
140
- },
141
- };
142
- const cachedResponse = {
143
- pk: `identity#${pixelId}#identity456`,
144
- pixelId,
145
- identityId: 'identity456',
146
- response: {
147
- identityId: 'identity456',
148
- traits: {
149
- emails: ['test@email.com'],
150
- addresses: [],
151
- },
152
- },
153
- updatedAt: new Date().toISOString(),
154
- };
155
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
156
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
157
- expect(result).toBeDefined();
158
- expect(result?.traits?.addresses).toHaveLength(1);
159
- expect(mockInvokeFunction).not.toHaveBeenCalled();
160
- expect(mockedDynamoDbClient.safeBatchWrite).toHaveBeenCalled();
161
- });
162
- it('should write to cache and skip Neptune when address with PII but no source is added', async () => {
163
- const incomingIdentity = {
164
- identityId: 'identity456',
165
- traits: {
166
- emails: ['test@email.com'],
167
- addresses: [{ city: 'Miami', lastName: 'Smith' }],
168
- },
169
- };
170
- const cachedResponse = {
171
- pk: `identity#${pixelId}#identity456`,
172
- pixelId,
173
- identityId: 'identity456',
174
- response: {
175
- identityId: 'identity456',
176
- traits: {
177
- emails: ['test@email.com'],
178
- addresses: [],
179
- },
180
- },
181
- updatedAt: new Date().toISOString(),
182
- };
183
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
184
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
185
- expect(result).toBeDefined();
186
- expect(mockInvokeFunction).not.toHaveBeenCalled();
187
- expect(mockedDynamoDbClient.safeBatchWrite).toHaveBeenCalled();
188
- });
189
- it('should log with writeReason ADDRESS_UPDATE and staleTraits', async () => {
190
- const incomingIdentity = {
191
- identityId: 'identity456',
192
- traits: {
193
- emails: ['test@email.com'],
194
- addresses: [{ city: 'Seattle', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT }],
195
- },
196
- };
197
- const cachedResponse = {
198
- pk: `identity#${pixelId}#identity456`,
199
- pixelId,
200
- identityId: 'identity456',
201
- response: {
202
- identityId: 'identity456',
203
- traits: {
204
- emails: ['test@email.com'],
205
- addresses: [],
206
- },
207
- },
208
- updatedAt: new Date().toISOString(),
209
- };
210
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
211
- await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
212
- expect(mockedLogger.info).toHaveBeenCalledWith('IdentityCache - identity cache write: address update only (skipping Neptune)', expect.objectContaining({
213
- pixelId,
214
- writeReason: identity_cache_dynamodb_service_1.CACHE_WRITE_REASON.ADDRESS_UPDATE,
215
- staleTier: identity_cache_helper_1.IDENTITY_CACHE_STALENESS_TIER.ADDRESS_UPDATE,
216
- }));
217
- });
218
- });
219
- describe('TIER: IDENTITY_STALE — Calls Neptune AND writes to cache', () => {
220
- it('should call Neptune and write to cache when emails are stale', async () => {
221
- const incomingIdentity = {
222
- identityId: 'identity456',
223
- traits: {
224
- emails: ['old@email.com', 'new@email.com'],
225
- },
226
- };
227
- const cachedResponse = {
228
- pk: `identity#${pixelId}#identity456`,
229
- pixelId,
230
- identityId: 'identity456',
231
- response: {
232
- identityId: 'identity456',
233
- traits: {
234
- emails: ['old@email.com'],
235
- },
236
- },
237
- updatedAt: new Date().toISOString(),
238
- };
239
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
240
- mockInvokeFunction.mockResolvedValueOnce({
241
- statusCode: 200,
242
- body: JSON.stringify({
243
- identity: {
244
- identityId: 'identity456',
245
- traits: { emails: ['old@email.com', 'new@email.com', 'resolved@email.com'] },
246
- },
247
- }),
248
- });
249
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
250
- expect(result).toBeDefined();
251
- expect(mockInvokeFunction).toHaveBeenCalled();
252
- expect(mockedDynamoDbClient.safeBatchWrite).toHaveBeenCalled();
253
- });
254
- it('should call Neptune and write to cache when phones are stale', async () => {
255
- const incomingIdentity = {
256
- identityId: 'identity456',
257
- traits: {
258
- phones: ['+1234567890', '+9876543210'],
259
- },
260
- };
261
- const cachedResponse = {
262
- pk: `identity#${pixelId}#identity456`,
263
- pixelId,
264
- identityId: 'identity456',
265
- response: {
266
- identityId: 'identity456',
267
- traits: {
268
- phones: ['+1234567890'],
269
- },
270
- },
271
- updatedAt: new Date().toISOString(),
272
- };
273
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
274
- mockInvokeFunction.mockResolvedValueOnce({
275
- statusCode: 200,
276
- body: JSON.stringify({
277
- identity: {
278
- identityId: 'identity456',
279
- traits: { phones: ['+1234567890', '+9876543210'] },
280
- },
281
- }),
282
- });
283
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
284
- expect(result).toBeDefined();
285
- expect(mockInvokeFunction).toHaveBeenCalled();
286
- expect(mockedDynamoDbClient.safeBatchWrite).toHaveBeenCalled();
287
- });
288
- it('should log with writeReason STALE_TRAITS, staleTraits, and traitChanges', async () => {
289
- const incomingIdentity = {
290
- identityId: 'identity456',
291
- traits: {
292
- emails: ['old@email.com', 'new@email.com'],
293
- },
294
- };
295
- const cachedResponse = {
296
- pk: `identity#${pixelId}#identity456`,
297
- pixelId,
298
- identityId: 'identity456',
299
- response: {
300
- identityId: 'identity456',
301
- traits: { emails: ['old@email.com'] },
302
- },
303
- updatedAt: new Date().toISOString(),
304
- };
305
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
306
- mockInvokeFunction.mockResolvedValueOnce({
307
- statusCode: 200,
308
- body: JSON.stringify({
309
- identity: {
310
- identityId: 'identity456',
311
- traits: { emails: ['old@email.com', 'new@email.com'] },
312
- },
313
- }),
314
- });
315
- await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
316
- expect(mockedLogger.info).toHaveBeenCalledWith('IdentityCache - identity cache write: stale_traits_update', expect.objectContaining({
317
- pixelId,
318
- writeReason: identity_cache_dynamodb_service_1.CACHE_WRITE_REASON.STALE_TRAITS,
319
- staleTier: identity_cache_helper_1.IDENTITY_CACHE_STALENESS_TIER.IDENTITY_STALE,
320
- staleTraits: expect.arrayContaining(['emails']),
321
- }));
322
- });
323
- });
324
- describe('CACHE_MISS — Calls Neptune AND writes to cache', () => {
325
- it('should call Neptune and write to cache on complete cache miss', async () => {
326
- const incomingIdentity = {
327
- identityId: 'identity456',
328
- traits: { emails: ['test@email.com'] },
329
- };
330
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(null);
331
- mockInvokeFunction.mockResolvedValueOnce({
332
- statusCode: 200,
333
- body: JSON.stringify({
334
- identity: {
335
- identityId: 'identity456',
336
- traits: { emails: ['test@email.com'] },
337
- },
338
- }),
339
- });
340
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
341
- expect(result).toBeDefined();
342
- expect(mockInvokeFunction).toHaveBeenCalled();
343
- expect(mockedDynamoDbClient.safeBatchWrite).toHaveBeenCalled();
344
- });
345
- it('should log with writeReason CACHE_MISS', async () => {
346
- const incomingIdentity = {
347
- identityId: 'identity456',
348
- traits: { emails: ['test@email.com'] },
349
- };
350
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(null);
351
- mockInvokeFunction.mockResolvedValueOnce({
352
- statusCode: 200,
353
- body: JSON.stringify({
354
- identity: {
355
- identityId: 'identity456',
356
- traits: { emails: ['test@email.com'] },
357
- },
358
- }),
359
- });
360
- await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
361
- expect(mockedLogger.info).toHaveBeenCalledWith('IdentityCache - identity cache write: new_identity', expect.objectContaining({
362
- pixelId,
363
- writeReason: identity_cache_dynamodb_service_1.CACHE_WRITE_REASON.CACHE_MISS,
364
- }));
365
- });
366
- });
367
- describe('EMAIL_POINTER_DISCOVERY — Calls Neptune with discovered identityId', () => {
368
- it('should call Neptune with discovered identityId from email pointer', async () => {
369
- const incomingIdentity = {
370
- traits: { emails: ['test@email.com'] },
371
- };
372
- const pointerRecord = {
373
- pk: `email#${pixelId}#test@email.com`,
374
- pixelId,
375
- identityId: 'discovered-id',
376
- gsi1pk: 'discovered-id',
377
- updatedAt: new Date().toISOString(),
378
- };
379
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(pointerRecord);
380
- mockInvokeFunction.mockResolvedValueOnce({
381
- statusCode: 200,
382
- body: JSON.stringify({
383
- identity: {
384
- identityId: 'discovered-id',
385
- traits: { emails: ['test@email.com', 'other@email.com'] },
386
- },
387
- }),
388
- });
389
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
390
- expect(result?.identityId).toBe('discovered-id');
391
- expect(mockInvokeFunction).toHaveBeenCalledWith(lambdaArn, expect.objectContaining({
392
- context: {
393
- identity: expect.objectContaining({
394
- identityId: 'discovered-id',
395
- }),
396
- },
397
- }));
398
- });
399
- });
400
- });
401
- describe('Identity Cache stripAddressTimestamps', () => {
402
- const pixelId = 'pixel123';
403
- const lambdaArn = 'arn:aws:lambda:us-east-1:123456789:function:identity-private';
404
- beforeEach(() => {
405
- jest.clearAllMocks();
406
- mockedDynamoDbClient.safeGet.mockReset();
407
- mockedDynamoDbClient.safeBatchWrite.mockReset();
408
- mockInvokeFunction.mockReset();
409
- identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.lambdaInvokeClient = undefined;
410
- });
411
- it('should strip cra and upa from addresses during comparison', async () => {
412
- const incomingIdentity = {
413
- identityId: 'identity456',
414
- traits: {
415
- emails: ['test@email.com'],
416
- addresses: [
417
- {
418
- city: 'Test City',
419
- s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
420
- cra: Date.parse('2024-01-01T00:00:00Z'),
421
- upa: Date.parse('2024-01-15T00:00:00Z'),
422
- },
423
- ],
424
- },
425
- };
426
- const cachedResponse = {
427
- pk: `identity#${pixelId}#identity456`,
428
- pixelId,
429
- identityId: 'identity456',
430
- response: {
431
- identityId: 'identity456',
432
- traits: {
433
- emails: ['test@email.com'],
434
- addresses: [
435
- {
436
- city: 'Test City',
437
- s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
438
- cra: Date.parse('2023-06-01T00:00:00Z'),
439
- upa: Date.parse('2023-12-01T00:00:00Z'),
440
- },
441
- ],
442
- },
443
- },
444
- updatedAt: new Date().toISOString(),
445
- };
446
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
447
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
448
- expect(result).toBeDefined();
449
- expect(mockInvokeFunction).not.toHaveBeenCalled();
450
- });
451
- it('should NOT mutate the original object when stripping timestamps', async () => {
452
- const originalAddress = {
453
- city: 'Original City',
454
- s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
455
- cra: Date.parse('2024-01-01T00:00:00Z'),
456
- upa: Date.parse('2024-01-15T00:00:00Z'),
457
- };
458
- const incomingIdentity = {
459
- identityId: 'identity456',
460
- traits: {
461
- emails: ['test@email.com'],
462
- addresses: [{ ...originalAddress }],
463
- },
464
- };
465
- const cachedResponse = {
466
- pk: `identity#${pixelId}#identity456`,
467
- pixelId,
468
- identityId: 'identity456',
469
- response: {
470
- identityId: 'identity456',
471
- traits: {
472
- emails: ['test@email.com'],
473
- addresses: [
474
- {
475
- city: 'Original City',
476
- s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
477
- cra: Date.parse('2023-06-01T00:00:00Z'),
478
- upa: Date.parse('2023-06-15T00:00:00Z'),
479
- },
480
- ],
481
- },
482
- },
483
- updatedAt: new Date().toISOString(),
484
- };
485
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
486
- await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
487
- expect(incomingIdentity.traits?.addresses?.[0]).toHaveProperty('cra');
488
- expect(incomingIdentity.traits?.addresses?.[0]).toHaveProperty('upa');
489
- expect(cachedResponse.response.traits?.addresses?.[0]).toHaveProperty('cra');
490
- expect(cachedResponse.response.traits?.addresses?.[0]).toHaveProperty('upa');
491
- });
492
- it('should handle identity with no addresses', async () => {
493
- const incomingIdentity = {
494
- identityId: 'identity456',
495
- traits: {
496
- emails: ['test@email.com'],
497
- },
498
- };
499
- const cachedResponse = {
500
- pk: `identity#${pixelId}#identity456`,
501
- pixelId,
502
- identityId: 'identity456',
503
- response: {
504
- identityId: 'identity456',
505
- traits: {
506
- emails: ['test@email.com'],
507
- },
508
- },
509
- updatedAt: new Date().toISOString(),
510
- };
511
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
512
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
513
- expect(result).toBeDefined();
514
- expect(mockInvokeFunction).not.toHaveBeenCalled();
515
- });
516
- it('should handle identity with empty addresses array', async () => {
517
- const incomingIdentity = {
518
- identityId: 'identity456',
519
- traits: {
520
- emails: ['test@email.com'],
521
- addresses: [],
522
- },
523
- };
524
- const cachedResponse = {
525
- pk: `identity#${pixelId}#identity456`,
526
- pixelId,
527
- identityId: 'identity456',
528
- response: {
529
- identityId: 'identity456',
530
- traits: {
531
- emails: ['test@email.com'],
532
- addresses: [],
533
- },
534
- },
535
- updatedAt: new Date().toISOString(),
536
- };
537
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
538
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
539
- expect(result).toBeDefined();
540
- expect(mockInvokeFunction).not.toHaveBeenCalled();
541
- });
542
- it('should handle addresses without cra/upa (no-op)', async () => {
543
- const incomingIdentity = {
544
- identityId: 'identity456',
545
- traits: {
546
- emails: ['test@email.com'],
547
- addresses: [{ city: 'Simple City', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT }],
548
- },
549
- };
550
- const cachedResponse = {
551
- pk: `identity#${pixelId}#identity456`,
552
- pixelId,
553
- identityId: 'identity456',
554
- response: {
555
- identityId: 'identity456',
556
- traits: {
557
- emails: ['test@email.com'],
558
- addresses: [{ city: 'Simple City', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT }],
559
- },
560
- },
561
- updatedAt: new Date().toISOString(),
562
- };
563
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
564
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
565
- expect(result).toBeDefined();
566
- expect(mockInvokeFunction).not.toHaveBeenCalled();
567
- });
568
- });
569
- describe('Identity Cache fastDeepEqual with Timestamp Stripping', () => {
570
- const pixelId = 'pixel123';
571
- const lambdaArn = 'arn:aws:lambda:us-east-1:123456789:function:identity-private';
572
- beforeEach(() => {
573
- jest.clearAllMocks();
574
- mockedDynamoDbClient.safeGet.mockReset();
575
- mockedDynamoDbClient.safeBatchWrite.mockReset();
576
- mockInvokeFunction.mockReset();
577
- identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.lambdaInvokeClient = undefined;
578
- });
579
- it('should treat cached and merged as equal when they differ ONLY in cra/upa', async () => {
580
- const incomingIdentity = {
581
- identityId: 'identity456',
582
- traits: {
583
- emails: ['test@email.com'],
584
- addresses: [
585
- {
586
- city: 'Same City',
587
- province: 'Same State',
588
- country: 'Same Country',
589
- s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
590
- cra: Date.parse('2024-06-01T00:00:00Z'),
591
- upa: Date.parse('2024-06-15T00:00:00Z'),
592
- },
593
- ],
594
- },
595
- };
596
- const cachedResponse = {
597
- pk: `identity#${pixelId}#identity456`,
598
- pixelId,
599
- identityId: 'identity456',
600
- response: {
601
- identityId: 'identity456',
602
- traits: {
603
- emails: ['test@email.com'],
604
- addresses: [
605
- {
606
- city: 'Same City',
607
- province: 'Same State',
608
- country: 'Same Country',
609
- s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
610
- cra: Date.parse('2024-01-01T00:00:00Z'),
611
- upa: Date.parse('2024-01-15T00:00:00Z'),
612
- },
613
- ],
614
- },
615
- },
616
- updatedAt: new Date().toISOString(),
617
- };
618
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
619
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
620
- expect(result).toBeDefined();
621
- expect(mockInvokeFunction).not.toHaveBeenCalled();
622
- expect(mockedDynamoDbClient.safeBatchWrite).not.toHaveBeenCalled();
623
- });
624
- it('should treat cached and merged as NOT equal when address content differs (plus cra/upa)', async () => {
625
- const incomingIdentity = {
626
- identityId: 'identity456',
627
- traits: {
628
- emails: ['test@email.com'],
629
- addresses: [
630
- {
631
- city: 'Different City',
632
- province: 'Same State',
633
- s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
634
- cra: Date.parse('2024-06-01T00:00:00Z'),
635
- upa: Date.parse('2024-06-15T00:00:00Z'),
636
- },
637
- ],
638
- },
639
- };
640
- const cachedResponse = {
641
- pk: `identity#${pixelId}#identity456`,
642
- pixelId,
643
- identityId: 'identity456',
644
- response: {
645
- identityId: 'identity456',
646
- traits: {
647
- emails: ['test@email.com'],
648
- addresses: [
649
- {
650
- city: 'Original City',
651
- province: 'Same State',
652
- s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
653
- cra: Date.parse('2024-01-01T00:00:00Z'),
654
- upa: Date.parse('2024-01-15T00:00:00Z'),
655
- },
656
- ],
657
- },
658
- },
659
- updatedAt: new Date().toISOString(),
660
- };
661
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
662
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
663
- expect(result).toBeDefined();
664
- });
665
- });
666
- describe('Identity Cache Logging Levels', () => {
667
- const pixelId = 'pixel123';
668
- const lambdaArn = 'arn:aws:lambda:us-east-1:123456789:function:identity-private';
669
- beforeEach(() => {
670
- jest.clearAllMocks();
671
- mockedDynamoDbClient.safeGet.mockReset();
672
- mockedDynamoDbClient.safeBatchWrite.mockReset();
673
- mockInvokeFunction.mockReset();
674
- identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.lambdaInvokeClient = undefined;
675
- });
676
- it('should log at debug level when cache is fresh (deep equal)', async () => {
677
- const incomingIdentity = {
678
- identityId: 'identity456',
679
- traits: {
680
- emails: ['test@email.com'],
681
- },
682
- };
683
- const cachedResponse = {
684
- pk: `identity#${pixelId}#identity456`,
685
- pixelId,
686
- identityId: 'identity456',
687
- response: {
688
- identityId: 'identity456',
689
- traits: {
690
- emails: ['test@email.com'],
691
- },
692
- },
693
- updatedAt: new Date().toISOString(),
694
- };
695
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
696
- await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
697
- expect(mockedLogger.debug).toHaveBeenCalledWith(expect.stringContaining('IdentityCache - '), expect.any(Object));
698
- expect(mockInvokeFunction).not.toHaveBeenCalled();
699
- expect(mockedDynamoDbClient.safeBatchWrite).not.toHaveBeenCalled();
700
- });
701
- it('should log at info level for ADDRESS_UPDATE path', async () => {
702
- const incomingIdentity = {
703
- identityId: 'identity456',
704
- traits: {
705
- emails: ['test@email.com'],
706
- addresses: [{ city: 'New City', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT }],
707
- },
708
- };
709
- const cachedResponse = {
710
- pk: `identity#${pixelId}#identity456`,
711
- pixelId,
712
- identityId: 'identity456',
713
- response: {
714
- identityId: 'identity456',
715
- traits: {
716
- emails: ['test@email.com'],
717
- addresses: [],
718
- },
719
- },
720
- updatedAt: new Date().toISOString(),
721
- };
722
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
723
- await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
724
- expect(mockedLogger.info).toHaveBeenCalledWith('IdentityCache - identity cache write: address update only (skipping Neptune)', expect.objectContaining({
725
- writeReason: identity_cache_dynamodb_service_1.CACHE_WRITE_REASON.ADDRESS_UPDATE,
726
- staleTraits: expect.any(Array),
727
- }));
728
- });
729
- it('should log at info level for IDENTITY_STALE path with traitChanges', async () => {
730
- const incomingIdentity = {
731
- identityId: 'identity456',
732
- traits: {
733
- emails: ['old@email.com', 'new@email.com'],
734
- },
735
- };
736
- const cachedResponse = {
737
- pk: `identity#${pixelId}#identity456`,
738
- pixelId,
739
- identityId: 'identity456',
740
- response: {
741
- identityId: 'identity456',
742
- traits: {
743
- emails: ['old@email.com'],
744
- },
745
- },
746
- updatedAt: new Date().toISOString(),
747
- };
748
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
749
- mockInvokeFunction.mockResolvedValueOnce({
750
- statusCode: 200,
751
- body: JSON.stringify({
752
- identity: {
753
- identityId: 'identity456',
754
- traits: { emails: ['old@email.com', 'new@email.com'] },
755
- },
756
- }),
757
- });
758
- await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
759
- expect(mockedLogger.info).toHaveBeenCalledWith('IdentityCache - identity cache write: stale_traits_update', expect.objectContaining({
760
- writeReason: identity_cache_dynamodb_service_1.CACHE_WRITE_REASON.STALE_TRAITS,
761
- staleTraits: expect.arrayContaining(['emails']),
762
- traitChanges: expect.any(Object),
763
- }));
764
- });
765
- });
766
- describe('getIdentityFromCache staleness tier results', () => {
767
- const pixelId = 'pixel123';
768
- beforeEach(() => {
769
- jest.clearAllMocks();
770
- mockedDynamoDbClient.safeGet.mockReset();
771
- });
772
- it('should return staleTier NONE when cached and incoming are deep equal', async () => {
773
- const incomingIdentity = {
774
- identityId: 'identity456',
775
- traits: {
776
- emails: ['test@email.com'],
777
- },
778
- };
779
- const cachedResponse = {
780
- pk: `identity#${pixelId}#identity456`,
781
- pixelId,
782
- identityId: 'identity456',
783
- response: {
784
- identityId: 'identity456',
785
- traits: {
786
- emails: ['test@email.com'],
787
- },
788
- },
789
- updatedAt: new Date().toISOString(),
790
- };
791
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
792
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityFromCache(pixelId, incomingIdentity);
793
- expect(result.staleTier).toBe(identity_cache_helper_1.IDENTITY_CACHE_STALENESS_TIER.NONE);
794
- expect(result.isCacheStale).toBe(false);
795
- expect(result.skipNeptune).toBeUndefined();
796
- });
797
- it('should return staleTier ADDRESS_UPDATE and skipNeptune=true for new non-IP address', async () => {
798
- const incomingIdentity = {
799
- identityId: 'identity456',
800
- traits: {
801
- emails: ['test@email.com'],
802
- addresses: [{ city: 'New City', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT }],
803
- },
804
- };
805
- const cachedResponse = {
806
- pk: `identity#${pixelId}#identity456`,
807
- pixelId,
808
- identityId: 'identity456',
809
- response: {
810
- identityId: 'identity456',
811
- traits: {
812
- emails: ['test@email.com'],
813
- addresses: [],
814
- },
815
- },
816
- updatedAt: new Date().toISOString(),
817
- };
818
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
819
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityFromCache(pixelId, incomingIdentity);
820
- expect(result.staleTier).toBe(identity_cache_helper_1.IDENTITY_CACHE_STALENESS_TIER.ADDRESS_UPDATE);
821
- expect(result.isCacheStale).toBe(true);
822
- expect(result.skipNeptune).toBe(true);
823
- expect(result.writeReason).toBe(identity_cache_dynamodb_service_1.CACHE_WRITE_REASON.ADDRESS_UPDATE);
824
- });
825
- it('should return staleTier IDENTITY_STALE and skipNeptune=false for new email', async () => {
826
- const incomingIdentity = {
827
- identityId: 'identity456',
828
- traits: {
829
- emails: ['old@email.com', 'new@email.com'],
830
- },
831
- };
832
- const cachedResponse = {
833
- pk: `identity#${pixelId}#identity456`,
834
- pixelId,
835
- identityId: 'identity456',
836
- response: {
837
- identityId: 'identity456',
838
- traits: {
839
- emails: ['old@email.com'],
840
- },
841
- },
842
- updatedAt: new Date().toISOString(),
843
- };
844
- mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
845
- const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityFromCache(pixelId, incomingIdentity);
846
- expect(result.staleTier).toBe(identity_cache_helper_1.IDENTITY_CACHE_STALENESS_TIER.IDENTITY_STALE);
847
- expect(result.isCacheStale).toBe(true);
848
- expect(result.skipNeptune).toBe(false);
849
- expect(result.writeReason).toBe(identity_cache_dynamodb_service_1.CACHE_WRITE_REASON.STALE_TRAITS);
850
- });
851
- });
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const at_tracking_event_types_1 = require("@adtrackify/at-tracking-event-types");
4
+ const identity_cache_helper_1 = require("../../helpers/identity-cache-helper");
5
+ const identity_cache_dynamodb_service_1 = require("../../services/db/identity-cache-dynamodb-service");
6
+ const mockInvokeFunction = jest.fn();
7
+ jest.mock('../../clients/index.js', () => ({
8
+ DynamoDbClient: {
9
+ safeGet: jest.fn(),
10
+ safePut: jest.fn(),
11
+ safeDelete: jest.fn(),
12
+ safeBatchGet: jest.fn(),
13
+ safeBatchWrite: jest.fn(),
14
+ safeQueryByGSI: jest.fn(),
15
+ batchWrite: jest.fn(),
16
+ },
17
+ LambdaInvokeClient: jest.fn().mockImplementation(() => ({
18
+ invokeFunction: mockInvokeFunction,
19
+ })),
20
+ }));
21
+ jest.mock('../../helpers/index.js', () => ({
22
+ Logger: {
23
+ debug: jest.fn(),
24
+ info: jest.fn(),
25
+ warn: jest.fn(),
26
+ error: jest.fn(),
27
+ },
28
+ }));
29
+ const identity_cache_dynamodb_service_js_1 = require("../../services/db/identity-cache-dynamodb-service.js");
30
+ const index_js_1 = require("../../clients/index.js");
31
+ const index_js_2 = require("../../helpers/index.js");
32
+ const mockedDynamoDbClient = index_js_1.DynamoDbClient;
33
+ const mockedLogger = index_js_2.Logger;
34
+ describe('Identity Cache Three-Tier Routing', () => {
35
+ const pixelId = 'pixel123';
36
+ const lambdaArn = 'arn:aws:lambda:us-east-1:123456789:function:identity-private';
37
+ beforeEach(() => {
38
+ jest.clearAllMocks();
39
+ mockedDynamoDbClient.safeGet.mockReset();
40
+ mockedDynamoDbClient.safePut.mockReset();
41
+ mockedDynamoDbClient.safeDelete.mockReset();
42
+ mockedDynamoDbClient.safeBatchGet.mockReset();
43
+ mockedDynamoDbClient.safeBatchWrite.mockReset();
44
+ mockedDynamoDbClient.safeQueryByGSI.mockReset();
45
+ mockedDynamoDbClient.batchWrite.mockReset();
46
+ mockInvokeFunction.mockReset();
47
+ identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.lambdaInvokeClient = undefined;
48
+ });
49
+ describe('TIER: VOLATILE_ONLY — Returns merged identity, NO Neptune, NO cache write', () => {
50
+ it('should return merged identity without Neptune call when only ipAddress is stale', async () => {
51
+ const incomingIdentity = {
52
+ identityId: 'identity456',
53
+ traits: {
54
+ emails: ['test@email.com'],
55
+ ipAddress: '10.0.0.1',
56
+ },
57
+ };
58
+ const cachedResponse = {
59
+ pk: `identity#${pixelId}#identity456`,
60
+ pixelId,
61
+ identityId: 'identity456',
62
+ response: {
63
+ identityId: 'identity456',
64
+ traits: {
65
+ emails: ['test@email.com'],
66
+ ipAddress: '192.168.1.1',
67
+ },
68
+ },
69
+ updatedAt: new Date().toISOString(),
70
+ };
71
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
72
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
73
+ expect(result).toBeDefined();
74
+ expect(result?.identityId).toBe('identity456');
75
+ expect(mockInvokeFunction).not.toHaveBeenCalled();
76
+ expect(mockedDynamoDbClient.safeBatchWrite).not.toHaveBeenCalled();
77
+ });
78
+ it('should return merged identity without Neptune call when only userAgent is stale', async () => {
79
+ const incomingIdentity = {
80
+ identityId: 'identity456',
81
+ traits: {
82
+ emails: ['test@email.com'],
83
+ userAgent: 'Chrome/120.0',
84
+ },
85
+ };
86
+ const cachedResponse = {
87
+ pk: `identity#${pixelId}#identity456`,
88
+ pixelId,
89
+ identityId: 'identity456',
90
+ response: {
91
+ identityId: 'identity456',
92
+ traits: {
93
+ emails: ['test@email.com'],
94
+ userAgent: 'Mozilla/5.0',
95
+ },
96
+ },
97
+ updatedAt: new Date().toISOString(),
98
+ };
99
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
100
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
101
+ expect(result).toBeDefined();
102
+ expect(mockInvokeFunction).not.toHaveBeenCalled();
103
+ expect(mockedDynamoDbClient.safeBatchWrite).not.toHaveBeenCalled();
104
+ });
105
+ it('should return merged identity without Neptune call when only IP-sourced addresses are stale', async () => {
106
+ const incomingIdentity = {
107
+ identityId: 'identity456',
108
+ traits: {
109
+ emails: ['test@email.com'],
110
+ addresses: [{ city: 'New York', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.IP_ADDRESS }],
111
+ },
112
+ };
113
+ const cachedResponse = {
114
+ pk: `identity#${pixelId}#identity456`,
115
+ pixelId,
116
+ identityId: 'identity456',
117
+ response: {
118
+ identityId: 'identity456',
119
+ traits: {
120
+ emails: ['test@email.com'],
121
+ addresses: [{ city: 'Chicago', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.IP_ADDRESS }],
122
+ },
123
+ },
124
+ updatedAt: new Date().toISOString(),
125
+ };
126
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
127
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
128
+ expect(result).toBeDefined();
129
+ expect(mockInvokeFunction).not.toHaveBeenCalled();
130
+ expect(mockedDynamoDbClient.safeBatchWrite).not.toHaveBeenCalled();
131
+ });
132
+ });
133
+ describe('TIER: ADDRESS_UPDATE — Writes to cache, NO Neptune call', () => {
134
+ it('should write to cache but skip Neptune when new non-IP address is added', async () => {
135
+ const incomingIdentity = {
136
+ identityId: 'identity456',
137
+ traits: {
138
+ emails: ['test@email.com'],
139
+ addresses: [{ city: 'Boston', firstName: 'John', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT }],
140
+ },
141
+ };
142
+ const cachedResponse = {
143
+ pk: `identity#${pixelId}#identity456`,
144
+ pixelId,
145
+ identityId: 'identity456',
146
+ response: {
147
+ identityId: 'identity456',
148
+ traits: {
149
+ emails: ['test@email.com'],
150
+ addresses: [],
151
+ },
152
+ },
153
+ updatedAt: new Date().toISOString(),
154
+ };
155
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
156
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
157
+ expect(result).toBeDefined();
158
+ expect(result?.traits?.addresses).toHaveLength(1);
159
+ expect(mockInvokeFunction).not.toHaveBeenCalled();
160
+ expect(mockedDynamoDbClient.safeBatchWrite).toHaveBeenCalled();
161
+ });
162
+ it('should write to cache and skip Neptune when address with PII but no source is added', async () => {
163
+ const incomingIdentity = {
164
+ identityId: 'identity456',
165
+ traits: {
166
+ emails: ['test@email.com'],
167
+ addresses: [{ city: 'Miami', lastName: 'Smith' }],
168
+ },
169
+ };
170
+ const cachedResponse = {
171
+ pk: `identity#${pixelId}#identity456`,
172
+ pixelId,
173
+ identityId: 'identity456',
174
+ response: {
175
+ identityId: 'identity456',
176
+ traits: {
177
+ emails: ['test@email.com'],
178
+ addresses: [],
179
+ },
180
+ },
181
+ updatedAt: new Date().toISOString(),
182
+ };
183
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
184
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
185
+ expect(result).toBeDefined();
186
+ expect(mockInvokeFunction).not.toHaveBeenCalled();
187
+ expect(mockedDynamoDbClient.safeBatchWrite).toHaveBeenCalled();
188
+ });
189
+ it('should log with writeReason ADDRESS_UPDATE and staleTraits', async () => {
190
+ const incomingIdentity = {
191
+ identityId: 'identity456',
192
+ traits: {
193
+ emails: ['test@email.com'],
194
+ addresses: [{ city: 'Seattle', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT }],
195
+ },
196
+ };
197
+ const cachedResponse = {
198
+ pk: `identity#${pixelId}#identity456`,
199
+ pixelId,
200
+ identityId: 'identity456',
201
+ response: {
202
+ identityId: 'identity456',
203
+ traits: {
204
+ emails: ['test@email.com'],
205
+ addresses: [],
206
+ },
207
+ },
208
+ updatedAt: new Date().toISOString(),
209
+ };
210
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
211
+ await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
212
+ expect(mockedLogger.info).toHaveBeenCalledWith('IdentityCache - identity cache write: address update only (skipping Neptune)', expect.objectContaining({
213
+ pixelId,
214
+ writeReason: identity_cache_dynamodb_service_1.CACHE_WRITE_REASON.ADDRESS_UPDATE,
215
+ staleTier: identity_cache_helper_1.IDENTITY_CACHE_STALENESS_TIER.ADDRESS_UPDATE,
216
+ }));
217
+ });
218
+ });
219
+ describe('TIER: IDENTITY_STALE — Calls Neptune AND writes to cache', () => {
220
+ it('should call Neptune and write to cache when emails are stale', async () => {
221
+ const incomingIdentity = {
222
+ identityId: 'identity456',
223
+ traits: {
224
+ emails: ['old@email.com', 'new@email.com'],
225
+ },
226
+ };
227
+ const cachedResponse = {
228
+ pk: `identity#${pixelId}#identity456`,
229
+ pixelId,
230
+ identityId: 'identity456',
231
+ response: {
232
+ identityId: 'identity456',
233
+ traits: {
234
+ emails: ['old@email.com'],
235
+ },
236
+ },
237
+ updatedAt: new Date().toISOString(),
238
+ };
239
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
240
+ mockInvokeFunction.mockResolvedValueOnce({
241
+ statusCode: 200,
242
+ body: JSON.stringify({
243
+ identity: {
244
+ identityId: 'identity456',
245
+ traits: { emails: ['old@email.com', 'new@email.com', 'resolved@email.com'] },
246
+ },
247
+ }),
248
+ });
249
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
250
+ expect(result).toBeDefined();
251
+ expect(mockInvokeFunction).toHaveBeenCalled();
252
+ expect(mockedDynamoDbClient.safeBatchWrite).toHaveBeenCalled();
253
+ });
254
+ it('should call Neptune and write to cache when phones are stale', async () => {
255
+ const incomingIdentity = {
256
+ identityId: 'identity456',
257
+ traits: {
258
+ phones: ['+1234567890', '+9876543210'],
259
+ },
260
+ };
261
+ const cachedResponse = {
262
+ pk: `identity#${pixelId}#identity456`,
263
+ pixelId,
264
+ identityId: 'identity456',
265
+ response: {
266
+ identityId: 'identity456',
267
+ traits: {
268
+ phones: ['+1234567890'],
269
+ },
270
+ },
271
+ updatedAt: new Date().toISOString(),
272
+ };
273
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
274
+ mockInvokeFunction.mockResolvedValueOnce({
275
+ statusCode: 200,
276
+ body: JSON.stringify({
277
+ identity: {
278
+ identityId: 'identity456',
279
+ traits: { phones: ['+1234567890', '+9876543210'] },
280
+ },
281
+ }),
282
+ });
283
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
284
+ expect(result).toBeDefined();
285
+ expect(mockInvokeFunction).toHaveBeenCalled();
286
+ expect(mockedDynamoDbClient.safeBatchWrite).toHaveBeenCalled();
287
+ });
288
+ it('should log with writeReason STALE_TRAITS, staleTraits, and traitChanges', async () => {
289
+ const incomingIdentity = {
290
+ identityId: 'identity456',
291
+ traits: {
292
+ emails: ['old@email.com', 'new@email.com'],
293
+ },
294
+ };
295
+ const cachedResponse = {
296
+ pk: `identity#${pixelId}#identity456`,
297
+ pixelId,
298
+ identityId: 'identity456',
299
+ response: {
300
+ identityId: 'identity456',
301
+ traits: { emails: ['old@email.com'] },
302
+ },
303
+ updatedAt: new Date().toISOString(),
304
+ };
305
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
306
+ mockInvokeFunction.mockResolvedValueOnce({
307
+ statusCode: 200,
308
+ body: JSON.stringify({
309
+ identity: {
310
+ identityId: 'identity456',
311
+ traits: { emails: ['old@email.com', 'new@email.com'] },
312
+ },
313
+ }),
314
+ });
315
+ await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
316
+ expect(mockedLogger.info).toHaveBeenCalledWith('IdentityCache - identity cache write: stale_traits_update', expect.objectContaining({
317
+ pixelId,
318
+ writeReason: identity_cache_dynamodb_service_1.CACHE_WRITE_REASON.STALE_TRAITS,
319
+ staleTier: identity_cache_helper_1.IDENTITY_CACHE_STALENESS_TIER.IDENTITY_STALE,
320
+ staleTraits: expect.arrayContaining(['emails']),
321
+ }));
322
+ });
323
+ });
324
+ describe('CACHE_MISS — Calls Neptune AND writes to cache', () => {
325
+ it('should call Neptune and write to cache on complete cache miss', async () => {
326
+ const incomingIdentity = {
327
+ identityId: 'identity456',
328
+ traits: { emails: ['test@email.com'] },
329
+ };
330
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(null);
331
+ mockInvokeFunction.mockResolvedValueOnce({
332
+ statusCode: 200,
333
+ body: JSON.stringify({
334
+ identity: {
335
+ identityId: 'identity456',
336
+ traits: { emails: ['test@email.com'] },
337
+ },
338
+ }),
339
+ });
340
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
341
+ expect(result).toBeDefined();
342
+ expect(mockInvokeFunction).toHaveBeenCalled();
343
+ expect(mockedDynamoDbClient.safeBatchWrite).toHaveBeenCalled();
344
+ });
345
+ it('should log with writeReason CACHE_MISS', async () => {
346
+ const incomingIdentity = {
347
+ identityId: 'identity456',
348
+ traits: { emails: ['test@email.com'] },
349
+ };
350
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(null);
351
+ mockInvokeFunction.mockResolvedValueOnce({
352
+ statusCode: 200,
353
+ body: JSON.stringify({
354
+ identity: {
355
+ identityId: 'identity456',
356
+ traits: { emails: ['test@email.com'] },
357
+ },
358
+ }),
359
+ });
360
+ await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
361
+ expect(mockedLogger.info).toHaveBeenCalledWith('IdentityCache - identity cache write: new_identity', expect.objectContaining({
362
+ pixelId,
363
+ writeReason: identity_cache_dynamodb_service_1.CACHE_WRITE_REASON.CACHE_MISS,
364
+ }));
365
+ });
366
+ });
367
+ describe('EMAIL_POINTER_DISCOVERY — Calls Neptune with discovered identityId', () => {
368
+ it('should call Neptune with discovered identityId from email pointer', async () => {
369
+ const incomingIdentity = {
370
+ traits: { emails: ['test@email.com'] },
371
+ };
372
+ const pointerRecord = {
373
+ pk: `email#${pixelId}#test@email.com`,
374
+ pixelId,
375
+ identityId: 'discovered-id',
376
+ gsi1pk: 'discovered-id',
377
+ updatedAt: new Date().toISOString(),
378
+ };
379
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(pointerRecord);
380
+ mockInvokeFunction.mockResolvedValueOnce({
381
+ statusCode: 200,
382
+ body: JSON.stringify({
383
+ identity: {
384
+ identityId: 'discovered-id',
385
+ traits: { emails: ['test@email.com', 'other@email.com'] },
386
+ },
387
+ }),
388
+ });
389
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
390
+ expect(result?.identityId).toBe('discovered-id');
391
+ expect(mockInvokeFunction).toHaveBeenCalledWith(lambdaArn, expect.objectContaining({
392
+ context: {
393
+ identity: expect.objectContaining({
394
+ identityId: 'discovered-id',
395
+ }),
396
+ },
397
+ }));
398
+ });
399
+ });
400
+ });
401
+ describe('Identity Cache stripAddressTimestamps', () => {
402
+ const pixelId = 'pixel123';
403
+ const lambdaArn = 'arn:aws:lambda:us-east-1:123456789:function:identity-private';
404
+ beforeEach(() => {
405
+ jest.clearAllMocks();
406
+ mockedDynamoDbClient.safeGet.mockReset();
407
+ mockedDynamoDbClient.safeBatchWrite.mockReset();
408
+ mockInvokeFunction.mockReset();
409
+ identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.lambdaInvokeClient = undefined;
410
+ });
411
+ it('should strip cra and upa from addresses during comparison', async () => {
412
+ const incomingIdentity = {
413
+ identityId: 'identity456',
414
+ traits: {
415
+ emails: ['test@email.com'],
416
+ addresses: [
417
+ {
418
+ city: 'Test City',
419
+ s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
420
+ cra: Date.parse('2024-01-01T00:00:00Z'),
421
+ upa: Date.parse('2024-01-15T00:00:00Z'),
422
+ },
423
+ ],
424
+ },
425
+ };
426
+ const cachedResponse = {
427
+ pk: `identity#${pixelId}#identity456`,
428
+ pixelId,
429
+ identityId: 'identity456',
430
+ response: {
431
+ identityId: 'identity456',
432
+ traits: {
433
+ emails: ['test@email.com'],
434
+ addresses: [
435
+ {
436
+ city: 'Test City',
437
+ s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
438
+ cra: Date.parse('2023-06-01T00:00:00Z'),
439
+ upa: Date.parse('2023-12-01T00:00:00Z'),
440
+ },
441
+ ],
442
+ },
443
+ },
444
+ updatedAt: new Date().toISOString(),
445
+ };
446
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
447
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
448
+ expect(result).toBeDefined();
449
+ expect(mockInvokeFunction).not.toHaveBeenCalled();
450
+ });
451
+ it('should NOT mutate the original object when stripping timestamps', async () => {
452
+ const originalAddress = {
453
+ city: 'Original City',
454
+ s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
455
+ cra: Date.parse('2024-01-01T00:00:00Z'),
456
+ upa: Date.parse('2024-01-15T00:00:00Z'),
457
+ };
458
+ const incomingIdentity = {
459
+ identityId: 'identity456',
460
+ traits: {
461
+ emails: ['test@email.com'],
462
+ addresses: [{ ...originalAddress }],
463
+ },
464
+ };
465
+ const cachedResponse = {
466
+ pk: `identity#${pixelId}#identity456`,
467
+ pixelId,
468
+ identityId: 'identity456',
469
+ response: {
470
+ identityId: 'identity456',
471
+ traits: {
472
+ emails: ['test@email.com'],
473
+ addresses: [
474
+ {
475
+ city: 'Original City',
476
+ s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
477
+ cra: Date.parse('2023-06-01T00:00:00Z'),
478
+ upa: Date.parse('2023-06-15T00:00:00Z'),
479
+ },
480
+ ],
481
+ },
482
+ },
483
+ updatedAt: new Date().toISOString(),
484
+ };
485
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
486
+ await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
487
+ expect(incomingIdentity.traits?.addresses?.[0]).toHaveProperty('cra');
488
+ expect(incomingIdentity.traits?.addresses?.[0]).toHaveProperty('upa');
489
+ expect(cachedResponse.response.traits?.addresses?.[0]).toHaveProperty('cra');
490
+ expect(cachedResponse.response.traits?.addresses?.[0]).toHaveProperty('upa');
491
+ });
492
+ it('should handle identity with no addresses', async () => {
493
+ const incomingIdentity = {
494
+ identityId: 'identity456',
495
+ traits: {
496
+ emails: ['test@email.com'],
497
+ },
498
+ };
499
+ const cachedResponse = {
500
+ pk: `identity#${pixelId}#identity456`,
501
+ pixelId,
502
+ identityId: 'identity456',
503
+ response: {
504
+ identityId: 'identity456',
505
+ traits: {
506
+ emails: ['test@email.com'],
507
+ },
508
+ },
509
+ updatedAt: new Date().toISOString(),
510
+ };
511
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
512
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
513
+ expect(result).toBeDefined();
514
+ expect(mockInvokeFunction).not.toHaveBeenCalled();
515
+ });
516
+ it('should handle identity with empty addresses array', async () => {
517
+ const incomingIdentity = {
518
+ identityId: 'identity456',
519
+ traits: {
520
+ emails: ['test@email.com'],
521
+ addresses: [],
522
+ },
523
+ };
524
+ const cachedResponse = {
525
+ pk: `identity#${pixelId}#identity456`,
526
+ pixelId,
527
+ identityId: 'identity456',
528
+ response: {
529
+ identityId: 'identity456',
530
+ traits: {
531
+ emails: ['test@email.com'],
532
+ addresses: [],
533
+ },
534
+ },
535
+ updatedAt: new Date().toISOString(),
536
+ };
537
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
538
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
539
+ expect(result).toBeDefined();
540
+ expect(mockInvokeFunction).not.toHaveBeenCalled();
541
+ });
542
+ it('should handle addresses without cra/upa (no-op)', async () => {
543
+ const incomingIdentity = {
544
+ identityId: 'identity456',
545
+ traits: {
546
+ emails: ['test@email.com'],
547
+ addresses: [{ city: 'Simple City', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT }],
548
+ },
549
+ };
550
+ const cachedResponse = {
551
+ pk: `identity#${pixelId}#identity456`,
552
+ pixelId,
553
+ identityId: 'identity456',
554
+ response: {
555
+ identityId: 'identity456',
556
+ traits: {
557
+ emails: ['test@email.com'],
558
+ addresses: [{ city: 'Simple City', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT }],
559
+ },
560
+ },
561
+ updatedAt: new Date().toISOString(),
562
+ };
563
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
564
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
565
+ expect(result).toBeDefined();
566
+ expect(mockInvokeFunction).not.toHaveBeenCalled();
567
+ });
568
+ });
569
+ describe('Identity Cache fastDeepEqual with Timestamp Stripping', () => {
570
+ const pixelId = 'pixel123';
571
+ const lambdaArn = 'arn:aws:lambda:us-east-1:123456789:function:identity-private';
572
+ beforeEach(() => {
573
+ jest.clearAllMocks();
574
+ mockedDynamoDbClient.safeGet.mockReset();
575
+ mockedDynamoDbClient.safeBatchWrite.mockReset();
576
+ mockInvokeFunction.mockReset();
577
+ identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.lambdaInvokeClient = undefined;
578
+ });
579
+ it('should treat cached and merged as equal when they differ ONLY in cra/upa', async () => {
580
+ const incomingIdentity = {
581
+ identityId: 'identity456',
582
+ traits: {
583
+ emails: ['test@email.com'],
584
+ addresses: [
585
+ {
586
+ city: 'Same City',
587
+ province: 'Same State',
588
+ country: 'Same Country',
589
+ s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
590
+ cra: Date.parse('2024-06-01T00:00:00Z'),
591
+ upa: Date.parse('2024-06-15T00:00:00Z'),
592
+ },
593
+ ],
594
+ },
595
+ };
596
+ const cachedResponse = {
597
+ pk: `identity#${pixelId}#identity456`,
598
+ pixelId,
599
+ identityId: 'identity456',
600
+ response: {
601
+ identityId: 'identity456',
602
+ traits: {
603
+ emails: ['test@email.com'],
604
+ addresses: [
605
+ {
606
+ city: 'Same City',
607
+ province: 'Same State',
608
+ country: 'Same Country',
609
+ s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
610
+ cra: Date.parse('2024-01-01T00:00:00Z'),
611
+ upa: Date.parse('2024-01-15T00:00:00Z'),
612
+ },
613
+ ],
614
+ },
615
+ },
616
+ updatedAt: new Date().toISOString(),
617
+ };
618
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
619
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
620
+ expect(result).toBeDefined();
621
+ expect(mockInvokeFunction).not.toHaveBeenCalled();
622
+ expect(mockedDynamoDbClient.safeBatchWrite).not.toHaveBeenCalled();
623
+ });
624
+ it('should treat cached and merged as NOT equal when address content differs (plus cra/upa)', async () => {
625
+ const incomingIdentity = {
626
+ identityId: 'identity456',
627
+ traits: {
628
+ emails: ['test@email.com'],
629
+ addresses: [
630
+ {
631
+ city: 'Different City',
632
+ province: 'Same State',
633
+ s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
634
+ cra: Date.parse('2024-06-01T00:00:00Z'),
635
+ upa: Date.parse('2024-06-15T00:00:00Z'),
636
+ },
637
+ ],
638
+ },
639
+ };
640
+ const cachedResponse = {
641
+ pk: `identity#${pixelId}#identity456`,
642
+ pixelId,
643
+ identityId: 'identity456',
644
+ response: {
645
+ identityId: 'identity456',
646
+ traits: {
647
+ emails: ['test@email.com'],
648
+ addresses: [
649
+ {
650
+ city: 'Original City',
651
+ province: 'Same State',
652
+ s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT,
653
+ cra: Date.parse('2024-01-01T00:00:00Z'),
654
+ upa: Date.parse('2024-01-15T00:00:00Z'),
655
+ },
656
+ ],
657
+ },
658
+ },
659
+ updatedAt: new Date().toISOString(),
660
+ };
661
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
662
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
663
+ expect(result).toBeDefined();
664
+ });
665
+ });
666
+ describe('Identity Cache Logging Levels', () => {
667
+ const pixelId = 'pixel123';
668
+ const lambdaArn = 'arn:aws:lambda:us-east-1:123456789:function:identity-private';
669
+ beforeEach(() => {
670
+ jest.clearAllMocks();
671
+ mockedDynamoDbClient.safeGet.mockReset();
672
+ mockedDynamoDbClient.safeBatchWrite.mockReset();
673
+ mockInvokeFunction.mockReset();
674
+ identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.lambdaInvokeClient = undefined;
675
+ });
676
+ it('should log at debug level when cache is fresh (deep equal)', async () => {
677
+ const incomingIdentity = {
678
+ identityId: 'identity456',
679
+ traits: {
680
+ emails: ['test@email.com'],
681
+ },
682
+ };
683
+ const cachedResponse = {
684
+ pk: `identity#${pixelId}#identity456`,
685
+ pixelId,
686
+ identityId: 'identity456',
687
+ response: {
688
+ identityId: 'identity456',
689
+ traits: {
690
+ emails: ['test@email.com'],
691
+ },
692
+ },
693
+ updatedAt: new Date().toISOString(),
694
+ };
695
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
696
+ await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
697
+ expect(mockedLogger.debug).toHaveBeenCalledWith(expect.stringContaining('IdentityCache - '), expect.any(Object));
698
+ expect(mockInvokeFunction).not.toHaveBeenCalled();
699
+ expect(mockedDynamoDbClient.safeBatchWrite).not.toHaveBeenCalled();
700
+ });
701
+ it('should log at info level for ADDRESS_UPDATE path', async () => {
702
+ const incomingIdentity = {
703
+ identityId: 'identity456',
704
+ traits: {
705
+ emails: ['test@email.com'],
706
+ addresses: [{ city: 'New City', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT }],
707
+ },
708
+ };
709
+ const cachedResponse = {
710
+ pk: `identity#${pixelId}#identity456`,
711
+ pixelId,
712
+ identityId: 'identity456',
713
+ response: {
714
+ identityId: 'identity456',
715
+ traits: {
716
+ emails: ['test@email.com'],
717
+ addresses: [],
718
+ },
719
+ },
720
+ updatedAt: new Date().toISOString(),
721
+ };
722
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
723
+ await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
724
+ expect(mockedLogger.info).toHaveBeenCalledWith('IdentityCache - identity cache write: address update only (skipping Neptune)', expect.objectContaining({
725
+ writeReason: identity_cache_dynamodb_service_1.CACHE_WRITE_REASON.ADDRESS_UPDATE,
726
+ staleTraits: expect.any(Array),
727
+ }));
728
+ });
729
+ it('should log at info level for IDENTITY_STALE path with traitChanges', async () => {
730
+ const incomingIdentity = {
731
+ identityId: 'identity456',
732
+ traits: {
733
+ emails: ['old@email.com', 'new@email.com'],
734
+ },
735
+ };
736
+ const cachedResponse = {
737
+ pk: `identity#${pixelId}#identity456`,
738
+ pixelId,
739
+ identityId: 'identity456',
740
+ response: {
741
+ identityId: 'identity456',
742
+ traits: {
743
+ emails: ['old@email.com'],
744
+ },
745
+ },
746
+ updatedAt: new Date().toISOString(),
747
+ };
748
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
749
+ mockInvokeFunction.mockResolvedValueOnce({
750
+ statusCode: 200,
751
+ body: JSON.stringify({
752
+ identity: {
753
+ identityId: 'identity456',
754
+ traits: { emails: ['old@email.com', 'new@email.com'] },
755
+ },
756
+ }),
757
+ });
758
+ await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityWithCaching(pixelId, incomingIdentity, lambdaArn);
759
+ expect(mockedLogger.info).toHaveBeenCalledWith('IdentityCache - identity cache write: stale_traits_update', expect.objectContaining({
760
+ writeReason: identity_cache_dynamodb_service_1.CACHE_WRITE_REASON.STALE_TRAITS,
761
+ staleTraits: expect.arrayContaining(['emails']),
762
+ traitChanges: expect.any(Object),
763
+ }));
764
+ });
765
+ });
766
+ describe('getIdentityFromCache staleness tier results', () => {
767
+ const pixelId = 'pixel123';
768
+ beforeEach(() => {
769
+ jest.clearAllMocks();
770
+ mockedDynamoDbClient.safeGet.mockReset();
771
+ });
772
+ it('should return staleTier NONE when cached and incoming are deep equal', async () => {
773
+ const incomingIdentity = {
774
+ identityId: 'identity456',
775
+ traits: {
776
+ emails: ['test@email.com'],
777
+ },
778
+ };
779
+ const cachedResponse = {
780
+ pk: `identity#${pixelId}#identity456`,
781
+ pixelId,
782
+ identityId: 'identity456',
783
+ response: {
784
+ identityId: 'identity456',
785
+ traits: {
786
+ emails: ['test@email.com'],
787
+ },
788
+ },
789
+ updatedAt: new Date().toISOString(),
790
+ };
791
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
792
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityFromCache(pixelId, incomingIdentity);
793
+ expect(result.staleTier).toBe(identity_cache_helper_1.IDENTITY_CACHE_STALENESS_TIER.NONE);
794
+ expect(result.isCacheStale).toBe(false);
795
+ expect(result.skipNeptune).toBeUndefined();
796
+ });
797
+ it('should return staleTier ADDRESS_UPDATE and skipNeptune=true for new non-IP address', async () => {
798
+ const incomingIdentity = {
799
+ identityId: 'identity456',
800
+ traits: {
801
+ emails: ['test@email.com'],
802
+ addresses: [{ city: 'New City', s: at_tracking_event_types_1.ADDRESS_INFO_SOURCE.INPUT }],
803
+ },
804
+ };
805
+ const cachedResponse = {
806
+ pk: `identity#${pixelId}#identity456`,
807
+ pixelId,
808
+ identityId: 'identity456',
809
+ response: {
810
+ identityId: 'identity456',
811
+ traits: {
812
+ emails: ['test@email.com'],
813
+ addresses: [],
814
+ },
815
+ },
816
+ updatedAt: new Date().toISOString(),
817
+ };
818
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
819
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityFromCache(pixelId, incomingIdentity);
820
+ expect(result.staleTier).toBe(identity_cache_helper_1.IDENTITY_CACHE_STALENESS_TIER.ADDRESS_UPDATE);
821
+ expect(result.isCacheStale).toBe(true);
822
+ expect(result.skipNeptune).toBe(true);
823
+ expect(result.writeReason).toBe(identity_cache_dynamodb_service_1.CACHE_WRITE_REASON.ADDRESS_UPDATE);
824
+ });
825
+ it('should return staleTier IDENTITY_STALE and skipNeptune=false for new email', async () => {
826
+ const incomingIdentity = {
827
+ identityId: 'identity456',
828
+ traits: {
829
+ emails: ['old@email.com', 'new@email.com'],
830
+ },
831
+ };
832
+ const cachedResponse = {
833
+ pk: `identity#${pixelId}#identity456`,
834
+ pixelId,
835
+ identityId: 'identity456',
836
+ response: {
837
+ identityId: 'identity456',
838
+ traits: {
839
+ emails: ['old@email.com'],
840
+ },
841
+ },
842
+ updatedAt: new Date().toISOString(),
843
+ };
844
+ mockedDynamoDbClient.safeGet.mockResolvedValueOnce(cachedResponse);
845
+ const result = await identity_cache_dynamodb_service_js_1.IdentityCacheDynamoDbService.getIdentityFromCache(pixelId, incomingIdentity);
846
+ expect(result.staleTier).toBe(identity_cache_helper_1.IDENTITY_CACHE_STALENESS_TIER.IDENTITY_STALE);
847
+ expect(result.isCacheStale).toBe(true);
848
+ expect(result.skipNeptune).toBe(false);
849
+ expect(result.writeReason).toBe(identity_cache_dynamodb_service_1.CACHE_WRITE_REASON.STALE_TRAITS);
850
+ });
851
+ });
852
852
  //# sourceMappingURL=identity-cache-tier-routing.spec.js.map