disco_app 0.12.5

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 (354) hide show
  1. checksums.yaml +7 -0
  2. data/Rakefile +37 -0
  3. data/app/assets/components/disco_app/buttons/model-destroy-button.es6.jsx +31 -0
  4. data/app/assets/components/disco_app/forms/model-form.es6.jsx +64 -0
  5. data/app/assets/components/embedded_app/bar.es6.jsx +31 -0
  6. data/app/assets/components/shopify/buttons/_buttons.scss +547 -0
  7. data/app/assets/components/shopify/buttons/button.es6.jsx +15 -0
  8. data/app/assets/components/shopify/card/_card.scss +342 -0
  9. data/app/assets/components/shopify/card/card-header.es6.jsx +34 -0
  10. data/app/assets/components/shopify/card/card-section.es6.jsx +26 -0
  11. data/app/assets/components/shopify/card/card.es6.jsx +16 -0
  12. data/app/assets/components/shopify/image/_image.scss +82 -0
  13. data/app/assets/components/shopify/table/_table.scss +18 -0
  14. data/app/assets/components/shopify/typography/_typography.scss +23 -0
  15. data/app/assets/components/shopify/typography/ui-heading.es6.jsx +16 -0
  16. data/app/assets/components/shopify/ui-layout/_ui-layout.scss +157 -0
  17. data/app/assets/components/shopify/ui-layout/ui-annotated-section.es6.jsx +29 -0
  18. data/app/assets/components/shopify/ui-layout/ui-empty-state.es6.jsx +35 -0
  19. data/app/assets/components/shopify/ui-layout/ui-footer-help.es6.jsx +13 -0
  20. data/app/assets/components/shopify/ui-layout/ui-layout-item.es6.jsx +11 -0
  21. data/app/assets/components/shopify/ui-layout/ui-layout-section.es6.jsx +19 -0
  22. data/app/assets/components/shopify/ui-layout/ui-layout-sections.es6.jsx +11 -0
  23. data/app/assets/components/shopify/ui-layout/ui-layout.es6.jsx +11 -0
  24. data/app/assets/components/shopify/ui-layout/ui-page-actions.es6.jsx +13 -0
  25. data/app/assets/components/shopify/ui-layout/ui-page-actions__buttons.es6.jsx +27 -0
  26. data/app/assets/components/shopify/ui-stack/_ui-stack.scss +39 -0
  27. data/app/assets/components/shopify/ui-stack/ui-stack-item.es6.jsx +21 -0
  28. data/app/assets/components/shopify/ui-stack/ui-stack.es6.jsx +24 -0
  29. data/app/assets/images/disco_app/icon.svg +1 -0
  30. data/app/assets/images/disco_app/icons.svg +0 -0
  31. data/app/assets/javascripts/disco_app/components/custom/filterable_shop_list.js.jsx +61 -0
  32. data/app/assets/javascripts/disco_app/components/custom/inline-radio-options.es6.jsx +59 -0
  33. data/app/assets/javascripts/disco_app/components/custom/rules-editor.es6.jsx +432 -0
  34. data/app/assets/javascripts/disco_app/components/custom/shop_filter_query.js.jsx +13 -0
  35. data/app/assets/javascripts/disco_app/components/custom/shop_filter_tab.js.jsx +34 -0
  36. data/app/assets/javascripts/disco_app/components/custom/shop_filter_tabs.js.jsx +21 -0
  37. data/app/assets/javascripts/disco_app/components/custom/shop_list.js.jsx +142 -0
  38. data/app/assets/javascripts/disco_app/components/custom/shop_row.js.jsx +43 -0
  39. data/app/assets/javascripts/disco_app/components/custom/shopify_admin_link.js.jsx +29 -0
  40. data/app/assets/javascripts/disco_app/components/ui-kit/forms/base_form.es6.jsx +72 -0
  41. data/app/assets/javascripts/disco_app/components/ui-kit/forms/base_input.es6.jsx +20 -0
  42. data/app/assets/javascripts/disco_app/components/ui-kit/forms/input-checkbox.es6.jsx +30 -0
  43. data/app/assets/javascripts/disco_app/components/ui-kit/forms/input-radio.es6.jsx +30 -0
  44. data/app/assets/javascripts/disco_app/components/ui-kit/forms/input-select.es6.jsx +45 -0
  45. data/app/assets/javascripts/disco_app/components/ui-kit/forms/input-text.es6.jsx +69 -0
  46. data/app/assets/javascripts/disco_app/components/ui-kit/forms/input-textarea.es6.jsx +48 -0
  47. data/app/assets/javascripts/disco_app/components/ui-kit/forms/input-time.es6.jsx +7 -0
  48. data/app/assets/javascripts/disco_app/components/ui-kit/forms/ui-form__element.es6.jsx +17 -0
  49. data/app/assets/javascripts/disco_app/components/ui-kit/forms/ui-form__group.es6.jsx +11 -0
  50. data/app/assets/javascripts/disco_app/components/ui-kit/forms/ui-form__section.es6.jsx +11 -0
  51. data/app/assets/javascripts/disco_app/components/ui-kit/icons/icon-chevron.es6.jsx +33 -0
  52. data/app/assets/javascripts/disco_app/components/ui-kit/icons/next-icon.es6.jsx +19 -0
  53. data/app/assets/javascripts/disco_app/components/ui-kit/input_select.es6.jsx +21 -0
  54. data/app/assets/javascripts/disco_app/components/ui-kit/tables/table.es6.jsx +27 -0
  55. data/app/assets/javascripts/disco_app/components.js +3 -0
  56. data/app/assets/javascripts/disco_app/disco_app.js +9 -0
  57. data/app/assets/javascripts/disco_app/frame.js +152 -0
  58. data/app/assets/javascripts/disco_app/shopify-turbolinks.js +7 -0
  59. data/app/assets/javascripts/disco_app/ui-kit.js +1 -0
  60. data/app/assets/stylesheets/disco_app/admin/_header.scss +75 -0
  61. data/app/assets/stylesheets/disco_app/admin/_layout.scss +32 -0
  62. data/app/assets/stylesheets/disco_app/admin/_nav.scss +184 -0
  63. data/app/assets/stylesheets/disco_app/admin.scss +11 -0
  64. data/app/assets/stylesheets/disco_app/disco_app.scss +26 -0
  65. data/app/assets/stylesheets/disco_app/frame/_buttons.scss +54 -0
  66. data/app/assets/stylesheets/disco_app/frame/_forms.scss +26 -0
  67. data/app/assets/stylesheets/disco_app/frame/_layout.scss +77 -0
  68. data/app/assets/stylesheets/disco_app/frame/_type.scss +25 -0
  69. data/app/assets/stylesheets/disco_app/frame.scss +10 -0
  70. data/app/assets/stylesheets/disco_app/mixins/_flexbox.scss +400 -0
  71. data/app/assets/stylesheets/disco_app/ui-kit/_ui-forms.scss +69 -0
  72. data/app/assets/stylesheets/disco_app/ui-kit/_ui-icons.scss +28 -0
  73. data/app/assets/stylesheets/disco_app/ui-kit/_ui-kit.scss +5121 -0
  74. data/app/assets/stylesheets/disco_app/ui-kit/_ui-layout.scss +15 -0
  75. data/app/assets/stylesheets/disco_app/ui-kit/_ui-tabs.scss +75 -0
  76. data/app/assets/stylesheets/disco_app/ui-kit/_ui-type.scss +13 -0
  77. data/app/clients/disco_app/api_client.rb +27 -0
  78. data/app/clients/disco_app/disco_api_error.rb +2 -0
  79. data/app/controllers/disco_app/admin/app_settings_controller.rb +3 -0
  80. data/app/controllers/disco_app/admin/application_controller.rb +10 -0
  81. data/app/controllers/disco_app/admin/concerns/app_settings_controller.rb +24 -0
  82. data/app/controllers/disco_app/admin/concerns/authenticated_controller.rb +20 -0
  83. data/app/controllers/disco_app/admin/concerns/plans_controller.rb +54 -0
  84. data/app/controllers/disco_app/admin/concerns/shops_controller.rb +7 -0
  85. data/app/controllers/disco_app/admin/concerns/subscriptions_controller.rb +29 -0
  86. data/app/controllers/disco_app/admin/plans_controller.rb +3 -0
  87. data/app/controllers/disco_app/admin/resources/shops_controller.rb +3 -0
  88. data/app/controllers/disco_app/admin/shops_controller.rb +3 -0
  89. data/app/controllers/disco_app/admin/subscriptions_controller.rb +3 -0
  90. data/app/controllers/disco_app/charges_controller.rb +47 -0
  91. data/app/controllers/disco_app/concerns/app_proxy_controller.rb +40 -0
  92. data/app/controllers/disco_app/concerns/authenticated_controller.rb +71 -0
  93. data/app/controllers/disco_app/concerns/carrier_request_controller.rb +35 -0
  94. data/app/controllers/disco_app/frame_controller.rb +9 -0
  95. data/app/controllers/disco_app/install_controller.rb +27 -0
  96. data/app/controllers/disco_app/subscriptions_controller.rb +32 -0
  97. data/app/controllers/disco_app/webhooks_controller.rb +46 -0
  98. data/app/controllers/sessions_controller.rb +33 -0
  99. data/app/helpers/disco_app/application_helper.rb +68 -0
  100. data/app/jobs/disco_app/app_installed_job.rb +3 -0
  101. data/app/jobs/disco_app/app_uninstalled_job.rb +3 -0
  102. data/app/jobs/disco_app/concerns/app_installed_job.rb +39 -0
  103. data/app/jobs/disco_app/concerns/app_uninstalled_job.rb +21 -0
  104. data/app/jobs/disco_app/concerns/render_asset_group_job.rb +8 -0
  105. data/app/jobs/disco_app/concerns/shop_update_job.rb +13 -0
  106. data/app/jobs/disco_app/concerns/subscription_changed_job.rb +8 -0
  107. data/app/jobs/disco_app/concerns/synchronise_carrier_service_job.rb +55 -0
  108. data/app/jobs/disco_app/concerns/synchronise_resources_job.rb +12 -0
  109. data/app/jobs/disco_app/concerns/synchronise_webhooks_job.rb +52 -0
  110. data/app/jobs/disco_app/render_asset_group_job.rb +3 -0
  111. data/app/jobs/disco_app/send_subscription_job.rb +7 -0
  112. data/app/jobs/disco_app/shop_job.rb +27 -0
  113. data/app/jobs/disco_app/shop_update_job.rb +3 -0
  114. data/app/jobs/disco_app/subscription_changed_job.rb +3 -0
  115. data/app/jobs/disco_app/synchronise_carrier_service_job.rb +3 -0
  116. data/app/jobs/disco_app/synchronise_resources_job.rb +3 -0
  117. data/app/jobs/disco_app/synchronise_webhooks_job.rb +3 -0
  118. data/app/models/disco_app/app_settings.rb +3 -0
  119. data/app/models/disco_app/application_charge.rb +18 -0
  120. data/app/models/disco_app/concerns/app_settings.rb +7 -0
  121. data/app/models/disco_app/concerns/can_be_liquified.rb +45 -0
  122. data/app/models/disco_app/concerns/has_metafields.rb +48 -0
  123. data/app/models/disco_app/concerns/plan.rb +26 -0
  124. data/app/models/disco_app/concerns/plan_code.rb +15 -0
  125. data/app/models/disco_app/concerns/renders_assets.rb +166 -0
  126. data/app/models/disco_app/concerns/shop.rb +96 -0
  127. data/app/models/disco_app/concerns/subscription.rb +66 -0
  128. data/app/models/disco_app/concerns/synchronises.rb +58 -0
  129. data/app/models/disco_app/concerns/taggable.rb +16 -0
  130. data/app/models/disco_app/plan.rb +3 -0
  131. data/app/models/disco_app/plan_code.rb +3 -0
  132. data/app/models/disco_app/recurring_application_charge.rb +18 -0
  133. data/app/models/disco_app/session_storage.rb +18 -0
  134. data/app/models/disco_app/shop.rb +3 -0
  135. data/app/models/disco_app/subscription.rb +3 -0
  136. data/app/resources/disco_app/admin/resources/concerns/shop_resource.rb +100 -0
  137. data/app/resources/disco_app/admin/resources/shop_resource.rb +4 -0
  138. data/app/services/disco_app/carrier_request_service.rb +15 -0
  139. data/app/services/disco_app/charges_service.rb +81 -0
  140. data/app/services/disco_app/proxy_service.rb +17 -0
  141. data/app/services/disco_app/request_validation_service.rb +15 -0
  142. data/app/services/disco_app/subscription_service.rb +46 -0
  143. data/app/services/disco_app/webhook_service.rb +30 -0
  144. data/app/views/disco_app/admin/app_settings/edit.html.erb +5 -0
  145. data/app/views/disco_app/admin/plans/_form.html.erb +72 -0
  146. data/app/views/disco_app/admin/plans/_plan_code_fields.html.erb +15 -0
  147. data/app/views/disco_app/admin/plans/edit.html.erb +7 -0
  148. data/app/views/disco_app/admin/plans/index.html.erb +43 -0
  149. data/app/views/disco_app/admin/plans/new.html.erb +7 -0
  150. data/app/views/disco_app/admin/shops/index.html.erb +13 -0
  151. data/app/views/disco_app/admin/subscriptions/edit.html.erb +33 -0
  152. data/app/views/disco_app/charges/activate.html.erb +1 -0
  153. data/app/views/disco_app/charges/create.html.erb +1 -0
  154. data/app/views/disco_app/charges/new.html.erb +23 -0
  155. data/app/views/disco_app/frame/frame.html.erb +36 -0
  156. data/app/views/disco_app/install/installing.html.erb +22 -0
  157. data/app/views/disco_app/install/uninstalling.html.erb +1 -0
  158. data/app/views/disco_app/proxy_errors/404.html.erb +1 -0
  159. data/app/views/disco_app/shared/_card.html.erb +14 -0
  160. data/app/views/disco_app/shared/_icons.html.erb +1 -0
  161. data/app/views/disco_app/shared/_section.html.erb +17 -0
  162. data/app/views/disco_app/subscriptions/new.html.erb +25 -0
  163. data/app/views/layouts/admin/_nav_items.erb +20 -0
  164. data/app/views/layouts/admin.html.erb +67 -0
  165. data/app/views/layouts/application.html.erb +18 -0
  166. data/app/views/layouts/embedded_app.html.erb +50 -0
  167. data/app/views/layouts/embedded_app_modal.html.erb +28 -0
  168. data/app/views/shopify_app/sessions/new.html.erb +42 -0
  169. data/config/routes.rb +64 -0
  170. data/db/migrate/20150525000000_create_shops_if_not_existent.rb +110 -0
  171. data/lib/disco_app/configuration.rb +45 -0
  172. data/lib/disco_app/constants.rb +4 -0
  173. data/lib/disco_app/engine.rb +26 -0
  174. data/lib/disco_app/session.rb +14 -0
  175. data/lib/disco_app/support/file_fixtures.rb +23 -0
  176. data/lib/disco_app/test_help.rb +11 -0
  177. data/lib/disco_app/version.rb +3 -0
  178. data/lib/disco_app.rb +7 -0
  179. data/lib/generators/disco_app/USAGE +5 -0
  180. data/lib/generators/disco_app/disco_app_generator.rb +236 -0
  181. data/lib/generators/disco_app/templates/assets/javascripts/application.js +17 -0
  182. data/lib/generators/disco_app/templates/assets/javascripts/components.js +3 -0
  183. data/lib/generators/disco_app/templates/assets/stylesheets/application.scss +5 -0
  184. data/lib/generators/disco_app/templates/config/database.yml.tt +20 -0
  185. data/lib/generators/disco_app/templates/config/newrelic.yml +26 -0
  186. data/lib/generators/disco_app/templates/config/puma.rb +15 -0
  187. data/lib/generators/disco_app/templates/controllers/home_controller.rb +7 -0
  188. data/lib/generators/disco_app/templates/initializers/disco_app.rb +28 -0
  189. data/lib/generators/disco_app/templates/initializers/rollbar.rb +23 -0
  190. data/lib/generators/disco_app/templates/initializers/session_store.rb +2 -0
  191. data/lib/generators/disco_app/templates/initializers/shopify_app.rb +6 -0
  192. data/lib/generators/disco_app/templates/initializers/shopify_session_repository.rb +7 -0
  193. data/lib/generators/disco_app/templates/root/CHECKS +4 -0
  194. data/lib/generators/disco_app/templates/root/Procfile +2 -0
  195. data/lib/generators/disco_app/templates/views/home/index.html.erb +2 -0
  196. data/lib/tasks/api.rake +10 -0
  197. data/lib/tasks/carrier_service.rake +10 -0
  198. data/lib/tasks/database.rake +8 -0
  199. data/lib/tasks/sessions.rake +9 -0
  200. data/lib/tasks/shops.rake +10 -0
  201. data/lib/tasks/start.rake +3 -0
  202. data/lib/tasks/webhooks.rake +10 -0
  203. data/test/clients/disco_app/api_client_test.rb +22 -0
  204. data/test/controllers/disco_app/admin/shops_controller_test.rb +54 -0
  205. data/test/controllers/disco_app/charges_controller_test.rb +99 -0
  206. data/test/controllers/disco_app/install_controller_test.rb +50 -0
  207. data/test/controllers/disco_app/subscriptions_controller_test.rb +68 -0
  208. data/test/controllers/disco_app/webhooks_controller_test.rb +58 -0
  209. data/test/controllers/home_controller_test.rb +101 -0
  210. data/test/controllers/proxy_controller_test.rb +42 -0
  211. data/test/disco_app_test.rb +7 -0
  212. data/test/dummy/Rakefile +6 -0
  213. data/test/dummy/app/assets/javascripts/application.js +17 -0
  214. data/test/dummy/app/assets/stylesheets/application.scss +5 -0
  215. data/test/dummy/app/controllers/application_controller.rb +6 -0
  216. data/test/dummy/app/controllers/carrier_request_controller.rb +10 -0
  217. data/test/dummy/app/controllers/disco_app/admin/shops_controller.rb +8 -0
  218. data/test/dummy/app/controllers/home_controller.rb +7 -0
  219. data/test/dummy/app/controllers/proxy_controller.rb +8 -0
  220. data/test/dummy/app/helpers/application_helper.rb +2 -0
  221. data/test/dummy/app/jobs/carts_update_job.rb +7 -0
  222. data/test/dummy/app/jobs/disco_app/app_installed_job.rb +16 -0
  223. data/test/dummy/app/jobs/disco_app/app_uninstalled_job.rb +11 -0
  224. data/test/dummy/app/jobs/products_create_job.rb +7 -0
  225. data/test/dummy/app/jobs/products_delete_job.rb +7 -0
  226. data/test/dummy/app/jobs/products_update_job.rb +7 -0
  227. data/test/dummy/app/models/cart.rb +24 -0
  228. data/test/dummy/app/models/disco_app/shop.rb +20 -0
  229. data/test/dummy/app/models/js_configuration.rb +8 -0
  230. data/test/dummy/app/models/product.rb +9 -0
  231. data/test/dummy/app/models/widget_configuration.rb +10 -0
  232. data/test/dummy/app/views/assets/script_tag.js.erb +1 -0
  233. data/test/dummy/app/views/assets/test.js.erb +1 -0
  234. data/test/dummy/app/views/assets/widget.js.erb +2 -0
  235. data/test/dummy/app/views/assets/widget.scss.erb +3 -0
  236. data/test/dummy/app/views/home/index.html.erb +2 -0
  237. data/test/dummy/app/views/snippets/widget.liquid.erb +1 -0
  238. data/test/dummy/bin/bundle +3 -0
  239. data/test/dummy/bin/rails +4 -0
  240. data/test/dummy/bin/rake +4 -0
  241. data/test/dummy/bin/setup +29 -0
  242. data/test/dummy/config/application.rb +38 -0
  243. data/test/dummy/config/boot.rb +5 -0
  244. data/test/dummy/config/database.codeship.yml +23 -0
  245. data/test/dummy/config/database.gitlab-ci.yml +24 -0
  246. data/test/dummy/config/database.yml +20 -0
  247. data/test/dummy/config/environment.rb +5 -0
  248. data/test/dummy/config/environments/development.rb +41 -0
  249. data/test/dummy/config/environments/production.rb +85 -0
  250. data/test/dummy/config/environments/test.rb +42 -0
  251. data/test/dummy/config/initializers/assets.rb +11 -0
  252. data/test/dummy/config/initializers/backtrace_silencers.rb +7 -0
  253. data/test/dummy/config/initializers/cookies_serializer.rb +3 -0
  254. data/test/dummy/config/initializers/disco_app.rb +28 -0
  255. data/test/dummy/config/initializers/filter_parameter_logging.rb +4 -0
  256. data/test/dummy/config/initializers/inflections.rb +16 -0
  257. data/test/dummy/config/initializers/mime_types.rb +4 -0
  258. data/test/dummy/config/initializers/omniauth.rb +7 -0
  259. data/test/dummy/config/initializers/session_store.rb +2 -0
  260. data/test/dummy/config/initializers/shopify_app.rb +6 -0
  261. data/test/dummy/config/initializers/shopify_session_repository.rb +7 -0
  262. data/test/dummy/config/initializers/wrap_parameters.rb +14 -0
  263. data/test/dummy/config/locales/en.yml +23 -0
  264. data/test/dummy/config/routes.rb +11 -0
  265. data/test/dummy/config/secrets.yml +22 -0
  266. data/test/dummy/config.ru +4 -0
  267. data/test/dummy/db/migrate/20160307182229_create_products.rb +11 -0
  268. data/test/dummy/db/migrate/20160530160739_create_asset_models.rb +19 -0
  269. data/test/dummy/db/migrate/20161105054746_create_carts.rb +13 -0
  270. data/test/dummy/db/schema.rb +152 -0
  271. data/test/dummy/public/404.html +67 -0
  272. data/test/dummy/public/422.html +67 -0
  273. data/test/dummy/public/500.html +66 -0
  274. data/test/dummy/public/favicon.ico +0 -0
  275. data/test/fixtures/api/subscriptions/valid_request.json +40 -0
  276. data/test/fixtures/api/widget_store/assets/create_script_tag_js_request.json +6 -0
  277. data/test/fixtures/api/widget_store/assets/create_script_tag_js_response.json +12 -0
  278. data/test/fixtures/api/widget_store/assets/create_script_tag_request.json +6 -0
  279. data/test/fixtures/api/widget_store/assets/create_script_tag_response.json +10 -0
  280. data/test/fixtures/api/widget_store/assets/create_test_js_request.json +6 -0
  281. data/test/fixtures/api/widget_store/assets/create_test_js_response.json +12 -0
  282. data/test/fixtures/api/widget_store/assets/create_widget_js_request.json +6 -0
  283. data/test/fixtures/api/widget_store/assets/create_widget_js_response.json +12 -0
  284. data/test/fixtures/api/widget_store/assets/create_widget_liquid_request.json +6 -0
  285. data/test/fixtures/api/widget_store/assets/create_widget_liquid_response.json +12 -0
  286. data/test/fixtures/api/widget_store/assets/create_widget_scss_request.json +6 -0
  287. data/test/fixtures/api/widget_store/assets/create_widget_scss_response.json +12 -0
  288. data/test/fixtures/api/widget_store/assets/get_script_tags_empty_request.json +1 -0
  289. data/test/fixtures/api/widget_store/assets/get_script_tags_empty_response.json +1 -0
  290. data/test/fixtures/api/widget_store/assets/get_script_tags_preexisting_request.json +1 -0
  291. data/test/fixtures/api/widget_store/assets/get_script_tags_preexisting_response.json +12 -0
  292. data/test/fixtures/api/widget_store/assets/update_script_tag_request.json +10 -0
  293. data/test/fixtures/api/widget_store/assets/update_script_tag_response.json +10 -0
  294. data/test/fixtures/api/widget_store/carrier_services.json +1 -0
  295. data/test/fixtures/api/widget_store/carrier_services_create.json +8 -0
  296. data/test/fixtures/api/widget_store/charges/activate_application_charge_request.json +16 -0
  297. data/test/fixtures/api/widget_store/charges/activate_application_charge_response.json +1 -0
  298. data/test/fixtures/api/widget_store/charges/activate_recurring_application_charge_request.json +20 -0
  299. data/test/fixtures/api/widget_store/charges/activate_recurring_application_charge_response.json +1 -0
  300. data/test/fixtures/api/widget_store/charges/create_application_charge_request.json +9 -0
  301. data/test/fixtures/api/widget_store/charges/create_application_charge_response.json +16 -0
  302. data/test/fixtures/api/widget_store/charges/create_recurring_application_charge_request.json +9 -0
  303. data/test/fixtures/api/widget_store/charges/create_recurring_application_charge_response.json +20 -0
  304. data/test/fixtures/api/widget_store/charges/create_second_recurring_application_charge_request.json +9 -0
  305. data/test/fixtures/api/widget_store/charges/create_second_recurring_application_charge_response.json +20 -0
  306. data/test/fixtures/api/widget_store/charges/get_accepted_application_charge_response.json +16 -0
  307. data/test/fixtures/api/widget_store/charges/get_accepted_recurring_application_charge_response.json +20 -0
  308. data/test/fixtures/api/widget_store/charges/get_declined_application_charge_response.json +16 -0
  309. data/test/fixtures/api/widget_store/charges/get_declined_recurring_application_charge_response.json +20 -0
  310. data/test/fixtures/api/widget_store/charges/get_pending_application_charge_response.json +16 -0
  311. data/test/fixtures/api/widget_store/charges/get_pending_recurring_application_charge_response.json +20 -0
  312. data/test/fixtures/api/widget_store/products/write_metafields_multiple_namespaces_request.json +31 -0
  313. data/test/fixtures/api/widget_store/products/write_metafields_multiple_namespaces_response.json +1 -0
  314. data/test/fixtures/api/widget_store/products/write_metafields_single_namespace_request.json +19 -0
  315. data/test/fixtures/api/widget_store/products/write_metafields_single_namespace_response.json +1 -0
  316. data/test/fixtures/api/widget_store/shop.json +46 -0
  317. data/test/fixtures/api/widget_store/webhooks.json +1 -0
  318. data/test/fixtures/assets/test.js +1 -0
  319. data/test/fixtures/assets/test.min.js +1 -0
  320. data/test/fixtures/carts.yml +5 -0
  321. data/test/fixtures/disco_app/application_charges.yml +11 -0
  322. data/test/fixtures/disco_app/plan_codes.yml +13 -0
  323. data/test/fixtures/disco_app/plans.yml +37 -0
  324. data/test/fixtures/disco_app/recurring_application_charges.yml +13 -0
  325. data/test/fixtures/disco_app/shops.yml +12 -0
  326. data/test/fixtures/disco_app/subscriptions.yml +25 -0
  327. data/test/fixtures/js_configurations.yml +3 -0
  328. data/test/fixtures/liquid/model.liquid +8 -0
  329. data/test/fixtures/products.yml +4 -0
  330. data/test/fixtures/webhooks/app_uninstalled.json +46 -0
  331. data/test/fixtures/webhooks/cart_updated.json +28 -0
  332. data/test/fixtures/webhooks/product_created.json +167 -0
  333. data/test/fixtures/webhooks/product_deleted.json +3 -0
  334. data/test/fixtures/webhooks/product_updated.json +167 -0
  335. data/test/fixtures/widget_configurations.yml +4 -0
  336. data/test/integration/synchronises_test.rb +62 -0
  337. data/test/jobs/disco_app/app_installed_job_test.rb +57 -0
  338. data/test/jobs/disco_app/app_uninstalled_job_test.rb +30 -0
  339. data/test/jobs/disco_app/send_subscription_job_test.rb +24 -0
  340. data/test/jobs/disco_app/synchronise_carrier_service_job_test.rb +25 -0
  341. data/test/jobs/disco_app/synchronise_webhooks_job_test.rb +30 -0
  342. data/test/models/disco_app/can_be_liquified_test.rb +55 -0
  343. data/test/models/disco_app/has_metafields_test.rb +40 -0
  344. data/test/models/disco_app/plan_test.rb +5 -0
  345. data/test/models/disco_app/renders_assets_test.rb +109 -0
  346. data/test/models/disco_app/session_test.rb +31 -0
  347. data/test/models/disco_app/shop_test.rb +43 -0
  348. data/test/models/disco_app/subscription_test.rb +19 -0
  349. data/test/services/disco_app/charges_service_test.rb +112 -0
  350. data/test/services/disco_app/subscription_service_test.rb +60 -0
  351. data/test/support/test_file_fixtures.rb +29 -0
  352. data/test/support/test_shopify_api.rb +16 -0
  353. data/test/test_helper.rb +58 -0
  354. metadata +987 -0
@@ -0,0 +1,166 @@
1
+ require 'render_anywhere'
2
+ require 'uglifier'
3
+
4
+ module DiscoApp::Concerns::RendersAssets
5
+ extend ActiveSupport::Concern
6
+
7
+ included do
8
+ include RenderAnywhere
9
+ after_commit :queue_render_asset_group_job
10
+ end
11
+
12
+ class_methods do
13
+
14
+ # Ruby "macro" that allows the definition of a number of Shopify assets that
15
+ # should be rendered and uploaded when certain attributes on the including
16
+ # class change. This assumes that the including class (1) is an ActiveRecord
17
+ # model that supports an `after_commit` callback; and (2) that the model has
18
+ # a `shop` method (or attribute) that references the DiscoApp::Shop instance
19
+ # associated with the current model.
20
+ #
21
+ # Options
22
+ #
23
+ # assets: Required. A list of asset templates to be rendered and
24
+ # uploaded to Shopify. The order of assets matters only
25
+ # in that subsequent asset templates will have access to
26
+ # the public CDN url of earlier-rendered assets through
27
+ # a "@public_urls" context variable.
28
+ #
29
+ # triggered_by: Optional. A list of attributes that should trigger the
30
+ # re-rendering and upload of the assets defined by the
31
+ # `assets` option, provided as a list of string. If
32
+ # empty or nil, nothing will be triggered.
33
+ #
34
+ # script_tags: Optional. A list of assets that should have script
35
+ # tags created or updated after being rendered to the
36
+ # storefront.
37
+ #
38
+ # minify: Optional. Whether Javascript assets should be minified
39
+ # after being rendered. Defaults to true in production
40
+ # environments, false otherwise. Note that stylesheet
41
+ # assets, when uploaded as .scss files, are
42
+ # automatically minified by Shopify, so we don't need to
43
+ # do it on our end.
44
+ #
45
+ def renders_assets(*asset_groups)
46
+ options = asset_groups.last.is_a?(Hash) ? asset_groups.pop : {}
47
+ options = renders_assets_default_options.merge(options)
48
+
49
+ # Ensure assets, triggered by and script tag options are arrays.
50
+ options[:assets] = options[:assets] ? Array(options[:assets]).map(&:to_sym) : []
51
+ options[:triggered_by] = options[:triggered_by] ? Array(options[:triggered_by]).map(&:to_s) : []
52
+ options[:script_tags] = options[:script_tags] ? Array(options[:script_tags]).map(&:to_sym) : []
53
+
54
+ asset_groups.each do |asset_group|
55
+ renderable_asset_groups[asset_group.to_sym] = options
56
+ end
57
+ end
58
+
59
+ # Return a list of renderable asset groups, along with their options.
60
+ def renderable_asset_groups
61
+ @renderable_asset_groups ||= {}
62
+ end
63
+
64
+ # Return the default options for the `renders_assets` macro.
65
+ def renders_assets_default_options
66
+ {
67
+ assets: nil,
68
+ triggered_by: nil,
69
+ script_tags: nil,
70
+ minify: Rails.env.production?
71
+ }
72
+ end
73
+
74
+ end
75
+
76
+ # Callback, triggered after a model save. Iterates through each asset group
77
+ # defined on the model and queues a render job if any of the changed
78
+ # attributes are found in the asset group's triggered_by list.
79
+ def queue_render_asset_group_job
80
+ renderable_asset_groups.each do |asset_group, options|
81
+ unless (previous_changes.keys & options[:triggered_by]).empty?
82
+ DiscoApp::RenderAssetGroupJob.perform_later(shop, self, asset_group.to_s)
83
+ end
84
+ end
85
+ end
86
+
87
+ # Copies the class-level hash of assets with symbol asset name as keys and
88
+ # their corresponding options as values to the instance.
89
+ def renderable_asset_groups
90
+ @renderable_asset_groups ||= self.class.renderable_asset_groups.dup
91
+ end
92
+
93
+ # Render the specified asset group and upload the result to Shopify.
94
+ def render_asset_group(asset_group)
95
+ options = renderable_asset_groups[asset_group]
96
+ public_urls = {}.with_indifferent_access
97
+
98
+ options[:assets].each do |asset|
99
+ # Create/replace the asset via the Shopify API.
100
+ shopify_asset = shop.with_api_context {
101
+ ShopifyAPI::Asset.create(
102
+ key: asset,
103
+ value: render_asset_group_asset(asset, public_urls, options)
104
+ )
105
+ }
106
+
107
+ # Store the public URL to this asset, so that we're able to use it in
108
+ # subsequent template renders. Adds a .css suffix to .scss assets, so that
109
+ # we use the Shopify-compiled version of any SCSS stylesheets.
110
+ if shopify_asset.public_url.present?
111
+ public_urls[asset] = shopify_asset.public_url.gsub(/\.scss\?/, '.scss.css?')
112
+ end
113
+ end
114
+
115
+ # If we specified the creation of any script tags based on newly rendered
116
+ # assets, do that now.
117
+ unless options[:script_tags].empty?
118
+ render_asset_script_tags(options, public_urls)
119
+ end
120
+ end
121
+
122
+ private
123
+
124
+ # Render an individual asset within an asset group.
125
+ def render_asset_group_asset(asset, public_urls, options)
126
+ rendered_asset = render_asset_renderer.render_to_string(
127
+ template: asset,
128
+ layout: nil,
129
+ locals: {
130
+ :"@#{self.class.name.underscore}" => self,
131
+ :@public_urls => public_urls
132
+ }
133
+ )
134
+
135
+ if should_be_minified?(asset, options)
136
+ ::Uglifier.compile(rendered_asset)
137
+ else
138
+ rendered_asset
139
+ end
140
+ end
141
+
142
+ # Return true if the given asset should be minified with Uglifier.
143
+ def should_be_minified?(asset, options)
144
+ asset.to_s.end_with?('.js') and options[:minify]
145
+ end
146
+
147
+ def render_asset_renderer
148
+ @render_asset_renderer ||= self.class.const_get('RenderingController').new
149
+ end
150
+
151
+ # Render any script tags defined by the :script_tags options that we have
152
+ # a public URL for.
153
+ def render_asset_script_tags(options, public_urls)
154
+ # Fetch the current set of script tags for the store.
155
+ current_script_tags = shop.with_api_context { ShopifyAPI::ScriptTag.find(:all) }
156
+
157
+ # Iterate each script tag for which we have a known public URL and create
158
+ # or update the corresponding script tag resource.
159
+ public_urls.slice(*options[:script_tags]).each do |asset, public_url|
160
+ script_tag = current_script_tags.find(lambda { ShopifyAPI::ScriptTag.new(event: 'onload') }) { |script_tag| script_tag.src.include?("#{asset}?") }
161
+ script_tag.src = public_url
162
+ shop.with_api_context { script_tag.save }
163
+ end
164
+ end
165
+
166
+ end
@@ -0,0 +1,96 @@
1
+ module DiscoApp::Concerns::Shop
2
+ extend ActiveSupport::Concern
3
+
4
+ included do
5
+ include ShopifyApp::Shop
6
+ include ActionView::Helpers::DateHelper
7
+
8
+ # Define relationships to plans and subscriptions.
9
+ has_many :subscriptions
10
+ has_many :plans, through: :subscriptions
11
+
12
+ # Define relationship to sessions.
13
+ has_many :sessions, class_name: 'DiscoApp::Session', dependent: :destroy
14
+
15
+ # Define possible installation statuses as an enum.
16
+ enum status: [:never_installed, :awaiting_install, :installing, :installed, :awaiting_uninstall, :uninstalling, :uninstalled]
17
+
18
+ # Define some useful scopes.
19
+ scope :status, -> (status) { where status: status }
20
+ scope :installed, -> { where status: statuses[:installed] }
21
+ scope :has_active_shopify_plan, -> { where.not(plan_name: [:cancelled, :frozen, :fraudulent]) }
22
+
23
+ # Alias 'with_shopify_session' as 'with_api_context' for better readability, but also as 'temp' for
24
+ # backward compatibility.
25
+ alias_method :with_api_context, :with_shopify_session
26
+ alias_method :temp, :with_shopify_session
27
+
28
+ # Return true if the shop is considered as in development mode.
29
+ def development?
30
+ ['staff', 'custom', 'affiliate'].include?(plan_name)
31
+ end
32
+
33
+ # Convenience method to check if this shop has a current subscription.
34
+ def current_subscription?
35
+ current_subscription.present?
36
+ end
37
+
38
+ # Convenience method to get the current subscription for this shop, if any.
39
+ def current_subscription
40
+ subscriptions.current.first
41
+ end
42
+
43
+ # Convenience method to get the current plan for this shop, if any.
44
+ def current_plan
45
+ current_subscription&.plan
46
+ end
47
+
48
+ # Return the absolute URL to the shop's storefront.
49
+ def url
50
+ "#{protocol}://#{domain}"
51
+ end
52
+
53
+ # Return the protocol the shop's storefront uses. This should now always be
54
+ # https as all Shopify stores have SSL enabled.
55
+ def protocol
56
+ 'https'
57
+ end
58
+
59
+ # Return the absolute URL to the shop's admin.
60
+ def admin_url
61
+ "https://#{shopify_domain}/admin"
62
+ end
63
+
64
+ # Convenience method to get the email of the shop's admin, to display in Rollbar.
65
+ def email
66
+ self.data['email']
67
+ end
68
+
69
+ def installed_duration
70
+ distance_of_time_in_words_to_now(created_at.time)
71
+ end
72
+
73
+ # Return the shop's configured timezone. If none can be parsed from the
74
+ # shop's "data" hash, return the default Rails zone (which should be UTC).
75
+ def time_zone
76
+ @time_zone ||= begin
77
+ Time.find_zone!(data['timezone'].to_s.gsub(/^\(.+\)\s/, ''))
78
+ rescue ArgumentError
79
+ Time.zone
80
+ end
81
+ end
82
+
83
+ # Return the shop's configured locale as a symbol. If none exists for some
84
+ # reason, 'en' is returned.
85
+ def locale
86
+ (data['primary_locale'] || 'en').to_sym
87
+ end
88
+
89
+ # Return an instance of the Disco API client.
90
+ def disco_api_client
91
+ @api_client ||= DiscoApp::ApiClient.new(self, ENV['DISCO_API_URL'])
92
+ end
93
+
94
+ end
95
+
96
+ end
@@ -0,0 +1,66 @@
1
+ module DiscoApp::Concerns::Subscription
2
+ extend ActiveSupport::Concern
3
+
4
+ included do
5
+
6
+ belongs_to :shop
7
+ belongs_to :plan
8
+ belongs_to :plan_code
9
+
10
+ has_many :one_time_charges, class_name: 'DiscoApp::ApplicationCharge', dependent: :destroy
11
+ has_many :recurring_charges, class_name: 'DiscoApp::RecurringApplicationCharge', dependent: :destroy
12
+
13
+ enum status: [:trial, :active, :cancelled]
14
+ enum subscription_type: [:recurring, :one_time]
15
+
16
+ scope :current, -> { where status: [statuses[:trial], statuses[:active]] }
17
+
18
+ after_commit :cancel_charge
19
+
20
+ end
21
+
22
+ # Only require an active charge if the amount to be charged is > 0.
23
+ def requires_active_charge?
24
+ amount > 0
25
+ end
26
+
27
+ # Convenience method to check if this subscription has an active charge.
28
+ def active_charge?
29
+ active_charge.present?
30
+ end
31
+
32
+ # Convenience method to get the active charge for this subscription.
33
+ def active_charge
34
+ charges.active.first
35
+ end
36
+
37
+ # Return the appropriate set of charges for this subscription's type.
38
+ def charges
39
+ recurring? ? recurring_charges : one_time_charges
40
+ end
41
+
42
+ def charge_class
43
+ recurring? ? DiscoApp::RecurringApplicationCharge : DiscoApp::ApplicationCharge
44
+ end
45
+
46
+ def shopify_charge_class
47
+ recurring? ? ShopifyAPI::RecurringApplicationCharge : ShopifyAPI::ApplicationCharge
48
+ end
49
+
50
+ def as_json(options = {})
51
+ super.merge(
52
+ 'active_charge' => active_charge
53
+ )
54
+ end
55
+
56
+ private
57
+
58
+ # If the amount or trial period for this subscription changes, clear any
59
+ # active charge, as the user will need to re-authorize the charge.
60
+ def cancel_charge
61
+ return if (previous_changes.keys & ['amount', 'trial_period_days']).empty?
62
+ return unless active_charge?
63
+ active_charge.cancelled!
64
+ end
65
+
66
+ end
@@ -0,0 +1,58 @@
1
+ module DiscoApp::Concerns::Synchronises
2
+ extend ActiveSupport::Concern
3
+
4
+ class_methods do
5
+
6
+ # Define the number of resources per page to fetch.
7
+ SYNCHRONISES_PAGE_LIMIT = 250
8
+
9
+ def should_synchronise?(shop, data)
10
+ true
11
+ end
12
+
13
+ def synchronise_by(shop, data)
14
+ { id: data[:id] }
15
+ end
16
+
17
+ def synchronise(shop, data)
18
+ data = data.with_indifferent_access
19
+
20
+ return unless should_synchronise?(shop, data)
21
+
22
+ begin
23
+ instance = self.find_or_create_by!(self.synchronise_by(shop, data)) do |instance|
24
+ instance.shop = shop
25
+ instance.data = data
26
+ end
27
+ rescue ActiveRecord::RecordNotUnique, PG::UniqueViolation
28
+ retry
29
+ end
30
+
31
+ instance.update(data: data)
32
+
33
+ instance
34
+ end
35
+
36
+ def should_synchronise_deletion?(shop, data)
37
+ true
38
+ end
39
+
40
+ def synchronise_deletion(shop, data)
41
+ data = data.with_indifferent_access
42
+
43
+ return unless should_synchronise_deletion?(shop, data)
44
+
45
+ self.destroy_all(shop: shop, id: data[:id])
46
+ end
47
+
48
+ def synchronise_all(shop, params = {})
49
+ resource_count = shop.with_api_context { self::SHOPIFY_API_CLASS.count(params) }
50
+
51
+ (1..(resource_count / SYNCHRONISES_PAGE_LIMIT.to_f).ceil).each do |page|
52
+ DiscoApp::SynchroniseResourcesJob.perform_later(shop, self.name, params.merge(page: page, limit: SYNCHRONISES_PAGE_LIMIT))
53
+ end
54
+ end
55
+
56
+ end
57
+
58
+ end
@@ -0,0 +1,16 @@
1
+ module DiscoApp::Concerns::Taggable
2
+ extend ActiveSupport::Concern
3
+
4
+ def tags
5
+ data['tags'].split(',').map(&:strip)
6
+ end
7
+
8
+ def add_tag(tag)
9
+ data['tags'] = (tags + [tag]).uniq.join(',')
10
+ end
11
+
12
+ def remove_tag(tag)
13
+ data['tags'] = (tags - [tag]).uniq.join(',')
14
+ end
15
+
16
+ end
@@ -0,0 +1,3 @@
1
+ class DiscoApp::Plan < ActiveRecord::Base
2
+ include DiscoApp::Concerns::Plan
3
+ end
@@ -0,0 +1,3 @@
1
+ class DiscoApp::PlanCode < ActiveRecord::Base
2
+ include DiscoApp::Concerns::PlanCode
3
+ end
@@ -0,0 +1,18 @@
1
+ class DiscoApp::RecurringApplicationCharge < ActiveRecord::Base
2
+
3
+ belongs_to :shop
4
+ belongs_to :subscription
5
+
6
+ enum status: [:pending, :accepted, :declined, :active, :cancelled, :expired]
7
+
8
+ scope :active, -> { where status: statuses[:active] }
9
+
10
+ def recurring?
11
+ true
12
+ end
13
+
14
+ def activate_url
15
+ DiscoApp::Engine.routes.url_helpers.activate_subscription_charge_url(subscription, self)
16
+ end
17
+
18
+ end
@@ -0,0 +1,18 @@
1
+ module DiscoApp
2
+ class SessionStorage
3
+ def self.store(session)
4
+ shop = Shop.find_or_initialize_by(shopify_domain: session.url)
5
+ shop.shopify_token = session.token
6
+ shop.save!
7
+ shop.id
8
+ end
9
+
10
+ def self.retrieve(id)
11
+ return unless id
12
+ shop = Shop.find(id)
13
+ ShopifyAPI::Session.new(shop.shopify_domain, shop.shopify_token)
14
+ rescue ActiveRecord::RecordNotFound
15
+ nil
16
+ end
17
+ end
18
+ end
@@ -0,0 +1,3 @@
1
+ class DiscoApp::Shop < ActiveRecord::Base
2
+ include DiscoApp::Concerns::Shop
3
+ end
@@ -0,0 +1,3 @@
1
+ class DiscoApp::Subscription < ActiveRecord::Base
2
+ include DiscoApp::Concerns::Subscription
3
+ end
@@ -0,0 +1,100 @@
1
+ require 'jsonapi/resource'
2
+
3
+ module DiscoApp::Admin::Resources::Concerns::ShopResource
4
+ extend ActiveSupport::Concern
5
+
6
+ included do
7
+
8
+ attributes :domain, :status, :created_at
9
+ attributes :email, :country_name, :currency, :plan_display_name
10
+ attributes :current_subscription_id, :current_subscription_display_amount, :current_subscription_display_plan, :current_subscription_display_plan_code, :current_subscription_source
11
+ attributes :installed_duration
12
+
13
+ model_name 'DiscoApp::Shop'
14
+
15
+ filters :query, :status
16
+
17
+ # Adjust the base records method to ensure only models for the authenticated domain are retrieved.
18
+ def self.records(options = {})
19
+ records = DiscoApp::Shop.order(created_at: :desc)
20
+ records
21
+ end
22
+
23
+ # Apply filters.
24
+ def self.apply_filter(records, filter, value, options)
25
+ return records if value.blank?
26
+
27
+ # Perform appropriate filtering.
28
+ case filter
29
+ when :query
30
+ return records.where('name LIKE ? OR shopify_domain LIKE ? OR domain LIKE ?', "%#{value.first}%", "%#{value.first}%", "%#{value.first}%")
31
+ when :status
32
+ return records.where(status: value.map { |v| DiscoApp::Shop.statuses[v.to_sym] } )
33
+ else
34
+ return super(records, filter, value)
35
+ end
36
+ end
37
+
38
+ # Don't allow the update of any fields via the API.
39
+ def self.updatable_fields(context)
40
+ []
41
+ end
42
+
43
+ # Don't allow the creation of any fields via the API.
44
+ def self.creatable_fields(context)
45
+ []
46
+ end
47
+
48
+ def email
49
+ @model.data['email']
50
+ end
51
+
52
+ def country_name
53
+ @model.data['country_name']
54
+ end
55
+
56
+ def currency
57
+ @model.data['currency']
58
+ end
59
+
60
+ def plan_display_name
61
+ @model.data['plan_display_name']
62
+ end
63
+
64
+ def current_subscription_id
65
+ if @model.current_subscription?
66
+ @model.current_subscription.id
67
+ end
68
+ end
69
+
70
+ def current_subscription_display_amount
71
+ if @model.current_subscription?
72
+ @model.current_subscription.amount
73
+ else
74
+ '-'
75
+ end
76
+ end
77
+
78
+ def current_subscription_display_plan
79
+ if @model.current_subscription?
80
+ @model.current_plan.name
81
+ else
82
+ 'None'
83
+ end
84
+ end
85
+
86
+ def current_subscription_display_plan_code
87
+ @model.current_subscription&.plan_code&.code
88
+ end
89
+
90
+ def current_subscription_source
91
+ if @model.current_subscription?
92
+ @model.current_subscription.source || '-'
93
+ else
94
+ '-'
95
+ end
96
+ end
97
+
98
+ end
99
+
100
+ end
@@ -0,0 +1,4 @@
1
+ class DiscoApp::Admin::Resources::ShopResource < JSONAPI::Resource
2
+ include DiscoApp::Admin::Resources::Concerns::ShopResource
3
+
4
+ end
@@ -0,0 +1,15 @@
1
+ class DiscoApp::CarrierRequestService
2
+
3
+ # Return true iff the provided hmac_to_verify matches that calculated from the
4
+ # given data and secret.
5
+ def self.is_valid_hmac?(body, secret, hmac_to_verify)
6
+ ActiveSupport::SecurityUtils.secure_compare(self.calculated_hmac(body, secret), hmac_to_verify.to_s)
7
+ end
8
+
9
+ # Calculate the HMAC for the given data and secret.
10
+ def self.calculated_hmac(body, secret)
11
+ digest = OpenSSL::Digest.new('sha256')
12
+ Base64.encode64(OpenSSL::HMAC.digest(digest, secret, body)).strip
13
+ end
14
+
15
+ end
@@ -0,0 +1,81 @@
1
+ class DiscoApp::ChargesService
2
+
3
+ # Create the appropriate type of Shopify charge for the given subscription
4
+ # (either one-time or recurring) and return.
5
+ def self.create(shop, subscription)
6
+ # Create the charge object locally first.
7
+ charge = subscription.charge_class.create!(
8
+ shop: shop,
9
+ subscription: subscription,
10
+ )
11
+
12
+ # Create the charge object on Shopify.
13
+ shopify_charge = shop.with_api_context {
14
+ subscription.shopify_charge_class.create(
15
+ name: subscription.plan.name,
16
+ price: '%.2f' % (subscription.amount.to_f / 100.0),
17
+ trial_days: subscription.plan.has_trial? ? subscription.trial_period_days : nil,
18
+ return_url: charge.activate_url,
19
+ test: !DiscoApp.configuration.real_charges?
20
+ )
21
+ }
22
+
23
+ # If we couldn't create the charge on Shopify, return nil.
24
+ if shopify_charge.nil?
25
+ return nil
26
+ end
27
+
28
+ # Update the local record of the charge from Shopify's created charge, then
29
+ # return it.
30
+ charge.update(
31
+ shopify_id: shopify_charge.id,
32
+ confirmation_url: shopify_charge.confirmation_url
33
+ )
34
+ charge
35
+ end
36
+
37
+ # Attempt to activate the given Shopify charge for the given Shop using the
38
+ # Shopify API. Returns true on successful activation, false otherwise.
39
+ def self.activate(shop, charge)
40
+ begin
41
+ # Start by fetching the Shopify charge to check that it was accepted.
42
+ shopify_charge = shop.with_api_context {
43
+ charge.subscription.shopify_charge_class.find(charge.shopify_id)
44
+ }
45
+
46
+ # Update the status of the local charge based on the Shopify charge.
47
+ charge.send("#{shopify_charge.status}!") if charge.respond_to? "#{shopify_charge.status}!"
48
+
49
+ # If the charge wasn't accepted, fail and return.
50
+ return false unless charge.accepted?
51
+
52
+ # If the charge was indeed accepted, activate it via Shopify.
53
+ charge.shop.with_api_context {
54
+ shopify_charge.activate
55
+ }
56
+
57
+ # If the charge was recurring, make sure that all other local recurring
58
+ # charges are marked inactive.
59
+ if charge.recurring?
60
+ self.cancel_recurring_charges(shop, charge)
61
+ end
62
+
63
+ charge.active!
64
+
65
+ true
66
+ rescue
67
+ false
68
+ end
69
+ end
70
+
71
+ # Cancel all recurring charges for the given shop. If the optional charge
72
+ # parameter is given, it will be excluded from the cancellation.
73
+ def self.cancel_recurring_charges(shop, charge = nil)
74
+ charges = DiscoApp::RecurringApplicationCharge.where(shop: shop)
75
+ if charge.present?
76
+ charges = charges.where.not(id: charge)
77
+ end
78
+ charges.update_all(status: DiscoApp::RecurringApplicationCharge.statuses[:cancelled])
79
+ end
80
+
81
+ end
@@ -0,0 +1,17 @@
1
+ class DiscoApp::ProxyService
2
+
3
+ # Return true iff the signature provided in the given query string matches
4
+ # that calculated from the remaining query parameters and the given secret.
5
+ def self.proxy_signature_is_valid?(query_string, secret)
6
+ query_hash = Rack::Utils.parse_query(query_string)
7
+ signature = query_hash.delete('signature').to_s
8
+ ActiveSupport::SecurityUtils.variable_size_secure_compare(self.calculated_signature(query_hash, secret), signature)
9
+ end
10
+
11
+ # Return the calculated signature for the given query hash and secret.
12
+ def self.calculated_signature(query_hash, secret)
13
+ sorted_params = query_hash.collect{ |k, v| "#{k}=#{Array(v).join(',')}" }.sort.join
14
+ OpenSSL::HMAC.hexdigest(OpenSSL::Digest.new('sha256'), secret, sorted_params)
15
+ end
16
+
17
+ end