mirror of
https://github.com/we-promise/sure.git
synced 2026-04-18 19:44:09 +00:00
* Implement a setting to retrieve high res logos * Update _brand_fetch_settings.html.erb * Add fallback for stock tickers also to use Brandfetch * Update security.rb * Update toggle logic for high-res logos setting Signed-off-by: Juan José Mata <jjmata@jjmata.com> * Update security.rb * Update security.rb --------- Signed-off-by: Juan José Mata <jjmata@jjmata.com> Co-authored-by: Juan José Mata <jjmata@jjmata.com>
156 lines
5.3 KiB
Ruby
156 lines
5.3 KiB
Ruby
class Family::AutoMerchantDetector
|
|
Error = Class.new(StandardError)
|
|
|
|
def initialize(family, transaction_ids: [])
|
|
@family = family
|
|
@transaction_ids = transaction_ids
|
|
end
|
|
|
|
def auto_detect
|
|
raise "No LLM provider for auto-detecting merchants" unless llm_provider
|
|
|
|
if scope.none?
|
|
Rails.logger.info("No transactions to auto-detect merchants for family #{family.id}")
|
|
return 0
|
|
else
|
|
Rails.logger.info("Auto-detecting merchants for #{scope.count} transactions for family #{family.id}")
|
|
end
|
|
|
|
result = llm_provider.auto_detect_merchants(
|
|
transactions: transactions_input,
|
|
user_merchants: user_merchants_input,
|
|
family: family
|
|
)
|
|
|
|
unless result.success?
|
|
Rails.logger.error("Failed to auto-detect merchants for family #{family.id}: #{result.error.message}")
|
|
return 0
|
|
end
|
|
|
|
modified_count = 0
|
|
scope.each do |transaction|
|
|
auto_detection = result.data.find { |c| c.transaction_id == transaction.id }
|
|
next unless auto_detection&.business_name.present? && auto_detection&.business_url.present?
|
|
|
|
existing_merchant = transaction.merchant
|
|
|
|
if existing_merchant.nil?
|
|
# Case 1: No merchant - create/find AI merchant and assign
|
|
merchant_id = find_matching_user_merchant(auto_detection)
|
|
merchant_id ||= find_or_create_ai_merchant(auto_detection)&.id
|
|
|
|
if merchant_id.present?
|
|
was_modified = transaction.enrich_attribute(:merchant_id, merchant_id, source: "ai")
|
|
transaction.lock_attr!(:merchant_id)
|
|
modified_count += 1 if was_modified
|
|
end
|
|
|
|
elsif existing_merchant.is_a?(ProviderMerchant) && existing_merchant.source != "ai"
|
|
# Case 2: Has provider merchant (non-AI) - enhance it with AI data
|
|
if enhance_provider_merchant(existing_merchant, auto_detection)
|
|
transaction.lock_attr!(:merchant_id)
|
|
modified_count += 1
|
|
end
|
|
end
|
|
# Case 3: AI merchant or FamilyMerchant - skip (already good or user-set)
|
|
end
|
|
|
|
modified_count
|
|
end
|
|
|
|
private
|
|
attr_reader :family, :transaction_ids
|
|
|
|
# For now, OpenAI only, but this should work with any LLM concept provider
|
|
def llm_provider
|
|
Provider::Registry.get_provider(:openai)
|
|
end
|
|
|
|
def default_logo_provider_url
|
|
"https://cdn.brandfetch.io"
|
|
end
|
|
|
|
def user_merchants_input
|
|
family.merchants.map do |merchant|
|
|
{
|
|
id: merchant.id,
|
|
name: merchant.name
|
|
}
|
|
end
|
|
end
|
|
|
|
def transactions_input
|
|
scope.map do |transaction|
|
|
{
|
|
id: transaction.id,
|
|
amount: transaction.entry.amount.abs,
|
|
classification: transaction.entry.classification,
|
|
description: [ transaction.entry.name, transaction.entry.notes ].compact.reject(&:empty?).join(" "),
|
|
merchant: transaction.merchant&.name
|
|
}
|
|
end
|
|
end
|
|
|
|
def scope
|
|
family.transactions.where(id: transaction_ids)
|
|
.enrichable(:merchant_id)
|
|
.includes(:merchant, :entry)
|
|
end
|
|
|
|
def find_matching_user_merchant(auto_detection)
|
|
user_merchants_input.find { |m| m[:name] == auto_detection.business_name }&.dig(:id)
|
|
end
|
|
|
|
def find_or_create_ai_merchant(auto_detection)
|
|
# Strategy 1: Find existing merchant by website_url (most reliable for deduplication)
|
|
if auto_detection.business_url.present?
|
|
existing = ProviderMerchant.find_by(website_url: auto_detection.business_url)
|
|
return existing if existing
|
|
end
|
|
|
|
# Strategy 2: Find by exact name match
|
|
existing = ProviderMerchant.find_by(source: "ai", name: auto_detection.business_name)
|
|
return existing if existing
|
|
|
|
# Strategy 3: Create new merchant
|
|
ProviderMerchant.create!(
|
|
source: "ai",
|
|
name: auto_detection.business_name,
|
|
website_url: auto_detection.business_url,
|
|
logo_url: build_logo_url(auto_detection.business_url)
|
|
)
|
|
rescue ActiveRecord::RecordInvalid, ActiveRecord::RecordNotUnique
|
|
# Race condition: another process created the merchant between our find and create
|
|
ProviderMerchant.find_by(source: "ai", name: auto_detection.business_name)
|
|
end
|
|
|
|
def build_logo_url(business_url)
|
|
return nil unless Setting.brand_fetch_client_id.present? && business_url.present?
|
|
size = Setting.brand_fetch_logo_size
|
|
"#{default_logo_provider_url}/#{business_url}/icon/fallback/lettermark/w/#{size}/h/#{size}?c=#{Setting.brand_fetch_client_id}"
|
|
end
|
|
|
|
def enhance_provider_merchant(merchant, auto_detection)
|
|
updates = {}
|
|
|
|
# Add website_url if missing
|
|
if merchant.website_url.blank? && auto_detection.business_url.present?
|
|
updates[:website_url] = auto_detection.business_url
|
|
|
|
# Add logo if BrandFetch is configured
|
|
if Setting.brand_fetch_client_id.present?
|
|
size = Setting.brand_fetch_logo_size
|
|
updates[:logo_url] = "#{default_logo_provider_url}/#{auto_detection.business_url}/icon/fallback/lettermark/w/#{size}/h/#{size}?c=#{Setting.brand_fetch_client_id}"
|
|
end
|
|
end
|
|
|
|
return false if updates.empty?
|
|
|
|
merchant.update!(updates)
|
|
true
|
|
rescue ActiveRecord::RecordInvalid => e
|
|
Rails.logger.error("Failed to enhance merchant #{merchant.id}: #{e.message}")
|
|
false
|
|
end
|
|
end
|