solidus_subscriptions-alpha 0.0.3 → 0.0.4

Sign up to get free protection for your applications and to get access to all the features.
Files changed (99) hide show
  1. checksums.yaml +4 -4
  2. data/LICENSE +26 -26
  3. data/README.md +130 -130
  4. data/Rakefile +28 -28
  5. data/app/assets/javascripts/spree/backend/solidus_subscriptions.js +1 -1
  6. data/app/assets/javascripts/spree/frontend/solidus_subscriptions.js +1 -1
  7. data/app/assets/stylesheets/spree/backend/solidus_subscriptions.css +4 -4
  8. data/app/assets/stylesheets/spree/frontend/solidus_subscriptions.css +4 -4
  9. data/app/controllers/solidus_subscriptions/api/v1/line_items_controller.rb +35 -35
  10. data/app/controllers/solidus_subscriptions/api/v1/subscriptions_controller.rb +44 -44
  11. data/app/controllers/spree/admin/subscriptions_controller.rb +66 -66
  12. data/app/decorators/spree/controllers/api/line_items/create_subscription_line_items.rb +28 -28
  13. data/app/decorators/spree/controllers/orders/create_subscription_line_items.rb +33 -33
  14. data/app/decorators/spree/line_items/subscription_line_items_association.rb +22 -22
  15. data/app/decorators/spree/orders/after_create.rb +15 -15
  16. data/app/decorators/spree/orders/finalize_creates_subscriptions.rb +19 -19
  17. data/app/decorators/spree/orders/subscription_line_items_association.rb +15 -15
  18. data/app/decorators/spree/products/subscribable_decorator.rb +11 -11
  19. data/app/decorators/spree/users/have_many_subscriptions.rb +18 -18
  20. data/app/decorators/spree/variant_pretty_name.rb +13 -13
  21. data/app/jobs/solidus_subscriptions/process_installments_job.rb +22 -22
  22. data/app/models/solidus_subscriptions/checkout.rb +141 -141
  23. data/app/models/solidus_subscriptions/dispatcher.rb +32 -32
  24. data/app/models/solidus_subscriptions/failure_dispatcher.rb +19 -19
  25. data/app/models/solidus_subscriptions/installment.rb +126 -126
  26. data/app/models/solidus_subscriptions/installment_detail.rb +23 -23
  27. data/app/models/solidus_subscriptions/interval.rb +24 -24
  28. data/app/models/solidus_subscriptions/line_item.rb +98 -98
  29. data/app/models/solidus_subscriptions/line_item_builder.rb +44 -44
  30. data/app/models/solidus_subscriptions/order_builder.rb +40 -40
  31. data/app/models/solidus_subscriptions/out_of_stock_dispatcher.rb +19 -19
  32. data/app/models/solidus_subscriptions/payment_failed_dispatcher.rb +23 -23
  33. data/app/models/solidus_subscriptions/subscription.rb +217 -217
  34. data/app/models/solidus_subscriptions/subscription_generator.rb +60 -60
  35. data/app/models/solidus_subscriptions/subscription_line_item_builder.rb +21 -21
  36. data/app/models/solidus_subscriptions/subscription_order_promotion_rule.rb +25 -25
  37. data/app/models/solidus_subscriptions/subscription_promotion_rule.rb +38 -38
  38. data/app/models/solidus_subscriptions/success_dispatcher.rb +16 -16
  39. data/app/models/solidus_subscriptions/unsubscribable_error.rb +17 -17
  40. data/app/models/solidus_subscriptions/user_mismatch_error.rb +15 -15
  41. data/app/overrides/views/admin_subscribable_checkbox.rb +13 -13
  42. data/app/overrides/views/admin_subscriptions_menu_link.rb +8 -8
  43. data/app/overrides/views/subscription_line_item_fields.rb +6 -6
  44. data/app/views/spree/admin/products/_subscribable_checkbox.html.erb +8 -8
  45. data/app/views/spree/admin/promotions/rules/_subscription_order_promotion_rule.html.erb +0 -0
  46. data/app/views/spree/admin/promotions/rules/_subscription_promotion_rule.html.erb +0 -0
  47. data/app/views/spree/admin/shared/_no_objects_found.html.erb +4 -4
  48. data/app/views/spree/admin/shared/_subscription_tab.html.erb +3 -3
  49. data/app/views/spree/admin/solidus_subscriptions/subscriptions/_subscription.html.erb +66 -66
  50. data/app/views/spree/admin/subscriptions/_form.html.erb +81 -81
  51. data/app/views/spree/admin/subscriptions/_legacy_form.html.erb +81 -81
  52. data/app/views/spree/admin/subscriptions/_legacy_sidebar.html.erb +28 -28
  53. data/app/views/spree/admin/subscriptions/edit.html.erb +21 -21
  54. data/app/views/spree/admin/subscriptions/index.html.erb +119 -119
  55. data/app/views/spree/admin/subscriptions/new.html.erb +9 -9
  56. data/app/views/spree/admin/variants/_subscribable_checkbox.html.erb +6 -6
  57. data/app/views/spree/frontend/products/_subscription_line_item_fields.html.erb +30 -30
  58. data/config/locales/en.yml +91 -91
  59. data/config/routes.rb +25 -25
  60. data/db/migrate/20160825164850_create_solidus_subscriptions_subscriptions.rb +11 -11
  61. data/db/migrate/20160825173548_create_solidus_subscriptions_line_items.rb +17 -17
  62. data/db/migrate/20160825202248_create_solidus_subscriptions_installments.rb +23 -23
  63. data/db/migrate/20160825211202_create_solidus_subscriptions_installment_details.rb +22 -22
  64. data/db/migrate/20160825214240_add_subscribable_to_spree_variants.rb +5 -5
  65. data/db/migrate/20160829201653_change_subscription_line_items_installments_to_max_installments.rb +5 -5
  66. data/db/migrate/20160902220242_remove_state_from_solidus_susbscriptions_installment_details.rb +5 -5
  67. data/db/migrate/20160902220604_add_successful_to_solidus_subscriptions_installment_details.rb +5 -5
  68. data/db/migrate/20160902221218_add_message_to_solidus_subscriptions_installment_details.rb +5 -5
  69. data/db/migrate/20160922164101_add_interval_length_and_units_to_subscription_line_items.rb +8 -8
  70. data/db/migrate/20161006191003_add_skip_count_to_solidus_subscriptions_subscriptions.rb +5 -5
  71. data/db/migrate/20161006191127_add_successive_skip_count_to_solidus_subscriptions_subscriptions.rb +5 -5
  72. data/db/migrate/20161014212649_allow_spree_line_item_id_to_be_null.rb +5 -5
  73. data/db/migrate/20161017155749_add_order_id_to_solidus_subscriptions_installment_details.rb +6 -6
  74. data/db/migrate/20161017175509_remove_order_id_from_solidus_subscriptions_installments.rb +6 -6
  75. data/db/migrate/20161017201944_add_subscription_order_to_spree_orders.rb +5 -5
  76. data/db/migrate/20161221155142_add_store_to_solidus_subscriptions_subscriptions.rb +6 -6
  77. data/db/migrate/20161223152905_add_address_id_to_solidus_subscriptions_subscriptions.rb +7 -7
  78. data/db/migrate/20170106224713_change_line_item_max_installments_to_end_date.rb +6 -6
  79. data/db/migrate/20170111224458_change_subscription_actionable_date_to_datetime.rb +5 -5
  80. data/db/migrate/20170111232801_change_inteval_actionable_date_to_datetime.rb +5 -5
  81. data/db/migrate/20170112012407_add_config_options_to_subscriptions.rb +7 -7
  82. data/lib/generators/solidus_subscriptions/install/install_generator.rb +30 -30
  83. data/lib/solidus_subscriptions.rb +6 -6
  84. data/lib/solidus_subscriptions/ability.rb +19 -19
  85. data/lib/solidus_subscriptions/config.rb +97 -97
  86. data/lib/solidus_subscriptions/engine.rb +56 -56
  87. data/lib/solidus_subscriptions/permitted_attributes.rb +36 -36
  88. data/lib/solidus_subscriptions/processor.rb +108 -108
  89. data/lib/solidus_subscriptions/testing_support/factories.rb +5 -5
  90. data/lib/solidus_subscriptions/testing_support/factories/installment_detail_factory.rb +7 -7
  91. data/lib/solidus_subscriptions/testing_support/factories/installment_factory.rb +21 -21
  92. data/lib/solidus_subscriptions/testing_support/factories/line_item_factory.rb +18 -18
  93. data/lib/solidus_subscriptions/testing_support/factories/spree/line_item_factory.rb +17 -17
  94. data/lib/solidus_subscriptions/testing_support/factories/spree/order_factory.rb +18 -18
  95. data/lib/solidus_subscriptions/testing_support/factories/spree_modification_factory.rb +8 -8
  96. data/lib/solidus_subscriptions/testing_support/factories/subscription_factory.rb +43 -43
  97. data/lib/solidus_subscriptions/version.rb +3 -3
  98. data/lib/tasks/process_subscriptions.rake +6 -6
  99. metadata +3 -3
@@ -1,217 +1,217 @@
1
- # The subscription class is responsable for grouping together the
2
- # information required for the system to place a subscriptions order on
3
- # behalf of a specific user.
4
- module SolidusSubscriptions
5
- class Subscription < ActiveRecord::Base
6
- include Interval
7
-
8
- PROCESSING_STATES = [:pending, :failed, :success]
9
-
10
- belongs_to :user, class_name: Spree.user_class
11
- has_many :line_items, class_name: 'SolidusSubscriptions::LineItem', inverse_of: :subscription
12
- has_many :installments, class_name: 'SolidusSubscriptions::Installment'
13
- belongs_to :store, class_name: 'Spree::Store'
14
- belongs_to :shipping_address, class_name: 'Spree::Address'
15
-
16
- validates :user, presence: :true
17
- validates :skip_count, :successive_skip_count, presence: true, numericality: { greater_than_or_equal_to: 0 }
18
- validates :interval_length, numericality: { greater_than: 0 }
19
-
20
- accepts_nested_attributes_for :shipping_address
21
- accepts_nested_attributes_for :line_items, allow_destroy: true
22
-
23
- # The following methods are delegated to the associated
24
- # SolidusSubscriptions::LineItem
25
- #
26
- # :quantity, :subscribable_id
27
- delegate :quantity, :subscribable_id, to: :line_item
28
-
29
- # Find all subscriptions that are "actionable"; that is, ones that have an
30
- # actionable_date in the past and are not invalid or canceled.
31
- scope :actionable, (lambda do
32
- where("#{table_name}.actionable_date <= ?", Time.zone.now).
33
- where.not(state: ["canceled", "inactive"])
34
- end)
35
-
36
- # Find subscriptions based on their processing state. This state is not a
37
- # model attrubute.
38
- #
39
- # @param state [Symbol] One of :pending, :success, or failed
40
- #
41
- # pending: New subscriptions, never been processed
42
- # failed: Subscriptions which failed to be processed on the last attempt
43
- # success: Subscriptions which were successfully processed on the last attempt
44
- scope :in_processing_state, (lambda do |state|
45
- case state.to_sym
46
- when :success
47
- fulfilled.joins(:installments)
48
- when :failed
49
- fulfilled_ids = fulfilled.pluck(:id)
50
- where.not(id: fulfilled_ids)
51
- when :pending
52
- includes(:installments).where(solidus_subscriptions_installments: { id: nil })
53
- else
54
- raise ArgumentError.new("state must be one of: :success, :failed, :pending")
55
- end
56
- end)
57
-
58
- scope :fulfilled, (lambda do
59
- unfulfilled_ids = unfulfilled.pluck(:id)
60
- where.not(id: unfulfilled_ids)
61
- end)
62
-
63
- scope :unfulfilled, (lambda do
64
- joins(:installments).merge(Installment.unfulfilled)
65
- end)
66
-
67
- def self.ransackable_scopes(_auth_object = nil)
68
- [:in_processing_state]
69
- end
70
-
71
- def self.processing_states
72
- PROCESSING_STATES
73
- end
74
-
75
- # The subscription state determines the behaviours around when it is
76
- # processed. Here is a brief description of the states and how they affect
77
- # the subscription.
78
- #
79
- # [active] Default state when created. Subscription can be processed
80
- # [canceled] The user has ended their subscription. Subscription will not
81
- # be processed.
82
- # [pending_cancellation] The user has ended their subscription, but the
83
- # conditions for canceling the subscription have not been met. Subscription
84
- # will continue to be processed until the subscription is canceled and
85
- # the conditions are met.
86
- # [inactive] The number of installments has been fulfilled. The subscription
87
- # will no longer be processed
88
- state_machine :state, initial: :active do
89
- event :cancel do
90
- transition [:active, :pending_cancellation] => :canceled,
91
- if: ->(subscription) { subscription.can_be_canceled? }
92
-
93
- transition active: :pending_cancellation
94
- end
95
-
96
- after_transition to: :canceled, do: :advance_actionable_date
97
-
98
- event :deactivate do
99
- transition active: :inactive,
100
- if: ->(subscription) { subscription.can_be_deactivated? }
101
- end
102
-
103
- event :activate do
104
- transition any - [:active] => :active
105
- end
106
-
107
- after_transition to: :active, do: :advance_actionable_date
108
- end
109
-
110
- # This method determines if a subscription may be canceled. Canceled
111
- # subcriptions will not be processed. By default subscriptions may always be
112
- # canceled. If this method is overriden to return false, the subscription
113
- # will be moved to the :pending_cancellation state until it is canceled
114
- # again and this condition is true.
115
- #
116
- # USE CASE: Subscriptions can only be canceled more than 10 days before they
117
- # are processed. Override this method to be:
118
- #
119
- # def can_be_canceled?
120
- # return true if actionable_date.nil?
121
- # (actionable_date - 10.days.from_now.to_date) > 0
122
- # end
123
- #
124
- # If a user cancels this subscription less than 10 days before it will
125
- # be processed the subscription will be bumped into the
126
- # :pending_cancellation state instead of being canceled. Susbcriptions
127
- # pending cancellation will still be processed.
128
- def can_be_canceled?
129
- return true if actionable_date.nil?
130
- (actionable_date - Config.minimum_cancellation_notice).future?
131
- end
132
-
133
- def skip
134
- check_successive_skips_exceeded
135
- check_total_skips_exceeded
136
-
137
- return if errors.any?
138
-
139
- advance_actionable_date
140
- end
141
-
142
- # This method determines if a subscription can be deactivated. A deactivated
143
- # subscription will not be processed. By default a subscription can be
144
- # deactivated if the end_date defined on
145
- # subscription_line_item is less than the current date
146
- # In this case the subscription has been fulfilled and
147
- # should not be processed again. Subscriptions without an end_date
148
- # value cannot be deactivated.
149
- def can_be_deactivated?
150
- active? && line_item.end_date && actionable_date > line_item.end_date
151
- end
152
-
153
- # Get the date after the current actionable_date where this subscription
154
- # will be actionable again
155
- #
156
- # @return [Date] The current actionable_date plus 1 interval. The next
157
- # date after the current actionable_date this subscription will be
158
- # eligible to be processed.
159
- def next_actionable_date
160
- return nil unless active?
161
- new_date = (actionable_date || Time.zone.now)
162
- (new_date + interval).beginning_of_minute
163
- end
164
-
165
- # Advance the actionable date to the next_actionable_date value. Will modify
166
- # the record.
167
- #
168
- # @return [Date] The next date after the current actionable_date this
169
- # subscription will be eligible to be processed.
170
- def advance_actionable_date
171
- update! actionable_date: next_actionable_date
172
- actionable_date
173
- end
174
-
175
- # Get the builder for the subscription_line_item. This will be an
176
- # object that can generate the appropriate line item for the subscribable
177
- # object
178
- #
179
- # @return [SolidusSubscriptions::LineItemBuilder]
180
- def line_item_builder
181
- LineItemBuilder.new(line_items)
182
- end
183
-
184
- # The state of the last attempt to process an installment associated to
185
- # this subscrtipion
186
- #
187
- # @return [String] pending if the no installments have been processed,
188
- # failed if the last installment has not been fulfilled and, success
189
- # if the last installment was fulfilled.
190
- def processing_state
191
- return 'pending' if installments.empty?
192
- installments.last.fulfilled? ? 'success' : 'failed'
193
- end
194
-
195
- private
196
-
197
- def check_successive_skips_exceeded
198
- return unless Config.maximum_successive_skips
199
-
200
- if successive_skip_count >= Config.maximum_successive_skips
201
- errors.add(:successive_skip_count, :exceeded)
202
- end
203
- end
204
-
205
- def check_total_skips_exceeded
206
- return unless Config.maximum_total_skips
207
-
208
- if skip_count >= Config.maximum_total_skips
209
- errors.add(:skip_count, :exceeded)
210
- end
211
- end
212
-
213
- def line_item
214
- line_items.first
215
- end
216
- end
217
- end
1
+ # The subscription class is responsable for grouping together the
2
+ # information required for the system to place a subscriptions order on
3
+ # behalf of a specific user.
4
+ module SolidusSubscriptions
5
+ class Subscription < ActiveRecord::Base
6
+ include Interval
7
+
8
+ PROCESSING_STATES = [:pending, :failed, :success]
9
+
10
+ belongs_to :user, class_name: Spree.user_class
11
+ has_many :line_items, class_name: 'SolidusSubscriptions::LineItem', inverse_of: :subscription
12
+ has_many :installments, class_name: 'SolidusSubscriptions::Installment'
13
+ belongs_to :store, class_name: 'Spree::Store'
14
+ belongs_to :shipping_address, class_name: 'Spree::Address'
15
+
16
+ validates :user, presence: :true
17
+ validates :skip_count, :successive_skip_count, presence: true, numericality: { greater_than_or_equal_to: 0 }
18
+ validates :interval_length, numericality: { greater_than: 0 }
19
+
20
+ accepts_nested_attributes_for :shipping_address
21
+ accepts_nested_attributes_for :line_items, allow_destroy: true
22
+
23
+ # The following methods are delegated to the associated
24
+ # SolidusSubscriptions::LineItem
25
+ #
26
+ # :quantity, :subscribable_id
27
+ delegate :quantity, :subscribable_id, to: :line_item
28
+
29
+ # Find all subscriptions that are "actionable"; that is, ones that have an
30
+ # actionable_date in the past and are not invalid or canceled.
31
+ scope :actionable, (lambda do
32
+ where("#{table_name}.actionable_date <= ?", Time.zone.now).
33
+ where.not(state: ["canceled", "inactive"])
34
+ end)
35
+
36
+ # Find subscriptions based on their processing state. This state is not a
37
+ # model attrubute.
38
+ #
39
+ # @param state [Symbol] One of :pending, :success, or failed
40
+ #
41
+ # pending: New subscriptions, never been processed
42
+ # failed: Subscriptions which failed to be processed on the last attempt
43
+ # success: Subscriptions which were successfully processed on the last attempt
44
+ scope :in_processing_state, (lambda do |state|
45
+ case state.to_sym
46
+ when :success
47
+ fulfilled.joins(:installments)
48
+ when :failed
49
+ fulfilled_ids = fulfilled.pluck(:id)
50
+ where.not(id: fulfilled_ids)
51
+ when :pending
52
+ includes(:installments).where(solidus_subscriptions_installments: { id: nil })
53
+ else
54
+ raise ArgumentError.new("state must be one of: :success, :failed, :pending")
55
+ end
56
+ end)
57
+
58
+ scope :fulfilled, (lambda do
59
+ unfulfilled_ids = unfulfilled.pluck(:id)
60
+ where.not(id: unfulfilled_ids)
61
+ end)
62
+
63
+ scope :unfulfilled, (lambda do
64
+ joins(:installments).merge(Installment.unfulfilled)
65
+ end)
66
+
67
+ def self.ransackable_scopes(_auth_object = nil)
68
+ [:in_processing_state]
69
+ end
70
+
71
+ def self.processing_states
72
+ PROCESSING_STATES
73
+ end
74
+
75
+ # The subscription state determines the behaviours around when it is
76
+ # processed. Here is a brief description of the states and how they affect
77
+ # the subscription.
78
+ #
79
+ # [active] Default state when created. Subscription can be processed
80
+ # [canceled] The user has ended their subscription. Subscription will not
81
+ # be processed.
82
+ # [pending_cancellation] The user has ended their subscription, but the
83
+ # conditions for canceling the subscription have not been met. Subscription
84
+ # will continue to be processed until the subscription is canceled and
85
+ # the conditions are met.
86
+ # [inactive] The number of installments has been fulfilled. The subscription
87
+ # will no longer be processed
88
+ state_machine :state, initial: :active do
89
+ event :cancel do
90
+ transition [:active, :pending_cancellation] => :canceled,
91
+ if: ->(subscription) { subscription.can_be_canceled? }
92
+
93
+ transition active: :pending_cancellation
94
+ end
95
+
96
+ after_transition to: :canceled, do: :advance_actionable_date
97
+
98
+ event :deactivate do
99
+ transition active: :inactive,
100
+ if: ->(subscription) { subscription.can_be_deactivated? }
101
+ end
102
+
103
+ event :activate do
104
+ transition any - [:active] => :active
105
+ end
106
+
107
+ after_transition to: :active, do: :advance_actionable_date
108
+ end
109
+
110
+ # This method determines if a subscription may be canceled. Canceled
111
+ # subcriptions will not be processed. By default subscriptions may always be
112
+ # canceled. If this method is overriden to return false, the subscription
113
+ # will be moved to the :pending_cancellation state until it is canceled
114
+ # again and this condition is true.
115
+ #
116
+ # USE CASE: Subscriptions can only be canceled more than 10 days before they
117
+ # are processed. Override this method to be:
118
+ #
119
+ # def can_be_canceled?
120
+ # return true if actionable_date.nil?
121
+ # (actionable_date - 10.days.from_now.to_date) > 0
122
+ # end
123
+ #
124
+ # If a user cancels this subscription less than 10 days before it will
125
+ # be processed the subscription will be bumped into the
126
+ # :pending_cancellation state instead of being canceled. Susbcriptions
127
+ # pending cancellation will still be processed.
128
+ def can_be_canceled?
129
+ return true if actionable_date.nil?
130
+ (actionable_date - Config.minimum_cancellation_notice).future?
131
+ end
132
+
133
+ def skip
134
+ check_successive_skips_exceeded
135
+ check_total_skips_exceeded
136
+
137
+ return if errors.any?
138
+
139
+ advance_actionable_date
140
+ end
141
+
142
+ # This method determines if a subscription can be deactivated. A deactivated
143
+ # subscription will not be processed. By default a subscription can be
144
+ # deactivated if the end_date defined on
145
+ # subscription_line_item is less than the current date
146
+ # In this case the subscription has been fulfilled and
147
+ # should not be processed again. Subscriptions without an end_date
148
+ # value cannot be deactivated.
149
+ def can_be_deactivated?
150
+ active? && line_item.end_date && actionable_date > line_item.end_date
151
+ end
152
+
153
+ # Get the date after the current actionable_date where this subscription
154
+ # will be actionable again
155
+ #
156
+ # @return [Date] The current actionable_date plus 1 interval. The next
157
+ # date after the current actionable_date this subscription will be
158
+ # eligible to be processed.
159
+ def next_actionable_date
160
+ return nil unless active?
161
+ new_date = (actionable_date || Time.zone.now)
162
+ (new_date + interval).beginning_of_minute
163
+ end
164
+
165
+ # Advance the actionable date to the next_actionable_date value. Will modify
166
+ # the record.
167
+ #
168
+ # @return [Date] The next date after the current actionable_date this
169
+ # subscription will be eligible to be processed.
170
+ def advance_actionable_date
171
+ update! actionable_date: next_actionable_date
172
+ actionable_date
173
+ end
174
+
175
+ # Get the builder for the subscription_line_item. This will be an
176
+ # object that can generate the appropriate line item for the subscribable
177
+ # object
178
+ #
179
+ # @return [SolidusSubscriptions::LineItemBuilder]
180
+ def line_item_builder
181
+ LineItemBuilder.new(line_items)
182
+ end
183
+
184
+ # The state of the last attempt to process an installment associated to
185
+ # this subscrtipion
186
+ #
187
+ # @return [String] pending if the no installments have been processed,
188
+ # failed if the last installment has not been fulfilled and, success
189
+ # if the last installment was fulfilled.
190
+ def processing_state
191
+ return 'pending' if installments.empty?
192
+ installments.last.fulfilled? ? 'success' : 'failed'
193
+ end
194
+
195
+ private
196
+
197
+ def check_successive_skips_exceeded
198
+ return unless Config.maximum_successive_skips
199
+
200
+ if successive_skip_count >= Config.maximum_successive_skips
201
+ errors.add(:successive_skip_count, :exceeded)
202
+ end
203
+ end
204
+
205
+ def check_total_skips_exceeded
206
+ return unless Config.maximum_total_skips
207
+
208
+ if skip_count >= Config.maximum_total_skips
209
+ errors.add(:skip_count, :exceeded)
210
+ end
211
+ end
212
+
213
+ def line_item
214
+ line_items.first
215
+ end
216
+ end
217
+ end
@@ -1,60 +1,60 @@
1
- # This module is responsible for taking SolidusSubscriptions::LineItem
2
- # objects and creating SolidusSubscriptions::Subscription Objects
3
- module SolidusSubscriptions
4
- module SubscriptionGenerator
5
- extend self
6
-
7
- SubscriptionConfiguration = Struct.new(:interval_length, :interval_units, :end_date)
8
-
9
- # Create and persist a subscription for a collection of subscription
10
- # line items
11
- #
12
- # @param subscription_line_items [Array<SolidusSubscriptions::LineItem>] The
13
- # subscription_line_items to be activated
14
- #
15
- # @return [SolidusSubscriptions::Subscription]
16
- def activate(subscription_line_items)
17
- return if subscription_line_items.empty?
18
-
19
- order = subscription_line_items.first.order
20
- configuration = subscription_configuration(subscription_line_items.first)
21
-
22
- subscription_attributes = {
23
- user: order.user,
24
- line_items: subscription_line_items,
25
- store: order.store,
26
- shipping_address: order.ship_address,
27
- **configuration.to_h
28
- }
29
-
30
- Subscription.create!(subscription_attributes) do |sub|
31
- sub.actionable_date = sub.next_actionable_date
32
- end
33
- end
34
-
35
- # Group a collection of line items by common subscription configuration
36
- # options. Grouped subscription_line_items can belong to a single
37
- # subscription.
38
- #
39
- # @param subscription_line_items [Array<SolidusSubscriptions::LineItem>] The
40
- # subscription_line_items to be grouped.
41
- #
42
- # @return [Array<Array<SolidusSubscriptions::LineItem>>]
43
- def group(subscription_line_items)
44
- subscription_line_items.group_by do |li|
45
- subscription_configuration(li)
46
- end.
47
- values
48
- end
49
-
50
- private
51
-
52
- def subscription_configuration(subscription_line_item)
53
- SubscriptionConfiguration.new(
54
- subscription_line_item.interval_length,
55
- subscription_line_item.interval_units,
56
- subscription_line_item.end_date
57
- )
58
- end
59
- end
60
- end
1
+ # This module is responsible for taking SolidusSubscriptions::LineItem
2
+ # objects and creating SolidusSubscriptions::Subscription Objects
3
+ module SolidusSubscriptions
4
+ module SubscriptionGenerator
5
+ extend self
6
+
7
+ SubscriptionConfiguration = Struct.new(:interval_length, :interval_units, :end_date)
8
+
9
+ # Create and persist a subscription for a collection of subscription
10
+ # line items
11
+ #
12
+ # @param subscription_line_items [Array<SolidusSubscriptions::LineItem>] The
13
+ # subscription_line_items to be activated
14
+ #
15
+ # @return [SolidusSubscriptions::Subscription]
16
+ def activate(subscription_line_items)
17
+ return if subscription_line_items.empty?
18
+
19
+ order = subscription_line_items.first.order
20
+ configuration = subscription_configuration(subscription_line_items.first)
21
+
22
+ subscription_attributes = {
23
+ user: order.user,
24
+ line_items: subscription_line_items,
25
+ store: order.store,
26
+ shipping_address: order.ship_address,
27
+ **configuration.to_h
28
+ }
29
+
30
+ Subscription.create!(subscription_attributes) do |sub|
31
+ sub.actionable_date = sub.next_actionable_date
32
+ end
33
+ end
34
+
35
+ # Group a collection of line items by common subscription configuration
36
+ # options. Grouped subscription_line_items can belong to a single
37
+ # subscription.
38
+ #
39
+ # @param subscription_line_items [Array<SolidusSubscriptions::LineItem>] The
40
+ # subscription_line_items to be grouped.
41
+ #
42
+ # @return [Array<Array<SolidusSubscriptions::LineItem>>]
43
+ def group(subscription_line_items)
44
+ subscription_line_items.group_by do |li|
45
+ subscription_configuration(li)
46
+ end.
47
+ values
48
+ end
49
+
50
+ private
51
+
52
+ def subscription_configuration(subscription_line_item)
53
+ SubscriptionConfiguration.new(
54
+ subscription_line_item.interval_length,
55
+ subscription_line_item.interval_units,
56
+ subscription_line_item.end_date
57
+ )
58
+ end
59
+ end
60
+ end