spree_cm_commissioner 2.5.0.pre.pre6 → 2.5.0.pre.pre8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (100) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/test_and_build_gem.yml +1 -1
  3. data/.tool-versions +1 -1
  4. data/Gemfile.lock +1 -1
  5. data/app/controllers/spree/api/v2/storefront/popular_route_places_controller.rb +7 -1
  6. data/app/controllers/spree/api/v2/storefront/route_places_controller.rb +9 -9
  7. data/app/finders/spree_cm_commissioner/places/find_with_route.rb +10 -10
  8. data/app/finders/spree_cm_commissioner/routes/find_popular.rb +10 -14
  9. data/app/interactors/spree_cm_commissioner/stock/inventory_item_resetter.rb +1 -1
  10. data/app/jobs/spree_cm_commissioner/stock/inventory_items_adjuster_job.rb +4 -4
  11. data/app/models/concerns/spree_cm_commissioner/option_type_attr_type.rb +3 -15
  12. data/app/models/concerns/spree_cm_commissioner/order_state_machine.rb +0 -2
  13. data/app/models/concerns/spree_cm_commissioner/store_metadata.rb +2 -14
  14. data/app/models/concerns/spree_cm_commissioner/tenant_preference.rb +3 -0
  15. data/app/models/concerns/spree_cm_commissioner/variant_options_concern.rb +7 -1
  16. data/app/models/spree_cm_commissioner/guest.rb +0 -13
  17. data/app/models/spree_cm_commissioner/inventory_item.rb +1 -5
  18. data/app/models/spree_cm_commissioner/line_item_decorator.rb +1 -13
  19. data/app/models/spree_cm_commissioner/option_type_decorator.rb +0 -8
  20. data/app/models/spree_cm_commissioner/option_value_decorator.rb +0 -34
  21. data/app/models/spree_cm_commissioner/order_decorator.rb +0 -1
  22. data/app/models/spree_cm_commissioner/product_decorator.rb +2 -3
  23. data/app/models/spree_cm_commissioner/redis_stock/cached_inventory_items_builder.rb +2 -2
  24. data/app/models/spree_cm_commissioner/redis_stock/inventory_updater.rb +2 -2
  25. data/app/models/spree_cm_commissioner/taxon_decorator.rb +0 -1
  26. data/app/models/spree_cm_commissioner/taxonomy_decorator.rb +1 -13
  27. data/app/models/spree_cm_commissioner/variant_decorator.rb +4 -7
  28. data/app/models/spree_cm_commissioner/vendor_decorator.rb +0 -4
  29. data/app/presenters/spree/variants/{visible_options_presenter.rb → visable_options_presenter.rb} +4 -2
  30. data/app/request_schemas/spree_cm_commissioner/popular_route_places_request_schema.rb +12 -0
  31. data/app/request_schemas/spree_cm_commissioner/route_places_request_schema.rb +5 -0
  32. data/app/views/spree/admin/tenants/_form.html.erb +18 -0
  33. data/app/views/spree/admin/tenants/form/_footer.html.erb +31 -0
  34. data/app/views/spree/admin/tenants/form/_social.html.erb +31 -0
  35. data/app/views/spree/order_mailer/confirm_email.html.erb +1 -1
  36. data/app/views/spree_cm_commissioner/layouts/order_mailer.html.erb +1 -1
  37. data/app/views/spree_cm_commissioner/order_mailer/tenant/_footer.html.erb +13 -6
  38. data/app/views/spree_cm_commissioner/order_mailer/tenant/_support_contact.html.erb +23 -24
  39. data/config/locales/en.yml +0 -8
  40. data/config/locales/km.yml +0 -8
  41. data/config/routes.rb +0 -8
  42. data/db/migrate/20251209022924_add_contact_fields_to_cm_tenants.rb +9 -0
  43. data/lib/cm_app_logger.rb +0 -1
  44. data/lib/spree_cm_commissioner/version.rb +1 -1
  45. data/lib/spree_cm_commissioner.rb +7 -8
  46. metadata +7 -57
  47. data/app/controllers/spree/admin/integration_mappings_controller.rb +0 -21
  48. data/app/controllers/spree/admin/integration_sessions_controller.rb +0 -21
  49. data/app/controllers/spree/admin/integrations_controller.rb +0 -83
  50. data/app/controllers/spree/api/v2/storefront/event_matches_controller.rb +0 -15
  51. data/app/errors/spree_cm_commissioner/integrations/external_client_error.rb +0 -10
  52. data/app/errors/spree_cm_commissioner/integrations/sync_error.rb +0 -4
  53. data/app/finders/spree_cm_commissioner/events/find_matches.rb +0 -15
  54. data/app/helpers/spree_cm_commissioner/external_integrations_helper.rb +0 -58
  55. data/app/jobs/spree_cm_commissioner/integrations/base_job.rb +0 -39
  56. data/app/jobs/spree_cm_commissioner/integrations/polling_job.rb +0 -53
  57. data/app/jobs/spree_cm_commissioner/integrations/polling_scheduler_job.rb +0 -30
  58. data/app/jobs/spree_cm_commissioner/telegram_alerts/integration_sync_failure_job.rb +0 -17
  59. data/app/models/concerns/spree_cm_commissioner/integrations/integration_mappable.rb +0 -61
  60. data/app/models/concerns/spree_cm_commissioner/line_item_integration.rb +0 -36
  61. data/app/models/concerns/spree_cm_commissioner/order_integration.rb +0 -33
  62. data/app/models/spree_cm_commissioner/integration.rb +0 -29
  63. data/app/models/spree_cm_commissioner/integration_mapping.rb +0 -41
  64. data/app/models/spree_cm_commissioner/integration_sync_session.rb +0 -15
  65. data/app/models/spree_cm_commissioner/integrations/stadium_x_v1.rb +0 -37
  66. data/app/overrides/spree/admin/shared/sub_menu/_integrations/external_integrations.html.erb.deface +0 -6
  67. data/app/services/spree_cm_commissioner/integrations/base/sync_manager.rb +0 -69
  68. data/app/services/spree_cm_commissioner/integrations/base/sync_result.rb +0 -183
  69. data/app/services/spree_cm_commissioner/integrations/polling.rb +0 -70
  70. data/app/services/spree_cm_commissioner/integrations/polling_scheduler.rb +0 -79
  71. data/app/services/spree_cm_commissioner/integrations/stadium_x_v1/external_client/client.rb +0 -152
  72. data/app/services/spree_cm_commissioner/integrations/stadium_x_v1/inventory/unstock_inventory.rb +0 -83
  73. data/app/services/spree_cm_commissioner/integrations/stadium_x_v1/polling/sync_matches.rb +0 -113
  74. data/app/services/spree_cm_commissioner/integrations/stadium_x_v1/polling/sync_zones.rb +0 -215
  75. data/app/services/spree_cm_commissioner/integrations/stadium_x_v1/resources/base.rb +0 -20
  76. data/app/services/spree_cm_commissioner/integrations/stadium_x_v1/resources/league.rb +0 -19
  77. data/app/services/spree_cm_commissioner/integrations/stadium_x_v1/resources/match.rb +0 -95
  78. data/app/services/spree_cm_commissioner/integrations/stadium_x_v1/resources/ticket.rb +0 -81
  79. data/app/services/spree_cm_commissioner/integrations/stadium_x_v1/resources/ticket_image.rb +0 -19
  80. data/app/services/spree_cm_commissioner/integrations/stadium_x_v1/resources/zone.rb +0 -90
  81. data/app/services/spree_cm_commissioner/integrations/stadium_x_v1/sync_manager.rb +0 -35
  82. data/app/services/spree_cm_commissioner/integrations/stadium_x_v1/sync_strategies/full_sync_strategy.rb +0 -38
  83. data/app/services/spree_cm_commissioner/integrations/stadium_x_v1/sync_strategies/incremental_sync_strategy.rb +0 -44
  84. data/app/services/spree_cm_commissioner/integrations/stadium_x_v1/sync_strategies/webhook_sync_strategy.rb +0 -16
  85. data/app/services/spree_cm_commissioner/telegram_alerts/integration_sync_failure.rb +0 -49
  86. data/app/views/spree/admin/integration_mappings/_integration_mappings.html.erb +0 -107
  87. data/app/views/spree/admin/integration_mappings/index.html.erb +0 -33
  88. data/app/views/spree/admin/integration_sessions/_integration_sync_sessions.html.erb +0 -116
  89. data/app/views/spree/admin/integration_sessions/index.html.erb +0 -42
  90. data/app/views/spree/admin/integrations/_form.html.erb +0 -104
  91. data/app/views/spree/admin/integrations/_stadium_x_v1_fields.html.erb +0 -29
  92. data/app/views/spree/admin/integrations/edit.html.erb +0 -45
  93. data/app/views/spree/admin/integrations/index.html.erb +0 -75
  94. data/app/views/spree/admin/integrations/new.html.erb +0 -25
  95. data/db/migrate/20251017094845_create_cm_integrations.rb +0 -22
  96. data/db/migrate/20251017101555_create_cm_integration_sync_sessions.rb +0 -68
  97. data/db/migrate/20251017101605_create_cm_integration_mappings.rb +0 -52
  98. data/lib/spree_cm_commissioner/test_helper/factories/integration_factory.rb +0 -25
  99. data/lib/spree_cm_commissioner/test_helper/factories/integration_mapping_factory.rb +0 -14
  100. data/lib/spree_cm_commissioner/test_helper/factories/integration_sync_session_factory.rb +0 -7
@@ -1,21 +0,0 @@
1
- module Spree
2
- module Admin
3
- class IntegrationMappingsController < Spree::Admin::BaseController
4
- helper SpreeCmCommissioner::ExternalIntegrationsHelper
5
- before_action :load_integration
6
-
7
- def index
8
- @mappings = @integration.integration_mappings
9
- .order(updated_at: :desc)
10
- .page(params[:page])
11
- .per(params[:per_page] || 25)
12
- end
13
-
14
- private
15
-
16
- def load_integration
17
- @integration ||= SpreeCmCommissioner::Integration.find(params[:integration_id])
18
- end
19
- end
20
- end
21
- end
@@ -1,21 +0,0 @@
1
- module Spree
2
- module Admin
3
- class IntegrationSessionsController < Spree::Admin::BaseController
4
- helper SpreeCmCommissioner::ExternalIntegrationsHelper
5
- before_action :load_integration
6
-
7
- def index
8
- @sync_sessions = @integration.integration_sync_sessions
9
- .order(created_at: :desc)
10
- .page(params[:page])
11
- .per(params[:per_page] || 25)
12
- end
13
-
14
- private
15
-
16
- def load_integration
17
- @integration ||= SpreeCmCommissioner::Integration.find(params[:integration_id])
18
- end
19
- end
20
- end
21
- end
@@ -1,83 +0,0 @@
1
- module Spree
2
- module Admin
3
- class IntegrationsController < Spree::Admin::ResourceController
4
- helper SpreeCmCommissioner::ExternalIntegrationsHelper
5
- before_action :load_form_collections, only: %i[new create edit update]
6
- before_action :load_integration, only: %i[enqueue_polling]
7
-
8
- def index
9
- @integrations = model_class.all
10
- end
11
-
12
- def enqueue_polling
13
- sync_type = params[:sync_type].presence || :full
14
-
15
- if @integration.active?
16
- SpreeCmCommissioner::Integrations::PollingJob.perform_later(
17
- integration_id: @integration.id,
18
- sync_type: sync_type
19
- )
20
- flash[:success] = Spree.t('admin.integration_sync_enqueued')
21
- else
22
- flash[:error] = Spree.t('admin.integration_inactive')
23
- end
24
-
25
- redirect_to admin_integration_sessions_path(@integration)
26
- end
27
-
28
- # override
29
- def model_class
30
- SpreeCmCommissioner::Integration
31
- end
32
-
33
- # override
34
- def collection_url(options = {})
35
- admin_integrations_url(options)
36
- end
37
-
38
- private
39
-
40
- def permitted_resource_params
41
- params.require(:spree_cm_commissioner_integration).permit(
42
- :name,
43
- :type,
44
- :status,
45
- :conflict_strategy,
46
- :incremental_sync_interval_seconds,
47
- :full_sync_interval_hours,
48
- :tenant_id,
49
- :vendor_id,
50
- public_metadata: {},
51
- private_metadata: {}
52
- )
53
- end
54
-
55
- def load_integration
56
- @integration ||= SpreeCmCommissioner::Integration.find(params[:id])
57
- end
58
-
59
- def load_form_collections
60
- @integration_type_options ||= integration_type_options
61
- @status_options ||= enum_to_options(SpreeCmCommissioner::Integration.statuses)
62
- @conflict_strategy_options ||= enum_to_options(SpreeCmCommissioner::Integration.conflict_strategies)
63
- @tenants ||= SpreeCmCommissioner::Tenant.select(:id, :name).order(:name)
64
- @vendors ||= Spree::Vendor.select(:id, :name).order(:name)
65
- end
66
-
67
- def enum_to_options(enum_hash)
68
- enum_hash.keys.map { |key| [key.humanize, key] }
69
- end
70
-
71
- def integration_type_options
72
- files = Dir[SpreeCmCommissioner::Engine.root.join('app/models/spree_cm_commissioner/integrations/*.rb')]
73
-
74
- return [[Spree.t(:type), 'SpreeCmCommissioner::Integration']] if files.blank?
75
-
76
- files.map do |path|
77
- class_name = "SpreeCmCommissioner::Integrations::#{File.basename(path, '.rb').camelize}"
78
- [class_name, class_name]
79
- end.sort_by(&:first)
80
- end
81
- end
82
- end
83
- end
@@ -1,15 +0,0 @@
1
- module Spree
2
- module Api
3
- module V2
4
- module Storefront
5
- class EventMatchesController < TaxonsController
6
- private
7
-
8
- def collection_finder
9
- SpreeCmCommissioner::Events::FindMatches
10
- end
11
- end
12
- end
13
- end
14
- end
15
- end
@@ -1,10 +0,0 @@
1
- module SpreeCmCommissioner::Integrations
2
- class ExternalClientError < StandardError
3
- attr_reader :status_code
4
-
5
- def initialize(message, status_code = nil)
6
- super(message)
7
- @status_code = status_code
8
- end
9
- end
10
- end
@@ -1,4 +0,0 @@
1
- module SpreeCmCommissioner::Integrations
2
- class SyncError < StandardError
3
- end
4
- end
@@ -1,15 +0,0 @@
1
- module SpreeCmCommissioner
2
- module Events
3
- class FindMatches < ::Spree::BaseFinder
4
- # currently, there is no way to determine if an event is a match yet.
5
- # this finder returns all events that have active integration mappings (synced from external systems like StadiumX)
6
- # as a temporary workaround to filter out non-match events.
7
- def execute
8
- scope.joins(:integration_mappings).where(
9
- kind: :event,
10
- integration_mappings: { status: :active }
11
- ).distinct
12
- end
13
- end
14
- end
15
- end
@@ -1,58 +0,0 @@
1
- module SpreeCmCommissioner
2
- module ExternalIntegrationsHelper
3
- def integration_status_badge(status)
4
- badge_class = case status.to_s
5
- when 'active' then 'badge-success'
6
- when 'paused' then 'badge-warning'
7
- else 'badge-secondary'
8
- end
9
-
10
- content_tag(:span, status.to_s.humanize, class: "badge badge-pill #{badge_class}")
11
- end
12
-
13
- def integration_conflict_strategy_badge(strategy)
14
- content_tag(:span, strategy.to_s.humanize, class: 'badge badge-pill badge-secondary')
15
- end
16
-
17
- def integration_interval_badge(value, unit)
18
- content_tag(:span, "#{value}#{unit}", class: 'badge badge-pill badge-info')
19
- end
20
-
21
- def integration_base_url_link(url)
22
- if url.present?
23
- link_to url, url, target: '_blank', rel: 'noopener', class: 'text-decoration-underline'
24
- else
25
- content_tag(:span, '—', class: 'text-muted')
26
- end
27
- end
28
-
29
- def integration_mapping_internal_link(mapping)
30
- return mapping.internal_id if mapping.internal.blank?
31
-
32
- begin
33
- case mapping.internal_type
34
- when 'Spree::Taxon'
35
- taxon = mapping.internal
36
- taxonomy_id = taxon.taxonomy_id || taxon.taxonomy&.id
37
- if taxonomy_id
38
- link_to mapping.internal_id,
39
- edit_admin_taxonomy_taxon_path(taxonomy_id, taxon.id),
40
- class: 'text-primary'
41
- end
42
- when 'Spree::Product'
43
- link_to mapping.internal_id, edit_admin_product_path(mapping.internal), class: 'text-primary'
44
- when 'Spree::Variant'
45
- link_to mapping.internal_id, edit_admin_product_variant_path(mapping.internal.product_id, mapping.internal), class: 'text-primary'
46
- when 'Spree::Vendor'
47
- link_to mapping.internal_id, edit_admin_vendor_path(mapping.internal), class: 'text-primary'
48
- else
49
- # Try polymorphic route as fallback
50
- link_to mapping.internal_id, [:edit, :admin, mapping.internal], class: 'text-primary'
51
- end
52
- rescue StandardError => e
53
- Rails.logger.warn("Failed to generate link for mapping #{mapping.id}: #{e.message}")
54
- mapping.internal_id
55
- end
56
- end
57
- end
58
- end
@@ -1,39 +0,0 @@
1
- module SpreeCmCommissioner
2
- module Integrations
3
- # Base class for integration jobs
4
- #
5
- # Provides common error handling and alerting functionality for all integration jobs.
6
- # Subclasses should implement their own perform logic and call the appropriate
7
- # error handler methods.
8
- class BaseJob < ApplicationJob
9
- queue_as :integration
10
-
11
- protected
12
-
13
- # Handle error by logging and sending alert
14
- # @param label [String] The error label for logging
15
- # @param error [StandardError] The error that occurred
16
- # @param data [Hash] Additional context data (e.g., sync_type, match_id, event_type)
17
- def handle_error(label:, error:, data: {})
18
- error_message = "#{error.class.name}: #{error.message}"
19
-
20
- # Log the error with structured data
21
- CmAppLogger.error(
22
- label: label,
23
- data: data.merge(
24
- message: error_message,
25
- backtrace: error.backtrace
26
- )
27
- )
28
-
29
- # Send alert via Telegram
30
- TelegramAlerts::IntegrationSyncFailureJob.perform_later(
31
- error_message: error_message,
32
- data: data
33
- )
34
-
35
- error_message
36
- end
37
- end
38
- end
39
- end
@@ -1,53 +0,0 @@
1
- module SpreeCmCommissioner
2
- module Integrations
3
- # Job that performs the actual polling (syncing) for a specific integration
4
- #
5
- # This job is enqueued by PollingSchedulerJob and handles:
6
- # - Full syncs: Complete refresh of all data
7
- # - Incremental syncs: Fetch only recent changes
8
- # - Webhook syncs: Real-time event processing (triggered externally)
9
- #
10
- # Each integration type (StadiumXV1, LarrytaV1, etc.) must implement
11
- # a sync_manager that responds to sync_full!, sync_incremental!, and sync_webhook!
12
- class PollingJob < BaseJob
13
- # Perform the integration pull
14
- # @param integration_id [Integer] The integration ID
15
- # @param sync_type [String] The sync type: 'full', 'incremental', or 'webhook_triggered'
16
- # @param event_type [String] Optional event type for webhook syncs
17
- # @param event_data [Hash] Optional event data for webhook syncs
18
- def perform(options)
19
- integration_id = options[:integration_id]
20
- sync_type = options[:sync_type]
21
- event_type = options[:event_type]
22
- event_data = options[:event_data]
23
- integration = SpreeCmCommissioner::Integration.find(integration_id)
24
-
25
- # integration is no longer active, skip processing.
26
- return unless integration.active?
27
-
28
- SpreeCmCommissioner::Integrations::Polling.new.call(
29
- integration: integration,
30
- sync_type: sync_type,
31
- event_type: event_type,
32
- event_data: event_data
33
- )
34
- rescue StandardError => e
35
- handle_sync_error(e, options)
36
- raise
37
- end
38
-
39
- private
40
-
41
- # Handle sync errors and alert
42
- # @param error [StandardError] The error that occurred
43
- # @param options [Hash] The options passed to perform
44
- def handle_sync_error(error, options)
45
- handle_error(
46
- label: 'Integrations::IntegrationPullJob#perform Sync failed',
47
- error: error,
48
- data: options
49
- )
50
- end
51
- end
52
- end
53
- end
@@ -1,30 +0,0 @@
1
- module SpreeCmCommissioner
2
- module Integrations
3
- # Scheduler job that runs periodically to enqueue integration polling jobs
4
- #
5
- # This job is responsible for:
6
- # - Running full syncs every 24 hours
7
- # - Running incremental syncs every 10 seconds
8
- # - Webhook syncs are triggered externally (not scheduled here)
9
- #
10
- # The job queries all active integrations and enqueues appropriate pull jobs
11
- # based on their last sync time and sync type configuration.
12
- class PollingSchedulerJob < BaseJob
13
- def perform
14
- SpreeCmCommissioner::Integrations::PollingScheduler.new.call
15
- rescue StandardError => e
16
- handle_scheduler_error(e)
17
- raise
18
- end
19
-
20
- private
21
-
22
- def handle_scheduler_error(error)
23
- handle_error(
24
- label: 'Integrations::PollingSchedulerJob#perform Error scheduling integration pulls',
25
- error: error
26
- )
27
- end
28
- end
29
- end
30
- end
@@ -1,17 +0,0 @@
1
- module SpreeCmCommissioner
2
- module TelegramAlerts
3
- class IntegrationSyncFailureJob < ApplicationJob
4
- queue_as :telegram_bot
5
-
6
- # Handle error by logging and sending alert
7
- # @param error_message [String] The error message
8
- # @param data [Hash] Additional context data (e.g., sync_type, match_id, event_type)
9
- def perform(options)
10
- SpreeCmCommissioner::TelegramAlerts::IntegrationSyncFailure.call(
11
- error_message: options[:error_message],
12
- data: options[:data] || {}
13
- )
14
- end
15
- end
16
- end
17
- end
@@ -1,61 +0,0 @@
1
- module SpreeCmCommissioner
2
- module Integrations
3
- # IntegrationMappable provides a polymorphic mapping layer between internal records and external integration IDs.
4
- #
5
- # Design Rationale: Why use a separate mapping table instead of adding external_id columns directly?
6
- #
7
- # 1. **No Schema Changes Required**: Adding a new integration doesn't require migrations to add columns to every table.
8
- # The mapping table is reusable across all models that include this concern.
9
- #
10
- # 2. **Polymorphic Scalability**: A single mapping table handles multiple record types (Spree::Product, Spree::Taxon,
11
- # SeatLayout, etc.) without duplicating external_id columns across the schema.
12
- #
13
- # 3. **Multiple Source Support**: Records can have mappings to multiple integrations simultaneously. A single product
14
- # might be synced from StadiumXV1, LarrytaV1, and other sources. Direct columns would require one per integration.
15
- #
16
- # 4. **Acceptable Performance Trade-off**: While joins are slightly slower than direct column lookups, the performance
17
- # impact is negligible in practice because:
18
- # - Lookups occur primarily during background jobs (async, not blocking user requests)
19
- # - Integration sync operations (incremental, full, webhook)
20
- # - Occasional user-triggered operations (not on every page load)
21
- # The flexibility and maintainability gains far outweigh the minimal performance cost.
22
- module IntegrationMappable
23
- extend ActiveSupport::Concern
24
-
25
- included do
26
- has_many :integration_mappings, as: :internal, class_name: 'SpreeCmCommissioner::IntegrationMapping', dependent: :destroy
27
- has_many :external_wins_integration_mappings, lambda {
28
- joins(:integration).where(integration: { conflict_strategy: :external_wins })
29
- }, as: :internal, class_name: 'SpreeCmCommissioner::IntegrationMapping'
30
- end
31
-
32
- class_methods do
33
- def find_or_initialize_integration_mapping(integration_id:, external_id:)
34
- IntegrationMapping.includes(:internal).find_or_initialize_by(
35
- integration_id: integration_id,
36
- internal_type: name,
37
- external_id: external_id
38
- ) do |mapping|
39
- mapping.internal = new
40
- mapping.external_id = external_id
41
- end
42
- end
43
-
44
- def find_integration_mapping(integration_id:, external_id:)
45
- IntegrationMapping.includes(:internal).find_by(
46
- integration_id: integration_id,
47
- internal_type: name,
48
- external_id: external_id
49
- )
50
- end
51
-
52
- def find_oldest_active_mapping(integration_id:)
53
- IntegrationMapping.where(
54
- integration_id: integration_id,
55
- internal_type: name
56
- ).active.order(:last_synced_at).first
57
- end
58
- end
59
- end
60
- end
61
- end
@@ -1,36 +0,0 @@
1
- # Caches the integration status of a line item's variant in private_metadata.
2
- #
3
- # Why cache?
4
- # Most products don't have integrations, so checking `variant.vendor.integration.present?`
5
- # on every read would cause N+1 queries. Caching avoids this performance hit.
6
- #
7
- # Behavior:
8
- # The flag is set once at line item creation and remains immutable. This preserves
9
- # the historical integration state of the order, even if the vendor's integration
10
- # status changes later. We don't have use cases requiring dynamic updates of this yet,
11
- # so we keep it constant for simplicity and consistency.
12
- module SpreeCmCommissioner
13
- module LineItemIntegration
14
- extend ActiveSupport::Concern
15
-
16
- included do
17
- before_create :set_integration_flag
18
- end
19
-
20
- def integration?
21
- return false if private_metadata['integration'].nil?
22
-
23
- ActiveModel::Type::Boolean.new.cast(private_metadata['integration'])
24
- end
25
-
26
- private
27
-
28
- def set_integration_flag
29
- return if private_metadata.key?('integration')
30
- return if variant.nil? || variant.vendor.nil?
31
-
32
- # We use variant.vendor instead of .vendor directly as on create, the association may not be set to line item yet.
33
- private_metadata['integration'] ||= variant.vendor.integration.present?
34
- end
35
- end
36
- end
@@ -1,33 +0,0 @@
1
- module SpreeCmCommissioner
2
- module OrderIntegration
3
- # Checks if any line item in this order has an integration (i.e., is from a vendor with an integration).
4
- # This is used to determine if inventory management operations should be delegated to an external system.
5
- def integration?
6
- line_items.any?(&:integration?)
7
- end
8
-
9
- # Attempts to unstock inventory for all integration line items in this order.
10
- # Groups line items by vendor and delegates to the vendor's integration service.
11
- def unstock_inventory_from_external_system!
12
- integration_line_items = line_items.select(&:integration?)
13
- return if integration_line_items.empty?
14
-
15
- integration_line_items.group_by(&:vendor_id).each do |_, line_items|
16
- integration = line_items.first&.vendor&.integration
17
- integration.unstock_external_inventory!(self, line_items) if integration.present?
18
- end
19
- end
20
-
21
- # Attempts to restock inventory for all integration line items in this order.
22
- # Groups line items by vendor and delegates to the vendor's integration service.
23
- def restock_inventory_from_external_system!
24
- integration_line_items = line_items.select(&:integration?)
25
- return if integration_line_items.empty?
26
-
27
- integration_line_items.group_by(&:vendor_id).each do |_, line_items|
28
- integration = line_items.first&.vendor&.integration
29
- integration.restock_external_inventory!(self, line_items) if integration.present?
30
- end
31
- end
32
- end
33
- end
@@ -1,29 +0,0 @@
1
- module SpreeCmCommissioner
2
- class Integration < Base
3
- include StoreMetadata
4
-
5
- enum status: { inactive: 0, active: 1, paused: 2 }
6
- enum conflict_strategy: { newest_wins: 0, internal_wins: 1, external_wins: 2, manual_resolution: 3 }
7
-
8
- belongs_to :tenant, class_name: 'SpreeCmCommissioner::Tenant', optional: true
9
- belongs_to :vendor, class_name: 'Spree::Vendor', optional: false, inverse_of: :integration
10
-
11
- has_many :integration_mappings, class_name: 'SpreeCmCommissioner::IntegrationMapping', dependent: :destroy, inverse_of: :integration
12
- has_many :integration_sync_sessions, class_name: 'SpreeCmCommissioner::IntegrationSyncSession', dependent: :destroy, inverse_of: :integration
13
-
14
- validates :incremental_sync_interval_seconds, presence: true, numericality: { only_integer: true, greater_than: 0, less_than_or_equal_to: 3600 }
15
- validates :full_sync_interval_hours, presence: true, numericality: { only_integer: true, greater_than: 0, less_than_or_equal_to: 168 }
16
-
17
- def sync_manager
18
- raise NotImplementedError, 'Subclasses must implement the sync_manager method'
19
- end
20
-
21
- def restock_external_inventory!(_order, _line_items)
22
- raise NotImplementedError, 'Subclasses must implement the restock_external_inventory! method'
23
- end
24
-
25
- def unstock_external_inventory!(_order, _line_items)
26
- raise NotImplementedError, 'Subclasses must implement the unstock_external_inventory! method'
27
- end
28
- end
29
- end
@@ -1,41 +0,0 @@
1
- module SpreeCmCommissioner
2
- class IntegrationMapping < Base
3
- include StoreMetadata
4
-
5
- enum status: { active: 0, archived: 1 }
6
-
7
- # polymorphic (Spree::Taxon, Spree::Product, Spree::Variant, Spree::Vendor)
8
- belongs_to :internal, polymorphic: true, optional: false
9
- belongs_to :integration, class_name: 'SpreeCmCommissioner::Integration', inverse_of: :integration_mappings, optional: false
10
-
11
- validates :external_id, presence: true, uniqueness: { scope: %i[integration_id internal_type internal_id date] }
12
- validate :validate_internal_exists, if: -> { internal_type.present? && internal_id.present? }
13
-
14
- # QR data for check-in. If external_wins? is true, use their QR data (they have their own check-in system).
15
- # Otherwise, use our system's QR data. Only applicable to map with internal that has QR support (e.g., line_item, guest).
16
- store_public_metadata :external_qr_data, :string, default: nil
17
-
18
- def mark_as_archived!
19
- self.status = :archived
20
- self.last_synced_at = Time.zone.now
21
-
22
- save!
23
- end
24
-
25
- def mark_as_active!(external_payload:)
26
- self.external_payload = external_payload
27
- self.last_synced_at = Time.zone.now
28
- self.status = :active
29
-
30
- save!
31
- end
32
-
33
- private
34
-
35
- def validate_internal_exists
36
- return if internal_type.safe_constantize&.exists?(id: internal_id)
37
-
38
- errors.add(:internal, "record (#{internal_type}##{internal_id}) does not exist")
39
- end
40
- end
41
- end
@@ -1,15 +0,0 @@
1
- module SpreeCmCommissioner
2
- class IntegrationSyncSession < Base
3
- include StoreMetadata
4
-
5
- enum status: { pending: 0, in_progress: 1, completed: 2, failed: 3, canceled: 4 }
6
- enum :sync_type, {
7
- full: 0, # on initial setup or periodic complete refresh
8
- incremental: 1, # on scheduled intervals to fetch recent changes
9
- webhook_triggered: 2 # immediately when an external webhook event is received
10
- }, prefix: true
11
-
12
- belongs_to :tenant, class_name: 'SpreeCmCommissioner::Tenant', optional: true
13
- belongs_to :integration, class_name: 'SpreeCmCommissioner::Integration', inverse_of: :integration_sync_sessions, optional: false
14
- end
15
- end
@@ -1,37 +0,0 @@
1
- class SpreeCmCommissioner::Integrations::StadiumXV1 < SpreeCmCommissioner::Integration
2
- store_private_metadata :public_key, :string
3
- store_private_metadata :private_key, :string
4
- store_private_metadata :base_url, :string # e.g. 'https://api.stadiumx.com'
5
-
6
- # override
7
- def sync_manager
8
- SpreeCmCommissioner::Integrations::StadiumXV1::SyncManager.new(
9
- integration: self,
10
- client: client
11
- )
12
- end
13
-
14
- # override
15
- def unstock_external_inventory!(order, line_items)
16
- result = SpreeCmCommissioner::Integrations::StadiumXV1::Inventory::UnstockInventory.new.call(
17
- integration: self,
18
- order: order,
19
- line_items: line_items
20
- )
21
-
22
- raise SpreeCmCommissioner::Integrations::SyncError, result.error unless result.success?
23
- end
24
-
25
- # override
26
- def restock_external_inventory!(_order, _line_items)
27
- raise SpreeCmCommissioner::Integrations::SyncError, 'Ticket cannot be cancelled'
28
- end
29
-
30
- def client
31
- SpreeCmCommissioner::Integrations::StadiumXV1::ExternalClient::Client.new(
32
- public_key: public_key,
33
- private_key: private_key,
34
- base_url: base_url
35
- )
36
- end
37
- end
@@ -1,6 +0,0 @@
1
- <!-- insert_bottom "[data-hook='admin_integrations']" -->
2
-
3
- <%= tab :integrations,
4
- match_path: '/integrations',
5
- label: Spree.t('admin.external_integrations_title'),
6
- if: can?(:manage, SpreeCmCommissioner::Integration) %>