mirror of
https://github.com/we-promise/sure.git
synced 2026-05-12 15:15:01 +00:00
* 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>
115 lines
3.3 KiB
Ruby
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
|