@reactionary/source 0.0.52 → 0.2.16

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 (293) hide show
  1. package/.env-template +19 -0
  2. package/.github/workflows/pull-request.yml +3 -1
  3. package/.github/workflows/release.yml +9 -0
  4. package/.vscode/extensions.json +0 -2
  5. package/LICENSE +21 -0
  6. package/README.md +175 -23
  7. package/core/package.json +6 -3
  8. package/core/src/cache/cache.interface.ts +1 -0
  9. package/core/src/cache/index.ts +4 -0
  10. package/core/src/cache/memory-cache.ts +30 -2
  11. package/core/src/cache/noop-cache.ts +15 -1
  12. package/core/src/cache/redis-cache.ts +20 -0
  13. package/core/src/client/client-builder.ts +71 -54
  14. package/core/src/client/client.ts +9 -47
  15. package/core/src/client/index.ts +2 -0
  16. package/core/src/decorators/index.ts +1 -0
  17. package/core/src/decorators/reactionary.decorator.ts +203 -34
  18. package/core/src/index.ts +6 -19
  19. package/core/src/initialization.ts +1 -18
  20. package/core/src/metrics/metrics.ts +67 -0
  21. package/core/src/providers/analytics.provider.ts +1 -6
  22. package/core/src/providers/base.provider.ts +5 -69
  23. package/core/src/providers/cart.provider.ts +15 -55
  24. package/core/src/providers/category.provider.ts +7 -11
  25. package/core/src/providers/checkout.provider.ts +17 -15
  26. package/core/src/providers/identity.provider.ts +6 -8
  27. package/core/src/providers/index.ts +2 -1
  28. package/core/src/providers/inventory.provider.ts +15 -5
  29. package/core/src/providers/order-search.provider.ts +29 -0
  30. package/core/src/providers/order.provider.ts +47 -15
  31. package/core/src/providers/price.provider.ts +30 -36
  32. package/core/src/providers/product-search.provider.ts +61 -0
  33. package/core/src/providers/product.provider.ts +71 -12
  34. package/core/src/providers/profile.provider.ts +74 -14
  35. package/core/src/providers/store.provider.ts +3 -5
  36. package/core/src/schemas/capabilities.schema.ts +10 -3
  37. package/core/src/schemas/errors/generic.error.ts +9 -0
  38. package/core/src/schemas/errors/index.ts +4 -0
  39. package/core/src/schemas/errors/invalid-input.error.ts +9 -0
  40. package/core/src/schemas/errors/invalid-output.error.ts +9 -0
  41. package/core/src/schemas/errors/not-found.error.ts +9 -0
  42. package/core/src/schemas/index.ts +7 -0
  43. package/core/src/schemas/models/analytics.model.ts +2 -1
  44. package/core/src/schemas/models/base.model.ts +6 -24
  45. package/core/src/schemas/models/cart.model.ts +5 -8
  46. package/core/src/schemas/models/category.model.ts +4 -9
  47. package/core/src/schemas/models/checkout.model.ts +6 -7
  48. package/core/src/schemas/models/cost.model.ts +4 -3
  49. package/core/src/schemas/models/currency.model.ts +2 -1
  50. package/core/src/schemas/models/identifiers.model.ts +106 -62
  51. package/core/src/schemas/models/identity.model.ts +10 -19
  52. package/core/src/schemas/models/index.ts +2 -1
  53. package/core/src/schemas/models/inventory.model.ts +8 -5
  54. package/core/src/schemas/models/order-search.model.ts +28 -0
  55. package/core/src/schemas/models/order.model.ts +20 -26
  56. package/core/src/schemas/models/payment.model.ts +14 -17
  57. package/core/src/schemas/models/price.model.ts +11 -11
  58. package/core/src/schemas/models/product-search.model.ts +42 -0
  59. package/core/src/schemas/models/product.model.ts +64 -22
  60. package/core/src/schemas/models/profile.model.ts +19 -22
  61. package/core/src/schemas/models/shipping-method.model.ts +24 -29
  62. package/core/src/schemas/models/store.model.ts +9 -5
  63. package/core/src/schemas/mutations/analytics.mutation.ts +8 -7
  64. package/core/src/schemas/mutations/base.mutation.ts +2 -1
  65. package/core/src/schemas/mutations/cart.mutation.ts +33 -33
  66. package/core/src/schemas/mutations/checkout.mutation.ts +23 -30
  67. package/core/src/schemas/mutations/identity.mutation.ts +4 -3
  68. package/core/src/schemas/mutations/profile.mutation.ts +38 -3
  69. package/core/src/schemas/queries/base.query.ts +2 -1
  70. package/core/src/schemas/queries/cart.query.ts +3 -3
  71. package/core/src/schemas/queries/category.query.ts +18 -18
  72. package/core/src/schemas/queries/checkout.query.ts +7 -9
  73. package/core/src/schemas/queries/identity.query.ts +2 -1
  74. package/core/src/schemas/queries/index.ts +2 -1
  75. package/core/src/schemas/queries/inventory.query.ts +5 -5
  76. package/core/src/schemas/queries/order-search.query.ts +10 -0
  77. package/core/src/schemas/queries/order.query.ts +3 -2
  78. package/core/src/schemas/queries/price.query.ts +10 -4
  79. package/core/src/schemas/queries/product-search.query.ts +16 -0
  80. package/core/src/schemas/queries/product.query.ts +13 -6
  81. package/core/src/schemas/queries/profile.query.ts +5 -2
  82. package/core/src/schemas/queries/store.query.ts +6 -5
  83. package/core/src/schemas/result.ts +107 -0
  84. package/core/src/schemas/session.schema.ts +4 -4
  85. package/core/src/test/reactionary.decorator.spec.ts +249 -0
  86. package/core/src/zod-utils.ts +19 -0
  87. package/core/tsconfig.json +1 -1
  88. package/core/tsconfig.spec.json +2 -26
  89. package/core/vitest.config.ts +14 -0
  90. package/documentation/1-purpose.md +114 -0
  91. package/documentation/2-getting-started.md +229 -0
  92. package/documentation/3-querying-and-changing-data.md +74 -0
  93. package/documentation/4-product-data.md +107 -0
  94. package/documentation/5-cart-and-checkout.md +211 -0
  95. package/documentation/6-product-search.md +143 -0
  96. package/documentation/7-marketing.md +3 -0
  97. package/eslint.config.mjs +1 -0
  98. package/examples/node/eslint.config.mjs +1 -4
  99. package/examples/node/package.json +10 -3
  100. package/examples/node/project.json +4 -1
  101. package/examples/node/src/basic/basic-node-provider-model-extension.spec.ts +22 -23
  102. package/examples/node/src/basic/basic-node-provider-query-extension.spec.ts +15 -11
  103. package/examples/node/src/basic/basic-node-setup.spec.ts +44 -28
  104. package/examples/node/src/basic/client-creation.spec.ts +53 -0
  105. package/examples/node/src/capabilities/cart.spec.ts +255 -0
  106. package/examples/node/src/capabilities/category.spec.ts +193 -0
  107. package/examples/node/src/capabilities/checkout.spec.ts +341 -0
  108. package/examples/node/src/capabilities/identity.spec.ts +93 -0
  109. package/examples/node/src/capabilities/inventory.spec.ts +66 -0
  110. package/examples/node/src/capabilities/order-search.spec.ts +265 -0
  111. package/examples/node/src/capabilities/order.spec.ts +91 -0
  112. package/examples/node/src/capabilities/price.spec.ts +51 -0
  113. package/examples/node/src/capabilities/product-search.spec.ts +293 -0
  114. package/examples/node/src/capabilities/product.spec.ts +122 -0
  115. package/examples/node/src/capabilities/profile.spec.ts +316 -0
  116. package/examples/node/src/capabilities/store.spec.ts +26 -0
  117. package/examples/node/src/utils.ts +147 -0
  118. package/examples/node/tsconfig.json +9 -12
  119. package/examples/node/tsconfig.lib.json +1 -2
  120. package/examples/node/tsconfig.spec.json +2 -14
  121. package/examples/node/vitest.config.ts +14 -0
  122. package/migrations.json +22 -5
  123. package/nx.json +8 -47
  124. package/package.json +24 -96
  125. package/providers/algolia/README.md +39 -2
  126. package/providers/algolia/package.json +2 -1
  127. package/providers/algolia/src/core/initialize.ts +7 -14
  128. package/providers/algolia/src/index.ts +2 -4
  129. package/providers/algolia/src/providers/index.ts +1 -0
  130. package/providers/algolia/src/providers/product-search.provider.ts +241 -0
  131. package/providers/algolia/src/schema/capabilities.schema.ts +2 -3
  132. package/providers/algolia/src/schema/index.ts +3 -0
  133. package/providers/algolia/src/schema/search.schema.ts +8 -8
  134. package/providers/algolia/tsconfig.json +1 -1
  135. package/providers/algolia/tsconfig.lib.json +1 -1
  136. package/providers/algolia/tsconfig.spec.json +2 -14
  137. package/providers/algolia/vitest.config.ts +14 -0
  138. package/providers/commercetools/README.md +30 -3
  139. package/providers/commercetools/package.json +2 -1
  140. package/providers/commercetools/src/core/client.ts +178 -99
  141. package/providers/commercetools/src/core/initialize.ts +130 -74
  142. package/providers/commercetools/src/core/token-cache.ts +45 -0
  143. package/providers/commercetools/src/index.ts +3 -2
  144. package/providers/commercetools/src/providers/cart.provider.ts +281 -341
  145. package/providers/commercetools/src/providers/category.provider.ts +223 -138
  146. package/providers/commercetools/src/providers/checkout.provider.ts +631 -449
  147. package/providers/commercetools/src/providers/identity.provider.ts +50 -29
  148. package/providers/commercetools/src/providers/index.ts +2 -2
  149. package/providers/commercetools/src/providers/inventory.provider.ts +76 -74
  150. package/providers/commercetools/src/providers/order-search.provider.ts +220 -0
  151. package/providers/commercetools/src/providers/order.provider.ts +96 -61
  152. package/providers/commercetools/src/providers/price.provider.ts +147 -117
  153. package/providers/commercetools/src/providers/product-search.provider.ts +528 -0
  154. package/providers/commercetools/src/providers/product.provider.ts +249 -74
  155. package/providers/commercetools/src/providers/profile.provider.ts +445 -28
  156. package/providers/commercetools/src/providers/store.provider.ts +54 -40
  157. package/providers/commercetools/src/schema/capabilities.schema.ts +3 -1
  158. package/providers/commercetools/src/schema/commercetools.schema.ts +17 -3
  159. package/providers/commercetools/src/schema/configuration.schema.ts +1 -0
  160. package/providers/commercetools/src/schema/session.schema.ts +7 -0
  161. package/providers/commercetools/src/test/caching.spec.ts +82 -0
  162. package/providers/commercetools/src/test/identity.spec.ts +109 -0
  163. package/providers/commercetools/src/test/test-utils.ts +21 -19
  164. package/providers/commercetools/tsconfig.json +1 -1
  165. package/providers/commercetools/tsconfig.lib.json +1 -1
  166. package/providers/commercetools/tsconfig.spec.json +2 -14
  167. package/providers/commercetools/vitest.config.ts +15 -0
  168. package/providers/fake/README.md +20 -4
  169. package/providers/fake/package.json +2 -1
  170. package/providers/fake/src/core/initialize.ts +47 -49
  171. package/providers/fake/src/providers/analytics.provider.ts +5 -7
  172. package/providers/fake/src/providers/cart.provider.ts +163 -92
  173. package/providers/fake/src/providers/category.provider.ts +78 -50
  174. package/providers/fake/src/providers/checkout.provider.ts +254 -0
  175. package/providers/fake/src/providers/identity.provider.ts +57 -65
  176. package/providers/fake/src/providers/index.ts +6 -2
  177. package/providers/fake/src/providers/inventory.provider.ts +40 -36
  178. package/providers/fake/src/providers/order-search.provider.ts +78 -0
  179. package/providers/fake/src/providers/order.provider.ts +106 -0
  180. package/providers/fake/src/providers/price.provider.ts +93 -41
  181. package/providers/fake/src/providers/product-search.provider.ts +206 -0
  182. package/providers/fake/src/providers/product.provider.ts +56 -41
  183. package/providers/fake/src/providers/profile.provider.ts +147 -0
  184. package/providers/fake/src/providers/store.provider.ts +30 -20
  185. package/providers/fake/src/schema/capabilities.schema.ts +5 -1
  186. package/providers/fake/src/test/cart.provider.spec.ts +59 -80
  187. package/providers/fake/src/test/category.provider.spec.ts +145 -87
  188. package/providers/fake/src/test/checkout.provider.spec.ts +222 -0
  189. package/providers/fake/src/test/order-search.provider.spec.ts +50 -0
  190. package/providers/fake/src/test/order.provider.spec.ts +44 -0
  191. package/providers/fake/src/test/price.provider.spec.ts +50 -45
  192. package/providers/fake/src/test/product.provider.spec.ts +15 -7
  193. package/providers/fake/src/test/profile.provider.spec.ts +167 -0
  194. package/providers/fake/tsconfig.json +1 -1
  195. package/providers/fake/tsconfig.lib.json +1 -1
  196. package/providers/fake/tsconfig.spec.json +2 -12
  197. package/providers/fake/vitest.config.ts +14 -0
  198. package/providers/medusa/README.md +30 -0
  199. package/providers/medusa/TESTING.md +98 -0
  200. package/providers/medusa/eslint.config.mjs +19 -0
  201. package/providers/medusa/package.json +22 -0
  202. package/providers/medusa/project.json +34 -0
  203. package/providers/medusa/src/core/client.ts +370 -0
  204. package/providers/medusa/src/core/initialize.ts +78 -0
  205. package/providers/medusa/src/index.ts +13 -0
  206. package/providers/medusa/src/providers/cart.provider.ts +575 -0
  207. package/providers/medusa/src/providers/category.provider.ts +247 -0
  208. package/providers/medusa/src/providers/checkout.provider.ts +636 -0
  209. package/providers/medusa/src/providers/identity.provider.ts +137 -0
  210. package/providers/medusa/src/providers/inventory.provider.ts +173 -0
  211. package/providers/medusa/src/providers/order-search.provider.ts +202 -0
  212. package/providers/medusa/src/providers/order.provider.ts +226 -0
  213. package/providers/medusa/src/providers/price.provider.ts +140 -0
  214. package/providers/medusa/src/providers/product-search.provider.ts +243 -0
  215. package/providers/medusa/src/providers/product.provider.ts +261 -0
  216. package/providers/medusa/src/providers/profile.provider.ts +392 -0
  217. package/providers/medusa/src/schema/capabilities.schema.ts +18 -0
  218. package/providers/medusa/src/schema/configuration.schema.ts +11 -0
  219. package/providers/medusa/src/schema/medusa.schema.ts +31 -0
  220. package/providers/medusa/src/test/cart.provider.spec.ts +240 -0
  221. package/providers/medusa/src/test/category.provider.spec.ts +231 -0
  222. package/providers/medusa/src/test/checkout.spec.ts +349 -0
  223. package/providers/medusa/src/test/identity.provider.spec.ts +122 -0
  224. package/providers/medusa/src/test/inventory.provider.spec.ts +88 -0
  225. package/providers/medusa/src/test/large-cart.provider.spec.ts +103 -0
  226. package/providers/medusa/src/test/price.provider.spec.ts +104 -0
  227. package/providers/medusa/src/test/product.provider.spec.ts +146 -0
  228. package/providers/medusa/src/test/search.provider.spec.ts +203 -0
  229. package/providers/medusa/src/test/test-utils.ts +13 -0
  230. package/providers/medusa/src/utils/medusa-helpers.ts +89 -0
  231. package/providers/medusa/tsconfig.json +21 -0
  232. package/providers/medusa/tsconfig.lib.json +9 -0
  233. package/providers/medusa/tsconfig.spec.json +4 -0
  234. package/providers/medusa/vitest.config.ts +15 -0
  235. package/providers/meilisearch/README.md +48 -0
  236. package/providers/meilisearch/eslint.config.mjs +22 -0
  237. package/providers/meilisearch/package.json +13 -0
  238. package/providers/meilisearch/project.json +34 -0
  239. package/providers/meilisearch/src/core/initialize.ts +21 -0
  240. package/providers/meilisearch/src/index.ts +6 -0
  241. package/providers/meilisearch/src/providers/index.ts +1 -0
  242. package/providers/meilisearch/src/providers/order-search.provider.ts +222 -0
  243. package/providers/meilisearch/src/providers/product-search.provider.ts +251 -0
  244. package/providers/meilisearch/src/schema/capabilities.schema.ts +10 -0
  245. package/providers/meilisearch/src/schema/configuration.schema.ts +11 -0
  246. package/providers/meilisearch/src/schema/index.ts +3 -0
  247. package/providers/meilisearch/src/schema/search.schema.ts +14 -0
  248. package/providers/meilisearch/tsconfig.json +24 -0
  249. package/providers/meilisearch/tsconfig.lib.json +10 -0
  250. package/providers/meilisearch/tsconfig.spec.json +4 -0
  251. package/providers/meilisearch/vitest.config.ts +14 -0
  252. package/providers/posthog/package.json +2 -1
  253. package/providers/posthog/tsconfig.json +1 -1
  254. package/tsconfig.base.json +5 -0
  255. package/vitest.config.ts +10 -0
  256. package/core/src/providers/search.provider.ts +0 -18
  257. package/core/src/schemas/models/search.model.ts +0 -36
  258. package/core/src/schemas/queries/search.query.ts +0 -9
  259. package/examples/next/.swcrc +0 -30
  260. package/examples/next/eslint.config.mjs +0 -21
  261. package/examples/next/index.d.ts +0 -6
  262. package/examples/next/next-env.d.ts +0 -5
  263. package/examples/next/next.config.js +0 -31
  264. package/examples/next/project.json +0 -9
  265. package/examples/next/public/.gitkeep +0 -0
  266. package/examples/next/public/favicon.ico +0 -0
  267. package/examples/next/src/app/global.css +0 -0
  268. package/examples/next/src/app/layout.tsx +0 -18
  269. package/examples/next/src/app/page.module.scss +0 -2
  270. package/examples/next/src/app/page.tsx +0 -47
  271. package/examples/next/src/instrumentation.ts +0 -9
  272. package/examples/next/tsconfig.json +0 -44
  273. package/examples/node/jest.config.ts +0 -10
  274. package/jest.config.ts +0 -6
  275. package/jest.preset.js +0 -3
  276. package/providers/algolia/jest.config.ts +0 -10
  277. package/providers/algolia/src/providers/product.provider.ts +0 -66
  278. package/providers/algolia/src/providers/search.provider.ts +0 -106
  279. package/providers/algolia/src/test/search.provider.spec.ts +0 -91
  280. package/providers/commercetools/jest.config.cjs +0 -10
  281. package/providers/commercetools/src/providers/search.provider.ts +0 -96
  282. package/providers/commercetools/src/test/cart.provider.spec.ts +0 -199
  283. package/providers/commercetools/src/test/category.provider.spec.ts +0 -168
  284. package/providers/commercetools/src/test/checkout.provider.spec.ts +0 -312
  285. package/providers/commercetools/src/test/identity.provider.spec.ts +0 -88
  286. package/providers/commercetools/src/test/inventory.provider.spec.ts +0 -41
  287. package/providers/commercetools/src/test/price.provider.spec.ts +0 -81
  288. package/providers/commercetools/src/test/product.provider.spec.ts +0 -80
  289. package/providers/commercetools/src/test/profile.provider.spec.ts +0 -49
  290. package/providers/commercetools/src/test/search.provider.spec.ts +0 -61
  291. package/providers/commercetools/src/test/store.provider.spec.ts +0 -37
  292. package/providers/fake/jest.config.cjs +0 -10
  293. package/providers/fake/src/providers/search.provider.ts +0 -132
@@ -0,0 +1,349 @@
1
+ import type { Cart, Checkout, RequestContext, ShippingInstruction } from '@reactionary/core';
2
+ import {
3
+ ClientBuilder,
4
+ createInitialRequestContext,
5
+ NoOpCache,
6
+ PaymentInstructionSchema,
7
+ ShippingInstructionSchema,
8
+ unwrapValue,
9
+ } from '@reactionary/core';
10
+ import 'dotenv/config';
11
+ import { assert, beforeEach, describe, expect, it } from 'vitest';
12
+ import { withMedusaCapabilities } from '../core/initialize.js';
13
+ import type { MedusaConfiguration } from '../schema/configuration.schema.js';
14
+ import { getMedusaTestConfiguration } from './test-utils.js';
15
+
16
+ const testData = {
17
+ skuWithoutTiers: '0766623301831',
18
+ skuWithTiers: '0766623360203',
19
+ };
20
+
21
+ function createMedusaClient(config: MedusaConfiguration, reqCtx: RequestContext) {
22
+ const builder = new ClientBuilder(reqCtx)
23
+ .withCache(new NoOpCache())
24
+ .withCapability(
25
+ withMedusaCapabilities(config, {
26
+ cart: true,
27
+ product: true,
28
+ category: true,
29
+ checkout: true,
30
+ identity: true,
31
+ inventory: true,
32
+ order: true,
33
+ price: true,
34
+ productSearch: true,
35
+ })
36
+ );
37
+
38
+ return builder.build();
39
+ }
40
+
41
+ describe.each(['Medusa'])('Checkout Capability - %s', (provider) => {
42
+ let client: ReturnType<typeof createMedusaClient>;
43
+
44
+ beforeEach(() => {
45
+ const reqCtx = createInitialRequestContext();
46
+ const config = getMedusaTestConfiguration();
47
+ client = createMedusaClient(config, reqCtx);
48
+ });
49
+
50
+ describe('anonymous sessions', () => {
51
+ let cart: Cart;
52
+
53
+ beforeEach(async () => {
54
+ cart = unwrapValue(await client.cart.add(
55
+ {
56
+ variant: {
57
+ sku: testData.skuWithoutTiers,
58
+ },
59
+ quantity: 1,
60
+ },
61
+ ));
62
+ });
63
+
64
+ it('can create a checkout session from a cart', async () => {
65
+ // we have either an anonymous user, or an authenticated user.
66
+ // if it is anonymous, we assume you will have collected some basic info by now ?
67
+
68
+ const checkout = await client.checkout.initiateCheckoutForCart(
69
+ {
70
+ cart: cart,
71
+ billingAddress: {
72
+ countryCode: 'DK',
73
+ firstName: 'John',
74
+ lastName: 'Doe',
75
+ streetAddress: '123 Main St',
76
+ streetNumber: '1A',
77
+ postalCode: '12345',
78
+ city: 'Anytown',
79
+ region: '',
80
+ },
81
+ notificationEmail: 'sample@example.com',
82
+ notificationPhone: '+4512345678',
83
+ }
84
+ );
85
+
86
+ if (!checkout.success) {
87
+ assert.fail();
88
+ }
89
+
90
+ expect(checkout.value.identifier.key).toBeDefined();
91
+ expect(checkout.value.originalCartReference.key).toBe(cart.identifier.key);
92
+ expect(checkout.value.billingAddress?.firstName).toBe('John');
93
+ expect(checkout.value.items.length).toBe(1);
94
+ expect(checkout.value.items[0].variant.sku).toBe(testData.skuWithoutTiers);
95
+ });
96
+
97
+ describe('checkout actions', () => {
98
+ let checkout: Checkout;
99
+ beforeEach(async () => {
100
+ checkout = unwrapValue(await client.checkout.initiateCheckoutForCart(
101
+ {
102
+ cart: cart,
103
+ billingAddress: {
104
+ countryCode: 'DK',
105
+ firstName: 'John',
106
+ lastName: 'Doe',
107
+ streetAddress: '123 Main St',
108
+ streetNumber: '1A',
109
+ postalCode: '12345',
110
+ city: 'Anytown',
111
+ region: '',
112
+ },
113
+ notificationEmail: 'sample@example.com',
114
+ notificationPhone: '+4512345678',
115
+ }
116
+ ));
117
+ });
118
+
119
+ it('can list payment methods', async () => {
120
+ const paymentMethods = await client.checkout.getAvailablePaymentMethods(
121
+ {
122
+ checkout: checkout.identifier,
123
+ }
124
+ );
125
+
126
+ if (!paymentMethods.success) {
127
+ assert.fail();
128
+ }
129
+
130
+ expect(paymentMethods.value.length).toBeGreaterThan(0);
131
+ expect(
132
+ paymentMethods.value.find((x) => x.identifier.method === 'pp_stripe_stripe')
133
+ ).toBeDefined();
134
+ });
135
+
136
+ it('can list shipping methods', async () => {
137
+ const shippingMethods = await client.checkout.getAvailableShippingMethods(
138
+ {
139
+ checkout: checkout.identifier,
140
+ }
141
+ );
142
+
143
+ if (!shippingMethods.success) {
144
+ assert.fail();
145
+ }
146
+
147
+ expect(shippingMethods.value.length).toBeGreaterThan(0);
148
+ expect(
149
+ shippingMethods.value.find((x) => x.name === 'Standard Shipping')
150
+ ).toBeDefined();
151
+ });
152
+
153
+ it('can add a payment instruction', async () => {
154
+ const paymentMethods = await client.checkout.getAvailablePaymentMethods(
155
+ {
156
+ checkout: checkout.identifier,
157
+ }
158
+ );
159
+
160
+ if (!paymentMethods.success) {
161
+ assert.fail();
162
+ }
163
+
164
+ const pm = paymentMethods.value.find((x) => x.identifier.method === 'pp_stripe_stripe');
165
+ expect(pm).toBeDefined();
166
+
167
+ const checkoutWithPi = await client.checkout.addPaymentInstruction(
168
+ {
169
+ checkout: checkout.identifier,
170
+ paymentInstruction: {
171
+ paymentMethod: pm!.identifier,
172
+ amount: checkout.price.grandTotal,
173
+ protocolData: [{ key: 'test-key', value: 'test-value' }],
174
+ },
175
+ }
176
+ );
177
+
178
+ if (!checkoutWithPi.success) {
179
+ assert.fail();
180
+ }
181
+
182
+ expect(checkoutWithPi.value.paymentInstructions.length).toBe(1);
183
+ expect(checkoutWithPi.value.paymentInstructions[0].paymentMethod.method).toBe(
184
+ 'pp_stripe_stripe'
185
+ );
186
+ expect(checkoutWithPi.value.paymentInstructions[0].protocolData.find(x => x.key === 'client_secret')?.value).toBeDefined();
187
+
188
+ });
189
+
190
+ it.skip('can cancel an in-progress payment', async () => {
191
+ const paymentMethods = await client.checkout.getAvailablePaymentMethods(
192
+ {
193
+ checkout: checkout.identifier,
194
+ }
195
+ );
196
+
197
+ if (!paymentMethods.success) {
198
+ assert.fail();
199
+ }
200
+
201
+ const pm = paymentMethods.value.find((x) => x.identifier.method === 'pp_stripe_stripe');
202
+ expect(pm).toBeDefined();
203
+
204
+ const checkoutWithPi = await client.checkout.addPaymentInstruction(
205
+ {
206
+ checkout: checkout.identifier,
207
+ paymentInstruction: {
208
+ paymentMethod: pm!.identifier,
209
+ amount: checkout.price.grandTotal,
210
+ protocolData: [{ key: 'test-key', value: 'test-value' }],
211
+ },
212
+ }
213
+ );
214
+
215
+ if (!checkoutWithPi.success) {
216
+ assert.fail();
217
+ }
218
+
219
+ expect(checkoutWithPi.value.paymentInstructions.length).toBe(1);
220
+
221
+ const checkoutAfterCancel = await client.checkout.removePaymentInstruction(
222
+ {
223
+ checkout: checkout.identifier,
224
+ paymentInstruction:
225
+ checkoutWithPi.value.paymentInstructions[0].identifier,
226
+ }
227
+ );
228
+
229
+ if (!checkoutAfterCancel.success) {
230
+ assert.fail();
231
+ }
232
+
233
+ expect(checkoutAfterCancel.value.paymentInstructions.length).toBe(0);
234
+ });
235
+
236
+ it('can set shipping address', async () => {
237
+ const checkoutWithShipping = await client.checkout.setShippingAddress(
238
+ {
239
+ checkout: checkout.identifier,
240
+ shippingAddress: {
241
+ countryCode: 'DK',
242
+ firstName: 'Jane',
243
+ lastName: 'Doe',
244
+ streetAddress: '456 Other St',
245
+ streetNumber: '2B',
246
+ postalCode: '54321',
247
+ city: 'Othertown',
248
+ region: '',
249
+ },
250
+ }
251
+ );
252
+
253
+ if (!checkoutWithShipping.success) {
254
+ assert.fail();
255
+ }
256
+
257
+ expect(checkoutWithShipping.value.shippingAddress).toBeDefined();
258
+ expect(checkoutWithShipping.value.shippingAddress?.firstName).toBe('Jane');
259
+ });
260
+
261
+ it('can set shipping instructions', async () => {
262
+ const shippingMethods = await client.checkout.getAvailableShippingMethods(
263
+ {
264
+ checkout: checkout.identifier,
265
+ }
266
+ );
267
+
268
+ if (!shippingMethods.success) {
269
+ assert.fail();
270
+ }
271
+
272
+ const sm = shippingMethods.value.find((x) => x.name === 'Standard Shipping');
273
+ expect(sm).toBeDefined();
274
+
275
+ const shippingInstruction: ShippingInstruction = {
276
+ shippingMethod: sm?.identifier || { key: '' },
277
+ instructions: 'Leave at front door if not home',
278
+ consentForUnattendedDelivery: true,
279
+ pickupPoint: '4190asx141',
280
+ };
281
+
282
+ const checkoutWithShipping = await client.checkout.setShippingInstruction(
283
+ {
284
+ checkout: checkout.identifier,
285
+ shippingInstruction,
286
+ }
287
+ );
288
+
289
+ if (!checkoutWithShipping.success) {
290
+ assert.fail();
291
+ }
292
+
293
+ expect(checkout.price.totalShipping.value).toBe(0);
294
+ expect(checkoutWithShipping.value.price.totalShipping.value).toBeGreaterThan(0);
295
+ expect(checkoutWithShipping.value.shippingInstruction).toBeDefined();
296
+ expect(
297
+ checkoutWithShipping.value.shippingInstruction?.shippingMethod.key
298
+ ).toBe(sm?.identifier.key);
299
+ expect(checkoutWithShipping.value.shippingInstruction?.instructions).toBe(
300
+ 'Leave at front door if not home'
301
+ );
302
+ expect(checkoutWithShipping.value.shippingInstruction?.pickupPoint).toBe(
303
+ '4190asx141'
304
+ );
305
+ expect(
306
+ checkoutWithShipping.value.shippingInstruction?.consentForUnattendedDelivery
307
+ ).toBe(true);
308
+ });
309
+
310
+ it.skip('wont report it finalizable until everything is paid/authorized', async () => {
311
+ expect(checkout.readyForFinalization).toBe(false);
312
+ const pm = unwrapValue(
313
+ await client.checkout.getAvailablePaymentMethods(
314
+ {
315
+ checkout: checkout.identifier,
316
+ }
317
+ )
318
+ ).find((x) => x.identifier.method === 'stripe');
319
+
320
+
321
+ expect(pm).toBeDefined();
322
+
323
+ const checkoutWithPi = await client.checkout.addPaymentInstruction(
324
+ {
325
+ checkout: checkout.identifier,
326
+ paymentInstruction: {
327
+ paymentMethod: pm!.identifier,
328
+ amount: checkout.price.grandTotal,
329
+ protocolData: [{ key: 'test-key', value: 'test-value' }],
330
+ },
331
+ }
332
+ );
333
+
334
+ if (!checkoutWithPi.success) {
335
+ assert.fail();
336
+ }
337
+
338
+ // do something to simulate payment authorization ?
339
+ const checkoutReady = await client.checkout.getById(
340
+ { identifier: checkoutWithPi.value.identifier },
341
+ );
342
+ if (!checkoutReady.success || !checkoutReady) {
343
+ expect.fail('checkout not found');
344
+ }
345
+ expect(checkoutReady.value.readyForFinalization).toBe(true);
346
+ });
347
+ });
348
+ });
349
+ });
@@ -0,0 +1,122 @@
1
+ import { describe, it, expect, beforeEach, assert } from 'vitest';
2
+ import { MedusaIdentityProvider } from '../providers/identity.provider.js';
3
+ import { IdentitySchema, NoOpCache, createInitialRequestContext, type RequestContext } from '@reactionary/core';
4
+ import { getMedusaTestConfiguration } from './test-utils.js';
5
+ import { MedusaAPI } from '../core/client.js';
6
+
7
+ const testData = {
8
+ testEmail: 'test@example.com',
9
+ testPassword: 'TestPassword123!',
10
+ testFirstName: 'Test',
11
+ testLastName: 'User'
12
+ };
13
+
14
+ describe('Medusa Identity Provider', () => {
15
+ let provider: MedusaIdentityProvider;
16
+ let reqCtx: RequestContext;
17
+
18
+ beforeEach(() => {
19
+ reqCtx = createInitialRequestContext();
20
+ const config = getMedusaTestConfiguration();
21
+ const client = new MedusaAPI(config, reqCtx);
22
+
23
+ provider = new MedusaIdentityProvider(config, new NoOpCache(), reqCtx, client);
24
+ });
25
+
26
+ it('should return anonymous identity when not authenticated', async () => {
27
+ const result = await provider.getSelf({});
28
+
29
+ if (!result.success) {
30
+ assert.fail();
31
+ }
32
+
33
+ expect(result.value).toBeTruthy();
34
+ expect(result.value.type).toBe('Anonymous');
35
+ });
36
+
37
+ it('should be able to register a new user', async () => {
38
+ const testEmail = `test-user+${new Date().getTime()}@example.com`;
39
+ const result = await provider.register({
40
+ username: testEmail,
41
+ password: testData.testPassword,
42
+ });
43
+
44
+ if (!result.success) {
45
+ assert.fail();
46
+ }
47
+
48
+ expect(result).toBeTruthy();
49
+ // Registration should auto-login and return Registered identity
50
+ expect(['Registered', 'Anonymous']).toContain(result.value.type);
51
+ });
52
+
53
+ it('should be able to login with valid credentials', async () => {
54
+ try {
55
+ const result = await provider.register({
56
+ username: testData.testEmail,
57
+ password: testData.testPassword,
58
+ });
59
+ await provider.logout({});
60
+
61
+ } catch (err) {
62
+ // already exists, its fine
63
+ }
64
+
65
+ const result = await provider.login({
66
+ username: testData.testEmail,
67
+ password: testData.testPassword,
68
+ });
69
+
70
+ if (!result.success) {
71
+ assert.fail();
72
+ }
73
+
74
+ expect(result).toBeTruthy();
75
+ expect(['Registered', 'Anonymous']).toContain(result.value.type);
76
+ });
77
+
78
+ it('should be able to logout', async () => {
79
+ // First login
80
+ await provider.login({
81
+ username: testData.testEmail,
82
+ password: testData.testPassword,
83
+ });
84
+
85
+ // Then logout
86
+ const result = await provider.logout({});
87
+
88
+ if (!result.success) {
89
+ assert.fail();
90
+ }
91
+
92
+ expect(result).toBeTruthy();
93
+ expect(result.value.type).toBe('Anonymous');
94
+ });
95
+
96
+ it('should return registered identity when authenticated', async () => {
97
+ // Login first
98
+ await provider.login({
99
+ username: testData.testEmail,
100
+ password: testData.testPassword,
101
+ });
102
+
103
+ // Check identity
104
+ const result = await provider.getSelf({});
105
+
106
+ if (!result.success) {
107
+ assert.fail();
108
+ }
109
+
110
+ expect(result).toBeTruthy();
111
+ expect(['Registered', 'Anonymous']).toContain(result.value.type);
112
+ });
113
+
114
+ it('should handle login errors gracefully', async () => {
115
+ await expect(async () => {
116
+ await provider.login({
117
+ username: 'nonexistent@example.com',
118
+ password: 'wrongpassword',
119
+ });
120
+ }).rejects.toThrow();
121
+ });
122
+ });
@@ -0,0 +1,88 @@
1
+ import { describe, it, expect, beforeEach, assert } from 'vitest';
2
+ import { MedusaInventoryProvider } from '../providers/inventory.provider.js';
3
+ import {
4
+ InventorySchema,
5
+ NoOpCache,
6
+ createInitialRequestContext,
7
+ type RequestContext,
8
+ } from '@reactionary/core';
9
+ import { getMedusaTestConfiguration } from './test-utils.js';
10
+ import { MedusaAPI } from '../core/client.js';
11
+
12
+ const testData = {
13
+ skuInStock: ' 4007249524126',
14
+ skuOutOfStock: '4007249524645',
15
+ fulfillmentCenter: 'European Warehouse',
16
+ };
17
+
18
+ // FIXME: Currently broken in terms of actually looking up anything...
19
+ describe('Medusa Inventory Provider', () => {
20
+ let provider: MedusaInventoryProvider;
21
+ let reqCtx: RequestContext;
22
+
23
+ beforeEach(() => {
24
+ reqCtx = createInitialRequestContext();
25
+ const config = getMedusaTestConfiguration();
26
+ const client = new MedusaAPI(config, reqCtx);
27
+
28
+ provider = new MedusaInventoryProvider(
29
+ config,
30
+ new NoOpCache(),
31
+ reqCtx,
32
+ client
33
+ );
34
+ });
35
+
36
+ it('should be able to get inventory for a SKU with stock', async () => {
37
+ const result = await provider.getBySKU({
38
+ variant: { sku: testData.skuInStock },
39
+ fulfilmentCenter: { key: testData.fulfillmentCenter },
40
+ });
41
+
42
+ if (!result.success) {
43
+ assert.fail();
44
+ }
45
+
46
+ expect(result.value.identifier.variant.sku).toBe(testData.skuInStock);
47
+ expect(result.value.identifier.fulfillmentCenter.key).toBe(
48
+ testData.fulfillmentCenter
49
+ );
50
+ expect(result.value.quantity).toBeGreaterThanOrEqual(0);
51
+ if (result.value.quantity > 0) {
52
+ expect(result.value.status).toBe('inStock');
53
+ }
54
+ });
55
+
56
+ it('should return out of stock for a SKU without inventory', async () => {
57
+ const result = await provider.getBySKU({
58
+ variant: { sku: testData.skuOutOfStock },
59
+ fulfilmentCenter: { key: testData.fulfillmentCenter },
60
+ });
61
+
62
+ if (!result.success) {
63
+ assert.fail();
64
+ }
65
+
66
+ expect(result.value.identifier.variant.sku).toBe(testData.skuOutOfStock);
67
+ expect(result.value.identifier.fulfillmentCenter.key).toBe(
68
+ testData.fulfillmentCenter
69
+ );
70
+ expect(result.value.quantity).toBe(0);
71
+ expect(result.value.status).toBe('outOfStock');
72
+ });
73
+
74
+ it('should return placeholder inventory for an unknown SKU', async () => {
75
+ const result = await provider.getBySKU({
76
+ variant: { sku: 'unknown-sku' },
77
+ fulfilmentCenter: { key: testData.fulfillmentCenter },
78
+ });
79
+
80
+ if (!result.success) {
81
+ assert.fail();
82
+ }
83
+
84
+ expect(result.value.identifier.variant.sku).toBe('unknown-sku');
85
+ expect(result.value.quantity).toBe(0);
86
+ expect(result.value.status).toBe('outOfStock');
87
+ });
88
+ });
@@ -0,0 +1,103 @@
1
+ import {
2
+ CartSchema,
3
+ NoOpCache,
4
+ ProductSearchQueryByTermSchema,
5
+ ProductSearchResultItemSchema,
6
+ createInitialRequestContext,
7
+ type ProductSearchQueryByTerm,
8
+ type RequestContext,
9
+ } from '@reactionary/core';
10
+ import { assert, beforeAll, beforeEach, describe, expect, it } from 'vitest';
11
+ import { MedusaCartProvider } from '../providers/cart.provider.js';
12
+ import { MedusaSearchProvider } from '../providers/product-search.provider.js';
13
+ import { getMedusaTestConfiguration } from './test-utils.js';
14
+ import { MedusaAPI } from '../core/client.js';
15
+
16
+ /**
17
+ const testData = {
18
+ skuWithoutTiers: '8719514435254',
19
+ skuWithTiers: '8719514435377'
20
+ }
21
+ */
22
+
23
+ describe('Medusa Cart Provider - Large Scenarios', () => {
24
+ let provider: MedusaCartProvider;
25
+ let searchProvider: MedusaSearchProvider;
26
+ let reqCtx: RequestContext;
27
+
28
+ beforeEach(() => {
29
+ reqCtx = createInitialRequestContext();
30
+ const client = new MedusaAPI(getMedusaTestConfiguration(), reqCtx);
31
+ provider = new MedusaCartProvider(
32
+ getMedusaTestConfiguration(),
33
+ new NoOpCache(),
34
+ reqCtx,
35
+ client
36
+ );
37
+ searchProvider = new MedusaSearchProvider(
38
+ getMedusaTestConfiguration(),
39
+ new NoOpCache(),
40
+ reqCtx,
41
+ client
42
+ );
43
+ });
44
+
45
+ describe('large carts', () => {
46
+ it('should be able to add an 50 items to a cart in less than 30 seconds', async () => {
47
+ const searchResult = await searchProvider.queryByTerm(
48
+ ProductSearchQueryByTermSchema.parse({
49
+ search: {
50
+ term: 'phil',
51
+ paginationOptions: {
52
+ pageNumber: 1,
53
+ pageSize: 50,
54
+ },
55
+ filters: [],
56
+ facets: [],
57
+ },
58
+ } satisfies ProductSearchQueryByTerm)
59
+ );
60
+
61
+ if (!searchResult.success) {
62
+ assert.fail();
63
+ }
64
+
65
+ expect(searchResult.value.items.length).toBeGreaterThanOrEqual(50);
66
+
67
+ let cartIdentifier = undefined;
68
+ let cart;
69
+ for (const product of searchResult.value.items) {
70
+ cart = await provider.add({
71
+ cart: cartIdentifier,
72
+ variant: {
73
+ sku: product.variants[0].variant.sku,
74
+ },
75
+ quantity: 1,
76
+ });
77
+
78
+ if (!cart.success) {
79
+ assert.fail();
80
+ }
81
+
82
+ cartIdentifier = cart.value.identifier;
83
+ }
84
+
85
+ if (!cart) {
86
+ assert.fail();
87
+ }
88
+
89
+ expect(cart.value.identifier.key).toBeDefined();
90
+ expect(cart.value.items.length).toBe(50);
91
+
92
+ expect(cart.value.items[0].price.totalPrice.value).toBeGreaterThan(0);
93
+ expect(cart.value.items[0].price.totalPrice.currency).toBe(
94
+ reqCtx.languageContext.currencyCode
95
+ );
96
+
97
+ expect(cart.value.price.grandTotal.value).toBeGreaterThan(0);
98
+ expect(cart.value.price.grandTotal.currency).toBe(
99
+ reqCtx.languageContext.currencyCode
100
+ );
101
+ }, 30000);
102
+ });
103
+ });