shopify_api 9.2.0 → 9.3.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (190) hide show
  1. checksums.yaml +4 -4
  2. data/.github/ISSUE_TEMPLATE.md +36 -0
  3. data/.github/workflows/build.yml +41 -0
  4. data/.gitignore +0 -1
  5. data/.rubocop.yml +23 -3
  6. data/.rubocop_todo.yml +75 -0
  7. data/CHANGELOG.md +15 -0
  8. data/Gemfile +1 -0
  9. data/Gemfile.lock +151 -0
  10. data/Gemfile_ar51 +1 -1
  11. data/README.md +40 -30
  12. data/RELEASING +10 -6
  13. data/Rakefile +10 -6
  14. data/docs/graphql.md +1 -1
  15. data/docs/index.md +1 -1
  16. data/lib/active_resource/connection_ext.rb +1 -0
  17. data/lib/active_resource/detailed_log_subscriber.rb +6 -7
  18. data/lib/active_resource/json_errors.rb +8 -2
  19. data/lib/shopify_api.rb +8 -1
  20. data/lib/shopify_api/api_access.rb +57 -0
  21. data/lib/shopify_api/api_version.rb +6 -5
  22. data/lib/shopify_api/connection.rb +1 -0
  23. data/lib/shopify_api/countable.rb +3 -2
  24. data/lib/shopify_api/disable_prefix_check.rb +2 -2
  25. data/lib/shopify_api/events.rb +2 -1
  26. data/lib/shopify_api/graphql.rb +8 -6
  27. data/lib/shopify_api/limits.rb +3 -2
  28. data/lib/shopify_api/message_enricher.rb +11 -9
  29. data/lib/shopify_api/meta.rb +0 -1
  30. data/lib/shopify_api/metafields.rb +5 -4
  31. data/lib/shopify_api/pagination_link_headers.rb +4 -3
  32. data/lib/shopify_api/resources.rb +1 -0
  33. data/lib/shopify_api/resources/access_scope.rb +1 -1
  34. data/lib/shopify_api/resources/access_token.rb +1 -0
  35. data/lib/shopify_api/resources/address.rb +1 -0
  36. data/lib/shopify_api/resources/announcement.rb +2 -1
  37. data/lib/shopify_api/resources/application_charge.rb +1 -0
  38. data/lib/shopify_api/resources/application_credit.rb +1 -0
  39. data/lib/shopify_api/resources/article.rb +3 -2
  40. data/lib/shopify_api/resources/asset.rb +6 -5
  41. data/lib/shopify_api/resources/assigned_fulfillment_order.rb +1 -1
  42. data/lib/shopify_api/resources/base.rb +11 -6
  43. data/lib/shopify_api/resources/billing_address.rb +1 -0
  44. data/lib/shopify_api/resources/blog.rb +2 -1
  45. data/lib/shopify_api/resources/carrier_service.rb +1 -0
  46. data/lib/shopify_api/resources/cart.rb +2 -1
  47. data/lib/shopify_api/resources/collect.rb +1 -0
  48. data/lib/shopify_api/resources/collection_listing.rb +1 -0
  49. data/lib/shopify_api/resources/comment.rb +20 -5
  50. data/lib/shopify_api/resources/country.rb +1 -0
  51. data/lib/shopify_api/resources/custom_collection.rb +4 -3
  52. data/lib/shopify_api/resources/customer.rb +2 -1
  53. data/lib/shopify_api/resources/customer_group.rb +1 -0
  54. data/lib/shopify_api/resources/customer_invite.rb +1 -0
  55. data/lib/shopify_api/resources/customer_saved_search.rb +2 -1
  56. data/lib/shopify_api/resources/discount_code.rb +1 -0
  57. data/lib/shopify_api/resources/discount_code_batch.rb +4 -2
  58. data/lib/shopify_api/resources/draft_order.rb +1 -0
  59. data/lib/shopify_api/resources/draft_order_invoice.rb +1 -0
  60. data/lib/shopify_api/resources/event.rb +1 -0
  61. data/lib/shopify_api/resources/fulfillment.rb +12 -3
  62. data/lib/shopify_api/resources/fulfillment_event.rb +1 -0
  63. data/lib/shopify_api/resources/fulfillment_order.rb +30 -16
  64. data/lib/shopify_api/resources/fulfillment_order_locations_for_move.rb +1 -0
  65. data/lib/shopify_api/resources/fulfillment_request.rb +1 -0
  66. data/lib/shopify_api/resources/fulfillment_service.rb +1 -0
  67. data/lib/shopify_api/resources/fulfillment_v2.rb +3 -2
  68. data/lib/shopify_api/resources/gift_card.rb +1 -0
  69. data/lib/shopify_api/resources/image.rb +2 -1
  70. data/lib/shopify_api/resources/inventory_level.rb +2 -3
  71. data/lib/shopify_api/resources/line_item.rb +4 -3
  72. data/lib/shopify_api/resources/location.rb +1 -1
  73. data/lib/shopify_api/resources/marketing_event.rb +1 -0
  74. data/lib/shopify_api/resources/metafield.rb +1 -0
  75. data/lib/shopify_api/resources/note_attribute.rb +1 -0
  76. data/lib/shopify_api/resources/option.rb +1 -0
  77. data/lib/shopify_api/resources/order.rb +2 -1
  78. data/lib/shopify_api/resources/order_risk.rb +1 -0
  79. data/lib/shopify_api/resources/page.rb +1 -0
  80. data/lib/shopify_api/resources/payment_details.rb +1 -0
  81. data/lib/shopify_api/resources/policy.rb +1 -0
  82. data/lib/shopify_api/resources/price_rule.rb +1 -1
  83. data/lib/shopify_api/resources/product.rb +14 -11
  84. data/lib/shopify_api/resources/product_listing.rb +1 -0
  85. data/lib/shopify_api/resources/province.rb +1 -0
  86. data/lib/shopify_api/resources/receipt.rb +1 -0
  87. data/lib/shopify_api/resources/recurring_application_charge.rb +4 -1
  88. data/lib/shopify_api/resources/redirect.rb +1 -0
  89. data/lib/shopify_api/resources/refund.rb +2 -1
  90. data/lib/shopify_api/resources/report.rb +1 -0
  91. data/lib/shopify_api/resources/resource_feedback.rb +1 -1
  92. data/lib/shopify_api/resources/rule.rb +1 -0
  93. data/lib/shopify_api/resources/script_tag.rb +1 -0
  94. data/lib/shopify_api/resources/shipping_address.rb +1 -0
  95. data/lib/shopify_api/resources/shipping_line.rb +1 -0
  96. data/lib/shopify_api/resources/shipping_zone.rb +1 -0
  97. data/lib/shopify_api/resources/shop.rb +2 -1
  98. data/lib/shopify_api/resources/smart_collection.rb +2 -2
  99. data/lib/shopify_api/resources/storefront_access_token.rb +1 -0
  100. data/lib/shopify_api/resources/tax_line.rb +1 -0
  101. data/lib/shopify_api/resources/tax_service.rb +1 -0
  102. data/lib/shopify_api/resources/theme.rb +1 -0
  103. data/lib/shopify_api/resources/transaction.rb +1 -0
  104. data/lib/shopify_api/resources/usage_charge.rb +1 -0
  105. data/lib/shopify_api/resources/user.rb +1 -0
  106. data/lib/shopify_api/resources/variant.rb +16 -18
  107. data/lib/shopify_api/resources/webhook.rb +1 -0
  108. data/lib/shopify_api/session.rb +20 -12
  109. data/lib/shopify_api/version.rb +2 -1
  110. data/lib/verify_docs.rb +1 -0
  111. data/service.yml +1 -1
  112. data/shopify_api.gemspec +10 -4
  113. data/test/access_token_test.rb +6 -5
  114. data/test/active_resource/json_errors_test.rb +6 -6
  115. data/test/api_access_test.rb +153 -0
  116. data/test/api_version_test.rb +3 -3
  117. data/test/application_charge_test.rb +30 -27
  118. data/test/application_credit_test.rb +10 -9
  119. data/test/article_test.rb +34 -35
  120. data/test/asset_test.rb +14 -6
  121. data/test/assigned_fulfillment_order_test.rb +5 -4
  122. data/test/base_test.rb +55 -56
  123. data/test/blog_test.rb +4 -3
  124. data/test/carrier_service_test.rb +7 -6
  125. data/test/cart_test.rb +2 -1
  126. data/test/collect_test.rb +4 -3
  127. data/test/collection_listing_test.rb +21 -16
  128. data/test/collection_publication_test.rb +8 -8
  129. data/test/collection_test.rb +20 -19
  130. data/test/countable_test.rb +3 -2
  131. data/test/currency_test.rb +5 -5
  132. data/test/custom_collection_test.rb +4 -3
  133. data/test/customer_saved_search_test.rb +18 -8
  134. data/test/customer_test.rb +22 -14
  135. data/test/detailed_log_subscriber_test.rb +15 -11
  136. data/test/discount_code_batch_test.rb +11 -10
  137. data/test/discount_code_test.rb +21 -15
  138. data/test/draft_order_test.rb +68 -52
  139. data/test/fixtures/assigned_fulfillment_orders.json +2 -0
  140. data/test/fixtures/fulfillment_order.json +1 -0
  141. data/test/fixtures/fulfillment_orders.json +2 -0
  142. data/test/fulfillment_event_test.rb +31 -26
  143. data/test/fulfillment_order_test.rb +215 -147
  144. data/test/fulfillment_order_test_helper.rb +1 -0
  145. data/test/fulfillment_request_test.rb +10 -8
  146. data/test/fulfillment_service_test.rb +13 -12
  147. data/test/fulfillment_test.rb +81 -66
  148. data/test/fulfillment_v2_test.rb +16 -12
  149. data/test/gift_card_test.rb +6 -4
  150. data/test/graphql_test.rb +27 -27
  151. data/test/image_test.rb +19 -17
  152. data/test/inventory_level_test.rb +24 -15
  153. data/test/lib/webmock_extensions/last_request.rb +1 -1
  154. data/test/limits_test.rb +2 -1
  155. data/test/location_test.rb +2 -1
  156. data/test/marketing_event_test.rb +20 -20
  157. data/test/message_enricher_test.rb +6 -6
  158. data/test/meta_test.rb +7 -9
  159. data/test/metafield_test.rb +30 -20
  160. data/test/order_risk_test.rb +17 -16
  161. data/test/order_test.rb +43 -28
  162. data/test/pagination_test.rb +89 -56
  163. data/test/policy_test.rb +6 -5
  164. data/test/price_rule_test.rb +20 -15
  165. data/test/product_listing_test.rb +20 -20
  166. data/test/product_publication_test.rb +8 -8
  167. data/test/product_test.rb +44 -32
  168. data/test/publication_test.rb +3 -3
  169. data/test/recurring_application_charge_test.rb +104 -42
  170. data/test/redirect_test.rb +4 -3
  171. data/test/refund_test.rb +22 -17
  172. data/test/report_test.rb +12 -10
  173. data/test/resource_feedback_test.rb +14 -13
  174. data/test/script_tag_test.rb +10 -9
  175. data/test/session_test.rb +74 -46
  176. data/test/shipping_zone_test.rb +4 -3
  177. data/test/shop_test.rb +47 -33
  178. data/test/smart_collection_test.rb +5 -4
  179. data/test/storefront_access_token_test.rb +13 -15
  180. data/test/tax_service_test.rb +7 -4
  181. data/test/tender_transaction_test.rb +3 -3
  182. data/test/test_helper.rb +13 -11
  183. data/test/transaction_test.rb +4 -3
  184. data/test/usage_charge_test.rb +12 -8
  185. data/test/user_test.rb +4 -3
  186. data/test/variant_test.rb +23 -54
  187. data/test/webhook_test.rb +10 -7
  188. metadata +14 -9
  189. data/.rubocop-https---shopify-github-io-ruby-style-guide-rubocop-yml +0 -1027
  190. data/.travis.yml +0 -28
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module ShopifyAPI
2
3
  class Webhook < Base
3
4
  end
@@ -1,12 +1,14 @@
1
+ # frozen_string_literal: true
1
2
  require 'openssl'
2
3
  require 'rack'
3
4
 
4
5
  module ShopifyAPI
5
-
6
6
  class ValidationException < StandardError
7
7
  end
8
8
 
9
9
  class Session
10
+ SECONDS_IN_A_DAY = 24 * 60 * 60
11
+
10
12
  cattr_accessor :api_key, :secret, :myshopify_domain
11
13
  self.myshopify_domain = 'myshopify.com'
12
14
 
@@ -15,9 +17,8 @@ module ShopifyAPI
15
17
  alias_method :url, :domain
16
18
 
17
19
  class << self
18
-
19
20
  def setup(params)
20
- params.each { |k,value| public_send("#{k}=", value) }
21
+ params.each { |k, value| public_send("#{k}=", value) }
21
22
  end
22
23
 
23
24
  def temp(domain:, token:, api_version: ShopifyAPI::Base.api_version, &block)
@@ -56,7 +57,7 @@ module ShopifyAPI
56
57
  # extract host, removing any username, password or path
57
58
  shop = URI.parse("https://#{domain}").host
58
59
  # extract subdomain of .myshopify.com
59
- if idx = shop.index(".")
60
+ if (idx = shop.index("."))
60
61
  shop = shop.slice(0, idx)
61
62
  end
62
63
  return nil if shop.empty?
@@ -67,9 +68,11 @@ module ShopifyAPI
67
68
 
68
69
  def validate_signature(params)
69
70
  params = (params.respond_to?(:to_unsafe_hash) ? params.to_unsafe_hash : params).with_indifferent_access
70
- return false unless signature = params[:hmac]
71
+ return false unless (signature = params[:hmac])
71
72
 
72
- calculated_signature = OpenSSL::HMAC.hexdigest(OpenSSL::Digest::SHA256.new(), secret, encoded_params_for_signature(params))
73
+ calculated_signature = OpenSSL::HMAC.hexdigest(
74
+ OpenSSL::Digest.new('SHA256'), secret, encoded_params_for_signature(params)
75
+ )
73
76
 
74
77
  Rack::Utils.secure_compare(calculated_signature, signature)
75
78
  end
@@ -78,7 +81,7 @@ module ShopifyAPI
78
81
 
79
82
  def encoded_params_for_signature(params)
80
83
  params = params.except(:signature, :hmac, :action, :controller)
81
- params.map{|k,v| "#{URI.escape(k.to_s, '&=%')}=#{URI.escape(v.to_s, '&%')}"}.sort.join('&')
84
+ params.map { |k, v| "#{URI.escape(k.to_s, '&=%')}=#{URI.escape(v.to_s, '&%')}" }.sort.join('&')
82
85
  end
83
86
 
84
87
  def extract_current_session
@@ -97,7 +100,7 @@ module ShopifyAPI
97
100
  end
98
101
 
99
102
  def create_permission_url(scope, redirect_uri, options = {})
100
- params = { client_id: api_key, scope: scope.join(','), redirect_uri: redirect_uri }
103
+ params = { client_id: api_key, scope: ShopifyAPI::ApiAccess.new(scope).to_s, redirect_uri: redirect_uri }
101
104
  params[:state] = options[:state] if options[:state]
102
105
  construct_oauth_url("authorize", params)
103
106
  end
@@ -105,7 +108,8 @@ module ShopifyAPI
105
108
  def request_token(params)
106
109
  return token if token
107
110
 
108
- unless self.class.validate_signature(params) && params[:timestamp].to_i > 24.hours.ago.utc.to_i
111
+ twenty_four_hours_ago = Time.now.utc.to_i - SECONDS_IN_A_DAY
112
+ unless self.class.validate_signature(params) && params[:timestamp].to_i > twenty_four_hours_ago
109
113
  raise ShopifyAPI::ValidationException, "Invalid Signature: Possible malicious login"
110
114
  end
111
115
 
@@ -114,12 +118,12 @@ module ShopifyAPI
114
118
  self.extra = JSON.parse(response.body)
115
119
  self.token = extra.delete('access_token')
116
120
 
117
- if expires_in = extra.delete('expires_in')
121
+ if (expires_in = extra.delete('expires_in'))
118
122
  extra['expires_at'] = Time.now.utc.to_i + expires_in
119
123
  end
120
124
  token
121
125
  else
122
- raise RuntimeError, response.msg
126
+ raise response.msg
123
127
  end
124
128
  end
125
129
 
@@ -132,7 +136,11 @@ module ShopifyAPI
132
136
  end
133
137
 
134
138
  def api_version=(version)
135
- @api_version = ApiVersion::NullVersion.matches?(version) ? ApiVersion::NullVersion : ApiVersion.find_version(version)
139
+ @api_version = if ApiVersion::NullVersion.matches?(version)
140
+ ApiVersion::NullVersion
141
+ else
142
+ ApiVersion.find_version(version)
143
+ end
136
144
  end
137
145
 
138
146
  def valid?
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module ShopifyAPI
2
- VERSION = "9.2.0"
3
+ VERSION = "9.3.0"
3
4
  end
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  class VerifyDocs
2
3
  def self.call
3
4
  readme_content = File.read("README.md").lines[2..-1]
@@ -5,4 +5,4 @@ owners:
5
5
  - Shopify/app-partner-dev-tools-education
6
6
  slack_channels:
7
7
  - dev-tools-education
8
- - api-patterns-team
8
+ - help-api-patterns
@@ -1,4 +1,5 @@
1
- $:.push File.expand_path("../lib", __FILE__)
1
+ # frozen_string_literal: true
2
+ $:.push(File.expand_path("../lib", __FILE__))
2
3
  require "shopify_api/version"
3
4
 
4
5
  Gem::Specification.new do |s|
@@ -7,7 +8,12 @@ Gem::Specification.new do |s|
7
8
  s.author = "Shopify"
8
9
 
9
10
  s.summary = %q{The Shopify API gem is a lightweight gem for accessing the Shopify admin REST web services}
10
- s.description = %q{The Shopify API gem allows Ruby developers to programmatically access the admin section of Shopify stores. The API is implemented as JSON or XML over HTTP using all four verbs (GET/POST/PUT/DELETE). Each resource, like Order, Product, or Collection, has its own URL and is manipulated in isolation.}
11
+ s.description = <<~HERE
12
+ The Shopify API gem allows Ruby developers to programmatically access the admin
13
+ section of Shopify stores. The API is implemented as JSON or XML over HTTP using
14
+ all four verbs (GET/POST/PUT/DELETE). Each resource, like Order, Product, or
15
+ Collection, has its own URL and is manipulated in isolation.
16
+ HERE
11
17
  s.email = %q{developers@jadedpixel.com}
12
18
  s.homepage = %q{http://www.shopify.com/partners/apps}
13
19
 
@@ -15,7 +21,7 @@ Gem::Specification.new do |s|
15
21
 
16
22
  s.extra_rdoc_files = [
17
23
  "LICENSE",
18
- "README.md"
24
+ "README.md",
19
25
  ]
20
26
  s.files = `git ls-files`.split("\n")
21
27
  s.test_files = `git ls-files -- {test}/*`.split("\n")
@@ -35,7 +41,7 @@ Gem::Specification.new do |s|
35
41
  s.add_development_dependency("minitest", ">= 4.0")
36
42
  s.add_development_dependency("rake")
37
43
  s.add_development_dependency("timecop")
38
- s.add_development_dependency("rubocop")
44
+ s.add_development_dependency("rubocop-shopify")
39
45
  s.add_development_dependency("pry")
40
46
  s.add_development_dependency("pry-byebug")
41
47
  end
@@ -1,19 +1,20 @@
1
+ # frozen_string_literal: true
1
2
  require 'test_helper'
2
3
 
3
4
  class AccessTokenTest < Test::Unit::TestCase
4
-
5
5
  def test_delegate_access_token
6
- fake "access_tokens/delegate.json?delegate_access_scope%5B%5D=write_orders&" \
6
+ fake(
7
+ "access_tokens/delegate.json?delegate_access_scope%5B%5D=write_orders&" \
7
8
  "delegate_access_scope%5B%5D=read_products&expires_in=",
8
9
  method: :post,
9
10
  status: 201,
10
11
  body: load_fixture('access_token_delegate'),
11
12
  extension: false
12
-
13
+ )
13
14
  delegate_scope = ['write_orders', 'read_products']
14
15
  token = ShopifyAPI::AccessToken.delegate(delegate_scope)
15
16
 
16
- assert_equal 'abracadabra', token.access_token
17
- assert_equal 'write_orders,read_products', token.scope
17
+ assert_equal('abracadabra', token.access_token)
18
+ assert_equal('write_orders,read_products', token.scope)
18
19
  end
19
20
  end
@@ -1,19 +1,19 @@
1
+ # frozen_string_literal: true
1
2
  require 'test_helper'
2
3
 
3
4
  module ActiveResource
4
5
  class JsonErrorsTest < Test::Unit::TestCase
5
-
6
6
  def test_parsing_of_error_json_hash
7
7
  @model = ShopifyAPI::Order.new
8
- @model.errors.from_json({errors: {name: ['missing']}}.to_json)
9
- assert_equal ['missing'], @model.errors[:name]
8
+ @model.errors.from_json({ errors: { name: ['missing'] } }.to_json)
9
+ assert_equal(['missing'], @model.errors[:name])
10
10
  end
11
11
 
12
12
  def test_parsing_of_error_json_plain_string
13
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
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
17
  end
18
18
  end
19
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
@@ -45,7 +45,7 @@ class ApiVersionTest < Test::Unit::TestCase
45
45
  test "find_version does raise when coercing a string if no versions are defined when version_lookup_mode is :raise_on_unknown" do
46
46
  refute ShopifyAPI::ApiVersion.versions['made up version']
47
47
  ShopifyAPI::ApiVersion.version_lookup_mode = :raise_on_unknown
48
- assert_raises ShopifyAPI::ApiVersion::UnknownVersion do
48
+ assert_raises(ShopifyAPI::ApiVersion::UnknownVersion) do
49
49
  ShopifyAPI::ApiVersion.find_version('made up version')
50
50
  end
51
51
  end
@@ -54,7 +54,7 @@ class ApiVersionTest < Test::Unit::TestCase
54
54
  ShopifyAPI::ApiVersion.clear_known_versions
55
55
  ShopifyAPI::ApiVersion.version_lookup_mode = :define_on_unknown
56
56
  assert_equal :define_on_unknown, ShopifyAPI::ApiVersion.version_lookup_mode
57
- assert_raises ArgumentError do
57
+ assert_raises(ArgumentError) do
58
58
  ShopifyAPI::ApiVersion.find_version(ShopifyAPI::ApiVersion::NullVersion)
59
59
  end
60
60
  end
@@ -108,7 +108,6 @@ class ApiVersionTest < Test::Unit::TestCase
108
108
  }
109
109
  )
110
110
  silence_warnings do
111
-
112
111
  refute_equal(
113
112
  ShopifyAPI::ApiVersion.new(handle: '2019-01'),
114
113
  ShopifyAPI::ApiVersion.latest_stable_version
@@ -151,6 +150,7 @@ class ApiVersionTest < Test::Unit::TestCase
151
150
 
152
151
  class TestApiVersion < ShopifyAPI::ApiVersion
153
152
  def initialize(name)
153
+ super(name)
154
154
  @version_name = name
155
155
  end
156
156
  end
@@ -1,79 +1,82 @@
1
+ # frozen_string_literal: true
1
2
  require 'test_helper'
2
3
 
3
4
  class ApplicationChargeTest < Test::Unit::TestCase
4
-
5
5
  def test_application_charge_create
6
- fake "application_charges", :method => :post, :status => 201, :body => load_fixture('application_charge')
6
+ fake("application_charges", method: :post, status: 201, body: load_fixture('application_charge'))
7
7
 
8
8
  charge = ShopifyAPI::ApplicationCharge.create(
9
- name: "iPod Cleaning",
10
- price: 5.00,
11
- return_url: "http://google.com"
9
+ name: "iPod Cleaning",
10
+ price: 5.00,
11
+ return_url: "http://google.com"
12
12
  )
13
13
 
14
- assert_equal 'https://this-is-my-test-shop.myshopify.com/admin/charges/803742/confirm_application_charge?signature=BAhpA55DDA%3D%3D--55b44e274e438c619be4631c804abcbcb6ee915a', charge.confirmation_url
14
+ assert_equal(
15
+ 'https://this-is-my-test-shop.myshopify.com/admin/charges/803742/confirm_application_charge?' \
16
+ 'signature=BAhpA55DDA%3D%3D--55b44e274e438c619be4631c804abcbcb6ee915a',
17
+ charge.confirmation_url
18
+ )
15
19
  end
16
20
 
17
21
  def test_get_application_charge
18
- fake "application_charges/803742", :method => :get, :status => 201, :body => load_fixture('application_charge')
22
+ fake("application_charges/803742", method: :get, status: 201, body: load_fixture('application_charge'))
19
23
 
20
24
  charge = ShopifyAPI::ApplicationCharge.find(803742)
21
25
 
22
- assert_equal "iPod Cleaning", charge.name
26
+ assert_equal("iPod Cleaning", charge.name)
23
27
  end
24
28
 
25
29
  def test_list_application_charges
26
- fake "application_charges", :method => :get, :status => 201, :body => load_fixture('application_charges')
30
+ fake("application_charges", method: :get, status: 201, body: load_fixture('application_charges'))
27
31
 
28
32
  charges = ShopifyAPI::ApplicationCharge.find(:all)
29
33
 
30
- assert_equal 4, charges.size
31
- assert_equal "iPhone Case", charges.last.name
34
+ assert_equal(4, charges.size)
35
+ assert_equal("iPhone Case", charges.last.name)
32
36
  end
33
37
 
34
38
  def test_list_pending_application_charges
35
- fake "application_charges", :method => :get, :status => 201, :body => load_fixture('application_charges')
39
+ fake("application_charges", method: :get, status: 201, body: load_fixture('application_charges'))
36
40
 
37
41
  pending_charges = ShopifyAPI::ApplicationCharge.pending
38
42
 
39
- assert_equal 1, pending_charges.size
40
- assert_equal "Screen Replacement", pending_charges.first.name
43
+ assert_equal(1, pending_charges.size)
44
+ assert_equal("Screen Replacement", pending_charges.first.name)
41
45
  end
42
46
 
43
47
  def test_list_expired_application_charges
44
- fake "application_charges", :method => :get, :status => 201, :body => load_fixture('application_charges')
48
+ fake("application_charges", method: :get, status: 201, body: load_fixture('application_charges'))
45
49
 
46
50
  expired_charges = ShopifyAPI::ApplicationCharge.expired
47
51
 
48
- assert_equal 1, expired_charges.size
49
- assert_equal "iPod Cleaning", expired_charges.first.name
52
+ assert_equal(1, expired_charges.size)
53
+ assert_equal("iPod Cleaning", expired_charges.first.name)
50
54
  end
51
55
 
52
56
  def test_list_accepted_application_charges
53
- fake "application_charges", :method => :get, :status => 201, :body => load_fixture('application_charges')
57
+ fake("application_charges", method: :get, status: 201, body: load_fixture('application_charges'))
54
58
 
55
59
  accepted_charges = ShopifyAPI::ApplicationCharge.accepted
56
60
 
57
- assert_equal 1, accepted_charges.size
58
- assert_equal "iPhone Case", accepted_charges.first.name
61
+ assert_equal(1, accepted_charges.size)
62
+ assert_equal("iPhone Case", accepted_charges.first.name)
59
63
  end
60
64
 
61
65
  def test_list_declined_application_charges
62
- fake "application_charges", :method => :get, :status => 201, :body => load_fixture('application_charges')
66
+ fake("application_charges", method: :get, status: 201, body: load_fixture('application_charges'))
63
67
 
64
68
  declined_charges = ShopifyAPI::ApplicationCharge.declined
65
69
 
66
- assert_equal 1, declined_charges.size
67
- assert_equal "Magic Mouse", declined_charges.first.name
70
+ assert_equal(1, declined_charges.size)
71
+ assert_equal("Magic Mouse", declined_charges.first.name)
68
72
  end
69
73
 
70
74
  def test_activate_application_charge
71
- fake "application_charges", :method => :get, :status => 201, :body => load_fixture('application_charges')
72
- fake "application_charges/803740/activate", :method => :post, :status => 200, :body => "{}"
75
+ fake("application_charges", method: :get, status: 201, body: load_fixture('application_charges'))
76
+ fake("application_charges/803740/activate", method: :post, status: 200, body: "{}")
73
77
 
74
78
  charge = ShopifyAPI::ApplicationCharge.accepted
75
79
 
76
- assert charge.last.activate
80
+ assert(charge.last.activate)
77
81
  end
78
-
79
82
  end