Fix exchange rate parity across all document types

- Fix exchange-rate service types to match actual backend response shapes
  (exchangeRate array, activeProvider success/error, used currencies as strings)
- Add ExchangeRateConverter to payments, expenses, and recurring invoices
- Set currency_id from customer currency in invoice/estimate selectCustomer()
- Load globalStore.currencies in ExchangeRateConverter on mount
- Pass driver/key/driver_config params to getSupportedCurrencies in provider modal
- Fix OpenExchangeRateDriver validateConnection to use base=USD (free plan compat)
- Fix checkActiveCurrencies SQLite whereJsonContains with array values
- Remove broken currency/companyCurrency props from ExpenseCreateView, use stores
- Show base currency equivalent in document line items and totals when exchange
  rate is active
This commit is contained in:
Darko Gjorgjijoski
2026-04-06 21:07:50 +02:00
parent e64529468c
commit b0b7d40c73
12 changed files with 213 additions and 54 deletions

View File

@@ -8,15 +8,16 @@ export interface CreateExchangeRateProviderPayload {
key: string
active?: boolean
currencies?: string[]
driver_config?: Record<string, string>
}
// Normalized response types (what callers receive)
export interface ExchangeRateResponse {
exchange_rate: number
exchangeRate: number | null
}
export interface ActiveProviderResponse {
has_active_provider: boolean
exchange_rate: number | null
hasActiveProvider: boolean
}
export interface SupportedCurrenciesResponse {
@@ -24,7 +25,8 @@ export interface SupportedCurrenciesResponse {
}
export interface UsedCurrenciesResponse {
activeUsedCurrencies: Currency[]
activeUsedCurrencies: string[]
allUsedCurrencies: string[]
}
export interface BulkCurrenciesResponse {
@@ -38,8 +40,23 @@ export interface BulkUpdatePayload {
}>
}
export interface ConfigOption {
key: string
value: string
}
export interface ConfigDriversResponse {
exchange_rate_drivers: string[]
exchange_rate_drivers: ConfigOption[]
}
export interface ConfigServersResponse {
currency_converter_servers: ConfigOption[]
}
export interface SupportedCurrenciesParams {
driver: string
key: string
driver_config?: Record<string, string>
}
export const exchangeRateService = {
@@ -73,24 +90,35 @@ export const exchangeRateService = {
},
// Exchange Rates
// Backend returns { exchangeRate: [number] } or { error: string }
async getRate(currencyId: number): Promise<ExchangeRateResponse> {
const { data } = await client.get(`${API.CURRENCIES}/${currencyId}/exchange-rate`)
return data
const raw = data as Record<string, unknown>
if (raw.exchangeRate && Array.isArray(raw.exchangeRate)) {
return { exchangeRate: Number(raw.exchangeRate[0]) ?? null }
}
return { exchangeRate: null }
},
// Backend returns { success: true, message: "provider_active" } or { error: "no_active_provider" }
async getActiveProvider(currencyId: number): Promise<ActiveProviderResponse> {
const { data } = await client.get(`${API.CURRENCIES}/${currencyId}/active-provider`)
return data
const raw = data as Record<string, unknown>
return { hasActiveProvider: raw.success === true }
},
// Currency lists
async getSupportedCurrencies(): Promise<SupportedCurrenciesResponse> {
const { data } = await client.get(API.SUPPORTED_CURRENCIES)
async getSupportedCurrencies(params: SupportedCurrenciesParams): Promise<SupportedCurrenciesResponse> {
const { data } = await client.get(API.SUPPORTED_CURRENCIES, { params })
return data
},
async getUsedCurrencies(): Promise<UsedCurrenciesResponse> {
const { data } = await client.get(API.USED_CURRENCIES)
// Backend returns { activeUsedCurrencies: string[], allUsedCurrencies: string[] }
async getUsedCurrencies(params?: { provider_id?: number }): Promise<UsedCurrenciesResponse> {
const { data } = await client.get(API.USED_CURRENCIES, { params })
return data
},
@@ -105,12 +133,14 @@ export const exchangeRateService = {
},
// Config
// Backend returns { exchange_rate_drivers: Array<{ key, value }> }
async getDrivers(): Promise<ConfigDriversResponse> {
const { data } = await client.get(API.CONFIG, { params: { key: 'exchange_rate_drivers' } })
return data
},
async getCurrencyConverterServers(): Promise<Record<string, unknown>> {
// Backend returns { currency_converter_servers: Array<{ key, value }> }
async getCurrencyConverterServers(): Promise<ConfigServersResponse> {
const { data } = await client.get(API.CONFIG, { params: { key: 'currency_converter_servers' } })
return data
},