disco_app 0.8.9
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/Rakefile +37 -0
- data/app/assets/images/disco_app/icon.svg +1 -0
- data/app/assets/javascripts/disco_app/components/filterable_shop_list.js.jsx +60 -0
- data/app/assets/javascripts/disco_app/components/shop_filter_tab.js.jsx +34 -0
- data/app/assets/javascripts/disco_app/components/shop_filter_tabs.js.jsx +21 -0
- data/app/assets/javascripts/disco_app/components/shop_list.js.jsx +144 -0
- data/app/assets/javascripts/disco_app/components/shop_row.js.jsx +27 -0
- data/app/assets/javascripts/disco_app/components/shopify_admin_link.js.jsx +29 -0
- data/app/assets/javascripts/disco_app/components.js +5 -0
- data/app/assets/javascripts/disco_app/disco_app.js +7 -0
- data/app/assets/javascripts/disco_app/frame.js +152 -0
- data/app/assets/javascripts/disco_app/shopify-turbolinks.js +7 -0
- data/app/assets/stylesheets/disco_app/bootstrap/_custom.scss +54 -0
- data/app/assets/stylesheets/disco_app/bootstrap/_variables.scss +872 -0
- data/app/assets/stylesheets/disco_app/disco/_buttons.scss +31 -0
- data/app/assets/stylesheets/disco_app/disco/_cards.scss +52 -0
- data/app/assets/stylesheets/disco_app/disco/_forms.scss +23 -0
- data/app/assets/stylesheets/disco_app/disco/_grid.scss +58 -0
- data/app/assets/stylesheets/disco_app/disco/_sections.scss +61 -0
- data/app/assets/stylesheets/disco_app/disco/_tables.scss +57 -0
- data/app/assets/stylesheets/disco_app/disco/_tabs.scss +61 -0
- data/app/assets/stylesheets/disco_app/disco/_type.scss +39 -0
- data/app/assets/stylesheets/disco_app/disco/mixins/_flexbox.scss +394 -0
- data/app/assets/stylesheets/disco_app/disco_app.scss +16 -0
- data/app/assets/stylesheets/disco_app/frame/_buttons.scss +54 -0
- data/app/assets/stylesheets/disco_app/frame/_forms.scss +26 -0
- data/app/assets/stylesheets/disco_app/frame/_layout.scss +77 -0
- data/app/assets/stylesheets/disco_app/frame/_type.scss +32 -0
- data/app/assets/stylesheets/disco_app/frame.scss +9 -0
- data/app/controllers/disco_app/admin/app_settings_controller.rb +3 -0
- data/app/controllers/disco_app/admin/application_controller.rb +3 -0
- data/app/controllers/disco_app/admin/concerns/app_settings_controller.rb +24 -0
- data/app/controllers/disco_app/admin/concerns/authenticated_controller.rb +20 -0
- data/app/controllers/disco_app/admin/concerns/plans_controller.rb +51 -0
- data/app/controllers/disco_app/admin/concerns/shops_controller.rb +7 -0
- data/app/controllers/disco_app/admin/plans_controller.rb +3 -0
- data/app/controllers/disco_app/admin/resources/shops_controller.rb +3 -0
- data/app/controllers/disco_app/admin/shops_controller.rb +3 -0
- data/app/controllers/disco_app/charges_controller.rb +47 -0
- data/app/controllers/disco_app/concerns/app_proxy_controller.rb +40 -0
- data/app/controllers/disco_app/concerns/authenticated_controller.rb +56 -0
- data/app/controllers/disco_app/concerns/carrier_request_controller.rb +21 -0
- data/app/controllers/disco_app/frame_controller.rb +9 -0
- data/app/controllers/disco_app/install_controller.rb +27 -0
- data/app/controllers/disco_app/subscriptions_controller.rb +32 -0
- data/app/controllers/disco_app/webhooks_controller.rb +46 -0
- data/app/controllers/sessions_controller.rb +28 -0
- data/app/helpers/disco_app/application_helper.rb +28 -0
- data/app/jobs/disco_app/app_installed_job.rb +3 -0
- data/app/jobs/disco_app/app_uninstalled_job.rb +3 -0
- data/app/jobs/disco_app/concerns/app_installed_job.rb +39 -0
- data/app/jobs/disco_app/concerns/app_uninstalled_job.rb +20 -0
- data/app/jobs/disco_app/concerns/shop_update_job.rb +16 -0
- data/app/jobs/disco_app/concerns/subscription_changed_job.rb +7 -0
- data/app/jobs/disco_app/concerns/synchronise_carrier_service_job.rb +52 -0
- data/app/jobs/disco_app/concerns/synchronise_webhooks_job.rb +61 -0
- data/app/jobs/disco_app/shop_job.rb +27 -0
- data/app/jobs/disco_app/shop_update_job.rb +3 -0
- data/app/jobs/disco_app/subscription_changed_job.rb +3 -0
- data/app/jobs/disco_app/synchronise_carrier_service_job.rb +3 -0
- data/app/jobs/disco_app/synchronise_webhooks_job.rb +3 -0
- data/app/models/disco_app/app_settings.rb +3 -0
- data/app/models/disco_app/application_charge.rb +18 -0
- data/app/models/disco_app/concerns/app_settings.rb +7 -0
- data/app/models/disco_app/concerns/plan.rb +26 -0
- data/app/models/disco_app/concerns/plan_code.rb +15 -0
- data/app/models/disco_app/concerns/shop.rb +80 -0
- data/app/models/disco_app/concerns/subscription.rb +48 -0
- data/app/models/disco_app/concerns/synchronises.rb +39 -0
- data/app/models/disco_app/plan.rb +3 -0
- data/app/models/disco_app/plan_code.rb +3 -0
- data/app/models/disco_app/recurring_application_charge.rb +18 -0
- data/app/models/disco_app/session_storage.rb +18 -0
- data/app/models/disco_app/shop.rb +3 -0
- data/app/models/disco_app/subscription.rb +3 -0
- data/app/resources/disco_app/admin/resources/concerns/shop_resource.rb +45 -0
- data/app/resources/disco_app/admin/resources/shop_resource.rb +4 -0
- data/app/services/disco_app/carrier_request_service.rb +15 -0
- data/app/services/disco_app/charges_service.rb +81 -0
- data/app/services/disco_app/proxy_service.rb +17 -0
- data/app/services/disco_app/subscription_service.rb +46 -0
- data/app/services/disco_app/webhook_service.rb +30 -0
- data/app/views/disco_app/admin/app_settings/edit.html.erb +5 -0
- data/app/views/disco_app/admin/plans/_form.html.erb +75 -0
- data/app/views/disco_app/admin/plans/edit.html.erb +7 -0
- data/app/views/disco_app/admin/plans/index.html.erb +41 -0
- data/app/views/disco_app/admin/plans/new.html.erb +7 -0
- data/app/views/disco_app/admin/shops/index.html.erb +12 -0
- data/app/views/disco_app/charges/activate.html.erb +1 -0
- data/app/views/disco_app/charges/create.html.erb +1 -0
- data/app/views/disco_app/charges/new.html.erb +12 -0
- data/app/views/disco_app/frame/frame.html.erb +36 -0
- data/app/views/disco_app/install/installing.html.erb +7 -0
- data/app/views/disco_app/install/uninstalling.html.erb +1 -0
- data/app/views/disco_app/proxy_errors/404.html.erb +1 -0
- data/app/views/disco_app/shared/_card.html.erb +14 -0
- data/app/views/disco_app/shared/_section.html.erb +17 -0
- data/app/views/disco_app/subscriptions/new.html.erb +25 -0
- data/app/views/layouts/admin/_navbar.html.erb +25 -0
- data/app/views/layouts/admin.html.erb +23 -0
- data/app/views/layouts/application.html.erb +18 -0
- data/app/views/layouts/embedded_app.html.erb +41 -0
- data/app/views/layouts/embedded_app_modal.html.erb +17 -0
- data/app/views/sessions/new.html.erb +26 -0
- data/config/routes.rb +46 -0
- data/db/migrate/20150525000000_create_shops_if_not_existent.rb +15 -0
- data/db/migrate/20150525162112_add_status_to_shops.rb +5 -0
- data/db/migrate/20150525171422_add_meta_to_shops.rb +11 -0
- data/db/migrate/20150629210346_add_charge_status_to_shop.rb +5 -0
- data/db/migrate/20150814214025_add_more_meta_to_shops.rb +15 -0
- data/db/migrate/20151017231302_create_disco_app_plans.rb +13 -0
- data/db/migrate/20151017232027_create_disco_app_subscriptions.rb +15 -0
- data/db/migrate/20151017234409_move_shop_to_disco_app_engine.rb +5 -0
- data/db/migrate/20160112233706_create_disco_app_sessions.rb +12 -0
- data/db/migrate/20160113194418_add_shop_id_to_disco_app_sessions.rb +6 -0
- data/db/migrate/20160223111044_create_disco_app_settings.rb +8 -0
- data/db/migrate/20160301223215_update_plans.rb +22 -0
- data/db/migrate/20160301224558_update_subscriptions.rb +13 -0
- data/db/migrate/20160302104816_create_disco_app_recurring_application_charges.rb +14 -0
- data/db/migrate/20160302105259_create_disco_app_application_charges.rb +14 -0
- data/db/migrate/20160302134728_drop_charge_status_from_shops.rb +5 -0
- data/db/migrate/20160302142941_add_shopify_attributes_to_charges.rb +8 -0
- data/db/migrate/20160331093148_create_disco_app_plan_codes.rb +14 -0
- data/db/migrate/20160401044420_add_status_to_plan_codes.rb +5 -0
- data/db/migrate/20160401045551_add_amount_and_plan_code_to_disco_app_subscriptions.rb +7 -0
- data/db/migrate/20160425205211_add_source_to_disco_app_subscriptions.rb +5 -0
- data/db/migrate/20160426033520_add_trial_period_days_to_disco_app_subscriptions.rb +5 -0
- data/lib/disco_app/configuration.rb +39 -0
- data/lib/disco_app/constants.rb +4 -0
- data/lib/disco_app/engine.rb +26 -0
- data/lib/disco_app/session.rb +14 -0
- data/lib/disco_app/support/file_fixtures.rb +23 -0
- data/lib/disco_app/test_help.rb +11 -0
- data/lib/disco_app/version.rb +3 -0
- data/lib/disco_app.rb +7 -0
- data/lib/generators/disco_app/USAGE +5 -0
- data/lib/generators/disco_app/adminify/adminify_generator.rb +35 -0
- data/lib/generators/disco_app/disco_app_generator.rb +164 -0
- data/lib/generators/disco_app/mailify/mailify_generator.rb +54 -0
- data/lib/generators/disco_app/monitorify/monitorify_generator.rb +28 -0
- data/lib/generators/disco_app/monitorify/templates/config/newrelic.yml +26 -0
- data/lib/generators/disco_app/monitorify/templates/initializers/rollbar.rb +12 -0
- data/lib/generators/disco_app/reactify/reactify_generator.rb +45 -0
- data/lib/generators/disco_app/templates/assets/javascripts/application.js +17 -0
- data/lib/generators/disco_app/templates/assets/stylesheets/application.scss +5 -0
- data/lib/generators/disco_app/templates/config/puma.rb +15 -0
- data/lib/generators/disco_app/templates/controllers/home_controller.rb +7 -0
- data/lib/generators/disco_app/templates/initializers/disco_app.rb +19 -0
- data/lib/generators/disco_app/templates/initializers/session_store.rb +2 -0
- data/lib/generators/disco_app/templates/initializers/shopify_app.rb +7 -0
- data/lib/generators/disco_app/templates/initializers/shopify_session_repository.rb +7 -0
- data/lib/generators/disco_app/templates/root/Procfile +2 -0
- data/lib/generators/disco_app/templates/views/home/index.html.erb +2 -0
- data/lib/tasks/carrier_service.rake +10 -0
- data/lib/tasks/sessions.rake +9 -0
- data/lib/tasks/start.rake +3 -0
- data/lib/tasks/webhooks.rake +10 -0
- data/test/controllers/disco_app/admin/shops_controller_test.rb +54 -0
- data/test/controllers/disco_app/charges_controller_test.rb +92 -0
- data/test/controllers/disco_app/install_controller_test.rb +50 -0
- data/test/controllers/disco_app/subscriptions_controller_test.rb +68 -0
- data/test/controllers/disco_app/webhooks_controller_test.rb +58 -0
- data/test/controllers/home_controller_test.rb +92 -0
- data/test/controllers/proxy_controller_test.rb +42 -0
- data/test/disco_app_test.rb +7 -0
- data/test/dummy/Rakefile +6 -0
- data/test/dummy/app/assets/javascripts/application.js +17 -0
- data/test/dummy/app/assets/stylesheets/application.scss +5 -0
- data/test/dummy/app/controllers/application_controller.rb +6 -0
- data/test/dummy/app/controllers/disco_app/admin/shops_controller.rb +8 -0
- data/test/dummy/app/controllers/home_controller.rb +7 -0
- data/test/dummy/app/controllers/proxy_controller.rb +8 -0
- data/test/dummy/app/helpers/application_helper.rb +2 -0
- data/test/dummy/app/jobs/disco_app/app_installed_job.rb +16 -0
- data/test/dummy/app/jobs/disco_app/app_uninstalled_job.rb +11 -0
- data/test/dummy/app/jobs/products_create_job.rb +7 -0
- data/test/dummy/app/jobs/products_delete_job.rb +7 -0
- data/test/dummy/app/jobs/products_update_job.rb +7 -0
- data/test/dummy/app/models/disco_app/shop.rb +15 -0
- data/test/dummy/app/models/product.rb +6 -0
- data/test/dummy/app/views/home/index.html.erb +2 -0
- data/test/dummy/bin/bundle +3 -0
- data/test/dummy/bin/rails +4 -0
- data/test/dummy/bin/rake +4 -0
- data/test/dummy/bin/setup +29 -0
- data/test/dummy/config/application.rb +38 -0
- data/test/dummy/config/boot.rb +5 -0
- data/test/dummy/config/database.codeship.yml +23 -0
- data/test/dummy/config/database.yml +20 -0
- data/test/dummy/config/environment.rb +5 -0
- data/test/dummy/config/environments/development.rb +41 -0
- data/test/dummy/config/environments/production.rb +85 -0
- data/test/dummy/config/environments/test.rb +42 -0
- data/test/dummy/config/initializers/assets.rb +11 -0
- data/test/dummy/config/initializers/backtrace_silencers.rb +7 -0
- data/test/dummy/config/initializers/cookies_serializer.rb +3 -0
- data/test/dummy/config/initializers/disco_app.rb +19 -0
- data/test/dummy/config/initializers/filter_parameter_logging.rb +4 -0
- data/test/dummy/config/initializers/inflections.rb +16 -0
- data/test/dummy/config/initializers/mime_types.rb +4 -0
- data/test/dummy/config/initializers/omniauth.rb +9 -0
- data/test/dummy/config/initializers/session_store.rb +2 -0
- data/test/dummy/config/initializers/shopify_app.rb +7 -0
- data/test/dummy/config/initializers/shopify_session_repository.rb +7 -0
- data/test/dummy/config/initializers/wrap_parameters.rb +14 -0
- data/test/dummy/config/locales/en.yml +23 -0
- data/test/dummy/config/routes.rb +10 -0
- data/test/dummy/config/secrets.yml +22 -0
- data/test/dummy/config.ru +4 -0
- data/test/dummy/db/migrate/20160307182229_create_products.rb +11 -0
- data/test/dummy/db/schema.rb +140 -0
- data/test/dummy/public/404.html +67 -0
- data/test/dummy/public/422.html +67 -0
- data/test/dummy/public/500.html +66 -0
- data/test/dummy/public/favicon.ico +0 -0
- data/test/fixtures/api/widget_store/charges/activate_application_charge_request.json +16 -0
- data/test/fixtures/api/widget_store/charges/activate_application_charge_response.json +1 -0
- data/test/fixtures/api/widget_store/charges/activate_recurring_application_charge_request.json +20 -0
- data/test/fixtures/api/widget_store/charges/activate_recurring_application_charge_response.json +1 -0
- data/test/fixtures/api/widget_store/charges/create_application_charge_request.json +9 -0
- data/test/fixtures/api/widget_store/charges/create_application_charge_response.json +16 -0
- data/test/fixtures/api/widget_store/charges/create_recurring_application_charge_request.json +9 -0
- data/test/fixtures/api/widget_store/charges/create_recurring_application_charge_response.json +20 -0
- data/test/fixtures/api/widget_store/charges/create_second_recurring_application_charge_request.json +9 -0
- data/test/fixtures/api/widget_store/charges/create_second_recurring_application_charge_response.json +20 -0
- data/test/fixtures/api/widget_store/charges/get_accepted_application_charge_response.json +16 -0
- data/test/fixtures/api/widget_store/charges/get_accepted_recurring_application_charge_response.json +20 -0
- data/test/fixtures/api/widget_store/charges/get_declined_application_charge_response.json +16 -0
- data/test/fixtures/api/widget_store/charges/get_declined_recurring_application_charge_response.json +20 -0
- data/test/fixtures/api/widget_store/charges/get_pending_application_charge_response.json +16 -0
- data/test/fixtures/api/widget_store/charges/get_pending_recurring_application_charge_response.json +20 -0
- data/test/fixtures/api/widget_store/shop.json +46 -0
- data/test/fixtures/api/widget_store/webhooks.json +1 -0
- data/test/fixtures/disco_app/application_charges.yml +11 -0
- data/test/fixtures/disco_app/plan_codes.yml +13 -0
- data/test/fixtures/disco_app/plans.yml +37 -0
- data/test/fixtures/disco_app/recurring_application_charges.yml +11 -0
- data/test/fixtures/disco_app/shops.yml +10 -0
- data/test/fixtures/disco_app/subscriptions.yml +21 -0
- data/test/fixtures/products.yml +4 -0
- data/test/fixtures/webhooks/app_uninstalled.json +46 -0
- data/test/fixtures/webhooks/product_created.json +167 -0
- data/test/fixtures/webhooks/product_deleted.json +3 -0
- data/test/fixtures/webhooks/product_updated.json +167 -0
- data/test/integration/synchronises_test.rb +55 -0
- data/test/jobs/disco_app/app_installed_job_test.rb +55 -0
- data/test/jobs/disco_app/app_uninstalled_job_test.rb +30 -0
- data/test/models/disco_app/plan_test.rb +5 -0
- data/test/models/disco_app/session_test.rb +31 -0
- data/test/models/disco_app/shop_test.rb +27 -0
- data/test/services/disco_app/charges_service_test.rb +105 -0
- data/test/services/disco_app/subscription_service_test.rb +60 -0
- data/test/support/test_file_fixtures.rb +29 -0
- data/test/support/test_shopify_api.rb +16 -0
- data/test/test_helper.rb +52 -0
- metadata +663 -0
@@ -0,0 +1,18 @@
|
|
1
|
+
class DiscoApp::ApplicationCharge < ActiveRecord::Base
|
2
|
+
|
3
|
+
belongs_to :shop
|
4
|
+
belongs_to :subscription
|
5
|
+
|
6
|
+
enum status: [:pending, :accepted, :declined, :expired, :active]
|
7
|
+
|
8
|
+
scope :active, -> { where status: statuses[:active] }
|
9
|
+
|
10
|
+
def recurring?
|
11
|
+
false
|
12
|
+
end
|
13
|
+
|
14
|
+
def activate_url
|
15
|
+
DiscoApp::Engine.routes.url_helpers.activate_subscription_charge_url(subscription, self)
|
16
|
+
end
|
17
|
+
|
18
|
+
end
|
@@ -0,0 +1,26 @@
|
|
1
|
+
module DiscoApp::Concerns::Plan
|
2
|
+
extend ActiveSupport::Concern
|
3
|
+
|
4
|
+
included do
|
5
|
+
|
6
|
+
has_many :subscriptions
|
7
|
+
has_many :shops, through: :subscriptions
|
8
|
+
has_many :plan_codes
|
9
|
+
|
10
|
+
accepts_nested_attributes_for :plan_codes
|
11
|
+
|
12
|
+
enum status: [:available, :unavailable]
|
13
|
+
enum plan_type: [:recurring, :one_time]
|
14
|
+
enum interval: [:month, :year]
|
15
|
+
|
16
|
+
scope :available, -> { where status: statuses[:available] }
|
17
|
+
|
18
|
+
validates_presence_of :name
|
19
|
+
|
20
|
+
end
|
21
|
+
|
22
|
+
def has_trial?
|
23
|
+
trial_period_days.present? and trial_period_days > 0
|
24
|
+
end
|
25
|
+
|
26
|
+
end
|
@@ -0,0 +1,80 @@
|
|
1
|
+
module DiscoApp::Concerns::Shop
|
2
|
+
extend ActiveSupport::Concern
|
3
|
+
|
4
|
+
included do
|
5
|
+
include ShopifyApp::Shop
|
6
|
+
include ActionView::Helpers::DateHelper
|
7
|
+
|
8
|
+
# Define relationships to plans and subscriptions.
|
9
|
+
has_many :subscriptions
|
10
|
+
has_many :plans, through: :subscriptions
|
11
|
+
|
12
|
+
# Define relationship to sessions.
|
13
|
+
has_many :sessions, class_name: 'DiscoApp::Session', dependent: :destroy
|
14
|
+
|
15
|
+
# Define possible installation statuses as an enum.
|
16
|
+
enum status: [:never_installed, :awaiting_install, :installing, :installed, :awaiting_uninstall, :uninstalling, :uninstalled]
|
17
|
+
|
18
|
+
# Define some useful scopes.
|
19
|
+
scope :status, -> (status) { where status: status }
|
20
|
+
scope :installed, -> { where status: statuses[:installed] }
|
21
|
+
scope :has_active_shopify_plan, -> { where.not(plan_name: [:cancelled, :frozen, :fraudulent]) }
|
22
|
+
|
23
|
+
# Alias 'with_shopify_session' as 'temp', as per our existing conventions.
|
24
|
+
alias_method :temp, :with_shopify_session
|
25
|
+
|
26
|
+
# Return a hash of attributes that should be used to create a new charge for this shop.
|
27
|
+
# This method can be overridden by the inheriting Shop class in order to provide charges
|
28
|
+
# customised to a particular shop. Otherwise, the default settings configured in application.rb
|
29
|
+
# will be used.
|
30
|
+
def new_charge_attributes
|
31
|
+
{
|
32
|
+
type: Rails.configuration.x.shopify_charges_default_type,
|
33
|
+
name: DiscoApp.configuration.app_name,
|
34
|
+
price: Rails.configuration.x.shopify_charges_default_price,
|
35
|
+
trial_days: Rails.configuration.x.shopify_charges_default_trial_days,
|
36
|
+
}
|
37
|
+
end
|
38
|
+
|
39
|
+
# Convenience method to check if this shop has a current subscription.
|
40
|
+
def current_subscription?
|
41
|
+
current_subscription.present?
|
42
|
+
end
|
43
|
+
|
44
|
+
# Convenience method to get the current subscription for this shop, if any.
|
45
|
+
def current_subscription
|
46
|
+
subscriptions.current.first
|
47
|
+
end
|
48
|
+
|
49
|
+
# Convenience method to get the current plan for this shop, if any.
|
50
|
+
def current_plan
|
51
|
+
current_subscription&.plan
|
52
|
+
end
|
53
|
+
|
54
|
+
# Return the absolute URL to the shop's storefront.
|
55
|
+
def url
|
56
|
+
"#{protocol}://#{domain}"
|
57
|
+
end
|
58
|
+
|
59
|
+
# Return the protocol the shop's storefront uses. This should now always be
|
60
|
+
# https as all Shopify stores have SSL enabled.
|
61
|
+
def protocol
|
62
|
+
'https'
|
63
|
+
end
|
64
|
+
|
65
|
+
# Return the absolute URL to the shop's admin.
|
66
|
+
def admin_url
|
67
|
+
"https://#{shopify_domain}/admin"
|
68
|
+
end
|
69
|
+
|
70
|
+
def installed_duration
|
71
|
+
distance_of_time_in_words_to_now(created_at.time)
|
72
|
+
end
|
73
|
+
|
74
|
+
def pretty_created_at
|
75
|
+
created_at.strftime("%B %d, %Y")
|
76
|
+
end
|
77
|
+
|
78
|
+
end
|
79
|
+
|
80
|
+
end
|
@@ -0,0 +1,48 @@
|
|
1
|
+
module DiscoApp::Concerns::Subscription
|
2
|
+
extend ActiveSupport::Concern
|
3
|
+
|
4
|
+
included do
|
5
|
+
|
6
|
+
belongs_to :shop
|
7
|
+
belongs_to :plan
|
8
|
+
belongs_to :plan_code
|
9
|
+
|
10
|
+
has_many :one_time_charges, class_name: 'DiscoApp::ApplicationCharge', dependent: :destroy
|
11
|
+
has_many :recurring_charges, class_name: 'DiscoApp::RecurringApplicationCharge', dependent: :destroy
|
12
|
+
|
13
|
+
enum status: [:trial, :active, :cancelled]
|
14
|
+
enum subscription_type: [:recurring, :one_time]
|
15
|
+
|
16
|
+
scope :current, -> { where status: [statuses[:trial], statuses[:active]] }
|
17
|
+
|
18
|
+
end
|
19
|
+
|
20
|
+
# Only require an active charge if the amount to be charged is > 0.
|
21
|
+
def requires_active_charge?
|
22
|
+
amount > 0
|
23
|
+
end
|
24
|
+
|
25
|
+
# Convenience method to check if this subscription has an active charge.
|
26
|
+
def active_charge?
|
27
|
+
active_charge.present?
|
28
|
+
end
|
29
|
+
|
30
|
+
# Convenience method to get the active charge for this subscription.
|
31
|
+
def active_charge
|
32
|
+
charges.active.first
|
33
|
+
end
|
34
|
+
|
35
|
+
# Return the appropriate set of charges for this subscription's type.
|
36
|
+
def charges
|
37
|
+
recurring? ? recurring_charges : one_time_charges
|
38
|
+
end
|
39
|
+
|
40
|
+
def charge_class
|
41
|
+
recurring? ? DiscoApp::RecurringApplicationCharge : DiscoApp::ApplicationCharge
|
42
|
+
end
|
43
|
+
|
44
|
+
def shopify_charge_class
|
45
|
+
recurring? ? ShopifyAPI::RecurringApplicationCharge : ShopifyAPI::ApplicationCharge
|
46
|
+
end
|
47
|
+
|
48
|
+
end
|
@@ -0,0 +1,39 @@
|
|
1
|
+
module DiscoApp::Concerns::Synchronises
|
2
|
+
extend ActiveSupport::Concern
|
3
|
+
|
4
|
+
class_methods do
|
5
|
+
|
6
|
+
def should_synchronise?(shop, data)
|
7
|
+
true
|
8
|
+
end
|
9
|
+
|
10
|
+
def synchronise(shop, data)
|
11
|
+
data = data.with_indifferent_access
|
12
|
+
|
13
|
+
return unless should_synchronise?(shop, data)
|
14
|
+
|
15
|
+
instance = self.find_or_create_by!(id: data[:id]) do |instance|
|
16
|
+
instance.shop = shop
|
17
|
+
instance.data = data
|
18
|
+
end
|
19
|
+
|
20
|
+
instance.update(data: data)
|
21
|
+
|
22
|
+
instance
|
23
|
+
end
|
24
|
+
|
25
|
+
def should_synchronise_deletion?(shop, data)
|
26
|
+
true
|
27
|
+
end
|
28
|
+
|
29
|
+
def synchronise_deletion(shop, data)
|
30
|
+
data = data.with_indifferent_access
|
31
|
+
|
32
|
+
return unless should_synchronise_deletion?(shop, data)
|
33
|
+
|
34
|
+
self.destroy_all(shop: shop, id: data[:id])
|
35
|
+
end
|
36
|
+
|
37
|
+
end
|
38
|
+
|
39
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
class DiscoApp::RecurringApplicationCharge < ActiveRecord::Base
|
2
|
+
|
3
|
+
belongs_to :shop
|
4
|
+
belongs_to :subscription
|
5
|
+
|
6
|
+
enum status: [:pending, :accepted, :declined, :active, :cancelled, :expired]
|
7
|
+
|
8
|
+
scope :active, -> { where status: statuses[:active] }
|
9
|
+
|
10
|
+
def recurring?
|
11
|
+
true
|
12
|
+
end
|
13
|
+
|
14
|
+
def activate_url
|
15
|
+
DiscoApp::Engine.routes.url_helpers.activate_subscription_charge_url(subscription, self)
|
16
|
+
end
|
17
|
+
|
18
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
module DiscoApp
|
2
|
+
class SessionStorage
|
3
|
+
def self.store(session)
|
4
|
+
shop = Shop.find_or_initialize_by(shopify_domain: session.url)
|
5
|
+
shop.shopify_token = session.token
|
6
|
+
shop.save!
|
7
|
+
shop.id
|
8
|
+
end
|
9
|
+
|
10
|
+
def self.retrieve(id)
|
11
|
+
return unless id
|
12
|
+
shop = Shop.find(id)
|
13
|
+
ShopifyAPI::Session.new(shop.shopify_domain, shop.shopify_token)
|
14
|
+
rescue ActiveRecord::RecordNotFound
|
15
|
+
nil
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,45 @@
|
|
1
|
+
require 'jsonapi/resource'
|
2
|
+
|
3
|
+
module DiscoApp::Admin::Resources::Concerns::ShopResource
|
4
|
+
extend ActiveSupport::Concern
|
5
|
+
|
6
|
+
included do
|
7
|
+
|
8
|
+
attributes :shopify_domain, :status, :email, :country_name
|
9
|
+
attributes :currency, :domain, :plan_display_name, :created_at
|
10
|
+
attributes :pretty_created_at, :installed_duration
|
11
|
+
|
12
|
+
model_name 'DiscoApp::Shop'
|
13
|
+
|
14
|
+
filters :status
|
15
|
+
|
16
|
+
# Adjust the base records method to ensure only models for the authenticated domain are retrieved.
|
17
|
+
def self.records(options = {})
|
18
|
+
records = DiscoApp::Shop.order(created_at: :desc)
|
19
|
+
records
|
20
|
+
end
|
21
|
+
|
22
|
+
# Apply filters.
|
23
|
+
def self.apply_filter(records, filter, value, options)
|
24
|
+
return records if value.blank?
|
25
|
+
|
26
|
+
# Perform appropriate filtering.
|
27
|
+
case filter
|
28
|
+
when :status
|
29
|
+
return records.where(status: value.map { |v| DiscoApp::Shop.statuses[v.to_sym] } )
|
30
|
+
else
|
31
|
+
return super(records, filter, value)
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
# Don't allow the update of any fields via the API.
|
36
|
+
def self.updatable_fields(context)
|
37
|
+
[]
|
38
|
+
end
|
39
|
+
|
40
|
+
# Don't allow the creation of any fields via the API.
|
41
|
+
def self.creatable_fields(context)
|
42
|
+
[]
|
43
|
+
end
|
44
|
+
end
|
45
|
+
end
|
@@ -0,0 +1,15 @@
|
|
1
|
+
class DiscoApp::CarrierRequestService
|
2
|
+
|
3
|
+
# Return true iff the provided hmac_to_verify matches that calculated from the
|
4
|
+
# given data and secret.
|
5
|
+
def self.is_valid_hmac?(body, secret, hmac_to_verify)
|
6
|
+
ActiveSupport::SecurityUtils.secure_compare(self.calculated_hmac(body, secret), hmac_to_verify.to_s)
|
7
|
+
end
|
8
|
+
|
9
|
+
# Calculate the HMAC for the given data and secret.
|
10
|
+
def self.calculated_hmac(body, secret)
|
11
|
+
digest = OpenSSL::Digest.new('sha256')
|
12
|
+
Base64.encode64(OpenSSL::HMAC.digest(digest, secret, body)).strip
|
13
|
+
end
|
14
|
+
|
15
|
+
end
|
@@ -0,0 +1,81 @@
|
|
1
|
+
class DiscoApp::ChargesService
|
2
|
+
|
3
|
+
# Create the appropriate type of Shopify charge for the given subscription
|
4
|
+
# (either one-time or recurring) and return.
|
5
|
+
def self.create(shop, subscription)
|
6
|
+
# Create the charge object locally first.
|
7
|
+
charge = subscription.charge_class.create!(
|
8
|
+
shop: shop,
|
9
|
+
subscription: subscription,
|
10
|
+
)
|
11
|
+
|
12
|
+
# Create the charge object on Shopify.
|
13
|
+
shopify_charge = shop.temp {
|
14
|
+
subscription.shopify_charge_class.create(
|
15
|
+
name: subscription.plan.name,
|
16
|
+
price: '%.2f' % (subscription.amount.to_f / 100.0),
|
17
|
+
trial_days: subscription.plan.has_trial? ? subscription.trial_period_days : nil,
|
18
|
+
return_url: charge.activate_url,
|
19
|
+
test: !DiscoApp.configuration.real_charges?
|
20
|
+
)
|
21
|
+
}
|
22
|
+
|
23
|
+
# If we couldn't create the charge on Shopify, return nil.
|
24
|
+
if shopify_charge.nil?
|
25
|
+
return nil
|
26
|
+
end
|
27
|
+
|
28
|
+
# Update the local record of the charge from Shopify's created charge, then
|
29
|
+
# return it.
|
30
|
+
charge.update(
|
31
|
+
shopify_id: shopify_charge.id,
|
32
|
+
confirmation_url: shopify_charge.confirmation_url
|
33
|
+
)
|
34
|
+
charge
|
35
|
+
end
|
36
|
+
|
37
|
+
# Attempt to activate the given Shopify charge for the given Shop using the
|
38
|
+
# Shopify API. Returns true on successful activation, false otherwise.
|
39
|
+
def self.activate(shop, charge)
|
40
|
+
begin
|
41
|
+
# Start by fetching the Shopify charge to check that it was accepted.
|
42
|
+
shopify_charge = shop.temp {
|
43
|
+
charge.subscription.shopify_charge_class.find(charge.shopify_id)
|
44
|
+
}
|
45
|
+
|
46
|
+
# Update the status of the local charge based on the Shopify charge.
|
47
|
+
charge.send("#{shopify_charge.status}!") if charge.respond_to? "#{shopify_charge.status}!"
|
48
|
+
|
49
|
+
# If the charge wasn't accepted, fail and return.
|
50
|
+
return false unless charge.accepted?
|
51
|
+
|
52
|
+
# If the charge was indeed accepted, activate it via Shopify.
|
53
|
+
charge.shop.temp {
|
54
|
+
shopify_charge.activate
|
55
|
+
}
|
56
|
+
|
57
|
+
# If the charge was recurring, make sure that all other local recurring
|
58
|
+
# charges are marked inactive.
|
59
|
+
if charge.recurring?
|
60
|
+
self.cancel_recurring_charges(shop, charge)
|
61
|
+
end
|
62
|
+
|
63
|
+
charge.active!
|
64
|
+
|
65
|
+
true
|
66
|
+
rescue
|
67
|
+
false
|
68
|
+
end
|
69
|
+
end
|
70
|
+
|
71
|
+
# Cancel all recurring charges for the given shop. If the optional charge
|
72
|
+
# parameter is given, it will be excluded from the cancellation.
|
73
|
+
def self.cancel_recurring_charges(shop, charge = nil)
|
74
|
+
charges = DiscoApp::RecurringApplicationCharge.where(shop: shop)
|
75
|
+
if charge.present?
|
76
|
+
charges = charges.where.not(id: charge)
|
77
|
+
end
|
78
|
+
charges.update_all(status: DiscoApp::RecurringApplicationCharge.statuses[:cancelled])
|
79
|
+
end
|
80
|
+
|
81
|
+
end
|
@@ -0,0 +1,17 @@
|
|
1
|
+
class DiscoApp::ProxyService
|
2
|
+
|
3
|
+
# Return true iff the signature provided in the given query string matches
|
4
|
+
# that calculated from the remaining query parameters and the given secret.
|
5
|
+
def self.proxy_signature_is_valid?(query_string, secret)
|
6
|
+
query_hash = Rack::Utils.parse_query(query_string)
|
7
|
+
signature = query_hash.delete('signature').to_s
|
8
|
+
ActiveSupport::SecurityUtils.variable_size_secure_compare(self.calculated_signature(query_hash, secret), signature)
|
9
|
+
end
|
10
|
+
|
11
|
+
# Return the calculated signature for the given query hash and secret.
|
12
|
+
def self.calculated_signature(query_hash, secret)
|
13
|
+
sorted_params = query_hash.collect{ |k, v| "#{k}=#{Array(v).join(',')}" }.sort.join
|
14
|
+
OpenSSL::HMAC.hexdigest(OpenSSL::Digest.new('sha256'), secret, sorted_params)
|
15
|
+
end
|
16
|
+
|
17
|
+
end
|
@@ -0,0 +1,46 @@
|
|
1
|
+
class DiscoApp::SubscriptionService
|
2
|
+
|
3
|
+
# Subscribe the given shop to the given plan, optionally using the given plan
|
4
|
+
# code and optionally tracking the subscription source.
|
5
|
+
def self.subscribe(shop, plan, plan_code = nil, source = nil)
|
6
|
+
|
7
|
+
# If a plan code was provided, fetch it for the given plan.
|
8
|
+
plan_code_instance = nil
|
9
|
+
if plan_code.present?
|
10
|
+
plan_code_instance = DiscoApp::PlanCode.available.find_by(plan: plan, code: plan_code)
|
11
|
+
end
|
12
|
+
|
13
|
+
# Cancel any existing current subscriptions.
|
14
|
+
shop.subscriptions.current.update_all(
|
15
|
+
status: DiscoApp::Subscription.statuses[:cancelled],
|
16
|
+
cancelled_at: Time.now
|
17
|
+
)
|
18
|
+
|
19
|
+
# Get the amount that should be charged for the subscription.
|
20
|
+
subscription_amount = plan_code_instance.present? ? plan_code_instance.amount : plan.amount
|
21
|
+
|
22
|
+
# Get the date the subscription trial should end.
|
23
|
+
subscription_trial_period_days = plan_code_instance.present? ? plan_code_instance.trial_period_days : plan.trial_period_days
|
24
|
+
|
25
|
+
# Create the new subscription.
|
26
|
+
new_subscription = DiscoApp::Subscription.create!(
|
27
|
+
shop: shop,
|
28
|
+
plan: plan,
|
29
|
+
plan_code: plan_code_instance,
|
30
|
+
status: DiscoApp::Subscription.statuses[plan.has_trial? ? :trial : :active],
|
31
|
+
subscription_type: plan.plan_type,
|
32
|
+
amount: subscription_amount,
|
33
|
+
trial_period_days: plan.has_trial? ? subscription_trial_period_days : nil,
|
34
|
+
trial_start_at: plan.has_trial? ? Time.now : nil,
|
35
|
+
trial_end_at: plan.has_trial? ? subscription_trial_period_days.days.from_now : nil,
|
36
|
+
source: source
|
37
|
+
)
|
38
|
+
|
39
|
+
# Enqueue the subscription changed background job.
|
40
|
+
DiscoApp::SubscriptionChangedJob.perform_later(shop.shopify_domain, new_subscription)
|
41
|
+
|
42
|
+
# Return the new subscription.
|
43
|
+
new_subscription
|
44
|
+
end
|
45
|
+
|
46
|
+
end
|
@@ -0,0 +1,30 @@
|
|
1
|
+
class DiscoApp::WebhookService
|
2
|
+
|
3
|
+
# Return true iff the provided hmac_to_verify matches that calculated from the
|
4
|
+
# given data and secret.
|
5
|
+
def self.is_valid_hmac?(body, secret, hmac_to_verify)
|
6
|
+
ActiveSupport::SecurityUtils.secure_compare(self.calculated_hmac(body, secret), hmac_to_verify.to_s)
|
7
|
+
end
|
8
|
+
|
9
|
+
# Calculate the HMAC for the given data and secret.
|
10
|
+
def self.calculated_hmac(body, secret)
|
11
|
+
digest = OpenSSL::Digest.new('sha256')
|
12
|
+
Base64.encode64(OpenSSL::HMAC.digest(digest, secret, body)).strip
|
13
|
+
end
|
14
|
+
|
15
|
+
# Try to find a job class for the given webhook topic.
|
16
|
+
def self.find_job_class(topic)
|
17
|
+
begin
|
18
|
+
# First try to find a top-level matching job class.
|
19
|
+
"#{topic}_job".gsub('/', '_').classify.constantize
|
20
|
+
rescue NameError
|
21
|
+
# If that fails, try to find a DiscoApp:: prefixed job class.
|
22
|
+
begin
|
23
|
+
%Q{DiscoApp::#{"#{topic}_job".gsub('/', '_').classify}}.constantize
|
24
|
+
rescue NameError
|
25
|
+
nil
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
end
|
@@ -0,0 +1,75 @@
|
|
1
|
+
<section class="section">
|
2
|
+
<div class="layout-content">
|
3
|
+
<section class="layout-content__main">
|
4
|
+
<div class="row">
|
5
|
+
<div class="col-md-12">
|
6
|
+
<div class="next-card">
|
7
|
+
|
8
|
+
<header class="next-card__header">
|
9
|
+
<h3>Plan</h3>
|
10
|
+
</header>
|
11
|
+
|
12
|
+
<section class="next-card__section">
|
13
|
+
<div class="form-group">
|
14
|
+
<%= f.label(:name, 'Name') %>
|
15
|
+
<%= f.text_field(:name) %>
|
16
|
+
</div>
|
17
|
+
|
18
|
+
<div class="form-group">
|
19
|
+
<%= f.label(:status, 'Status') %>
|
20
|
+
<%= f.select(:status, DiscoApp::Plan.statuses.map { |s| [s.first.humanize, s.first] }) %>
|
21
|
+
</div>
|
22
|
+
|
23
|
+
<div class="form-group">
|
24
|
+
<%= f.label(:plan_type, 'Plan Type') %>
|
25
|
+
<%= f.select(:plan_type, DiscoApp::Plan.plan_types.map { |s| [s.first.humanize, s.first] }) %>
|
26
|
+
</div>
|
27
|
+
|
28
|
+
<div class="form-group">
|
29
|
+
<%= f.label(:trial_period_days, 'Trial Period Days') %>
|
30
|
+
<%= f.number_field(:trial_period_days) %>
|
31
|
+
</div>
|
32
|
+
|
33
|
+
<div class="form-group">
|
34
|
+
<%= f.label(:amount, 'Amount') %>
|
35
|
+
<%= f.number_field(:amount) %>
|
36
|
+
</div>
|
37
|
+
</section>
|
38
|
+
</div>
|
39
|
+
</div>
|
40
|
+
<div class="col-md-12">
|
41
|
+
<div class="next-card">
|
42
|
+
|
43
|
+
<header class="next-card__header">
|
44
|
+
<h3>Plan Codes</h3>
|
45
|
+
</header>
|
46
|
+
|
47
|
+
<section class="next-card__section">
|
48
|
+
<%= f.fields_for :plan_codes do |plan_code| %>
|
49
|
+
<div class="row">
|
50
|
+
<div class="col-md-24">
|
51
|
+
<%= plan_code.label(:code, 'Code') %>
|
52
|
+
<%= plan_code.text_field(:code) %>
|
53
|
+
|
54
|
+
<%= plan_code.label(:trial_period_days, 'Trial Period Days') %>
|
55
|
+
<%= plan_code.number_field(:trial_period_days) %>
|
56
|
+
|
57
|
+
<%= plan_code.label(:amount, 'Amount') %>
|
58
|
+
<%= plan_code.number_field(:amount) %>
|
59
|
+
<hr>
|
60
|
+
</div>
|
61
|
+
</div>
|
62
|
+
<% end %>
|
63
|
+
</section>
|
64
|
+
</div>
|
65
|
+
</div>
|
66
|
+
</div>
|
67
|
+
<hr />
|
68
|
+
<div class="row">
|
69
|
+
<div class="col-md-12">
|
70
|
+
<%= f.submit 'Save', { class: 'btn btn-primary' } %>
|
71
|
+
</div>
|
72
|
+
</div>
|
73
|
+
</section>
|
74
|
+
</div>
|
75
|
+
</section>
|