Files
sure/app/models/provider/sophtron_adapter.rb
Juan José Mata 81cdccb768 [codex] Complete Sophtron account mapping (#1698)
* Complete Sophtron account mapping

* Clarify Sophtron login challenge flow

* Add Sophtron connection UI timeout

* Treat Sophtron timeout jobs as failed

* Reset failed Sophtron connection state

* Handle stale Sophtron connection jobs

* Advance Sophtron polling timeout

* Shorten Sophtron connection timeout

* Fix Sophtron modal polling updates

* Stabilize Sophtron MFA polling

* Give Sophtron OTP challenges more time

* Clarify Sophtron institution login failures

* Extend Sophtron polling during login progress

* Probe Sophtron accounts after completed MFA step

* Align Sophtron dialogs with design system

* Start Sophtron initial load after linking accounts

* Fix Sophtron initial transaction load

* Fail Sophtron sync without institution connection

* Fix tests

* Wrap Sophtron account linking in transaction

* Wrap Sophtron provider responses

* Fix Sophtron MFA security tests

* Guard Sophtron MFA challenge arrays

* Respect Sophtron initial load window

* Use unique Sophtron MFA answer field ids

* Address Sophtron review follow-ups

* Fix Sophtron transaction sync refresh

* Avoid blocking Sophtron refresh polling

* Move Sophtron account helpers to model

* Keep Sophtron grouping provider-level

* Start new Sophtron institution links

* Isolate Sophtron institution connections

---------

Signed-off-by: Juan José Mata <juanjo.mata@gmail.com>
2026-05-08 15:15:23 +02:00

115 lines
3.3 KiB
Ruby

class Provider::SophtronAdapter < Provider::Base
include Provider::Syncable
include Provider::InstitutionMetadata
# Register this adapter with the factory
Provider::Factory.register("SophtronAccount", self)
# Define which account types this provider supports
def self.supported_account_types
%w[Depository CreditCard Loan Investment]
end
# Returns connection configurations for this provider
def self.connection_configs(family:)
return [] unless family.can_connect_sophtron?
[ {
key: "sophtron",
name: "Sophtron",
description: "Connect to your bank via Sophtron's secure API aggregation service.",
can_connect: true,
new_account_path: ->(accountable_type, return_to) {
Rails.application.routes.url_helpers.select_accounts_sophtron_items_path(
accountable_type: accountable_type,
return_to: return_to,
connect_new_institution: true
)
},
existing_account_path: ->(account_id) {
Rails.application.routes.url_helpers.select_existing_account_sophtron_items_path(
account_id: account_id
)
}
} ]
end
def provider_name
"sophtron"
end
# Build a Sophtron provider instance with family-specific credentials
# Sophtron is now fully per-family - no global credentials supported
# @param family [Family] The family to get credentials for (required)
# @return [Provider::Sophtron, nil] Returns nil if User ID and Access key is not configured
def self.build_provider(family: nil)
return nil unless family.present?
# Get family-specific credentials
sophtron_item = family.configured_sophtron_item
return nil unless sophtron_item&.credentials_configured?
Provider::Sophtron.new(
sophtron_item.user_id,
sophtron_item.access_key,
base_url: sophtron_item.effective_base_url
)
end
def sync_path
Rails.application.routes.url_helpers.sync_sophtron_item_path(item)
end
def item
provider_account.sophtron_item
end
def can_delete_holdings?
false
end
def institution_domain
# Sophtron may provide institution metadata in account data
metadata = provider_account.institution_metadata
return nil unless metadata.present?
domain = metadata["domain"]
url = metadata["url"]
# Derive domain from URL if missing
if domain.blank? && url.present?
begin
domain = URI.parse(url).host&.gsub(/^www\./, "")
rescue URI::InvalidURIError
Rails.logger.warn("Invalid institution URL for Sophtron account #{provider_account.id}: #{url}")
end
end
domain
end
def institution_name
metadata = provider_account.institution_metadata || {}
return nil unless metadata.present?
metadata_name = metadata["name"].presence || metadata["institution_name"].presence
return metadata_name if metadata_name.present?
metadata_user_institution_id = metadata["user_institution_id"].presence || metadata["UserInstitutionID"].presence
return item&.institution_name if metadata_user_institution_id.present? && metadata_user_institution_id == item&.user_institution_id
nil
end
def institution_url
metadata = provider_account.institution_metadata
return nil unless metadata.present?
metadata["url"] || item&.institution_url
end
def institution_color
item&.institution_color
end
end