shopify-app-js 0.0.1-security → 1.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of shopify-app-js might be problematic. Click here for more details.

Files changed (424) hide show
  1. package/.changeset/README.md +8 -0
  2. package/.changeset/config.json +14 -0
  3. package/.changeset/flat-clouds-camp.md +5 -0
  4. package/.eslintrc.js +36 -0
  5. package/.github/CODEOWNERS +1 -0
  6. package/.github/ISSUE_TEMPLATE/BUG_REPORT.md +36 -0
  7. package/.github/ISSUE_TEMPLATE/ENHANCEMENT.md +9 -0
  8. package/.github/ISSUE_TEMPLATE/FEATURE_REQUEST.md +9 -0
  9. package/.github/PULL_REQUEST_TEMPLATE.md +34 -0
  10. package/.github/dependabot.yml +9 -0
  11. package/.github/workflows/changelog.yml +30 -0
  12. package/.github/workflows/ci.yml +22 -0
  13. package/.github/workflows/cla.yml +22 -0
  14. package/.github/workflows/close-waiting-for-response-issues.yml +20 -0
  15. package/.github/workflows/main-release.yml +36 -0
  16. package/.github/workflows/markdown_link_check.yml +14 -0
  17. package/.github/workflows/markdown_link_checker_config.json +9 -0
  18. package/.github/workflows/publish-experimental-build.yml +30 -0
  19. package/.github/workflows/release-candidate.yml +35 -0
  20. package/.github/workflows/remove-labels-on-activity.yml +16 -0
  21. package/.github/workflows/stale.yml +26 -0
  22. package/.prettierignore +5 -0
  23. package/.prettierrc +1 -0
  24. package/CODE_OF_CONDUCT.md +46 -0
  25. package/CONTRIBUTING.md +39 -0
  26. package/LICENSE.md +9 -0
  27. package/README.md +62 -3
  28. package/RELEASING.md +142 -0
  29. package/loom.config.ts +115 -0
  30. package/package.json +44 -3
  31. package/packages/.prettierrc +1 -0
  32. package/packages/shopify-app-express/CHANGELOG.md +316 -0
  33. package/packages/shopify-app-express/LICENSE.md +9 -0
  34. package/packages/shopify-app-express/README.md +93 -0
  35. package/packages/shopify-app-express/docs/reference/README.md +3 -0
  36. package/packages/shopify-app-express/docs/reference/auth.md +76 -0
  37. package/packages/shopify-app-express/docs/reference/cspHeaders.md +24 -0
  38. package/packages/shopify-app-express/docs/reference/ensureInstalledOnShop.md +17 -0
  39. package/packages/shopify-app-express/docs/reference/migrating-app-v6-api-lib-to-express-lib.md +345 -0
  40. package/packages/shopify-app-express/docs/reference/processWebhooks.md +67 -0
  41. package/packages/shopify-app-express/docs/reference/redirectOutOfApp.md +77 -0
  42. package/packages/shopify-app-express/docs/reference/redirectToShopifyOrAppRoot.md +20 -0
  43. package/packages/shopify-app-express/docs/reference/shopifyApp.md +148 -0
  44. package/packages/shopify-app-express/docs/reference/validateAuthenticatedSession.md +30 -0
  45. package/packages/shopify-app-express/loom.config.ts +27 -0
  46. package/packages/shopify-app-express/package.json +54 -0
  47. package/packages/shopify-app-express/src/__tests__/index.test.ts +100 -0
  48. package/packages/shopify-app-express/src/__tests__/integration/oauth.test.ts +445 -0
  49. package/packages/shopify-app-express/src/__tests__/integration/responses.ts +135 -0
  50. package/packages/shopify-app-express/src/__tests__/integration/types.ts +22 -0
  51. package/packages/shopify-app-express/src/__tests__/integration/utils.ts +68 -0
  52. package/packages/shopify-app-express/src/__tests__/integration/webhooks.test.ts +208 -0
  53. package/packages/shopify-app-express/src/__tests__/redirect-to-auth.test.ts +103 -0
  54. package/packages/shopify-app-express/src/__tests__/setup-jest.ts +8 -0
  55. package/packages/shopify-app-express/src/__tests__/test-helper.ts +204 -0
  56. package/packages/shopify-app-express/src/app-installations.ts +42 -0
  57. package/packages/shopify-app-express/src/auth/__tests__/auth.test.ts +303 -0
  58. package/packages/shopify-app-express/src/auth/auth-callback.ts +131 -0
  59. package/packages/shopify-app-express/src/auth/index.ts +32 -0
  60. package/packages/shopify-app-express/src/auth/types.ts +13 -0
  61. package/packages/shopify-app-express/src/config-types.ts +38 -0
  62. package/packages/shopify-app-express/src/error.ts +8 -0
  63. package/packages/shopify-app-express/src/index.ts +171 -0
  64. package/packages/shopify-app-express/src/middlewares/__tests__/csp-headers.test.ts +69 -0
  65. package/packages/shopify-app-express/src/middlewares/__tests__/ensure-installed-on-shop.test.ts +193 -0
  66. package/packages/shopify-app-express/src/middlewares/__tests__/redirect-to-shopify-or-app-root.test.ts +59 -0
  67. package/packages/shopify-app-express/src/middlewares/__tests__/validate-authenticated-session.test.ts +319 -0
  68. package/packages/shopify-app-express/src/middlewares/csp-headers.ts +31 -0
  69. package/packages/shopify-app-express/src/middlewares/ensure-installed-on-shop.ts +176 -0
  70. package/packages/shopify-app-express/src/middlewares/has-valid-access-token.ts +25 -0
  71. package/packages/shopify-app-express/src/middlewares/index.ts +15 -0
  72. package/packages/shopify-app-express/src/middlewares/redirect-to-shopify-or-app-root.ts +39 -0
  73. package/packages/shopify-app-express/src/middlewares/types.ts +6 -0
  74. package/packages/shopify-app-express/src/middlewares/validate-authenticated-session.ts +142 -0
  75. package/packages/shopify-app-express/src/redirect-out-of-app.ts +80 -0
  76. package/packages/shopify-app-express/src/redirect-to-auth.ts +70 -0
  77. package/packages/shopify-app-express/src/types.ts +26 -0
  78. package/packages/shopify-app-express/src/version.ts +1 -0
  79. package/packages/shopify-app-express/src/webhooks/__tests__/process.test.ts +138 -0
  80. package/packages/shopify-app-express/src/webhooks/index.ts +54 -0
  81. package/packages/shopify-app-express/src/webhooks/process.ts +22 -0
  82. package/packages/shopify-app-express/src/webhooks/types.ts +24 -0
  83. package/packages/shopify-app-express/tsconfig.json +10 -0
  84. package/packages/shopify-app-remix/.eslintrc.js +3 -0
  85. package/packages/shopify-app-remix/CHANGELOG.md +569 -0
  86. package/packages/shopify-app-remix/LICENSE.md +9 -0
  87. package/packages/shopify-app-remix/README.md +223 -0
  88. package/packages/shopify-app-remix/docs/build-docs.sh +11 -0
  89. package/packages/shopify-app-remix/docs/generated/generated_docs_data.json +18650 -0
  90. package/packages/shopify-app-remix/docs/generated/generated_static_pages.json +540 -0
  91. package/packages/shopify-app-remix/docs/staticPages/admin.doc.ts +130 -0
  92. package/packages/shopify-app-remix/docs/staticPages/examples/guides/admin/auth-cors.example.tsx +11 -0
  93. package/packages/shopify-app-remix/docs/staticPages/examples/guides/admin/auth.example.tsx +19 -0
  94. package/packages/shopify-app-remix/docs/staticPages/examples/guides/admin/graphql.example.tsx +29 -0
  95. package/packages/shopify-app-remix/docs/staticPages/examples/guides/admin/headers.example.tsx +10 -0
  96. package/packages/shopify-app-remix/docs/staticPages/examples/guides/admin/rest-resources.example.tsx +9 -0
  97. package/packages/shopify-app-remix/docs/staticPages/examples/guides/admin/rest.example.tsx +17 -0
  98. package/packages/shopify-app-remix/docs/staticPages/examples/guides/future-flags/config.example.ts +8 -0
  99. package/packages/shopify-app-remix/docs/staticPages/examples/guides/future-flags/unstable.example.ts +9 -0
  100. package/packages/shopify-app-remix/docs/staticPages/examples/guides/graphql-types/.graphqlrc.ts +16 -0
  101. package/packages/shopify-app-remix/docs/staticPages/examples/guides/graphql-types/install.npm.example.sh +2 -0
  102. package/packages/shopify-app-remix/docs/staticPages/examples/guides/graphql-types/install.pnpm.example.sh +2 -0
  103. package/packages/shopify-app-remix/docs/staticPages/examples/guides/graphql-types/install.yarn.example.sh +2 -0
  104. package/packages/shopify-app-remix/docs/staticPages/examples/guides/graphql-types/package.json +5 -0
  105. package/packages/shopify-app-remix/docs/staticPages/examples/guides/graphql-types/run.npm.example.sh +1 -0
  106. package/packages/shopify-app-remix/docs/staticPages/examples/guides/graphql-types/run.pnpm.example.sh +1 -0
  107. package/packages/shopify-app-remix/docs/staticPages/examples/guides/graphql-types/run.yarn.example.sh +1 -0
  108. package/packages/shopify-app-remix/docs/staticPages/examples/guides/webhooks/config.example.ts +19 -0
  109. package/packages/shopify-app-remix/docs/staticPages/examples/guides/webhooks/endpoint.example.ts +24 -0
  110. package/packages/shopify-app-remix/docs/staticPages/examples/index/app-provider.example.ts +30 -0
  111. package/packages/shopify-app-remix/docs/staticPages/examples/index/boundaries.example.ts +9 -0
  112. package/packages/shopify-app-remix/docs/staticPages/examples/index/create.npm.example.sh +1 -0
  113. package/packages/shopify-app-remix/docs/staticPages/examples/index/create.pnpm.example.sh +1 -0
  114. package/packages/shopify-app-remix/docs/staticPages/examples/index/create.yarn.example.sh +1 -0
  115. package/packages/shopify-app-remix/docs/staticPages/examples/index/embedded-app-auth-strategy-config.example.ts +10 -0
  116. package/packages/shopify-app-remix/docs/staticPages/examples/index/entry-server.example.ts +12 -0
  117. package/packages/shopify-app-remix/docs/staticPages/examples/index/install.npm.example.sh +1 -0
  118. package/packages/shopify-app-remix/docs/staticPages/examples/index/install.pnpm.example.sh +1 -0
  119. package/packages/shopify-app-remix/docs/staticPages/examples/index/install.yarn.example.sh +1 -0
  120. package/packages/shopify-app-remix/docs/staticPages/examples/index/shopify-app.example.ts +14 -0
  121. package/packages/shopify-app-remix/docs/staticPages/examples/index/splat-route.example.ts +11 -0
  122. package/packages/shopify-app-remix/docs/staticPages/future-flags.doc.ts +94 -0
  123. package/packages/shopify-app-remix/docs/staticPages/graphql-types.doc.ts +148 -0
  124. package/packages/shopify-app-remix/docs/staticPages/index.doc.ts +227 -0
  125. package/packages/shopify-app-remix/docs/staticPages/webhooks.doc.ts +64 -0
  126. package/packages/shopify-app-remix/docs/tsconfig.docs.json +9 -0
  127. package/packages/shopify-app-remix/docs/typeOverride.json +1 -0
  128. package/packages/shopify-app-remix/docs/upcoming_changes.md +153 -0
  129. package/packages/shopify-app-remix/loom.config.ts +57 -0
  130. package/packages/shopify-app-remix/package.json +93 -0
  131. package/packages/shopify-app-remix/src/react/.eslintrc.js +16 -0
  132. package/packages/shopify-app-remix/src/react/__tests__/test-helper.ts +22 -0
  133. package/packages/shopify-app-remix/src/react/components/AppProvider/AppProvider.doc.ts +34 -0
  134. package/packages/shopify-app-remix/src/react/components/AppProvider/AppProvider.tsx +121 -0
  135. package/packages/shopify-app-remix/src/react/components/AppProvider/__tests__/AppProvider.test.tsx +68 -0
  136. package/packages/shopify-app-remix/src/react/components/AppProvider/index.ts +1 -0
  137. package/packages/shopify-app-remix/src/react/components/RemixPolarisLink.tsx +14 -0
  138. package/packages/shopify-app-remix/src/react/components/index.ts +1 -0
  139. package/packages/shopify-app-remix/src/react/const.ts +2 -0
  140. package/packages/shopify-app-remix/src/react/index.ts +1 -0
  141. package/packages/shopify-app-remix/src/server/.eslintrc.js +16 -0
  142. package/packages/shopify-app-remix/src/server/__test-helpers/__tests__/request-mock.test.ts +171 -0
  143. package/packages/shopify-app-remix/src/server/__test-helpers/const.ts +11 -0
  144. package/packages/shopify-app-remix/src/server/__test-helpers/expect-admin-api-client.ts +119 -0
  145. package/packages/shopify-app-remix/src/server/__test-helpers/expect-begin-auth-redirect.ts +22 -0
  146. package/packages/shopify-app-remix/src/server/__test-helpers/expect-document-request-headers.ts +25 -0
  147. package/packages/shopify-app-remix/src/server/__test-helpers/expect-exit-iframe.ts +37 -0
  148. package/packages/shopify-app-remix/src/server/__test-helpers/expect-login-redirect.ts +8 -0
  149. package/packages/shopify-app-remix/src/server/__test-helpers/expect-storefront-api-client.ts +50 -0
  150. package/packages/shopify-app-remix/src/server/__test-helpers/get-hmac.ts +10 -0
  151. package/packages/shopify-app-remix/src/server/__test-helpers/get-jwt.ts +31 -0
  152. package/packages/shopify-app-remix/src/server/__test-helpers/get-thrown-response.ts +17 -0
  153. package/packages/shopify-app-remix/src/server/__test-helpers/index.ts +14 -0
  154. package/packages/shopify-app-remix/src/server/__test-helpers/request-mock.ts +169 -0
  155. package/packages/shopify-app-remix/src/server/__test-helpers/setup-valid-session.ts +45 -0
  156. package/packages/shopify-app-remix/src/server/__test-helpers/sign-request-cookie.ts +21 -0
  157. package/packages/shopify-app-remix/src/server/__test-helpers/test-config.ts +112 -0
  158. package/packages/shopify-app-remix/src/server/__tests__/override-logger.test.ts +101 -0
  159. package/packages/shopify-app-remix/src/server/__tests__/shopify-app.test.ts +119 -0
  160. package/packages/shopify-app-remix/src/server/adapters/__tests__/node-app-bridge-url.test.ts +18 -0
  161. package/packages/shopify-app-remix/src/server/adapters/__tests__/node.test.ts +28 -0
  162. package/packages/shopify-app-remix/src/server/adapters/node/__tests__/setup-jest.ts +10 -0
  163. package/packages/shopify-app-remix/src/server/adapters/node/index.ts +20 -0
  164. package/packages/shopify-app-remix/src/server/adapters/vercel/__tests__/setup-jest.ts +12 -0
  165. package/packages/shopify-app-remix/src/server/adapters/vercel/index.ts +13 -0
  166. package/packages/shopify-app-remix/src/server/authenticate/admin/__tests__/admin-client.test.ts +181 -0
  167. package/packages/shopify-app-remix/src/server/authenticate/admin/__tests__/doc-request-path.test.ts +151 -0
  168. package/packages/shopify-app-remix/src/server/authenticate/admin/__tests__/exit-i-frame-path.test.ts +99 -0
  169. package/packages/shopify-app-remix/src/server/authenticate/admin/__tests__/patch-session-token-path.test.ts +58 -0
  170. package/packages/shopify-app-remix/src/server/authenticate/admin/__tests__/reject-bot.test.ts +23 -0
  171. package/packages/shopify-app-remix/src/server/authenticate/admin/__tests__/respond-to-options.test.ts +50 -0
  172. package/packages/shopify-app-remix/src/server/authenticate/admin/__tests__/session-token-header-path.test.ts +91 -0
  173. package/packages/shopify-app-remix/src/server/authenticate/admin/authenticate.admin.doc.ts +37 -0
  174. package/packages/shopify-app-remix/src/server/authenticate/admin/authenticate.ts +201 -0
  175. package/packages/shopify-app-remix/src/server/authenticate/admin/billing/__tests__/cancel.test.ts +258 -0
  176. package/packages/shopify-app-remix/src/server/authenticate/admin/billing/__tests__/check.test.ts +254 -0
  177. package/packages/shopify-app-remix/src/server/authenticate/admin/billing/__tests__/mock-responses.ts +65 -0
  178. package/packages/shopify-app-remix/src/server/authenticate/admin/billing/__tests__/request.test.ts +373 -0
  179. package/packages/shopify-app-remix/src/server/authenticate/admin/billing/__tests__/require.test.ts +316 -0
  180. package/packages/shopify-app-remix/src/server/authenticate/admin/billing/authenticate.admin.billing.doc.ts +28 -0
  181. package/packages/shopify-app-remix/src/server/authenticate/admin/billing/cancel.ts +37 -0
  182. package/packages/shopify-app-remix/src/server/authenticate/admin/billing/check.ts +38 -0
  183. package/packages/shopify-app-remix/src/server/authenticate/admin/billing/index.ts +4 -0
  184. package/packages/shopify-app-remix/src/server/authenticate/admin/billing/request.ts +101 -0
  185. package/packages/shopify-app-remix/src/server/authenticate/admin/billing/require.ts +56 -0
  186. package/packages/shopify-app-remix/src/server/authenticate/admin/billing/types.ts +383 -0
  187. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/__tests__/redirect.test.ts +313 -0
  188. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/__tests__/validate-redirect-url.test.ts +84 -0
  189. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/begin-auth.ts +17 -0
  190. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/create-admin-api-context.ts +22 -0
  191. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/ensure-app-is-embedded-if-required.ts +18 -0
  192. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/ensure-session-token-search-param-if-required.ts +25 -0
  193. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/handle-client-error.ts +43 -0
  194. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/index.ts +14 -0
  195. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/redirect-to-auth-page.ts +28 -0
  196. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/redirect-to-bounce-page.ts +23 -0
  197. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/redirect-to-shopify-or-app-root.ts +21 -0
  198. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/redirect-with-app-bridge-headers.ts +13 -0
  199. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/redirect-with-exitiframe.ts +28 -0
  200. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/redirect.ts +83 -0
  201. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/render-app-bridge.ts +46 -0
  202. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/trigger-after-auth-hook.ts +28 -0
  203. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/validate-redirect-url.ts +67 -0
  204. package/packages/shopify-app-remix/src/server/authenticate/admin/helpers/validate-shop-and-host-params.ts +28 -0
  205. package/packages/shopify-app-remix/src/server/authenticate/admin/strategies/__tests__/auth-code-flow/admin-client.test.ts +222 -0
  206. package/packages/shopify-app-remix/src/server/authenticate/admin/strategies/__tests__/auth-code-flow/auth-callback-path.test.ts +395 -0
  207. package/packages/shopify-app-remix/src/server/authenticate/admin/strategies/__tests__/auth-code-flow/auth-path.test.ts +83 -0
  208. package/packages/shopify-app-remix/src/server/authenticate/admin/strategies/__tests__/auth-code-flow/authenticate.test.ts +183 -0
  209. package/packages/shopify-app-remix/src/server/authenticate/admin/strategies/__tests__/auth-code-flow/ensure-installed-on-shop.test.ts +234 -0
  210. package/packages/shopify-app-remix/src/server/authenticate/admin/strategies/__tests__/auth-code-flow/session-token-header-path.test.ts +79 -0
  211. package/packages/shopify-app-remix/src/server/authenticate/admin/strategies/__tests__/token-exchange/admin-client.test.ts +189 -0
  212. package/packages/shopify-app-remix/src/server/authenticate/admin/strategies/__tests__/token-exchange/authenticate.test.ts +310 -0
  213. package/packages/shopify-app-remix/src/server/authenticate/admin/strategies/auth-code-flow.ts +333 -0
  214. package/packages/shopify-app-remix/src/server/authenticate/admin/strategies/token-exchange.ts +169 -0
  215. package/packages/shopify-app-remix/src/server/authenticate/admin/strategies/types.ts +29 -0
  216. package/packages/shopify-app-remix/src/server/authenticate/admin/types.ts +199 -0
  217. package/packages/shopify-app-remix/src/server/authenticate/const.ts +15 -0
  218. package/packages/shopify-app-remix/src/server/authenticate/flow/__tests__/authenticate.test.ts +142 -0
  219. package/packages/shopify-app-remix/src/server/authenticate/flow/authenticate.flow.doc.ts +34 -0
  220. package/packages/shopify-app-remix/src/server/authenticate/flow/authenticate.ts +71 -0
  221. package/packages/shopify-app-remix/src/server/authenticate/flow/types.ts +90 -0
  222. package/packages/shopify-app-remix/src/server/authenticate/helpers/__tests__/add-response-headers.test.ts +25 -0
  223. package/packages/shopify-app-remix/src/server/authenticate/helpers/__tests__/app-bridge-url.test.ts +21 -0
  224. package/packages/shopify-app-remix/src/server/authenticate/helpers/__tests__/idempotent-promise-handler.test.ts +104 -0
  225. package/packages/shopify-app-remix/src/server/authenticate/helpers/add-response-headers.ts +43 -0
  226. package/packages/shopify-app-remix/src/server/authenticate/helpers/app-bridge-url.ts +10 -0
  227. package/packages/shopify-app-remix/src/server/authenticate/helpers/ensure-cors-headers.ts +38 -0
  228. package/packages/shopify-app-remix/src/server/authenticate/helpers/get-session-token-header.ts +11 -0
  229. package/packages/shopify-app-remix/src/server/authenticate/helpers/idempotent-promise-handler.ts +45 -0
  230. package/packages/shopify-app-remix/src/server/authenticate/helpers/index.ts +8 -0
  231. package/packages/shopify-app-remix/src/server/authenticate/helpers/reject-bot-request.ts +13 -0
  232. package/packages/shopify-app-remix/src/server/authenticate/helpers/respond-to-invalid-session-token.ts +29 -0
  233. package/packages/shopify-app-remix/src/server/authenticate/helpers/respond-to-options-request.ts +26 -0
  234. package/packages/shopify-app-remix/src/server/authenticate/helpers/validate-session-token.ts +32 -0
  235. package/packages/shopify-app-remix/src/server/authenticate/login/__tests__/login.test.ts +157 -0
  236. package/packages/shopify-app-remix/src/server/authenticate/login/login.ts +53 -0
  237. package/packages/shopify-app-remix/src/server/authenticate/public/__tests__/factory.test.ts +224 -0
  238. package/packages/shopify-app-remix/src/server/authenticate/public/appProxy/__tests__/authenticate.test.ts +282 -0
  239. package/packages/shopify-app-remix/src/server/authenticate/public/appProxy/authenticate.public.app-proxy.doc.ts +36 -0
  240. package/packages/shopify-app-remix/src/server/authenticate/public/appProxy/authenticate.ts +90 -0
  241. package/packages/shopify-app-remix/src/server/authenticate/public/appProxy/types.ts +164 -0
  242. package/packages/shopify-app-remix/src/server/authenticate/public/checkout/__tests__/authenticate.test.ts +142 -0
  243. package/packages/shopify-app-remix/src/server/authenticate/public/checkout/authenticate.public.checkout.doc.ts +23 -0
  244. package/packages/shopify-app-remix/src/server/authenticate/public/checkout/authenticate.ts +45 -0
  245. package/packages/shopify-app-remix/src/server/authenticate/public/checkout/types.ts +65 -0
  246. package/packages/shopify-app-remix/src/server/authenticate/public/factory.ts +49 -0
  247. package/packages/shopify-app-remix/src/server/authenticate/public/index.ts +1 -0
  248. package/packages/shopify-app-remix/src/server/authenticate/public/types.ts +77 -0
  249. package/packages/shopify-app-remix/src/server/authenticate/webhooks/__tests__/authenticate.test.ts +207 -0
  250. package/packages/shopify-app-remix/src/server/authenticate/webhooks/__tests__/mock-responses.ts +45 -0
  251. package/packages/shopify-app-remix/src/server/authenticate/webhooks/__tests__/register.test.ts +210 -0
  252. package/packages/shopify-app-remix/src/server/authenticate/webhooks/authenticate.ts +113 -0
  253. package/packages/shopify-app-remix/src/server/authenticate/webhooks/authenticate.webhooks.doc.ts +28 -0
  254. package/packages/shopify-app-remix/src/server/authenticate/webhooks/index.ts +1 -0
  255. package/packages/shopify-app-remix/src/server/authenticate/webhooks/register.ts +48 -0
  256. package/packages/shopify-app-remix/src/server/authenticate/webhooks/types.ts +246 -0
  257. package/packages/shopify-app-remix/src/server/boundary/__tests__/error.test.tsx +36 -0
  258. package/packages/shopify-app-remix/src/server/boundary/__tests__/headers.test.ts +66 -0
  259. package/packages/shopify-app-remix/src/server/boundary/error.tsx +14 -0
  260. package/packages/shopify-app-remix/src/server/boundary/headers.ts +15 -0
  261. package/packages/shopify-app-remix/src/server/boundary/index.ts +39 -0
  262. package/packages/shopify-app-remix/src/server/boundary/types.ts +4 -0
  263. package/packages/shopify-app-remix/src/server/clients/admin/authenticate.admin.api.doc.ts +34 -0
  264. package/packages/shopify-app-remix/src/server/clients/admin/factory.ts +30 -0
  265. package/packages/shopify-app-remix/src/server/clients/admin/graphql.ts +39 -0
  266. package/packages/shopify-app-remix/src/server/clients/admin/index.ts +2 -0
  267. package/packages/shopify-app-remix/src/server/clients/admin/rest.ts +157 -0
  268. package/packages/shopify-app-remix/src/server/clients/admin/types.ts +178 -0
  269. package/packages/shopify-app-remix/src/server/clients/index.ts +2 -0
  270. package/packages/shopify-app-remix/src/server/clients/storefront/authenticate.storefront.api.doc.ts +34 -0
  271. package/packages/shopify-app-remix/src/server/clients/storefront/factory.ts +32 -0
  272. package/packages/shopify-app-remix/src/server/clients/storefront/index.ts +2 -0
  273. package/packages/shopify-app-remix/src/server/clients/storefront/types.ts +31 -0
  274. package/packages/shopify-app-remix/src/server/clients/types.ts +27 -0
  275. package/packages/shopify-app-remix/src/server/config-types.ts +293 -0
  276. package/packages/shopify-app-remix/src/server/errors.ts +3 -0
  277. package/packages/shopify-app-remix/src/server/future/flags.ts +57 -0
  278. package/packages/shopify-app-remix/src/server/index.ts +20 -0
  279. package/packages/shopify-app-remix/src/server/override-logger.ts +42 -0
  280. package/packages/shopify-app-remix/src/server/shopify-app.doc.ts +45 -0
  281. package/packages/shopify-app-remix/src/server/shopify-app.ts +198 -0
  282. package/packages/shopify-app-remix/src/server/types.ts +509 -0
  283. package/packages/shopify-app-remix/src/server/unauthenticated/admin/__tests__/factory.test.ts +30 -0
  284. package/packages/shopify-app-remix/src/server/unauthenticated/admin/factory.ts +29 -0
  285. package/packages/shopify-app-remix/src/server/unauthenticated/admin/index.ts +1 -0
  286. package/packages/shopify-app-remix/src/server/unauthenticated/admin/types.ts +117 -0
  287. package/packages/shopify-app-remix/src/server/unauthenticated/admin/unauthenticated.admin.doc.ts +40 -0
  288. package/packages/shopify-app-remix/src/server/unauthenticated/helpers/get-offline-session.ts +13 -0
  289. package/packages/shopify-app-remix/src/server/unauthenticated/helpers/index.ts +1 -0
  290. package/packages/shopify-app-remix/src/server/unauthenticated/storefront/__tests__/factory.test.ts +30 -0
  291. package/packages/shopify-app-remix/src/server/unauthenticated/storefront/factory.ts +28 -0
  292. package/packages/shopify-app-remix/src/server/unauthenticated/storefront/index.ts +1 -0
  293. package/packages/shopify-app-remix/src/server/unauthenticated/storefront/types.ts +39 -0
  294. package/packages/shopify-app-remix/src/server/unauthenticated/storefront/unauthenticated.storefront.doc.ts +40 -0
  295. package/packages/shopify-app-remix/src/server/unauthenticated/types.ts +68 -0
  296. package/packages/shopify-app-remix/src/server/version.ts +1 -0
  297. package/packages/shopify-app-remix/tsconfig.json +11 -0
  298. package/packages/shopify-app-session-storage/CHANGELOG.md +149 -0
  299. package/packages/shopify-app-session-storage/LICENSE.md +9 -0
  300. package/packages/shopify-app-session-storage/README.md +5 -0
  301. package/packages/shopify-app-session-storage/implementing-session-storage.md +214 -0
  302. package/packages/shopify-app-session-storage/loom.config.ts +45 -0
  303. package/packages/shopify-app-session-storage/package.json +44 -0
  304. package/packages/shopify-app-session-storage/src/__tests__/setup-jest.ts +8 -0
  305. package/packages/shopify-app-session-storage/src/abstract-migration-engine.ts +53 -0
  306. package/packages/shopify-app-session-storage/src/index.ts +3 -0
  307. package/packages/shopify-app-session-storage/src/rdbms-session-storage-migrator.ts +59 -0
  308. package/packages/shopify-app-session-storage/src/types.ts +174 -0
  309. package/packages/shopify-app-session-storage/tsconfig.json +10 -0
  310. package/packages/shopify-app-session-storage-dynamodb/CHANGELOG.md +150 -0
  311. package/packages/shopify-app-session-storage-dynamodb/LICENSE.md +9 -0
  312. package/packages/shopify-app-session-storage-dynamodb/README.md +75 -0
  313. package/packages/shopify-app-session-storage-dynamodb/loom.config.ts +27 -0
  314. package/packages/shopify-app-session-storage-dynamodb/package.json +51 -0
  315. package/packages/shopify-app-session-storage-dynamodb/src/__tests__/dynamodb.test.ts +93 -0
  316. package/packages/shopify-app-session-storage-dynamodb/src/__tests__/setup-jest.ts +8 -0
  317. package/packages/shopify-app-session-storage-dynamodb/src/dynamodb.ts +126 -0
  318. package/packages/shopify-app-session-storage-dynamodb/tsconfig.json +10 -0
  319. package/packages/shopify-app-session-storage-kv/CHANGELOG.md +204 -0
  320. package/packages/shopify-app-session-storage-kv/LICENSE.md +9 -0
  321. package/packages/shopify-app-session-storage-kv/README.md +41 -0
  322. package/packages/shopify-app-session-storage-kv/loom.config.ts +27 -0
  323. package/packages/shopify-app-session-storage-kv/package.json +52 -0
  324. package/packages/shopify-app-session-storage-kv/src/__tests__/kv-namespace-dummy-worker.ts +8 -0
  325. package/packages/shopify-app-session-storage-kv/src/__tests__/kv.test.ts +22 -0
  326. package/packages/shopify-app-session-storage-kv/src/__tests__/setup-jest.ts +8 -0
  327. package/packages/shopify-app-session-storage-kv/src/kv.ts +87 -0
  328. package/packages/shopify-app-session-storage-kv/tsconfig.json +11 -0
  329. package/packages/shopify-app-session-storage-memory/CHANGELOG.md +190 -0
  330. package/packages/shopify-app-session-storage-memory/LICENSE.md +9 -0
  331. package/packages/shopify-app-session-storage-memory/README.md +19 -0
  332. package/packages/shopify-app-session-storage-memory/loom.config.ts +27 -0
  333. package/packages/shopify-app-session-storage-memory/package.json +46 -0
  334. package/packages/shopify-app-session-storage-memory/src/__tests__/memory.test.ts +12 -0
  335. package/packages/shopify-app-session-storage-memory/src/__tests__/setup-jest.ts +8 -0
  336. package/packages/shopify-app-session-storage-memory/src/memory.ts +34 -0
  337. package/packages/shopify-app-session-storage-memory/tsconfig.json +10 -0
  338. package/packages/shopify-app-session-storage-mongodb/CHANGELOG.md +208 -0
  339. package/packages/shopify-app-session-storage-mongodb/LICENSE.md +9 -0
  340. package/packages/shopify-app-session-storage-mongodb/README.md +30 -0
  341. package/packages/shopify-app-session-storage-mongodb/loom.config.ts +27 -0
  342. package/packages/shopify-app-session-storage-mongodb/package.json +49 -0
  343. package/packages/shopify-app-session-storage-mongodb/src/__tests__/mongodb.test.ts +54 -0
  344. package/packages/shopify-app-session-storage-mongodb/src/__tests__/setup-jest.ts +8 -0
  345. package/packages/shopify-app-session-storage-mongodb/src/mongodb.ts +120 -0
  346. package/packages/shopify-app-session-storage-mongodb/tsconfig.json +10 -0
  347. package/packages/shopify-app-session-storage-mysql/CHANGELOG.md +220 -0
  348. package/packages/shopify-app-session-storage-mysql/LICENSE.md +9 -0
  349. package/packages/shopify-app-session-storage-mysql/README.md +41 -0
  350. package/packages/shopify-app-session-storage-mysql/loom.config.ts +27 -0
  351. package/packages/shopify-app-session-storage-mysql/package.json +50 -0
  352. package/packages/shopify-app-session-storage-mysql/src/__tests__/mysql.test.ts +137 -0
  353. package/packages/shopify-app-session-storage-mysql/src/__tests__/setup-jest.ts +8 -0
  354. package/packages/shopify-app-session-storage-mysql/src/migrations.ts +18 -0
  355. package/packages/shopify-app-session-storage-mysql/src/mysql-connection.ts +108 -0
  356. package/packages/shopify-app-session-storage-mysql/src/mysql-migrator.ts +53 -0
  357. package/packages/shopify-app-session-storage-mysql/src/mysql.ts +178 -0
  358. package/packages/shopify-app-session-storage-mysql/tsconfig.json +10 -0
  359. package/packages/shopify-app-session-storage-postgresql/CHANGELOG.md +210 -0
  360. package/packages/shopify-app-session-storage-postgresql/LICENSE.md +9 -0
  361. package/packages/shopify-app-session-storage-postgresql/README.md +39 -0
  362. package/packages/shopify-app-session-storage-postgresql/loom.config.ts +27 -0
  363. package/packages/shopify-app-session-storage-postgresql/package.json +52 -0
  364. package/packages/shopify-app-session-storage-postgresql/src/__tests__/migrate-to-case-sensitivity.test.ts +344 -0
  365. package/packages/shopify-app-session-storage-postgresql/src/__tests__/postgresql.test.ts +136 -0
  366. package/packages/shopify-app-session-storage-postgresql/src/__tests__/setup-jest.ts +8 -0
  367. package/packages/shopify-app-session-storage-postgresql/src/migrations.ts +70 -0
  368. package/packages/shopify-app-session-storage-postgresql/src/postgres-connection.ts +94 -0
  369. package/packages/shopify-app-session-storage-postgresql/src/postgres-migrator.ts +27 -0
  370. package/packages/shopify-app-session-storage-postgresql/src/postgresql.ts +178 -0
  371. package/packages/shopify-app-session-storage-postgresql/tsconfig.json +10 -0
  372. package/packages/shopify-app-session-storage-prisma/CHANGELOG.md +148 -0
  373. package/packages/shopify-app-session-storage-prisma/LICENSE.md +9 -0
  374. package/packages/shopify-app-session-storage-prisma/README.md +68 -0
  375. package/packages/shopify-app-session-storage-prisma/loom.config.ts +27 -0
  376. package/packages/shopify-app-session-storage-prisma/package.json +51 -0
  377. package/packages/shopify-app-session-storage-prisma/prisma/migrations/20230425184828_init/migration.sql +11 -0
  378. package/packages/shopify-app-session-storage-prisma/prisma/migrations/20230906155758_mySession/migration.sql +11 -0
  379. package/packages/shopify-app-session-storage-prisma/prisma/migrations/migration_lock.toml +3 -0
  380. package/packages/shopify-app-session-storage-prisma/prisma/schema.prisma +30 -0
  381. package/packages/shopify-app-session-storage-prisma/src/__tests__/prisma.test.ts +66 -0
  382. package/packages/shopify-app-session-storage-prisma/src/__tests__/setup-jest.ts +8 -0
  383. package/packages/shopify-app-session-storage-prisma/src/prisma.ts +145 -0
  384. package/packages/shopify-app-session-storage-prisma/tsconfig.json +10 -0
  385. package/packages/shopify-app-session-storage-redis/CHANGELOG.md +200 -0
  386. package/packages/shopify-app-session-storage-redis/LICENSE.md +9 -0
  387. package/packages/shopify-app-session-storage-redis/README.md +38 -0
  388. package/packages/shopify-app-session-storage-redis/loom.config.ts +27 -0
  389. package/packages/shopify-app-session-storage-redis/package.json +50 -0
  390. package/packages/shopify-app-session-storage-redis/src/__tests__/migration-test-data.ts +46 -0
  391. package/packages/shopify-app-session-storage-redis/src/__tests__/redis.conf +2 -0
  392. package/packages/shopify-app-session-storage-redis/src/__tests__/redis.test.ts +236 -0
  393. package/packages/shopify-app-session-storage-redis/src/__tests__/setup-jest.ts +8 -0
  394. package/packages/shopify-app-session-storage-redis/src/migrations.ts +36 -0
  395. package/packages/shopify-app-session-storage-redis/src/redis-connection.ts +64 -0
  396. package/packages/shopify-app-session-storage-redis/src/redis-migrator.ts +58 -0
  397. package/packages/shopify-app-session-storage-redis/src/redis.ts +167 -0
  398. package/packages/shopify-app-session-storage-redis/tsconfig.json +10 -0
  399. package/packages/shopify-app-session-storage-sqlite/CHANGELOG.md +202 -0
  400. package/packages/shopify-app-session-storage-sqlite/LICENSE.md +9 -0
  401. package/packages/shopify-app-session-storage-sqlite/README.md +25 -0
  402. package/packages/shopify-app-session-storage-sqlite/loom.config.ts +27 -0
  403. package/packages/shopify-app-session-storage-sqlite/package.json +51 -0
  404. package/packages/shopify-app-session-storage-sqlite/src/__tests__/setup-jest.ts +8 -0
  405. package/packages/shopify-app-session-storage-sqlite/src/__tests__/sqlite.test.ts +44 -0
  406. package/packages/shopify-app-session-storage-sqlite/src/migrations.ts +54 -0
  407. package/packages/shopify-app-session-storage-sqlite/src/sqlite-connection.ts +80 -0
  408. package/packages/shopify-app-session-storage-sqlite/src/sqlite-migrator.ts +34 -0
  409. package/packages/shopify-app-session-storage-sqlite/src/sqlite.ts +147 -0
  410. package/packages/shopify-app-session-storage-sqlite/tsconfig.json +10 -0
  411. package/packages/shopify-app-session-storage-test-utils/CHANGELOG.md +185 -0
  412. package/packages/shopify-app-session-storage-test-utils/LICENSE.md +9 -0
  413. package/packages/shopify-app-session-storage-test-utils/loom.config.ts +27 -0
  414. package/packages/shopify-app-session-storage-test-utils/package.json +48 -0
  415. package/packages/shopify-app-session-storage-test-utils/src/__tests__/session-test-utils.test.ts +273 -0
  416. package/packages/shopify-app-session-storage-test-utils/src/battery-of-tests.ts +250 -0
  417. package/packages/shopify-app-session-storage-test-utils/src/index.ts +2 -0
  418. package/packages/shopify-app-session-storage-test-utils/src/session-test-utils.ts +24 -0
  419. package/packages/shopify-app-session-storage-test-utils/src/utils.ts +71 -0
  420. package/packages/shopify-app-session-storage-test-utils/tsconfig.json +10 -0
  421. package/tests/setup/build.js +1 -0
  422. package/tests/setup/setup-jest.ts +8 -0
  423. package/tsconfig.base.json +41 -0
  424. package/tsconfig.json +19 -0
@@ -0,0 +1,136 @@
1
+ import * as child_process from 'child_process';
2
+ import {promisify} from 'util';
3
+
4
+ import pg from 'pg';
5
+ import {
6
+ batteryOfTests,
7
+ poll,
8
+ } from '@shopify/shopify-app-session-storage-test-utils';
9
+ import {Session} from '@shopify/shopify-api';
10
+
11
+ import {PostgreSQLSessionStorage} from '../postgresql';
12
+
13
+ const exec = promisify(child_process.exec);
14
+
15
+ const dbURL = new URL(
16
+ `postgres://${encodeURIComponent('shop&fy')}:${encodeURIComponent(
17
+ 'passify#$',
18
+ )}@localhost/${encodeURIComponent('shop&test')}`,
19
+ );
20
+ const dbURL2 = new URL(
21
+ `postgres://${encodeURIComponent('shop&fy')}:${encodeURIComponent(
22
+ 'passify#$',
23
+ )}@localhost/${encodeURIComponent('shop&test2')}`,
24
+ );
25
+
26
+ describe('PostgreSQLSessionStorage', () => {
27
+ let storage: PostgreSQLSessionStorage;
28
+ let storage2: PostgreSQLSessionStorage;
29
+
30
+ let containerId: string;
31
+ beforeAll(async () => {
32
+ const runCommand = await exec(
33
+ "podman run -d -e POSTGRES_DB='shop&test' -e POSTGRES_USER='shop&fy' -e POSTGRES_PASSWORD='passify#$' -p 5432:5432 postgres:15",
34
+ {encoding: 'utf8'},
35
+ );
36
+
37
+ containerId = runCommand.stdout.trim();
38
+
39
+ await poll(
40
+ async () => {
41
+ try {
42
+ const client = new pg.Client({
43
+ host: dbURL.hostname,
44
+ user: decodeURIComponent(dbURL.username),
45
+ password: decodeURIComponent(dbURL.password),
46
+ database: decodeURIComponent(dbURL.pathname.slice(1)),
47
+ });
48
+ await client.connect();
49
+ await client.query(`CREATE DATABASE "shop&test2"`, []);
50
+ await client.query(
51
+ `GRANT ALL PRIVILEGES ON DATABASE "shop&test2" TO "shop&fy"`,
52
+ [],
53
+ );
54
+ await client.end();
55
+ } catch (error) {
56
+ // console.error(error); // uncomment to see error for debugging tests
57
+ return false;
58
+ }
59
+ return true;
60
+ },
61
+ {interval: 500, timeout: 20000},
62
+ );
63
+ storage = new PostgreSQLSessionStorage(dbURL);
64
+ storage2 = new PostgreSQLSessionStorage(dbURL2);
65
+
66
+ await storage.ready;
67
+ await storage2.ready;
68
+ });
69
+
70
+ afterAll(async () => {
71
+ await storage.disconnect();
72
+ await storage2.disconnect();
73
+
74
+ await exec(`podman rm -f ${containerId}`);
75
+ });
76
+
77
+ const tests = [
78
+ {dbName: 'shop&test', sessionStorage: async () => storage},
79
+ {dbName: 'shop&test2', sessionStorage: async () => storage2},
80
+ ];
81
+
82
+ for (const {dbName, sessionStorage} of tests) {
83
+ describe(`with ${dbName}`, () => {
84
+ batteryOfTests(sessionStorage);
85
+ });
86
+ }
87
+
88
+ it(`one-time initialisation like migrations and table creations are run only once`, async () => {
89
+ const storageClone1 = new PostgreSQLSessionStorage(dbURL);
90
+ await storageClone1.ready;
91
+
92
+ const storageClone2 = new PostgreSQLSessionStorage(dbURL);
93
+ await storageClone2.ready;
94
+
95
+ storageClone1.disconnect();
96
+ storageClone2.disconnect();
97
+ });
98
+
99
+ it(`can disconnect and reconnect to make queries with the pooling clients`, async () => {
100
+ const storage = new PostgreSQLSessionStorage(dbURL);
101
+ await storage.ready;
102
+ const sessionId = '123';
103
+ const session = new Session({
104
+ id: sessionId,
105
+ shop: 'test-shop.myshopify.com',
106
+ state: 'test-state',
107
+ isOnline: false,
108
+ scope: 'fake_scope',
109
+ });
110
+
111
+ expect(await storage.storeSession(session)).toBeTruthy();
112
+ await storage.disconnect();
113
+
114
+ const loadedSession = await storage.loadSession(sessionId);
115
+ expect(loadedSession).toEqual(session);
116
+ await storage.disconnect();
117
+
118
+ expect(await storage.deleteSession(sessionId)).toBeTruthy();
119
+ await storage.disconnect();
120
+ });
121
+
122
+ it(`can successfully connect with a url string instead of a URL object`, async () => {
123
+ const storage = new PostgreSQLSessionStorage(dbURL.toString());
124
+ await storage.ready;
125
+ const session = new Session({
126
+ id: '456',
127
+ shop: 'test-shop.myshopify.com',
128
+ state: 'test-state',
129
+ isOnline: false,
130
+ scope: 'fake_scope',
131
+ });
132
+
133
+ expect(await storage.storeSession(session)).toBeTruthy();
134
+ await storage.disconnect();
135
+ });
136
+ });
@@ -0,0 +1,8 @@
1
+ import fetchMock from 'jest-fetch-mock';
2
+
3
+ // Globally disable fetch requests so we don't risk making real ones
4
+ fetchMock.enableMocks();
5
+
6
+ beforeEach(() => {
7
+ fetchMock.mockReset();
8
+ });
@@ -0,0 +1,70 @@
1
+ import {MigrationOperation} from '@shopify/shopify-app-session-storage';
2
+
3
+ import {PostgresConnection} from './postgres-connection';
4
+
5
+ export const migrationList = [
6
+ new MigrationOperation(
7
+ 'migrateScopeFieldToVarchar1024',
8
+ migrateScopeFieldToVarchar1024,
9
+ ),
10
+ new MigrationOperation('migrateToCaseSensitivity', migrateToCaseSensitivity),
11
+ ];
12
+
13
+ // need change the size of the scope column from 255 to 1024 char
14
+ export async function migrateScopeFieldToVarchar1024(
15
+ connection: PostgresConnection,
16
+ ): Promise<void> {
17
+ await connection.query(`ALTER TABLE "${connection.sessionStorageIdentifier}"
18
+ ALTER COLUMN "scope" TYPE varchar(1024)`);
19
+ }
20
+
21
+ export async function migrateToCaseSensitivity(
22
+ connection: PostgresConnection,
23
+ ): Promise<void> {
24
+ let queries: string[] = [];
25
+ if (
26
+ connection.sessionStorageIdentifier ===
27
+ connection.sessionStorageIdentifier.toLowerCase()
28
+ ) {
29
+ // tablename is lowercase anyway, only need to rename the relevant columns (if not already renamed)
30
+ const hasOldColumnsQuery = `
31
+ SELECT EXISTS (
32
+ SELECT column_name FROM information_schema.columns
33
+ WHERE table_schema='public' AND table_name = '${connection.sessionStorageIdentifier}' AND column_name='isonline'
34
+ )
35
+ `;
36
+ const rows = await connection.query(hasOldColumnsQuery);
37
+ if (rows[0].exists) {
38
+ queries = [
39
+ `ALTER TABLE "${connection.sessionStorageIdentifier}" RENAME COLUMN "isonline" TO "isOnline"`,
40
+ `ALTER TABLE "${connection.sessionStorageIdentifier}" RENAME COLUMN "onlineaccessinfo" TO "onlineAccessInfo"`,
41
+ `ALTER TABLE "${connection.sessionStorageIdentifier}" RENAME COLUMN "accesstoken" TO "accessToken"`,
42
+ ];
43
+ }
44
+ } else {
45
+ // may need to migrate the data from the old table (if it exists) to the new one
46
+ const hasOldSessionTable = await connection.hasTable(
47
+ connection.sessionStorageIdentifier.toLowerCase(),
48
+ );
49
+ if (hasOldSessionTable) {
50
+ queries = [
51
+ // 1. copy the data from the old table to the new one
52
+ `INSERT INTO "${connection.sessionStorageIdentifier}" (
53
+ "id", "shop", "state", "isOnline", "scope", "expires", "onlineAccessInfo", "accessToken"
54
+ )
55
+ SELECT id, shop, state, isonline, scope, expires, onlineaccessinfo, accesstoken
56
+ FROM "${connection.sessionStorageIdentifier.toLowerCase()}"`,
57
+ // 2. drop the old table
58
+ `DROP TABLE "${connection.sessionStorageIdentifier.toLowerCase()}"`,
59
+ ];
60
+ }
61
+ }
62
+
63
+ if (queries.length !== 0) {
64
+ // wrap in a transaction
65
+ queries.unshift(`BEGIN`);
66
+ queries.push(`COMMIT`);
67
+
68
+ await connection.transaction(queries);
69
+ }
70
+ }
@@ -0,0 +1,94 @@
1
+ import pg from 'pg';
2
+ import {RdbmsConnection} from '@shopify/shopify-app-session-storage';
3
+
4
+ export class PostgresConnection implements RdbmsConnection {
5
+ sessionStorageIdentifier: string;
6
+ private ready: Promise<void>;
7
+ private pool: pg.Pool;
8
+ private dbUrl: URL;
9
+
10
+ constructor(dbUrl: string, sessionStorageIdentifier: string) {
11
+ this.dbUrl = new URL(dbUrl);
12
+ this.ready = this.init();
13
+ this.sessionStorageIdentifier = sessionStorageIdentifier;
14
+ }
15
+
16
+ async query(query: string, params: any[] = []): Promise<any[]> {
17
+ await this.ready;
18
+ return (await this.pool.query(query, params)).rows;
19
+ }
20
+
21
+ /**
22
+ * Runs a series of queries in a transaction - requires the use of a SINGLE client,
23
+ * hence we can't use the query method above.
24
+ *
25
+ * @param queries an array of SQL queries to execute in a transaction
26
+ */
27
+ async transaction(queries: string[]): Promise<void> {
28
+ await this.ready;
29
+
30
+ // check if the first and last queries are BEGIN and COMMIT, if not, add them
31
+ if (queries[0] !== 'BEGIN') {
32
+ queries.unshift('BEGIN');
33
+ }
34
+ if (queries[queries.length - 1] !== 'COMMIT') {
35
+ queries.push('COMMIT');
36
+ }
37
+ const client = await this.pool.connect();
38
+ try {
39
+ for (const query of queries) {
40
+ await client.query(query);
41
+ }
42
+ } catch (error) {
43
+ // rollback if any of the queries fail
44
+ await client.query(`ROLLBACK`);
45
+ throw error;
46
+ } finally {
47
+ client.release();
48
+ }
49
+ }
50
+
51
+ async disconnect(): Promise<void> {
52
+ // Since no longer using individual client, use disconnect to reset the pool.
53
+ await this.ready;
54
+ await this.pool.end();
55
+ this.ready = this.init();
56
+ }
57
+
58
+ async connect(): Promise<void> {
59
+ await this.ready;
60
+ }
61
+
62
+ public getDatabase(): string | undefined {
63
+ return decodeURIComponent(this.dbUrl.pathname.slice(1));
64
+ }
65
+
66
+ async hasTable(tablename: string): Promise<boolean> {
67
+ await this.ready;
68
+ const query = `
69
+ SELECT EXISTS (
70
+ SELECT tablename FROM pg_catalog.pg_tables
71
+ WHERE tablename = ${this.getArgumentPlaceholder(1)}
72
+ )
73
+ `;
74
+
75
+ // Allow multiple apps to be on the same host with separate DB and querying the right
76
+ // DB for the session table exisitence
77
+ const rows = await this.query(query, [tablename]);
78
+ return rows[0].exists;
79
+ }
80
+
81
+ getArgumentPlaceholder(position: number): string {
82
+ return `$${position}`;
83
+ }
84
+
85
+ private async init(): Promise<void> {
86
+ this.pool = new pg.Pool({
87
+ host: this.dbUrl.hostname,
88
+ user: decodeURIComponent(this.dbUrl.username),
89
+ password: decodeURIComponent(this.dbUrl.password),
90
+ database: this.getDatabase(),
91
+ port: Number(this.dbUrl.port),
92
+ });
93
+ }
94
+ }
@@ -0,0 +1,27 @@
1
+ import {
2
+ RdbmsSessionStorageMigrator,
3
+ RdbmsSessionStorageMigratorOptions,
4
+ MigrationOperation,
5
+ } from '@shopify/shopify-app-session-storage';
6
+
7
+ import {PostgresConnection} from './postgres-connection';
8
+
9
+ export class PostgresSessionStorageMigrator extends RdbmsSessionStorageMigrator {
10
+ constructor(
11
+ dbConnection: PostgresConnection,
12
+ opts: Partial<RdbmsSessionStorageMigratorOptions> = {},
13
+ migrations: MigrationOperation[],
14
+ ) {
15
+ super(dbConnection, opts, migrations);
16
+ }
17
+
18
+ async initMigrationPersistence(): Promise<void> {
19
+ const migration = `
20
+ CREATE TABLE IF NOT EXISTS ${this.options.migrationDBIdentifier} (
21
+ ${
22
+ this.getOptions().migrationNameColumnName
23
+ } varchar(255) NOT NULL PRIMARY KEY
24
+ );`;
25
+ await this.connection.query(migration, []);
26
+ }
27
+ }
@@ -0,0 +1,178 @@
1
+ import {Session} from '@shopify/shopify-api';
2
+ import {
3
+ SessionStorage,
4
+ RdbmsSessionStorageOptions,
5
+ } from '@shopify/shopify-app-session-storage';
6
+
7
+ import {migrationList} from './migrations';
8
+ import {PostgresConnection} from './postgres-connection';
9
+ import {PostgresSessionStorageMigrator} from './postgres-migrator';
10
+
11
+ export interface PostgreSQLSessionStorageOptions
12
+ extends RdbmsSessionStorageOptions {
13
+ port: number;
14
+ }
15
+ const defaultPostgreSQLSessionStorageOptions: PostgreSQLSessionStorageOptions =
16
+ {
17
+ sessionTableName: 'shopify_sessions',
18
+ port: 3211,
19
+ migratorOptions: {
20
+ migrationDBIdentifier: 'shopify_sessions_migrations',
21
+ migrationNameColumnName: 'migration_name',
22
+ },
23
+ };
24
+
25
+ export class PostgreSQLSessionStorage implements SessionStorage {
26
+ static withCredentials(
27
+ host: string,
28
+ dbName: string,
29
+ username: string,
30
+ password: string,
31
+ opts: Partial<PostgreSQLSessionStorageOptions>,
32
+ ) {
33
+ return new PostgreSQLSessionStorage(
34
+ new URL(
35
+ `postgres://${encodeURIComponent(username)}:${encodeURIComponent(
36
+ password,
37
+ )}@${host}/${encodeURIComponent(dbName)}`,
38
+ ),
39
+ opts,
40
+ );
41
+ }
42
+
43
+ public readonly ready: Promise<void>;
44
+ private internalInit: Promise<void>;
45
+ private options: PostgreSQLSessionStorageOptions;
46
+ private client: PostgresConnection;
47
+ private migrator: PostgresSessionStorageMigrator;
48
+
49
+ constructor(
50
+ dbUrl: URL | string,
51
+ opts: Partial<PostgreSQLSessionStorageOptions> = {},
52
+ ) {
53
+ this.options = {...defaultPostgreSQLSessionStorageOptions, ...opts};
54
+ this.internalInit = this.init(
55
+ typeof dbUrl === 'string' ? dbUrl : dbUrl.toString(),
56
+ );
57
+ this.migrator = new PostgresSessionStorageMigrator(
58
+ this.client,
59
+ this.options.migratorOptions,
60
+ migrationList,
61
+ );
62
+ this.ready = this.migrator.applyMigrations(this.internalInit);
63
+ }
64
+
65
+ public async storeSession(session: Session): Promise<boolean> {
66
+ await this.ready;
67
+
68
+ // Note milliseconds to seconds conversion for `expires` property
69
+ const entries = session
70
+ .toPropertyArray()
71
+ .map(([key, value]) =>
72
+ key === 'expires'
73
+ ? [key, Math.floor((value as number) / 1000)]
74
+ : [key, value],
75
+ );
76
+ const query = `
77
+ INSERT INTO "${this.options.sessionTableName}"
78
+ (${entries.map(([key]) => `"${key}"`).join(', ')})
79
+ VALUES (${entries
80
+ .map((_, i) => `${this.client.getArgumentPlaceholder(i + 1)}`)
81
+ .join(', ')})
82
+ ON CONFLICT ("id") DO UPDATE SET ${entries
83
+ .map(([key]) => `"${key}" = Excluded."${key}"`)
84
+ .join(', ')};
85
+ `;
86
+ await this.client.query(
87
+ query,
88
+ entries.map(([_key, value]) => value),
89
+ );
90
+ return true;
91
+ }
92
+
93
+ public async loadSession(id: string): Promise<Session | undefined> {
94
+ await this.ready;
95
+ const query = `
96
+ SELECT * FROM "${this.options.sessionTableName}"
97
+ WHERE "id" = ${this.client.getArgumentPlaceholder(1)};
98
+ `;
99
+ const rows = await this.client.query(query, [id]);
100
+ if (!Array.isArray(rows) || rows?.length !== 1) return undefined;
101
+ const rawResult = rows[0] as any;
102
+ return this.databaseRowToSession(rawResult);
103
+ }
104
+
105
+ public async deleteSession(id: string): Promise<boolean> {
106
+ await this.ready;
107
+ const query = `
108
+ DELETE FROM "${this.options.sessionTableName}"
109
+ WHERE "id" = ${this.client.getArgumentPlaceholder(1)};
110
+ `;
111
+ await this.client.query(query, [id]);
112
+ return true;
113
+ }
114
+
115
+ public async deleteSessions(ids: string[]): Promise<boolean> {
116
+ await this.ready;
117
+ const query = `
118
+ DELETE FROM "${this.options.sessionTableName}"
119
+ WHERE "id" IN (${ids
120
+ .map((_, i) => `${this.client.getArgumentPlaceholder(i + 1)}`)
121
+ .join(', ')});
122
+ `;
123
+ await this.client.query(query, ids);
124
+ return true;
125
+ }
126
+
127
+ public async findSessionsByShop(shop: string): Promise<Session[]> {
128
+ await this.ready;
129
+
130
+ const query = `
131
+ SELECT * FROM "${this.options.sessionTableName}"
132
+ WHERE "shop" = ${this.client.getArgumentPlaceholder(1)};
133
+ `;
134
+ const rows = await this.client.query(query, [shop]);
135
+ if (!Array.isArray(rows) || rows?.length === 0) return [];
136
+
137
+ const results: Session[] = rows.map((row: any) => {
138
+ return this.databaseRowToSession(row);
139
+ });
140
+ return results;
141
+ }
142
+
143
+ public disconnect(): Promise<void> {
144
+ return this.client.disconnect();
145
+ }
146
+
147
+ private async init(dbUrl: string) {
148
+ this.client = new PostgresConnection(dbUrl, this.options.sessionTableName);
149
+ await this.connectClient();
150
+ await this.createTable();
151
+ }
152
+
153
+ private async connectClient(): Promise<void> {
154
+ await this.client.connect();
155
+ }
156
+
157
+ private async createTable() {
158
+ const query = `
159
+ CREATE TABLE IF NOT EXISTS "${this.options.sessionTableName}" (
160
+ "id" varchar(255) NOT NULL PRIMARY KEY,
161
+ "shop" varchar(255) NOT NULL,
162
+ "state" varchar(255) NOT NULL,
163
+ "isOnline" boolean NOT NULL,
164
+ "scope" varchar(255),
165
+ "expires" integer,
166
+ "onlineAccessInfo" varchar(255),
167
+ "accessToken" varchar(255)
168
+ )
169
+ `;
170
+ await this.client.query(query);
171
+ }
172
+
173
+ private databaseRowToSession(row: any): Session {
174
+ // convert seconds to milliseconds prior to creating Session object
175
+ if (row.expires) row.expires *= 1000;
176
+ return Session.fromPropertyArray(Object.entries(row));
177
+ }
178
+ }
@@ -0,0 +1,10 @@
1
+ {
2
+ "extends": "../../tsconfig.base.json",
3
+ "compilerOptions": {
4
+ "baseUrl": ".",
5
+ "outDir": "./build/ts",
6
+ "rootDir": "src"
7
+ },
8
+ "include": ["src/**/*.ts"],
9
+ "exclude": ["**/*.test.ts", "**/*.test.tsx", "**/test/*", "**/__tests__/*"]
10
+ }
@@ -0,0 +1,148 @@
1
+ # @shopify/shopify-app-session-storage-prisma
2
+
3
+ ## 4.0.1
4
+
5
+ ### Patch Changes
6
+
7
+ - 02a8341: Updated dependency on `@shopify/shopify-api` to 9.3.1
8
+ - 321d6a4: Update @shopify/shopify-api to 9.3.2
9
+ - 56c2198: Bumps @prisma/client from 5.8.1 to 5.9.1.
10
+ - 2d5181b: Add error cause when throwing MissingSessionTableError
11
+ - Updated dependencies [02a8341]
12
+ - Updated dependencies [321d6a4]
13
+ - @shopify/shopify-app-session-storage@2.1.1
14
+
15
+ ## 4.0.0
16
+
17
+ ### Minor Changes
18
+
19
+ - 64e0246: Update shopify-api version to 9.2.0
20
+
21
+ ### Patch Changes
22
+
23
+ - f5742c1: Updated dependency on `@shopify/shopify-api`
24
+ - Updated dependencies [f5742c1]
25
+ - Updated dependencies [64e0246]
26
+ - @shopify/shopify-app-session-storage@2.1.0
27
+
28
+ ## 3.0.0
29
+
30
+ ### Major Changes
31
+
32
+ - ea88df2: Updated the dependency on `prisma` to v5+. This package itself has no breaking changes, but you'll need to update your app's dependency on Prisma as well as this package.
33
+
34
+ ### Patch Changes
35
+
36
+ - b4eeb24: Improved and simplified package.json dependencies
37
+ - b998c30: Bump shopify-api version from 9.0.1 to 9.0.2
38
+ - Updated dependencies [b4eeb24]
39
+ - Updated dependencies [b998c30]
40
+ - @shopify/shopify-app-session-storage@2.0.4
41
+
42
+ ## 2.0.3
43
+
44
+ ### Patch Changes
45
+
46
+ - d3e4b5e: Updated the dependency on `@shopify/shopify-api`
47
+ - Updated dependencies [d3e4b5e]
48
+ - @shopify/shopify-app-session-storage@2.0.3
49
+
50
+ ## 2.0.2
51
+
52
+ ### Patch Changes
53
+
54
+ - 3685bd4: Bump shopify-api to ^8.1.1
55
+ - Updated dependencies [3685bd4]
56
+ - @shopify/shopify-app-session-storage@2.0.2
57
+
58
+ ## 2.0.1
59
+
60
+ ### Patch Changes
61
+
62
+ - 6d12840: Updating dependencies on @shopify/shopify-api
63
+ - Updated dependencies [6d12840]
64
+ - @shopify/shopify-app-session-storage@2.0.1
65
+
66
+ ## 2.0.0
67
+
68
+ ### Major Changes
69
+
70
+ - f837060: **Removed support for Node 14**
71
+
72
+ Node 14 has reached its [EOL](https://endoflife.date/nodejs), and dependencies to this package no longer work on Node 14.
73
+ Because of that, we can no longer support that version.
74
+
75
+ If your app is running on Node 14, you'll need to update to a more recent version before upgrading this package.
76
+
77
+ This upgrade does not require any code changes.
78
+
79
+ ### Patch Changes
80
+
81
+ - a69d6fc: Updating dependency on @shopify/shopify-api to v.8.0.1
82
+ - Updated dependencies [f837060]
83
+ - Updated dependencies [a69d6fc]
84
+ - @shopify/shopify-app-session-storage@2.0.0
85
+
86
+ ## 1.1.1
87
+
88
+ ### Patch Changes
89
+
90
+ - 616388d: Updating dependency on @shopify/shopify-api to 7.7.0
91
+ - Updated dependencies [616388d]
92
+ - @shopify/shopify-app-session-storage@1.1.10
93
+
94
+ ## 1.1.0
95
+
96
+ ### Minor Changes
97
+
98
+ - 695f829: Allow using custom tables for the Prisma session storage.
99
+
100
+ ### Patch Changes
101
+
102
+ - 5b862fe: Upgraded shopify-api dependency to 7.6.0
103
+ - Updated dependencies [5b862fe]
104
+ - @shopify/shopify-app-session-storage@1.1.9
105
+
106
+ ## 1.0.5
107
+
108
+ ### Patch Changes
109
+
110
+ - 346b623: Updating dependency on @shopify/shopify-api
111
+ - Updated dependencies [346b623]
112
+ - @shopify/shopify-app-session-storage@1.1.8
113
+
114
+ ## 1.0.4
115
+
116
+ ### Patch Changes
117
+
118
+ - 13b9048: Updating @shopify/shopify-api dependency to the latest version
119
+ - Updated dependencies [13b9048]
120
+ - @shopify/shopify-app-session-storage@1.1.7
121
+
122
+ ## 1.0.3
123
+
124
+ ### Patch Changes
125
+
126
+ - 90a62de: Throw a specific error when the database doesn't exist to help indicate that a migration might be needed
127
+
128
+ ## 1.0.2
129
+
130
+ ### Patch Changes
131
+
132
+ - 32296d7: Update @shopify/shopify-api dependency to 7.5.0
133
+ - Updated dependencies [32296d7]
134
+ - @shopify/shopify-app-session-storage@1.1.6
135
+
136
+ ## 1.0.1
137
+
138
+ ### Patch Changes
139
+
140
+ - 93e9126: Updating @shopify/shopify-api dependency
141
+ - Updated dependencies [93e9126]
142
+ - @shopify/shopify-app-session-storage@1.1.5
143
+
144
+ ## 1.0.0
145
+
146
+ ### Major Changes
147
+
148
+ - Initial release of @shopify/shopify-app-session-storage-prisma
@@ -0,0 +1,9 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2022-present, Shopify Inc.
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
6
+
7
+ The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
8
+
9
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.