ruby_shopify_api 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (351) hide show
  1. checksums.yaml +7 -0
  2. data/.document +5 -0
  3. data/.github/CODEOWNERS +1 -0
  4. data/.github/ISSUE_TEMPLATE.md +36 -0
  5. data/.github/probots.yml +2 -0
  6. data/.github/workflows/build.yml +43 -0
  7. data/.gitignore +15 -0
  8. data/.rubocop.yml +28 -0
  9. data/.rubocop_todo.yml +75 -0
  10. data/CHANGELOG-OLD.md +501 -0
  11. data/CHANGELOG.md +5 -0
  12. data/CONTRIBUTING.md +8 -0
  13. data/CONTRIBUTORS +3 -0
  14. data/Gemfile +10 -0
  15. data/Gemfile.lock +151 -0
  16. data/Gemfile_ar41 +5 -0
  17. data/Gemfile_ar50 +5 -0
  18. data/Gemfile_ar51 +5 -0
  19. data/Gemfile_ar60 +5 -0
  20. data/Gemfile_ar_main +5 -0
  21. data/LICENSE +20 -0
  22. data/README.md +649 -0
  23. data/RELEASING +17 -0
  24. data/Rakefile +55 -0
  25. data/SECURITY.md +59 -0
  26. data/dev.yml +11 -0
  27. data/docker-compose.yml +13 -0
  28. data/docs/_config.yml +1 -0
  29. data/docs/_includes/footer.html +28 -0
  30. data/docs/_includes/head.html +28 -0
  31. data/docs/_layouts/index.html +57 -0
  32. data/docs/graphql.md +241 -0
  33. data/docs/index.md +639 -0
  34. data/lib/active_resource/connection_ext.rb +11 -0
  35. data/lib/active_resource/detailed_log_subscriber.rb +55 -0
  36. data/lib/active_resource/json_errors.rb +37 -0
  37. data/lib/shopify_api/api_access.rb +57 -0
  38. data/lib/shopify_api/api_version.rb +206 -0
  39. data/lib/shopify_api/connection.rb +36 -0
  40. data/lib/shopify_api/countable.rb +15 -0
  41. data/lib/shopify_api/disable_prefix_check.rb +31 -0
  42. data/lib/shopify_api/events.rb +8 -0
  43. data/lib/shopify_api/graphql/http_client.rb +22 -0
  44. data/lib/shopify_api/graphql/railtie.rb +17 -0
  45. data/lib/shopify_api/graphql/task.rake +100 -0
  46. data/lib/shopify_api/graphql.rb +103 -0
  47. data/lib/shopify_api/hmac_params.rb +33 -0
  48. data/lib/shopify_api/limits.rb +77 -0
  49. data/lib/shopify_api/message_enricher.rb +25 -0
  50. data/lib/shopify_api/meta.rb +14 -0
  51. data/lib/shopify_api/metafields.rb +21 -0
  52. data/lib/shopify_api/paginated_collection.rb +69 -0
  53. data/lib/shopify_api/pagination_link_headers.rb +34 -0
  54. data/lib/shopify_api/resources/abandoned_checkout.rb +7 -0
  55. data/lib/shopify_api/resources/access_scope.rb +10 -0
  56. data/lib/shopify_api/resources/access_token.rb +9 -0
  57. data/lib/shopify_api/resources/address.rb +5 -0
  58. data/lib/shopify_api/resources/announcement.rb +5 -0
  59. data/lib/shopify_api/resources/api_permission.rb +9 -0
  60. data/lib/shopify_api/resources/application_charge.rb +16 -0
  61. data/lib/shopify_api/resources/application_credit.rb +5 -0
  62. data/lib/shopify_api/resources/array_base.rb +13 -0
  63. data/lib/shopify_api/resources/article.rb +22 -0
  64. data/lib/shopify_api/resources/asset.rb +101 -0
  65. data/lib/shopify_api/resources/assigned_fulfillment_order.rb +16 -0
  66. data/lib/shopify_api/resources/base.rb +166 -0
  67. data/lib/shopify_api/resources/billing_address.rb +5 -0
  68. data/lib/shopify_api/resources/blog.rb +11 -0
  69. data/lib/shopify_api/resources/carrier_service.rb +5 -0
  70. data/lib/shopify_api/resources/cart.rb +5 -0
  71. data/lib/shopify_api/resources/checkout.rb +30 -0
  72. data/lib/shopify_api/resources/collect.rb +7 -0
  73. data/lib/shopify_api/resources/collection.rb +14 -0
  74. data/lib/shopify_api/resources/collection_listing.rb +19 -0
  75. data/lib/shopify_api/resources/collection_publication.rb +10 -0
  76. data/lib/shopify_api/resources/comment.rb +24 -0
  77. data/lib/shopify_api/resources/country.rb +5 -0
  78. data/lib/shopify_api/resources/currency.rb +6 -0
  79. data/lib/shopify_api/resources/custom_collection.rb +20 -0
  80. data/lib/shopify_api/resources/customer.rb +30 -0
  81. data/lib/shopify_api/resources/customer_group.rb +6 -0
  82. data/lib/shopify_api/resources/customer_invite.rb +5 -0
  83. data/lib/shopify_api/resources/customer_saved_search.rb +12 -0
  84. data/lib/shopify_api/resources/discount_code.rb +10 -0
  85. data/lib/shopify_api/resources/discount_code_batch.rb +34 -0
  86. data/lib/shopify_api/resources/draft_order.rb +15 -0
  87. data/lib/shopify_api/resources/draft_order_invoice.rb +5 -0
  88. data/lib/shopify_api/resources/event.rb +9 -0
  89. data/lib/shopify_api/resources/fulfillment.rb +56 -0
  90. data/lib/shopify_api/resources/fulfillment_event.rb +16 -0
  91. data/lib/shopify_api/resources/fulfillment_order.rb +151 -0
  92. data/lib/shopify_api/resources/fulfillment_order_locations_for_move.rb +5 -0
  93. data/lib/shopify_api/resources/fulfillment_request.rb +16 -0
  94. data/lib/shopify_api/resources/fulfillment_service.rb +5 -0
  95. data/lib/shopify_api/resources/fulfillment_v2.rb +21 -0
  96. data/lib/shopify_api/resources/gift_card.rb +8 -0
  97. data/lib/shopify_api/resources/image.rb +17 -0
  98. data/lib/shopify_api/resources/inventory_item.rb +6 -0
  99. data/lib/shopify_api/resources/inventory_level.rb +54 -0
  100. data/lib/shopify_api/resources/line_item.rb +15 -0
  101. data/lib/shopify_api/resources/location.rb +8 -0
  102. data/lib/shopify_api/resources/marketing_event.rb +11 -0
  103. data/lib/shopify_api/resources/metafield.rb +14 -0
  104. data/lib/shopify_api/resources/note_attribute.rb +5 -0
  105. data/lib/shopify_api/resources/option.rb +5 -0
  106. data/lib/shopify_api/resources/order.rb +44 -0
  107. data/lib/shopify_api/resources/order_risk.rb +9 -0
  108. data/lib/shopify_api/resources/page.rb +7 -0
  109. data/lib/shopify_api/resources/payment.rb +7 -0
  110. data/lib/shopify_api/resources/payment_details.rb +5 -0
  111. data/lib/shopify_api/resources/ping.rb +3 -0
  112. data/lib/shopify_api/resources/policy.rb +8 -0
  113. data/lib/shopify_api/resources/price_rule.rb +8 -0
  114. data/lib/shopify_api/resources/product.rb +59 -0
  115. data/lib/shopify_api/resources/product_listing.rb +17 -0
  116. data/lib/shopify_api/resources/product_publication.rb +10 -0
  117. data/lib/shopify_api/resources/province.rb +6 -0
  118. data/lib/shopify_api/resources/publication.rb +5 -0
  119. data/lib/shopify_api/resources/receipt.rb +5 -0
  120. data/lib/shopify_api/resources/recurring_application_charge.rb +34 -0
  121. data/lib/shopify_api/resources/redirect.rb +5 -0
  122. data/lib/shopify_api/resources/refund.rb +15 -0
  123. data/lib/shopify_api/resources/report.rb +5 -0
  124. data/lib/shopify_api/resources/resource_feedback.rb +19 -0
  125. data/lib/shopify_api/resources/rule.rb +5 -0
  126. data/lib/shopify_api/resources/script_tag.rb +5 -0
  127. data/lib/shopify_api/resources/shipping_address.rb +5 -0
  128. data/lib/shopify_api/resources/shipping_line.rb +5 -0
  129. data/lib/shopify_api/resources/shipping_rate.rb +7 -0
  130. data/lib/shopify_api/resources/shipping_zone.rb +5 -0
  131. data/lib/shopify_api/resources/shop.rb +26 -0
  132. data/lib/shopify_api/resources/smart_collection.rb +15 -0
  133. data/lib/shopify_api/resources/storefront_access_token.rb +5 -0
  134. data/lib/shopify_api/resources/tax_line.rb +5 -0
  135. data/lib/shopify_api/resources/tax_service.rb +5 -0
  136. data/lib/shopify_api/resources/tender_transaction.rb +6 -0
  137. data/lib/shopify_api/resources/theme.rb +5 -0
  138. data/lib/shopify_api/resources/transaction.rb +6 -0
  139. data/lib/shopify_api/resources/usage_charge.rb +6 -0
  140. data/lib/shopify_api/resources/user.rb +5 -0
  141. data/lib/shopify_api/resources/variant.rb +43 -0
  142. data/lib/shopify_api/resources/webhook.rb +5 -0
  143. data/lib/shopify_api/resources.rb +4 -0
  144. data/lib/shopify_api/session.rb +203 -0
  145. data/lib/shopify_api/version.rb +4 -0
  146. data/lib/shopify_api.rb +45 -0
  147. data/lib/verify_docs.rb +8 -0
  148. data/service.yml +2 -0
  149. data/shipit.rubygems.yml +1 -0
  150. data/shopify_api.gemspec +47 -0
  151. data/test/abandoned_checkouts_test.rb +29 -0
  152. data/test/access_scope_test.rb +23 -0
  153. data/test/access_token_test.rb +20 -0
  154. data/test/active_resource/json_errors_test.rb +19 -0
  155. data/test/api_access_test.rb +153 -0
  156. data/test/api_permission_test.rb +9 -0
  157. data/test/api_version_test.rb +157 -0
  158. data/test/application_charge_test.rb +82 -0
  159. data/test/application_credit_test.rb +36 -0
  160. data/test/article_test.rb +72 -0
  161. data/test/asset_test.rb +26 -0
  162. data/test/assigned_fulfillment_order_test.rb +78 -0
  163. data/test/base_test.rb +213 -0
  164. data/test/blog_test.rb +9 -0
  165. data/test/carrier_service_test.rb +18 -0
  166. data/test/cart_test.rb +14 -0
  167. data/test/checkouts_test.rb +77 -0
  168. data/test/collect_test.rb +10 -0
  169. data/test/collection_listing_test.rb +84 -0
  170. data/test/collection_publication_test.rb +40 -0
  171. data/test/collection_test.rb +50 -0
  172. data/test/countable_test.rb +14 -0
  173. data/test/currency_test.rb +21 -0
  174. data/test/custom_collection_test.rb +10 -0
  175. data/test/customer_saved_search_test.rb +37 -0
  176. data/test/customer_test.rb +58 -0
  177. data/test/detailed_log_subscriber_test.rb +143 -0
  178. data/test/discount_code_batch_test.rb +41 -0
  179. data/test/discount_code_test.rb +59 -0
  180. data/test/draft_order_test.rb +167 -0
  181. data/test/fixtures/abandoned_checkout.json +184 -0
  182. data/test/fixtures/abandoned_checkouts.json +186 -0
  183. data/test/fixtures/access_scopes.json +10 -0
  184. data/test/fixtures/access_token_delegate.json +4 -0
  185. data/test/fixtures/api_versions.json +38 -0
  186. data/test/fixtures/apis.json +42 -0
  187. data/test/fixtures/application_charge.json +16 -0
  188. data/test/fixtures/application_charges.json +57 -0
  189. data/test/fixtures/application_credit.json +12 -0
  190. data/test/fixtures/application_credits.json +24 -0
  191. data/test/fixtures/article.json +15 -0
  192. data/test/fixtures/articles.json +39 -0
  193. data/test/fixtures/asset.json +9 -0
  194. data/test/fixtures/assets.json +136 -0
  195. data/test/fixtures/assigned_fulfillment_orders.json +80 -0
  196. data/test/fixtures/authors.json +1 -0
  197. data/test/fixtures/blog.json +13 -0
  198. data/test/fixtures/blogs.json +13 -0
  199. data/test/fixtures/carrier_service.json +9 -0
  200. data/test/fixtures/carts.json +43 -0
  201. data/test/fixtures/checkout.json +160 -0
  202. data/test/fixtures/checkouts.json +162 -0
  203. data/test/fixtures/collect.json +12 -0
  204. data/test/fixtures/collection.json +17 -0
  205. data/test/fixtures/collection_listing.json +11 -0
  206. data/test/fixtures/collection_listing_product_ids.json +1 -0
  207. data/test/fixtures/collection_listing_product_ids2.json +1 -0
  208. data/test/fixtures/collection_listings.json +13 -0
  209. data/test/fixtures/collection_products.json +47 -0
  210. data/test/fixtures/collection_publication.json +11 -0
  211. data/test/fixtures/collection_publications.json +13 -0
  212. data/test/fixtures/currencies.json +25 -0
  213. data/test/fixtures/custom_collection.json +17 -0
  214. data/test/fixtures/customer_invite.json +9 -0
  215. data/test/fixtures/customer_saved_search.json +9 -0
  216. data/test/fixtures/customer_saved_search_customers.json +60 -0
  217. data/test/fixtures/customers.json +59 -0
  218. data/test/fixtures/customers_account_activation_url.json +3 -0
  219. data/test/fixtures/customers_search.json +60 -0
  220. data/test/fixtures/discount_code.json +10 -0
  221. data/test/fixtures/discount_code_batch.json +14 -0
  222. data/test/fixtures/discount_code_batch_discount_codes.json +21 -0
  223. data/test/fixtures/discount_codes.json +12 -0
  224. data/test/fixtures/draft_order.json +159 -0
  225. data/test/fixtures/draft_order_completed.json +159 -0
  226. data/test/fixtures/draft_order_invoice.json +9 -0
  227. data/test/fixtures/draft_orders.json +161 -0
  228. data/test/fixtures/engagement.json +15 -0
  229. data/test/fixtures/events.json +31 -0
  230. data/test/fixtures/fulfillment.json +49 -0
  231. data/test/fixtures/fulfillment_event.json +12 -0
  232. data/test/fixtures/fulfillment_order.json +39 -0
  233. data/test/fixtures/fulfillment_order_locations_for_move.json +18 -0
  234. data/test/fixtures/fulfillment_orders.json +80 -0
  235. data/test/fixtures/fulfillment_request.json +28 -0
  236. data/test/fixtures/fulfillment_service.json +10 -0
  237. data/test/fixtures/fulfillments.json +53 -0
  238. data/test/fixtures/gift_card.json +20 -0
  239. data/test/fixtures/gift_card_disabled.json +20 -0
  240. data/test/fixtures/graphql/2019-10.json +1083 -0
  241. data/test/fixtures/graphql/dummy_schema.rb +16 -0
  242. data/test/fixtures/graphql/unstable.json +1083 -0
  243. data/test/fixtures/image.json +10 -0
  244. data/test/fixtures/images.json +20 -0
  245. data/test/fixtures/inventory_level.json +7 -0
  246. data/test/fixtures/inventory_levels.json +24 -0
  247. data/test/fixtures/marketing_event.json +28 -0
  248. data/test/fixtures/marketing_events.json +54 -0
  249. data/test/fixtures/metafield.json +12 -0
  250. data/test/fixtures/metafields.json +34 -0
  251. data/test/fixtures/order.json +297 -0
  252. data/test/fixtures/order_risk.json +14 -0
  253. data/test/fixtures/order_risks.json +28 -0
  254. data/test/fixtures/order_with_properties.json +373 -0
  255. data/test/fixtures/orders.json +299 -0
  256. data/test/fixtures/payment.json +7 -0
  257. data/test/fixtures/payments.json +9 -0
  258. data/test/fixtures/ping/conversation.json +1 -0
  259. data/test/fixtures/ping/failed_delivery_confirmation.json +1 -0
  260. data/test/fixtures/ping/message.json +1 -0
  261. data/test/fixtures/ping/successful_delivery_confirmation.json +1 -0
  262. data/test/fixtures/policies.json +8 -0
  263. data/test/fixtures/price_rule.json +27 -0
  264. data/test/fixtures/price_rules.json +28 -0
  265. data/test/fixtures/product.json +116 -0
  266. data/test/fixtures/product_listing.json +86 -0
  267. data/test/fixtures/product_listing_product_ids.json +1 -0
  268. data/test/fixtures/product_listing_product_ids2.json +1 -0
  269. data/test/fixtures/product_listings.json +174 -0
  270. data/test/fixtures/product_publication.json +11 -0
  271. data/test/fixtures/product_publications.json +13 -0
  272. data/test/fixtures/publications.json +9 -0
  273. data/test/fixtures/recurring_application_charge.json +22 -0
  274. data/test/fixtures/recurring_application_charge_adjustment.json +5 -0
  275. data/test/fixtures/recurring_application_charges.json +106 -0
  276. data/test/fixtures/redirect.json +7 -0
  277. data/test/fixtures/refund.json +112 -0
  278. data/test/fixtures/report.json +9 -0
  279. data/test/fixtures/reports.json +11 -0
  280. data/test/fixtures/script_tag.json +10 -0
  281. data/test/fixtures/script_tags.json +18 -0
  282. data/test/fixtures/shipping_rates.json +12 -0
  283. data/test/fixtures/shipping_zones.json +315 -0
  284. data/test/fixtures/shop.json +26 -0
  285. data/test/fixtures/smart_collection.json +21 -0
  286. data/test/fixtures/smart_collection_products.json +155 -0
  287. data/test/fixtures/storefront_access_token.json +9 -0
  288. data/test/fixtures/storefront_access_tokens.json +18 -0
  289. data/test/fixtures/tags.json +1 -0
  290. data/test/fixtures/tax_service.json +9 -0
  291. data/test/fixtures/tender_transactions.json +52 -0
  292. data/test/fixtures/transaction.json +29 -0
  293. data/test/fixtures/usage_charge.json +11 -0
  294. data/test/fixtures/usage_charges.json +23 -0
  295. data/test/fixtures/user.json +21 -0
  296. data/test/fixtures/users.json +42 -0
  297. data/test/fixtures/variant.json +23 -0
  298. data/test/fixtures/variants.json +88 -0
  299. data/test/fixtures/webhook.json +10 -0
  300. data/test/fixtures/webhooks.json +18 -0
  301. data/test/fulfillment_event_test.rb +74 -0
  302. data/test/fulfillment_order_test.rb +530 -0
  303. data/test/fulfillment_order_test_helper.rb +8 -0
  304. data/test/fulfillment_request_test.rb +35 -0
  305. data/test/fulfillment_service_test.rb +18 -0
  306. data/test/fulfillment_test.rb +239 -0
  307. data/test/fulfillment_v2_test.rb +66 -0
  308. data/test/gift_card_test.rb +24 -0
  309. data/test/graphql/http_client_test.rb +26 -0
  310. data/test/graphql_test.rb +190 -0
  311. data/test/hmac_params_test.rb +25 -0
  312. data/test/image_test.rb +41 -0
  313. data/test/inventory_level_test.rb +68 -0
  314. data/test/lib/webmock_extensions/last_request.rb +16 -0
  315. data/test/limits_test.rb +39 -0
  316. data/test/location_test.rb +15 -0
  317. data/test/marketing_event_test.rb +68 -0
  318. data/test/message_enricher_test.rb +45 -0
  319. data/test/meta_test.rb +47 -0
  320. data/test/metafield_test.rb +56 -0
  321. data/test/order_risk_test.rb +47 -0
  322. data/test/order_test.rb +140 -0
  323. data/test/pagination_test.rb +290 -0
  324. data/test/payment_test.rb +19 -0
  325. data/test/policy_test.rb +20 -0
  326. data/test/price_rule_test.rb +70 -0
  327. data/test/product_listing_test.rb +97 -0
  328. data/test/product_publication_test.rb +40 -0
  329. data/test/product_test.rb +111 -0
  330. data/test/publication_test.rb +12 -0
  331. data/test/recurring_application_charge_test.rb +204 -0
  332. data/test/redirect_test.rb +10 -0
  333. data/test/refund_test.rb +37 -0
  334. data/test/report_test.rb +37 -0
  335. data/test/resource_feedback_test.rb +43 -0
  336. data/test/script_tag_test.rb +31 -0
  337. data/test/session_test.rb +640 -0
  338. data/test/shipping_rate_test.rb +17 -0
  339. data/test/shipping_zone_test.rb +11 -0
  340. data/test/shop_test.rb +82 -0
  341. data/test/smart_collection_test.rb +11 -0
  342. data/test/storefront_access_token_test.rb +30 -0
  343. data/test/tax_service_test.rb +12 -0
  344. data/test/tender_transaction_test.rb +18 -0
  345. data/test/test_helper.rb +126 -0
  346. data/test/transaction_test.rb +18 -0
  347. data/test/usage_charge_test.rb +25 -0
  348. data/test/user_test.rb +18 -0
  349. data/test/variant_test.rb +73 -0
  350. data/test/webhook_test.rb +24 -0
  351. metadata +555 -0
@@ -0,0 +1,203 @@
1
+ # frozen_string_literal: true
2
+ require 'openssl'
3
+ require 'rack'
4
+
5
+ module ShopifyAPI
6
+ class ValidationException < StandardError
7
+ end
8
+
9
+ class Session
10
+ SECONDS_IN_A_DAY = 24 * 60 * 60
11
+
12
+ cattr_accessor :api_key, :secret, :myshopify_domain
13
+ self.myshopify_domain = 'myshopify.com'
14
+
15
+ attr_accessor :domain, :token, :name, :extra
16
+ attr_reader :api_version, :access_scopes
17
+ alias_method :url, :domain
18
+
19
+ class << self
20
+ def setup(params)
21
+ params.each { |k, value| public_send("#{k}=", value) }
22
+ end
23
+
24
+ def temp(domain:, token:, api_version: ShopifyAPI::Base.api_version, &block)
25
+ session = new(domain: domain, token: token, api_version: api_version)
26
+
27
+ with_session(session, &block)
28
+ end
29
+
30
+ def with_session(session, &_block)
31
+ original_session = extract_current_session
32
+ original_user = ShopifyAPI::Base.user
33
+ original_password = ShopifyAPI::Base.password
34
+
35
+ begin
36
+ ShopifyAPI::Base.clear_session
37
+ ShopifyAPI::Base.activate_session(session)
38
+ yield
39
+ ensure
40
+ ShopifyAPI::Base.activate_session(original_session)
41
+ ShopifyAPI::Base.user = original_user
42
+ ShopifyAPI::Base.password = original_password
43
+ end
44
+ end
45
+
46
+ def with_version(api_version, &block)
47
+ original_session = extract_current_session
48
+ session = new(domain: original_session.site, token: original_session.token, api_version: api_version)
49
+
50
+ with_session(session, &block)
51
+ end
52
+
53
+ def prepare_domain(domain)
54
+ return nil if domain.blank?
55
+ # remove http:// or https://
56
+ domain = domain.strip.gsub(%r{\Ahttps?://}, '')
57
+ # extract host, removing any username, password or path
58
+ shop = URI.parse("https://#{domain}").host
59
+ # extract subdomain of .myshopify.com
60
+ if (idx = shop.index("."))
61
+ shop = shop.slice(0, idx)
62
+ end
63
+ return nil if shop.empty?
64
+ "#{shop}.#{myshopify_domain}"
65
+ rescue URI::InvalidURIError
66
+ nil
67
+ end
68
+
69
+ def validate_signature(params)
70
+ params = (params.respond_to?(:to_unsafe_hash) ? params.to_unsafe_hash : params).with_indifferent_access
71
+ return false unless (signature = params[:hmac])
72
+
73
+ calculated_signature = OpenSSL::HMAC.hexdigest(
74
+ OpenSSL::Digest.new('SHA256'), secret, ShopifyAPI::HmacParams.encode(params)
75
+ )
76
+
77
+ Rack::Utils.secure_compare(calculated_signature, signature)
78
+ end
79
+
80
+ private
81
+
82
+ def extract_current_session
83
+ site = ShopifyAPI::Base.site.to_s
84
+ token = ShopifyAPI::Base.headers['X-Shopify-Access-Token']
85
+ version = ShopifyAPI::Base.api_version
86
+ new(domain: site, token: token, api_version: version)
87
+ end
88
+ end
89
+
90
+ def initialize(domain:, token:, access_scopes: nil, api_version: ShopifyAPI::Base.api_version, extra: {})
91
+ self.domain = self.class.prepare_domain(domain)
92
+ self.api_version = api_version
93
+ self.token = token
94
+ self.access_scopes = access_scopes
95
+ self.extra = extra
96
+ end
97
+
98
+ def create_permission_url(scope, redirect_uri, options = {})
99
+ params = { client_id: api_key, scope: ShopifyAPI::ApiAccess.new(scope).to_s, redirect_uri: redirect_uri }
100
+ params[:state] = options[:state] if options[:state]
101
+ params["grant_options[]".to_sym] = options[:grant_options] if options[:grant_options]
102
+ construct_oauth_url("authorize", params)
103
+ end
104
+
105
+ def request_token(params)
106
+ return token if token
107
+
108
+ twenty_four_hours_ago = Time.now.utc.to_i - SECONDS_IN_A_DAY
109
+ unless self.class.validate_signature(params) && params[:timestamp].to_i > twenty_four_hours_ago
110
+ raise ShopifyAPI::ValidationException, "Invalid Signature: Possible malicious login"
111
+ end
112
+
113
+ response = access_token_request(params[:code])
114
+ if response.code == "200"
115
+ self.extra = JSON.parse(response.body)
116
+ self.token = extra.delete('access_token')
117
+
118
+ if (expires_in = extra.delete('expires_in'))
119
+ extra['expires_at'] = Time.now.utc.to_i + expires_in
120
+ end
121
+ token
122
+ else
123
+ raise response.msg
124
+ end
125
+ end
126
+
127
+ def shop
128
+ Shop.current
129
+ end
130
+
131
+ def site
132
+ "https://#{domain}"
133
+ end
134
+
135
+ def api_version=(version)
136
+ @api_version = if ApiVersion::NullVersion.matches?(version)
137
+ ApiVersion::NullVersion
138
+ else
139
+ ApiVersion.find_version(version)
140
+ end
141
+ end
142
+
143
+ def valid?
144
+ domain.present? && token.present? && api_version.is_a?(ApiVersion)
145
+ end
146
+
147
+ def expires_in
148
+ return unless expires_at.present?
149
+ [0, expires_at.to_i - Time.now.utc.to_i].max
150
+ end
151
+
152
+ def expires_at
153
+ return unless extra.present?
154
+ @expires_at ||= Time.at(extra['expires_at']).utc
155
+ end
156
+
157
+ def expired?
158
+ return false if expires_in.nil?
159
+ expires_in <= 0
160
+ end
161
+
162
+ def hash
163
+ state.hash
164
+ end
165
+
166
+ def ==(other)
167
+ self.class == other.class && state == other.state
168
+ end
169
+
170
+ alias_method :eql?, :==
171
+
172
+ protected
173
+
174
+ def state
175
+ [domain, token, api_version, extra]
176
+ end
177
+
178
+ private
179
+
180
+ def access_scopes=(access_scopes)
181
+ return unless access_scopes
182
+ @access_scopes = ShopifyAPI::ApiAccess.new(access_scopes)
183
+ end
184
+
185
+ def parameterize(params)
186
+ URI.encode_www_form(params)
187
+ end
188
+
189
+ def access_token_request(code)
190
+ uri = URI.parse(construct_oauth_url('access_token'))
191
+ https = Net::HTTP.new(uri.host, uri.port)
192
+ https.use_ssl = true
193
+ request = Net::HTTP::Post.new(uri.request_uri)
194
+ request.set_form_data('client_id' => api_key, 'client_secret' => secret, 'code' => code)
195
+ https.request(request)
196
+ end
197
+
198
+ def construct_oauth_url(path, query_params = {})
199
+ query_string = "?#{parameterize(query_params)}" unless query_params.empty?
200
+ "https://#{domain}/admin/oauth/#{path}#{query_string}"
201
+ end
202
+ end
203
+ end
@@ -0,0 +1,4 @@
1
+ # frozen_string_literal: true
2
+ module ShopifyAPI
3
+ VERSION = "1.0.0"
4
+ end
@@ -0,0 +1,45 @@
1
+ # frozen_string_literal: true
2
+ $:.unshift(File.dirname(__FILE__))
3
+ require 'active_resource'
4
+ require 'active_support/core_ext/class/attribute_accessors'
5
+ require 'digest/md5'
6
+ require 'base64'
7
+ require 'active_resource/detailed_log_subscriber'
8
+ require 'shopify_api/limits'
9
+ require 'shopify_api/api_version'
10
+ require 'shopify_api/meta'
11
+ require 'active_resource/json_errors'
12
+ require 'shopify_api/paginated_collection'
13
+ require 'shopify_api/disable_prefix_check'
14
+
15
+ if Gem::Version.new(RUBY_VERSION) < Gem::Version.new("2.6")
16
+ puts("\nshopify_api: NOTE: Support for Ruby #{RUBY_VERSION} will be dropped in the next major release. Please update to Ruby 2.6 or newer before updating this gem.\n\n")
17
+ end
18
+
19
+ module ShopifyAPI
20
+ include Limits
21
+ end
22
+
23
+ require 'shopify_api/events'
24
+ require 'shopify_api/metafields'
25
+ require 'shopify_api/countable'
26
+ require 'shopify_api/resources'
27
+ require 'shopify_api/session'
28
+ require 'shopify_api/hmac_params'
29
+ require 'shopify_api/api_access'
30
+ require 'shopify_api/message_enricher'
31
+ require 'shopify_api/connection'
32
+ require 'shopify_api/pagination_link_headers'
33
+ require 'shopify_api/graphql'
34
+ require 'shopify_api/graphql/railtie' if defined?(Rails)
35
+
36
+ if ShopifyAPI::Base.respond_to?(:connection_class)
37
+ ShopifyAPI::Base.connection_class = ShopifyAPI::Connection
38
+ else
39
+ require 'active_resource/connection_ext'
40
+ end
41
+
42
+ if ENV["SHOPIFY_LOG_PATH"]
43
+ ActiveResource::Base.logger = Logger.new(ENV["SHOPIFY_LOG_PATH"])
44
+ ActiveResource::DetailedLogSubscriber.attach_to(:active_resource_detailed)
45
+ end
@@ -0,0 +1,8 @@
1
+ # frozen_string_literal: true
2
+ class VerifyDocs
3
+ def self.call
4
+ readme_content = File.read("README.md").lines[2..-1]
5
+ docs_content = File.read("docs/index.md").lines[4..-1]
6
+ readme_content == docs_content
7
+ end
8
+ end
data/service.yml ADDED
@@ -0,0 +1,2 @@
1
+ audience: partner
2
+ classification: library
@@ -0,0 +1 @@
1
+ # using the default shipit config
@@ -0,0 +1,47 @@
1
+ # frozen_string_literal: true
2
+ $:.push(File.expand_path("../lib", __FILE__))
3
+ require "shopify_api/version"
4
+
5
+ Gem::Specification.new do |s|
6
+ s.name = %q{ruby_shopify_api}
7
+ s.version = ShopifyAPI::VERSION
8
+ s.authors = ["Hopper Gee"]
9
+ s.email = ["hopper.gee@hey.com"]
10
+
11
+ s.summary = %q{The Shopify API gem is a lightweight gem for accessing the Shopify admin REST web services}
12
+ s.description = <<~HERE
13
+ The Shopify API gem allows Ruby developers to programmatically access the admin
14
+ section of Shopify stores. The API is implemented as JSON or XML over HTTP using
15
+ all four verbs (GET/POST/PUT/DELETE). Each resource, like Order, Product, or
16
+ Collection, has its own URL and is manipulated in isolation.
17
+ HERE
18
+ s.homepage = %q{https://github.com/RetroShopifyDev/ruby-shopify-api}
19
+
20
+ s.metadata['allowed_push_host'] = 'https://rubygems.org'
21
+
22
+ s.extra_rdoc_files = [
23
+ "LICENSE",
24
+ "README.md",
25
+ ]
26
+ s.files = `git ls-files`.split("\n")
27
+ s.test_files = `git ls-files -- {test}/*`.split("\n")
28
+
29
+ s.rdoc_options = ["--charset=UTF-8"]
30
+ s.summary = %q{ShopifyAPI is a lightweight gem for accessing the Shopify admin REST web services}
31
+ s.license = "MIT"
32
+
33
+ s.required_ruby_version = ">= 2.4"
34
+
35
+ s.add_runtime_dependency("activeresource", ">= 4.1.0")
36
+ s.add_runtime_dependency("rack")
37
+ s.add_runtime_dependency("graphql-client")
38
+
39
+ s.add_development_dependency("mocha", ">= 1.4.0")
40
+ s.add_development_dependency("webmock")
41
+ s.add_development_dependency("minitest", ">= 5.14")
42
+ s.add_development_dependency("rake")
43
+ s.add_development_dependency("timecop")
44
+ s.add_development_dependency("rubocop-shopify")
45
+ s.add_development_dependency("pry")
46
+ s.add_development_dependency("pry-byebug")
47
+ end
@@ -0,0 +1,29 @@
1
+ # frozen_string_literal: true
2
+ require 'test_helper'
3
+
4
+ class AbandonedCheckoutsTest < Test::Unit::TestCase
5
+ def setup
6
+ super
7
+
8
+ @expected_checkouts = JSON.parse(load_fixture('abandoned_checkouts'))['checkouts']
9
+ @expected_checkout_id = JSON.parse(load_fixture('abandoned_checkout'))['checkout']['id']
10
+ end
11
+
12
+ test ":create creates a checkout" do
13
+ fake 'checkouts', method: :post, status: 201, body: load_fixture('abandoned_checkout')
14
+
15
+ checkout = ShopifyAPI::AbandonedCheckout.create
16
+
17
+ assert_equal @expected_checkout_id, checkout.id
18
+ assert_equal true, checkout.attributes.include?(:abandoned_checkout_url)
19
+ end
20
+
21
+ test "get all checkouts indexed by token" do
22
+ fake 'checkouts', method: :get, status: 200, body: load_fixture('abandoned_checkouts')
23
+
24
+ checkouts = ShopifyAPI::AbandonedCheckout.all
25
+
26
+ assert_equal @expected_checkout_id, checkouts.first.id
27
+ assert_equal @expected_checkouts.size, checkouts.size
28
+ end
29
+ end
@@ -0,0 +1,23 @@
1
+ # frozen_string_literal: true
2
+ require 'test_helper'
3
+
4
+ class AccessScopeTest < Test::Unit::TestCase
5
+ test 'access scope does not use the versioned resource urls' do
6
+ fake(
7
+ 'access_scopes',
8
+ url: 'https://shop2.myshopify.com/admin/oauth/access_scopes.json',
9
+ method: :get,
10
+ status: 201,
11
+ body: load_fixture('access_scopes'),
12
+ extension: false
13
+ )
14
+
15
+ unstable_version = ShopifyAPI::Session.new(domain: 'shop2.myshopify.com', token: 'token2', api_version: :unstable)
16
+
17
+ ShopifyAPI::Base.activate_session(unstable_version)
18
+
19
+ scope_handles = ShopifyAPI::AccessScope.find(:all).map(&:handle)
20
+
21
+ assert_equal(['write_product_listings', 'read_shipping'], scope_handles)
22
+ end
23
+ end
@@ -0,0 +1,20 @@
1
+ # frozen_string_literal: true
2
+ require 'test_helper'
3
+
4
+ class AccessTokenTest < Test::Unit::TestCase
5
+ def test_delegate_access_token
6
+ fake(
7
+ "access_tokens/delegate.json?delegate_access_scope%5B%5D=write_orders&" \
8
+ "delegate_access_scope%5B%5D=read_products&expires_in=",
9
+ method: :post,
10
+ status: 201,
11
+ body: load_fixture('access_token_delegate'),
12
+ extension: false
13
+ )
14
+ delegate_scope = ['write_orders', 'read_products']
15
+ token = ShopifyAPI::AccessToken.delegate(delegate_scope)
16
+
17
+ assert_equal('abracadabra', token.access_token)
18
+ assert_equal('write_orders,read_products', token.scope)
19
+ end
20
+ end
@@ -0,0 +1,19 @@
1
+ # frozen_string_literal: true
2
+ require 'test_helper'
3
+
4
+ module ActiveResource
5
+ class JsonErrorsTest < Test::Unit::TestCase
6
+ def test_parsing_of_error_json_hash
7
+ @model = ShopifyAPI::Order.new
8
+ @model.errors.from_json({ errors: { name: ['missing'] } }.to_json)
9
+ assert_equal(['missing'], @model.errors[:name])
10
+ end
11
+
12
+ def test_parsing_of_error_json_plain_string
13
+ @model = ShopifyAPI::Order.new
14
+ @model.errors.from_json({ errors: 'some generic error' }.to_json)
15
+ assert_equal(['some generic error'], @model.errors[:base])
16
+ assert_equal('some generic error', @model.errors.full_messages.to_sentence)
17
+ end
18
+ end
19
+ end
@@ -0,0 +1,153 @@
1
+ # frozen_string_literal: true
2
+ require 'test_helper'
3
+
4
+ class ApiAccessTest < Minitest::Test
5
+ def test_write_is_the_same_access_as_read_write_on_the_same_resource
6
+ read_write_orders = ShopifyAPI::ApiAccess.new(%w(read_orders write_orders))
7
+ write_orders = ShopifyAPI::ApiAccess.new(%w(write_orders))
8
+
9
+ assert_equal write_orders, read_write_orders
10
+ end
11
+
12
+ def test_write_is_the_same_access_as_read_write_on_the_same_unauthenticated_resource
13
+ unauthenticated_read_write_orders = ShopifyAPI::ApiAccess.new(%w(unauthenticated_read_orders unauthenticated_write_orders))
14
+ unauthenticated_write_orders = ShopifyAPI::ApiAccess.new(%w(unauthenticated_write_orders))
15
+
16
+ assert_equal unauthenticated_write_orders, unauthenticated_read_write_orders
17
+ end
18
+
19
+ def test_read_is_not_the_same_as_read_write_on_the_same_resource
20
+ read_orders = ShopifyAPI::ApiAccess.new(%w(read_orders))
21
+ read_write_orders = ShopifyAPI::ApiAccess.new(%w(write_orders read_orders))
22
+
23
+ refute_equal read_write_orders, read_orders
24
+ end
25
+
26
+ def test_two_different_resources_are_not_equal
27
+ read_orders = ShopifyAPI::ApiAccess.new(%w(read_orders))
28
+ read_products = ShopifyAPI::ApiAccess.new(%w(read_products))
29
+
30
+ refute_equal read_orders, read_products
31
+ end
32
+
33
+ def test_two_identical_scopes_are_equal
34
+ read_orders = ShopifyAPI::ApiAccess.new(%w(read_orders))
35
+ read_orders_identical = ShopifyAPI::ApiAccess.new(%w(read_orders))
36
+
37
+ assert_equal read_orders_identical, read_orders
38
+ end
39
+
40
+ def test_unauthenticated_is_not_implied_by_authenticated_access
41
+ unauthenticated_orders = ShopifyAPI::ApiAccess.new(%w(unauthenticated_read_orders))
42
+ authenticated_read_orders = ShopifyAPI::ApiAccess.new(%w(read_orders))
43
+ authenticated_write_orders = ShopifyAPI::ApiAccess.new(%w(write_orders))
44
+
45
+ refute_equal unauthenticated_orders, authenticated_read_orders
46
+ refute_equal unauthenticated_orders, authenticated_write_orders
47
+ end
48
+
49
+ def test_scopes_covers_is_truthy_for_same_scopes
50
+ read_orders = ShopifyAPI::ApiAccess.new(%w(read_orders))
51
+ read_orders_identical = ShopifyAPI::ApiAccess.new(%w(read_orders))
52
+
53
+ assert read_orders.covers?(read_orders_identical)
54
+ end
55
+
56
+ def test_covers_is_falsy_for_different_scopes
57
+ read_orders = ShopifyAPI::ApiAccess.new(%w(read_orders))
58
+ read_products = ShopifyAPI::ApiAccess.new(%w(read_products))
59
+
60
+ refute read_orders.covers?(read_products)
61
+ end
62
+
63
+ def test_covers_is_truthy_for_read_when_the_set_has_read_write
64
+ write_products = ShopifyAPI::ApiAccess.new(%w(write_products))
65
+ read_products = ShopifyAPI::ApiAccess.new(%w(read_products))
66
+
67
+ assert write_products.covers?(read_products)
68
+ end
69
+
70
+ def test_covers_is_truthy_for_read_when_the_set_has_read_write_for_that_resource_and_others
71
+ write_products_and_orders = ShopifyAPI::ApiAccess.new(%w(write_products, write_orders))
72
+ read_orders = ShopifyAPI::ApiAccess.new(%w(read_orders))
73
+
74
+ assert write_products_and_orders.covers?(read_orders)
75
+ end
76
+
77
+ def test_covers_is_truthy_for_write_when_the_set_has_read_write_for_that_resource_and_others
78
+ write_products_and_orders = ShopifyAPI::ApiAccess.new(%w(write_products, write_orders))
79
+ write_orders = ShopifyAPI::ApiAccess.new(%w(write_orders))
80
+
81
+ assert write_products_and_orders.covers?(write_orders)
82
+ end
83
+
84
+ def test_covers_is_truthy_for_subset_of_scopes
85
+ write_products_orders_customers = ShopifyAPI::ApiAccess.new(%w(write_products write_orders write_customers))
86
+ write_orders_products = ShopifyAPI::ApiAccess.new(%w(write_orders read_products))
87
+
88
+ assert write_products_orders_customers.covers?(write_orders_products)
89
+ end
90
+
91
+ def test_covers_is_falsy_for_sets_of_scopes_that_have_no_common_elements
92
+ write_products_orders_customers = ShopifyAPI::ApiAccess.new(%w(write_products write_orders write_customers))
93
+ write_images_read_content = ShopifyAPI::ApiAccess.new(%w(write_images read_content))
94
+
95
+ refute write_products_orders_customers.covers?(write_images_read_content)
96
+ end
97
+
98
+ def test_covers_is_falsy_for_sets_of_scopes_that_have_only_some_common_access
99
+ write_products_orders_customers = ShopifyAPI::ApiAccess.new(%w(write_products write_orders write_customers))
100
+ write_products_read_content = ShopifyAPI::ApiAccess.new(%w(write_products read_content))
101
+
102
+ refute write_products_orders_customers.covers?(write_products_read_content)
103
+ end
104
+
105
+ def test_duplicate_scopes_resolve_to_one_scope
106
+ read_orders_duplicated = ShopifyAPI::ApiAccess.new(%w(read_orders read_orders read_orders read_orders))
107
+ read_orders = ShopifyAPI::ApiAccess.new(%w(read_orders))
108
+
109
+ assert_equal read_orders, read_orders_duplicated
110
+ end
111
+
112
+ def test_to_s_outputs_scopes_as_a_comma_separated_list_without_implied_read_scopes
113
+ serialized_read_products_write_orders = "read_products,write_orders"
114
+ read_products_write_orders = ShopifyAPI::ApiAccess.new(%w(read_products read_orders write_orders))
115
+
116
+ assert_equal read_products_write_orders.to_s, serialized_read_products_write_orders
117
+ end
118
+
119
+ def test_to_a_outputs_scopes_as_an_array_of_strings_without_implied_read_scopes
120
+ serialized_read_products_write_orders = %w(write_orders read_products)
121
+ read_products_write_orders = ShopifyAPI::ApiAccess.new(%w(read_products read_orders write_orders))
122
+
123
+ assert_equal read_products_write_orders.to_a.sort, serialized_read_products_write_orders.sort
124
+ end
125
+
126
+ def test_creating_scopes_removes_extra_whitespace_from_scope_name_and_blank_scope_names
127
+ deserialized_read_products_write_orders = ShopifyAPI::ApiAccess.new([' read_products', ' ', 'write_orders '])
128
+ serialized_read_products_write_orders = deserialized_read_products_write_orders.to_s
129
+ expected_read_products_write_orders = ShopifyAPI::ApiAccess.new(%w(read_products write_orders))
130
+
131
+ assert_equal expected_read_products_write_orders, ShopifyAPI::ApiAccess.new(serialized_read_products_write_orders)
132
+ end
133
+
134
+ def test_creating_scopes_from_a_string_works_with_a_comma_separated_list
135
+ deserialized_read_products_write_orders = ShopifyAPI::ApiAccess.new("read_products,write_orders")
136
+ serialized_read_products_write_orders = deserialized_read_products_write_orders.to_s
137
+ expected_read_products_write_orders = ShopifyAPI::ApiAccess.new(%w(read_products write_orders))
138
+
139
+ assert_equal expected_read_products_write_orders, ShopifyAPI::ApiAccess.new(serialized_read_products_write_orders)
140
+ end
141
+
142
+ def test_using_to_s_from_one_scopes_to_construct_another_will_be_equal
143
+ read_products_write_orders = ShopifyAPI::ApiAccess.new(%w(read_products write_orders))
144
+
145
+ assert_equal read_products_write_orders, ShopifyAPI::ApiAccess.new(read_products_write_orders.to_s)
146
+ end
147
+
148
+ def test_using_to_a_from_one_scopes_to_construct_another_will_be_equal
149
+ read_products_write_orders = ShopifyAPI::ApiAccess.new(%w(read_products write_orders))
150
+
151
+ assert_equal read_products_write_orders, ShopifyAPI::ApiAccess.new(read_products_write_orders.to_a)
152
+ end
153
+ end
@@ -0,0 +1,9 @@
1
+ # frozen_string_literal: true
2
+ require 'test_helper'
3
+
4
+ class ApiPermissionTest < Test::Unit::TestCase
5
+ test "revoke access token" do
6
+ fake "api_permissions/current", method: :delete, status: 200, body: "{}"
7
+ assert ShopifyAPI::ApiPermission.destroy
8
+ end
9
+ end