spree_api 4.3.0.rc1 → 4.4.0.rc1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (217) hide show
  1. checksums.yaml +4 -4
  2. data/app/controllers/concerns/spree/api/v2/caching.rb +37 -0
  3. data/app/controllers/concerns/spree/api/v2/coupon_codes_helper.rb +29 -0
  4. data/app/controllers/concerns/spree/api/v2/number_resource.rb +11 -0
  5. data/app/controllers/concerns/spree/api/v2/platform/nested_set_reposition_concern.rb +37 -0
  6. data/app/controllers/concerns/spree/api/v2/platform/promotion_calculator_params.rb +17 -0
  7. data/app/controllers/concerns/spree/api/v2/platform/promotion_rule_params.rb +16 -0
  8. data/app/controllers/concerns/spree/api/v2/storefront/metadata_controller_concern.rb +18 -0
  9. data/app/controllers/spree/api/v1/checkouts_controller.rb +1 -1
  10. data/app/controllers/spree/api/v2/base_controller.rb +9 -6
  11. data/app/controllers/spree/api/v2/platform/adjustments_controller.rb +19 -0
  12. data/app/controllers/spree/api/v2/platform/classifications_controller.rb +1 -22
  13. data/app/controllers/spree/api/v2/platform/cms_pages_controller.rb +4 -0
  14. data/app/controllers/spree/api/v2/platform/cms_sections_controller.rb +13 -18
  15. data/app/controllers/spree/api/v2/platform/digital_links_controller.rb +25 -0
  16. data/app/controllers/spree/api/v2/platform/digitals_controller.rb +19 -0
  17. data/app/controllers/spree/api/v2/platform/line_items_controller.rb +59 -0
  18. data/app/controllers/spree/api/v2/platform/menu_items_controller.rb +5 -19
  19. data/app/controllers/spree/api/v2/platform/orders_controller.rb +163 -0
  20. data/app/controllers/spree/api/v2/platform/payment_methods_controller.rb +27 -0
  21. data/app/controllers/spree/api/v2/platform/payments_controller.rb +17 -0
  22. data/app/controllers/spree/api/v2/platform/products_controller.rb +12 -0
  23. data/app/controllers/spree/api/v2/platform/promotion_actions_controller.rb +30 -0
  24. data/app/controllers/spree/api/v2/platform/promotion_categories_controller.rb +19 -0
  25. data/app/controllers/spree/api/v2/platform/promotion_rules_controller.rb +25 -0
  26. data/app/controllers/spree/api/v2/platform/promotions_controller.rb +31 -0
  27. data/app/controllers/spree/api/v2/platform/resource_controller.rb +48 -14
  28. data/app/controllers/spree/api/v2/platform/roles_controller.rb +15 -0
  29. data/app/controllers/spree/api/v2/platform/shipments_controller.rb +143 -0
  30. data/app/controllers/spree/api/v2/platform/shipping_categories_controller.rb +15 -0
  31. data/app/controllers/spree/api/v2/platform/shipping_methods_controller.rb +24 -0
  32. data/app/controllers/spree/api/v2/platform/states_controller.rb +19 -0
  33. data/app/controllers/spree/api/v2/platform/stock_items_controller.rb +19 -0
  34. data/app/controllers/spree/api/v2/platform/stock_locations_controller.rb +19 -0
  35. data/app/controllers/spree/api/v2/platform/store_credit_categories_controller.rb +15 -0
  36. data/app/controllers/spree/api/v2/platform/store_credit_types_controller.rb +15 -0
  37. data/app/controllers/spree/api/v2/platform/store_credits_controller.rb +19 -0
  38. data/app/controllers/spree/api/v2/platform/tax_categories_controller.rb +19 -0
  39. data/app/controllers/spree/api/v2/platform/tax_rates_controller.rb +23 -0
  40. data/app/controllers/spree/api/v2/platform/taxonomies_controller.rb +19 -0
  41. data/app/controllers/spree/api/v2/platform/taxons_controller.rb +25 -0
  42. data/app/controllers/spree/api/v2/platform/users_controller.rb +4 -0
  43. data/app/controllers/spree/api/v2/platform/variants_controller.rb +19 -0
  44. data/app/controllers/spree/api/v2/platform/webhooks/events_controller.rb +21 -0
  45. data/app/controllers/spree/api/v2/platform/webhooks/subscribers_controller.rb +21 -0
  46. data/app/controllers/spree/api/v2/platform/wished_items_controller.rb +19 -0
  47. data/app/controllers/spree/api/v2/platform/wishlists_controller.rb +19 -0
  48. data/app/controllers/spree/api/v2/platform/zones_controller.rb +19 -0
  49. data/app/controllers/spree/api/v2/resource_controller.rb +9 -4
  50. data/app/controllers/spree/api/v2/storefront/account/addresses_controller.rb +5 -3
  51. data/app/controllers/spree/api/v2/storefront/account/credit_cards_controller.rb +4 -1
  52. data/app/controllers/spree/api/v2/storefront/account/orders_controller.rb +4 -0
  53. data/app/controllers/spree/api/v2/storefront/cart_controller.rb +57 -29
  54. data/app/controllers/spree/api/v2/storefront/checkout_controller.rb +24 -0
  55. data/app/controllers/spree/api/v2/storefront/countries_controller.rb +7 -0
  56. data/app/controllers/spree/api/v2/storefront/digitals_controller.rb +54 -0
  57. data/app/controllers/spree/api/v2/storefront/products_controller.rb +12 -0
  58. data/app/controllers/spree/api/v2/storefront/wishlists_controller.rb +171 -0
  59. data/app/helpers/spree/api/v2/collection_options_helpers.rb +1 -1
  60. data/app/jobs/spree/webhooks/subscribers/make_request_job.rb +17 -0
  61. data/app/models/concerns/spree/webhooks/has_webhooks.rb +58 -0
  62. data/app/models/spree/api/webhooks/order_decorator.rb +43 -0
  63. data/app/models/spree/api/webhooks/payment_decorator.rb +26 -0
  64. data/app/models/spree/api/webhooks/product_decorator.rb +27 -0
  65. data/app/models/spree/api/webhooks/shipment_decorator.rb +21 -0
  66. data/app/models/spree/api/webhooks/stock_item_decorator.rb +43 -0
  67. data/app/models/spree/api/webhooks/stock_movement_decorator.rb +52 -0
  68. data/app/models/spree/api/webhooks/variant_decorator.rb +26 -0
  69. data/app/models/spree/api_configuration.rb +4 -2
  70. data/app/models/spree/oauth_access_grant.rb +7 -0
  71. data/app/models/spree/oauth_access_token.rb +7 -0
  72. data/app/models/spree/oauth_application.rb +15 -0
  73. data/app/models/spree/webhooks/base.rb +11 -0
  74. data/app/models/spree/webhooks/event.rb +12 -0
  75. data/app/models/spree/webhooks/subscriber.rb +57 -0
  76. data/app/presenters/spree/api/products/filters_presenter.rb +39 -0
  77. data/app/serializers/concerns/spree/api/v2/image_transformation_concern.rb +15 -0
  78. data/app/serializers/concerns/spree/api/v2/resource_serializer_concern.rb +19 -1
  79. data/app/serializers/concerns/spree/api/v2/taxon_image_transformation_concern.rb +15 -0
  80. data/app/serializers/spree/api/v2/base_serializer.rb +12 -5
  81. data/app/serializers/spree/api/v2/platform/address_serializer.rb +1 -1
  82. data/app/serializers/spree/api/v2/platform/adjustment_serializer.rb +20 -0
  83. data/app/serializers/spree/api/v2/platform/asset_serializer.rb +13 -0
  84. data/app/serializers/spree/api/v2/platform/calculator_serializer.rb +17 -0
  85. data/app/serializers/spree/api/v2/platform/classification_serializer.rb +1 -1
  86. data/app/serializers/spree/api/v2/platform/cms_page_serializer.rb +1 -1
  87. data/app/serializers/spree/api/v2/platform/cms_section_serializer.rb +8 -1
  88. data/app/serializers/spree/api/v2/platform/country_serializer.rb +1 -1
  89. data/app/serializers/spree/api/v2/platform/credit_card_serializer.rb +14 -0
  90. data/app/serializers/spree/api/v2/platform/customer_return_serializer.rb +17 -0
  91. data/app/serializers/spree/api/v2/platform/digital_link_serializer.rb +16 -0
  92. data/app/serializers/spree/api/v2/platform/digital_serializer.rb +30 -0
  93. data/app/serializers/spree/api/v2/platform/feature_page_serializer.rb +11 -0
  94. data/app/serializers/spree/api/v2/platform/homepage_serializer.rb +11 -0
  95. data/app/serializers/spree/api/v2/platform/icon_serializer.rb +16 -0
  96. data/app/serializers/spree/api/v2/platform/image_serializer.rb +3 -1
  97. data/app/serializers/spree/api/v2/platform/inventory_unit_serializer.rb +19 -0
  98. data/app/serializers/spree/api/v2/platform/line_item_serializer.rb +19 -0
  99. data/app/serializers/spree/api/v2/platform/log_entry_serializer.rb +13 -0
  100. data/app/serializers/spree/api/v2/platform/menu_item_serializer.rb +29 -6
  101. data/app/serializers/spree/api/v2/platform/menu_serializer.rb +1 -1
  102. data/app/serializers/spree/api/v2/platform/option_type_serializer.rb +1 -1
  103. data/app/serializers/spree/api/v2/platform/option_value_serializer.rb +1 -1
  104. data/app/serializers/spree/api/v2/platform/order_promotion_serializer.rb +14 -0
  105. data/app/serializers/spree/api/v2/platform/order_serializer.rb +31 -0
  106. data/app/serializers/spree/api/v2/platform/payment_capture_event_serializer.rb +13 -0
  107. data/app/serializers/spree/api/v2/platform/payment_method_serializer.rb +18 -0
  108. data/app/serializers/spree/api/v2/platform/payment_serializer.rb +21 -0
  109. data/app/serializers/spree/api/v2/platform/price_serializer.rb +19 -0
  110. data/app/serializers/spree/api/v2/platform/product_property_serializer.rb +1 -1
  111. data/app/serializers/spree/api/v2/platform/product_serializer.rb +18 -14
  112. data/app/serializers/spree/api/v2/platform/promotion_action_line_item_serializer.rb +14 -0
  113. data/app/serializers/spree/api/v2/platform/promotion_action_serializer.rb +19 -0
  114. data/app/serializers/spree/api/v2/platform/promotion_category_serializer.rb +13 -0
  115. data/app/serializers/spree/api/v2/platform/promotion_rule_serializer.rb +21 -0
  116. data/app/serializers/spree/api/v2/platform/promotion_serializer.rb +17 -0
  117. data/app/serializers/spree/api/v2/platform/property_serializer.rb +11 -0
  118. data/app/serializers/spree/api/v2/platform/prototype_serializer.rb +15 -0
  119. data/app/serializers/spree/api/v2/platform/refund_reason_serializer.rb +11 -0
  120. data/app/serializers/spree/api/v2/platform/refund_serializer.rb +16 -0
  121. data/app/serializers/spree/api/v2/platform/reimbursement_credit_serializer.rb +10 -0
  122. data/app/serializers/spree/api/v2/platform/reimbursement_serializer.rb +18 -0
  123. data/app/serializers/spree/api/v2/platform/reimbursement_type_serializer.rb +11 -0
  124. data/app/serializers/spree/api/v2/platform/return_authorization_reason_serializer.rb +11 -0
  125. data/app/serializers/spree/api/v2/platform/return_authorization_serializer.rb +17 -0
  126. data/app/serializers/spree/api/v2/platform/return_item_serializer.rb +16 -0
  127. data/app/serializers/spree/api/v2/platform/role_serializer.rb +11 -0
  128. data/app/serializers/spree/api/v2/platform/shipment_serializer.rb +22 -0
  129. data/app/serializers/spree/api/v2/platform/shipping_category_serializer.rb +11 -0
  130. data/app/serializers/spree/api/v2/platform/shipping_method_serializer.rb +16 -0
  131. data/app/serializers/spree/api/v2/platform/shipping_rate_serializer.rb +15 -0
  132. data/app/serializers/spree/api/v2/platform/standard_page_serializer.rb +11 -0
  133. data/app/serializers/spree/api/v2/platform/state_change_serializer.rb +13 -0
  134. data/app/serializers/spree/api/v2/platform/state_serializer.rb +1 -1
  135. data/app/serializers/spree/api/v2/platform/stock_item_serializer.rb +1 -3
  136. data/app/serializers/spree/api/v2/platform/stock_location_serializer.rb +2 -4
  137. data/app/serializers/spree/api/v2/platform/stock_movement_serializer.rb +11 -0
  138. data/app/serializers/spree/api/v2/platform/stock_transfer_serializer.rb +15 -0
  139. data/app/serializers/spree/api/v2/platform/store_credit_category_serializer.rb +12 -0
  140. data/app/serializers/spree/api/v2/platform/store_credit_event_serializer.rb +14 -0
  141. data/app/serializers/spree/api/v2/platform/store_credit_serializer.rb +18 -0
  142. data/app/serializers/spree/api/v2/platform/store_credit_type_serializer.rb +12 -0
  143. data/app/serializers/spree/api/v2/platform/store_serializer.rb +1 -1
  144. data/app/serializers/spree/api/v2/platform/tax_category_serializer.rb +2 -2
  145. data/app/serializers/spree/api/v2/platform/tax_rate_serializer.rb +14 -0
  146. data/app/serializers/spree/api/v2/platform/taxon_image_serializer.rb +3 -1
  147. data/app/serializers/spree/api/v2/platform/taxon_serializer.rb +1 -1
  148. data/app/serializers/spree/api/v2/platform/taxonomy_serializer.rb +1 -1
  149. data/app/serializers/spree/api/v2/platform/user_serializer.rb +17 -1
  150. data/app/serializers/spree/api/v2/platform/variant_serializer.rb +3 -2
  151. data/app/serializers/spree/api/v2/platform/webhooks/event_serializer.rb +15 -0
  152. data/app/serializers/spree/api/v2/platform/webhooks/subscriber_serializer.rb +13 -0
  153. data/app/serializers/spree/api/v2/platform/wished_item_serializer.rb +29 -0
  154. data/app/serializers/spree/api/v2/platform/wishlist_serializer.rb +19 -0
  155. data/app/serializers/spree/api/v2/platform/zone_member_serializer.rb +13 -0
  156. data/app/serializers/spree/api/v2/platform/zone_serializer.rb +13 -0
  157. data/app/serializers/spree/v2/storefront/address_serializer.rb +1 -1
  158. data/app/serializers/spree/v2/storefront/cart_serializer.rb +1 -1
  159. data/app/serializers/spree/v2/storefront/cms_section_serializer.rb +5 -1
  160. data/app/serializers/spree/v2/storefront/credit_card_serializer.rb +1 -1
  161. data/app/serializers/spree/v2/storefront/digital_link_serializer.rb +11 -0
  162. data/app/serializers/spree/v2/storefront/estimated_shipping_rate_serializer.rb +2 -2
  163. data/app/serializers/spree/v2/storefront/icon_serializer.rb +14 -0
  164. data/app/serializers/spree/v2/storefront/image_serializer.rb +3 -1
  165. data/app/serializers/spree/v2/storefront/line_item_serializer.rb +2 -1
  166. data/app/serializers/spree/v2/storefront/menu_item_serializer.rb +12 -6
  167. data/app/serializers/spree/v2/storefront/option_type_serializer.rb +1 -1
  168. data/app/serializers/spree/v2/storefront/option_value_serializer.rb +1 -1
  169. data/app/serializers/spree/v2/storefront/payment_method_serializer.rb +5 -1
  170. data/app/serializers/spree/v2/storefront/payment_serializer.rb +1 -1
  171. data/app/serializers/spree/v2/storefront/product_serializer.rb +19 -13
  172. data/app/serializers/spree/v2/storefront/promotion_serializer.rb +1 -1
  173. data/app/serializers/spree/v2/storefront/shipment_serializer.rb +2 -1
  174. data/app/serializers/spree/v2/storefront/store_credit_serializer.rb +1 -1
  175. data/app/serializers/spree/v2/storefront/store_serializer.rb +1 -1
  176. data/app/serializers/spree/v2/storefront/taxon_image_serializer.rb +3 -1
  177. data/app/serializers/spree/v2/storefront/taxon_serializer.rb +1 -1
  178. data/app/serializers/spree/v2/storefront/taxonomy_serializer.rb +1 -1
  179. data/app/serializers/spree/v2/storefront/user_serializer.rb +1 -1
  180. data/app/serializers/spree/v2/storefront/variant_serializer.rb +1 -1
  181. data/app/serializers/spree/v2/storefront/wished_item_serializer.rb +29 -0
  182. data/app/serializers/spree/v2/storefront/wishlist_serializer.rb +17 -0
  183. data/app/services/spree/webhooks/subscribers/handle_request.rb +73 -0
  184. data/app/services/spree/webhooks/subscribers/make_request.rb +82 -0
  185. data/app/services/spree/webhooks/subscribers/queue_requests.rb +17 -0
  186. data/app/services/spree/webhooks.rb +13 -0
  187. data/config/i18n-tasks.yml +40 -0
  188. data/config/initializers/doorkeeper.rb +12 -12
  189. data/config/initializers/rabl.rb +2 -2
  190. data/config/locales/en.yml +29 -27
  191. data/config/routes.rb +86 -60
  192. data/db/migrate/20210902162826_create_spree_webhooks_tables.rb +16 -0
  193. data/db/migrate/20210919183228_enable_polymorphic_resource_owner.rb +21 -0
  194. data/db/migrate/20211025162826_create_spree_webhooks_events.rb +14 -0
  195. data/docs/oauth/index.yml +126 -33
  196. data/docs/v2/platform/index.yaml +20086 -524
  197. data/docs/v2/storefront/index.yaml +15094 -4517
  198. data/{app/models/spree → lib/spree/api}/api_dependencies.rb +61 -5
  199. data/lib/spree/api/engine.rb +24 -1
  200. data/lib/spree/api/testing_support/factories/oauth_application_factory.rb +6 -0
  201. data/lib/spree/api/testing_support/factories/webhook_event_factory.rb +27 -0
  202. data/lib/spree/api/testing_support/factories/webhook_subscriber_factory.rb +13 -0
  203. data/lib/spree/api/testing_support/factories.rb +3 -0
  204. data/lib/spree/api/testing_support/helpers.rb +1 -1
  205. data/lib/spree/api/testing_support/jobs.rb +18 -0
  206. data/lib/spree/api/testing_support/matchers/webhooks.rb +67 -0
  207. data/lib/spree/api/testing_support/serializers.rb +25 -0
  208. data/lib/spree/api/testing_support/spree_webhooks.rb +9 -0
  209. data/lib/spree/api/testing_support/v2/base.rb +1 -1
  210. data/lib/spree/api/testing_support/v2/current_order.rb +34 -1
  211. data/lib/spree/api/testing_support/v2/platform_contexts.rb +110 -51
  212. data/lib/spree/api/testing_support/v2/serializers_params.rb +3 -1
  213. data/lib/spree/api.rb +1 -0
  214. data/spec/fixtures/files/icon_256x256.jpg +0 -0
  215. data/spree_api.gemspec +16 -15
  216. metadata +182 -29
  217. data/app/controllers/spree/api/errors_controller.rb +0 -9
@@ -0,0 +1,171 @@
1
+ module Spree
2
+ module Api
3
+ module V2
4
+ module Storefront
5
+ class WishlistsController < ::Spree::Api::V2::ResourceController
6
+ before_action :require_spree_current_user, except: [:show]
7
+ before_action :ensure_valid_quantity, only: [:add_item, :set_item_quantity]
8
+
9
+ def show
10
+ spree_authorize! :show, resource
11
+ super
12
+ end
13
+
14
+ def create
15
+ spree_authorize! :create, Spree::Wishlist
16
+
17
+ @wishlist = spree_current_user.wishlists.new(wishlist_attributes)
18
+
19
+ ensure_current_store(@wishlist)
20
+
21
+ @wishlist.save
22
+
23
+ if @wishlist.persisted?
24
+ render_serialized_payload(201) { serialize_resource(@wishlist) }
25
+ else
26
+ render_error_payload(@wishlist.errors.full_messages.to_sentence)
27
+ end
28
+ end
29
+
30
+ def update
31
+ authorize! :update, resource
32
+
33
+ resource.update wishlist_attributes
34
+
35
+ if resource.errors.empty?
36
+ render_serialized_payload { serialize_resource(resource) }
37
+ else
38
+ render_error_payload(resource.errors.full_messages.to_sentence)
39
+ end
40
+ end
41
+
42
+ def destroy
43
+ authorize! :destroy, resource
44
+
45
+ if resource.destroy
46
+ head 204
47
+ else
48
+ render_error_payload(I18n.t('spree.api.v2.wishlist.errors.the_wishlist_could_not_be_destroyed'))
49
+ end
50
+ end
51
+
52
+ def default
53
+ spree_authorize! :create, Spree::Wishlist
54
+
55
+ @default_wishlist = spree_current_user.default_wishlist_for_store(current_store)
56
+
57
+ render_serialized_payload { serialize_resource(@default_wishlist) }
58
+ end
59
+
60
+ def add_item
61
+ spree_authorize! :create, Spree::WishedItem
62
+
63
+ if resource.wished_items.present? && resource.wished_items.detect { |wv| wv.variant_id.to_s == params[:variant_id].to_s }.present?
64
+ @wished_item = resource.wished_items.detect { |wi| wi.variant_id.to_s == params[:variant_id].to_s }
65
+ @wished_item.quantity = params[:quantity]
66
+ else
67
+ @wished_item = Spree::WishedItem.new(params.permit(:quantity, :variant_id))
68
+ @wished_item.wishlist = resource
69
+ @wished_item.save
70
+ end
71
+
72
+ resource.reload
73
+
74
+ if @wished_item.persisted?
75
+ render_serialized_payload { serialize_wished_item(@wished_item) }
76
+ else
77
+ render_error_payload(resource.errors.full_messages.to_sentence)
78
+ end
79
+ end
80
+
81
+ def set_item_quantity
82
+ spree_authorize! :update, wished_item
83
+
84
+ wished_item.update(params.permit(:quantity))
85
+
86
+ if wished_item.errors.empty?
87
+ render_serialized_payload { serialize_wished_item(wished_item) }
88
+ else
89
+ render_error_payload(resource.errors.full_messages.to_sentence)
90
+ end
91
+ end
92
+
93
+ def remove_item
94
+ spree_authorize! :destroy, wished_item
95
+
96
+ if wished_item.destroy
97
+ render_serialized_payload { serialize_wished_item(wished_item) }
98
+ else
99
+ render_error_payload(resource.errors.full_messages.to_sentence)
100
+ end
101
+ end
102
+
103
+ private
104
+
105
+ def scope(skip_cancancan: true)
106
+ if action_name == 'show'
107
+ super
108
+ else
109
+ super.where(user: spree_current_user)
110
+ end
111
+ end
112
+
113
+ def resource
114
+ @resource ||= scope.find_by(token: params[:id])
115
+ end
116
+
117
+ def model_class
118
+ Spree::Wishlist
119
+ end
120
+
121
+ def resource_serializer
122
+ ::Spree::V2::Storefront::WishlistSerializer
123
+ end
124
+
125
+ def collection_serializer
126
+ resource_serializer
127
+ end
128
+
129
+ def wishlist_attributes
130
+ params.require(:wishlist).permit(permitted_wishlist_attributes)
131
+ end
132
+
133
+ def wished_item_attributes
134
+ params.permit(permitted_wished_item_attributes)
135
+ end
136
+
137
+ def wished_item
138
+ @wished_item ||= resource.wished_items.find(params[:item_id])
139
+ end
140
+
141
+ def serialize_wished_item(wished_item)
142
+ ::Spree::V2::Storefront::WishedItemSerializer.new(
143
+ wished_item,
144
+ params: serializer_params,
145
+ include: resource_includes,
146
+ fields: sparse_fields
147
+ ).serializable_hash
148
+ end
149
+
150
+ def serializer_params
151
+ super.merge(is_variant_included: params[:is_variant_included])
152
+ end
153
+
154
+ def render_error_item_quantity
155
+ render json: { error: I18n.t('spree.api.v2.wishlist.wrong_quantity') }, status: 422
156
+ end
157
+
158
+ def ensure_valid_quantity
159
+ return render_error_item_quantity if params[:quantity].present? && params[:quantity].to_i <= 0
160
+
161
+ params[:quantity] = if params[:quantity].present?
162
+ params[:quantity].to_i
163
+ else
164
+ 1
165
+ end
166
+ end
167
+ end
168
+ end
169
+ end
170
+ end
171
+ end
@@ -23,7 +23,7 @@ module Spree
23
23
  # leaving this method in public scope so it's still possible to modify
24
24
  # those params to support non-standard non-JSON API parameters
25
25
  def collection_permitted_params
26
- params.permit(:format, :page, :per_page, :sort, :include, :fields, filter: {})
26
+ params.permit(:format, :page, :per_page, :sort, :include, fields: {}, filter: {})
27
27
  end
28
28
 
29
29
  private
@@ -0,0 +1,17 @@
1
+ module Spree
2
+ module Webhooks
3
+ module Subscribers
4
+ class MakeRequestJob < Spree::BaseJob
5
+ queue_as :spree_webhooks
6
+
7
+ def perform(webhook_payload_body, event_name, subscriber)
8
+ Spree::Webhooks::Subscribers::HandleRequest.new(
9
+ event_name: event_name,
10
+ subscriber: subscriber,
11
+ webhook_payload_body: webhook_payload_body
12
+ ).call
13
+ end
14
+ end
15
+ end
16
+ end
17
+ end
@@ -0,0 +1,58 @@
1
+ module Spree
2
+ module Webhooks
3
+ module HasWebhooks
4
+ extend ActiveSupport::Concern
5
+
6
+ included do
7
+ after_create_commit(proc { queue_webhooks_requests!(inferred_event_name(:create)) })
8
+ after_destroy_commit(proc { queue_webhooks_requests!(inferred_event_name(:delete)) })
9
+ after_update_commit(proc { queue_webhooks_requests!(inferred_event_name(:update)) })
10
+
11
+ def queue_webhooks_requests!(event_name)
12
+ return if disable_spree_webhooks? || webhook_payload_body.blank?
13
+ return if update_event?(event_name) && updating_only_timestamps?
14
+
15
+ Spree::Webhooks::Subscribers::QueueRequests.call(event_name: event_name, webhook_payload_body: webhook_payload_body)
16
+ end
17
+
18
+ def self.default_webhook_events
19
+ model_name = name.demodulize.tableize.singularize
20
+ %W[#{model_name}.create #{model_name}.delete #{model_name}.update]
21
+ end
22
+
23
+ def self.supported_webhook_events
24
+ events = default_webhook_events
25
+ events += custom_webhook_events if respond_to?(:custom_webhook_events)
26
+ events
27
+ end
28
+ end
29
+
30
+ private
31
+
32
+ def webhook_payload_body
33
+ resource_serializer.new(self).serializable_hash.to_json
34
+ end
35
+
36
+ def inferred_event_name(operation)
37
+ "#{self.class.name.demodulize.tableize.singularize}.#{operation}"
38
+ end
39
+
40
+ def resource_serializer
41
+ demodulized_class_name = self.class.to_s.demodulize
42
+ "Spree::Api::V2::Platform::#{demodulized_class_name}Serializer".constantize
43
+ end
44
+
45
+ def updating_only_timestamps?
46
+ (saved_changes.keys - %w[created_at updated_at deleted_at]).empty?
47
+ end
48
+
49
+ def update_event?(event_name)
50
+ event_name.end_with?('.update')
51
+ end
52
+
53
+ def disable_spree_webhooks?
54
+ ENV['DISABLE_SPREE_WEBHOOKS'] == 'true'
55
+ end
56
+ end
57
+ end
58
+ end
@@ -0,0 +1,43 @@
1
+ module Spree
2
+ module Api
3
+ module Webhooks
4
+ module OrderDecorator
5
+ def self.prepended(base)
6
+ def base.custom_webhook_events
7
+ %w[order.canceled order.placed order.resumed order.shipped]
8
+ end
9
+
10
+ base.after_update_commit :queue_webhooks_requests_for_order_resumed!
11
+ end
12
+
13
+ def after_cancel
14
+ super
15
+ queue_webhooks_requests!('order.canceled')
16
+ end
17
+
18
+ def finalize!
19
+ super
20
+ queue_webhooks_requests!('order.placed')
21
+ end
22
+
23
+ def after_resume
24
+ super
25
+ queue_webhooks_requests!('order.resumed')
26
+ self.state_machine_resumed = false
27
+ end
28
+
29
+ private
30
+
31
+ def queue_webhooks_requests_for_order_resumed!
32
+ return if state_machine_resumed?
33
+ return unless state_previously_changed?
34
+ return unless state_previous_change&.last == 'resumed'
35
+
36
+ queue_webhooks_requests!('order.resumed')
37
+ end
38
+ end
39
+ end
40
+ end
41
+ end
42
+
43
+ Spree::Order.prepend(Spree::Api::Webhooks::OrderDecorator)
@@ -0,0 +1,26 @@
1
+ module Spree
2
+ module Api
3
+ module Webhooks
4
+ module PaymentDecorator
5
+ def self.prepended(base)
6
+ def base.custom_webhook_events
7
+ %w[payment.paid payment.voided]
8
+ end
9
+ end
10
+
11
+ def after_void
12
+ super
13
+ queue_webhooks_requests!('payment.voided')
14
+ end
15
+
16
+ def after_completed
17
+ super
18
+ queue_webhooks_requests!('payment.paid')
19
+ order.queue_webhooks_requests!('order.paid') if order.paid?
20
+ end
21
+ end
22
+ end
23
+ end
24
+ end
25
+
26
+ Spree::Payment.prepend(Spree::Api::Webhooks::PaymentDecorator)
@@ -0,0 +1,27 @@
1
+ module Spree
2
+ module Api
3
+ module Webhooks
4
+ module ProductDecorator
5
+ def self.prepended(base)
6
+ def base.custom_webhook_events
7
+ %w[product.back_in_stock product.backorderable product.discontinued product.out_of_stock]
8
+ end
9
+
10
+ base.after_update_commit :queue_webhooks_requests_for_product_discontinued!
11
+ end
12
+
13
+ private
14
+
15
+ def queue_webhooks_requests_for_product_discontinued!
16
+ return unless discontinue_on_previously_changed?
17
+ return if (change = discontinue_on_previous_change).blank? || change.last.blank?
18
+
19
+ queue_webhooks_requests!('product.discontinued')
20
+ end
21
+ end
22
+ end
23
+ end
24
+ end
25
+
26
+ Spree::Product.prepend(Spree::Api::Webhooks::ProductDecorator)
27
+
@@ -0,0 +1,21 @@
1
+ module Spree
2
+ module Api
3
+ module Webhooks
4
+ module ShipmentDecorator
5
+ def self.prepended(base)
6
+ def base.custom_webhook_events
7
+ %w[shipment.shipped]
8
+ end
9
+ end
10
+
11
+ def after_ship
12
+ super
13
+ queue_webhooks_requests!('shipment.shipped')
14
+ order.queue_webhooks_requests!('order.shipped') if order.fully_shipped?
15
+ end
16
+ end
17
+ end
18
+ end
19
+ end
20
+
21
+ Spree::Shipment.prepend(Spree::Api::Webhooks::ShipmentDecorator)
@@ -0,0 +1,43 @@
1
+ module Spree
2
+ module Api
3
+ module Webhooks
4
+ module StockItemDecorator
5
+ def self.prepended(base)
6
+ base.around_save :queue_webhooks_requests_for_variant_backorderable!
7
+ base.around_save :queue_webhooks_requests_for_product_backorderable!
8
+ end
9
+
10
+ private
11
+
12
+ def queue_webhooks_requests_for_variant_backorderable!
13
+ was_out_of_stock = !variant.in_stock_or_backorderable?
14
+ was_not_backorderable = !variant_backorderable?
15
+ yield
16
+ if was_out_of_stock && was_not_backorderable && variant_backorderable?
17
+ reload
18
+ variant.queue_webhooks_requests!('variant.backorderable')
19
+ end
20
+ end
21
+
22
+ def queue_webhooks_requests_for_product_backorderable!
23
+ product_was_out_of_stock = !product.any_variant_in_stock_or_backorderable?
24
+ product_was_not_backorderable = !product_backorderable?
25
+ yield
26
+ if product_was_out_of_stock && product_was_not_backorderable && product_backorderable?
27
+ variant.product.queue_webhooks_requests!('product.backorderable')
28
+ end
29
+ end
30
+
31
+ def product_backorderable?
32
+ Spree::StockItem.exists?(backorderable: true, variant_id: variant.product.variants.ids)
33
+ end
34
+
35
+ def variant_backorderable?
36
+ variant.stock_items.exists?(backorderable: true)
37
+ end
38
+ end
39
+ end
40
+ end
41
+ end
42
+
43
+ Spree::StockItem.prepend(Spree::Api::Webhooks::StockItemDecorator)
@@ -0,0 +1,52 @@
1
+ module Spree
2
+ module Api
3
+ module Webhooks
4
+ module StockMovementDecorator
5
+ def self.prepended(base)
6
+ base.around_save :queue_webhooks_requests_for_variant_out_of_stock!
7
+ base.around_save :queue_webhooks_requests_for_variant_back_in_stock!
8
+ base.around_save :queue_webhooks_requests_for_product_out_of_stock!
9
+ base.around_save :queue_webhooks_requests_for_product_back_in_stock!
10
+ end
11
+
12
+ private
13
+
14
+ def queue_webhooks_requests_for_variant_out_of_stock!
15
+ variant_in_stock_before_update = variant.in_stock_or_backorderable?
16
+ yield
17
+ if variant_in_stock_before_update && !variant.in_stock_or_backorderable?
18
+ reload
19
+ stock_item.variant.queue_webhooks_requests!('variant.out_of_stock')
20
+ end
21
+ end
22
+
23
+ def queue_webhooks_requests_for_variant_back_in_stock!
24
+ variant_was_out_of_stock = !variant.in_stock_or_backorderable?
25
+ yield
26
+ if variant_was_out_of_stock && variant.in_stock_or_backorderable?
27
+ reload
28
+ variant.queue_webhooks_requests!('variant.back_in_stock')
29
+ end
30
+ end
31
+
32
+ def queue_webhooks_requests_for_product_back_in_stock!
33
+ product_was_out_of_stock = !product.any_variant_in_stock_or_backorderable?
34
+ yield
35
+ if product_was_out_of_stock && product.any_variant_in_stock_or_backorderable?
36
+ product.queue_webhooks_requests!('product.back_in_stock')
37
+ end
38
+ end
39
+
40
+ def queue_webhooks_requests_for_product_out_of_stock!
41
+ product_was_in_stock = product.any_variant_in_stock_or_backorderable?
42
+ yield
43
+ if product_was_in_stock && !product.any_variant_in_stock_or_backorderable?
44
+ product.queue_webhooks_requests!('product.out_of_stock')
45
+ end
46
+ end
47
+ end
48
+ end
49
+ end
50
+ end
51
+
52
+ Spree::StockMovement.prepend(Spree::Api::Webhooks::StockMovementDecorator)
@@ -0,0 +1,26 @@
1
+ module Spree
2
+ module Api
3
+ module Webhooks
4
+ module VariantDecorator
5
+ def self.prepended(base)
6
+ def base.custom_webhook_events
7
+ %w[variant.back_in_stock variant.backorderable variant.discontinued variant.out_of_stock]
8
+ end
9
+
10
+ base.after_update_commit :queue_webhooks_requests_for_variant_discontinued!
11
+ end
12
+
13
+ private
14
+
15
+ def queue_webhooks_requests_for_variant_discontinued!
16
+ return unless discontinue_on_previously_changed?
17
+ return if (change = discontinue_on_previous_change).blank? || change.last.blank?
18
+
19
+ queue_webhooks_requests!('variant.discontinued')
20
+ end
21
+ end
22
+ end
23
+ end
24
+ end
25
+
26
+ Spree::Variant.prepend(Spree::Api::Webhooks::VariantDecorator)
@@ -1,8 +1,10 @@
1
1
  module Spree
2
2
  class ApiConfiguration < Preferences::Configuration
3
3
  preference :requires_authentication, :boolean, default: true
4
- preference :api_v2_cache_ttl, :integer, default: 3600 # 1 hour in seconds
4
+ preference :api_v2_serializers_cache_ttl, :integer, default: 3600 # 1 hour in seconds
5
+ preference :api_v2_collection_cache_ttl, :integer, default: 3600 # 1 hour in seconds
6
+ preference :api_v2_collection_cache_namespace, :string, default: 'api_v2_collection_cache'
5
7
  preference :api_v2_content_type, :string, default: 'application/vnd.api+json'
6
- preference :api_v2_per_page_limit, :integer, default: 100
8
+ preference :api_v2_per_page_limit, :integer, default: 500
7
9
  end
8
10
  end
@@ -0,0 +1,7 @@
1
+ module Spree
2
+ class OauthAccessGrant < Base
3
+ include ::Doorkeeper::Orm::ActiveRecord::Mixins::AccessGrant
4
+
5
+ self.table_name = 'spree_oauth_access_grants'
6
+ end
7
+ end
@@ -0,0 +1,7 @@
1
+ module Spree
2
+ class OauthAccessToken < Base
3
+ include ::Doorkeeper::Orm::ActiveRecord::Mixins::AccessToken
4
+
5
+ self.table_name = 'spree_oauth_access_tokens'
6
+ end
7
+ end
@@ -0,0 +1,15 @@
1
+ module Spree
2
+ class OauthApplication < Base
3
+ include ::Doorkeeper::Orm::ActiveRecord::Mixins::Application
4
+
5
+ self.table_name = 'spree_oauth_applications'
6
+
7
+ before_validation :set_blank_for_redirect_uri
8
+
9
+ private
10
+
11
+ def set_blank_for_redirect_uri
12
+ self.redirect_uri = '' if redirect_uri.nil?
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,11 @@
1
+ module Spree
2
+ module Webhooks
3
+ def self.table_name_prefix
4
+ 'spree_webhooks_'
5
+ end
6
+
7
+ class Base < Spree::Base
8
+ self.abstract_class = true
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,12 @@
1
+ module Spree
2
+ module Webhooks
3
+ class Event < Spree::Webhooks::Base
4
+ validates :name, :subscriber, presence: true
5
+
6
+ belongs_to :subscriber, inverse_of: :events, optional: false
7
+
8
+ self.whitelisted_ransackable_associations = %w[subscriber]
9
+ self.whitelisted_ransackable_attributes = %w[name request_errors response_code success url]
10
+ end
11
+ end
12
+ end
@@ -0,0 +1,57 @@
1
+ module Spree
2
+ module Webhooks
3
+ class Subscriber < Spree::Webhooks::Base
4
+ has_many :events, inverse_of: :subscriber
5
+
6
+ validates :url, 'spree/url': true, presence: true
7
+
8
+ validate :check_uri_path
9
+
10
+ self.whitelisted_ransackable_attributes = %w[active subscriptions url]
11
+ self.whitelisted_ransackable_associations = %w[event]
12
+
13
+ scope :active, -> { where(active: true) }
14
+ scope :inactive, -> { where(active: false) }
15
+
16
+ before_save :parse_subscriptions
17
+
18
+ def self.with_urls_for(event)
19
+ where(
20
+ case ActiveRecord::Base.connection.adapter_name
21
+ when 'Mysql2'
22
+ ["('*' MEMBER OF(subscriptions) OR ? MEMBER OF(subscriptions))", event]
23
+ when 'PostgreSQL'
24
+ ["subscriptions @> '[\"*\"]' OR subscriptions @> ?", [event].to_json]
25
+ end
26
+ )
27
+ end
28
+
29
+ def self.supported_events
30
+ Spree::Base.descendants.
31
+ select { |model| model.included_modules.include? Spree::Webhooks::HasWebhooks }.
32
+ to_h do |model|
33
+ model_name = model.name.demodulize.underscore.to_sym
34
+ [model_name, model.supported_webhook_events]
35
+ end
36
+ end
37
+
38
+ private
39
+
40
+ def check_uri_path
41
+ uri = begin
42
+ URI.parse(url)
43
+ rescue URI::InvalidURIError
44
+ return false
45
+ end
46
+
47
+ errors.add(:url, 'the URL must have a path') if uri.blank? || uri.path.blank?
48
+ end
49
+
50
+ def parse_subscriptions
51
+ return if subscriptions.blank? || subscriptions.is_a?(Array)
52
+
53
+ self.subscriptions = JSON.parse(subscriptions)
54
+ end
55
+ end
56
+ end
57
+ end
@@ -0,0 +1,39 @@
1
+ module Spree
2
+ module Api
3
+ module Products
4
+ class FiltersPresenter
5
+ def initialize(current_store, current_currency, params)
6
+ @products_for_filters = find_products_for_filters(current_store, current_currency, params)
7
+ end
8
+
9
+ def to_h
10
+ option_values = Spree::OptionValues::FindAvailable.new(products_scope: products_for_filters).execute
11
+ option_values_presenters = Spree::Filters::OptionsPresenter.new(option_values_scope: option_values).to_a
12
+ product_properties = Spree::ProductProperties::FindAvailable.new(products_scope: products_for_filters).execute
13
+ product_properties_presenters = Spree::Filters::PropertiesPresenter.new(product_properties_scope: product_properties).to_a
14
+ {
15
+ option_types: option_values_presenters.map(&:to_h),
16
+ product_properties: product_properties_presenters.map(&:to_h)
17
+ }
18
+ end
19
+
20
+ private
21
+
22
+ attr_reader :products_for_filters
23
+
24
+ def find_products_for_filters(current_store, current_currency, params)
25
+ current_taxons = find_current_taxons(current_store, params)
26
+ current_store.products.active(current_currency).in_taxons(current_taxons)
27
+ end
28
+
29
+ def find_current_taxons(current_store, params)
30
+ taxons_param = params.dig(:filter, :taxons)
31
+ return nil if taxons_param.nil? || taxons_param.to_s.blank?
32
+
33
+ taxon_ids = taxons_param.to_s.split(',')
34
+ current_store.taxons.where(id: taxon_ids)
35
+ end
36
+ end
37
+ end
38
+ end
39
+ end