UNPKG

ynkap-payment

Version:

Module de paiement Y-Nkap pour Angular - Intégration simple des paiements mobiles (Orange Money, MTN Mobile Money)

1 lines 158 kB
{"version":3,"file":"ynkap-payment.mjs","sources":["../../../projects/ynkap/src/lib/ynkap.service.ts","../../../projects/ynkap/src/lib/configuration/configuration.service.ts","../../../projects/ynkap/src/lib/ynkap.component.ts","../../../projects/ynkap/src/lib/error-handling/models/error.model.ts","../../../projects/ynkap/src/lib/error-handling/error-handling.service.ts","../../../projects/ynkap/src/lib/auth/auth.service.ts","../../../projects/ynkap/src/lib/payment/retry/retry.service.ts","../../../projects/ynkap/src/lib/shared/logos-base64.ts","../../../projects/ynkap/src/lib/shared/logo.service.ts","../../../projects/ynkap/src/lib/payment/payment.service.ts","../../../projects/ynkap/src/lib/configuration/configuration/configuration.component.ts","../../../projects/ynkap/src/lib/configuration/configuration/configuration.component.html","../../../projects/ynkap/src/lib/configuration/configuration.module.ts","../../../projects/ynkap/src/lib/payment/payment/payment.animations.ts","../../../projects/ynkap/src/lib/shared/pipes/fcfa.pipe.ts","../../../projects/ynkap/src/lib/payment/payment/payment.component.ts","../../../projects/ynkap/src/lib/payment/payment/payment.component.html","../../../projects/ynkap/src/lib/shared/shared.module.ts","../../../projects/ynkap/src/lib/payment/payment.module.ts","../../../projects/ynkap/src/lib/error-handling/error-handling/error-handling.component.ts","../../../projects/ynkap/src/lib/error-handling/error-handling/error-handling.component.html","../../../projects/ynkap/src/lib/error-handling/error-handling.module.ts","../../../projects/ynkap/src/lib/shared/modal-stability.service.ts","../../../projects/ynkap/src/lib/pay-button/pay-button.component.ts","../../../projects/ynkap/src/lib/pay-button/pay-button.component.html","../../../projects/ynkap/src/lib/pay-button/pay-button.module.ts","../../../projects/ynkap/src/lib/ynkap.module.ts","../../../projects/ynkap/src/public-api.ts","../../../projects/ynkap/src/ynkap-payment.ts"],"sourcesContent":["import { Injectable } from '@angular/core';\n\n@Injectable({\n providedIn: 'root'\n})\nexport class YnkapService {\n\n constructor() { }\n}\n","import { Injectable } from '@angular/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport { ApiConfig } from './models/api-config.model';\n\n@Injectable({\n providedIn: 'root'\n})\nexport class ConfigurationService {\n private readonly defaultApiBaseUrl = 'https://api.ynkap.com/v1';\n private readonly sandboxApiBaseUrl = 'https://sandbox.api.ynkap.com/v1';\n \n private configSubject = new BehaviorSubject<ApiConfig | null>(null);\n private initialized = false;\n private _apiUrl = 'https://api.ynkap.com/v1';\n\n /**\n * Get the current API configuration\n */\n get config$(): Observable<ApiConfig | null> {\n return this.configSubject.asObservable();\n }\n\n /**\n * Get the current API configuration synchronously\n */\n get currentConfig(): ApiConfig | null {\n return this.configSubject.getValue();\n }\n\n get apiUrl(): string {\n return this._apiUrl;\n }\n\n set apiUrl(url: string) {\n this._apiUrl = url;\n }\n\n constructor() { }\n\n /**\n * Initialize the module with API configuration\n * @param config The API configuration\n * @returns True if initialization was successful, false otherwise\n */\n initialize(config: ApiConfig): boolean {\n if (!this.validateConfig(config)) {\n console.error('Y-Nkap: Invalid configuration provided');\n return false;\n }\n\n // Set defaults if not provided\n const completeConfig: ApiConfig = {\n ...config,\n environment: config.environment || 'production',\n apiBaseUrl: config.apiBaseUrl || (config.environment === 'sandbox' ? this.sandboxApiBaseUrl : this.defaultApiBaseUrl)\n };\n\n this.configSubject.next(completeConfig);\n this.initialized = true;\n return true;\n }\n\n /**\n * Check if the module has been initialized\n */\n isInitialized(): boolean {\n return this.initialized;\n }\n\n /**\n * Reset the configuration\n */\n reset(): void {\n this.configSubject.next(null);\n this.initialized = false;\n }\n\n /**\n * Update part of the configuration\n * @param partialConfig Partial configuration to update\n * @returns True if update was successful, false otherwise\n */\n updateConfig(partialConfig: Partial<ApiConfig>): boolean {\n const currentConfig = this.currentConfig;\n if (!currentConfig) {\n console.error('Y-Nkap: Cannot update config before initialization');\n return false;\n }\n\n const updatedConfig = { ...currentConfig, ...partialConfig };\n if (!this.validateConfig(updatedConfig)) {\n console.error('Y-Nkap: Invalid configuration update');\n return false;\n }\n\n this.configSubject.next(updatedConfig);\n return true;\n }\n\n /**\n * Validate API configuration\n * @param config The API configuration to validate\n * @returns True if configuration is valid, false otherwise\n */\n private validateConfig(config: ApiConfig): boolean {\n // Check if required fields are present and not empty\n if (!config.apiKey || config.apiKey.trim() === '') {\n console.error('Y-Nkap: API Key is required');\n return false;\n }\n\n if (!config.apiSecret || config.apiSecret.trim() === '') {\n console.error('Y-Nkap: API Secret is required');\n return false;\n }\n\n if (!config.merchantId || config.merchantId.trim() === '') {\n console.error('Y-Nkap: Merchant ID is required');\n return false;\n }\n\n // Validate environment if provided\n if (config.environment && !['production', 'sandbox'].includes(config.environment)) {\n console.error('Y-Nkap: Environment must be either \"production\" or \"sandbox\"');\n return false;\n }\n\n return true;\n }\n}\n","import { Component, Inject, Input, OnInit, Optional } from '@angular/core';\nimport { ConfigurationService } from './configuration/configuration.service';\nimport { ApiConfig } from './configuration/models/api-config.model';\n\n@Component({\n selector: 'lib-ynkap',\n template: `\n <div class=\"ynkap-container\">\n <!-- Message d'accueil ou d'erreur -->\n <div *ngIf=\"!isInitialized\" class=\"ynkap-error-message\">\n Y-Nkap n'est pas initialisé. Veuillez fournir une configuration valide avec vos clés d'API.\n </div>\n \n <!-- Message de bienvenue si initialisé -->\n <div *ngIf=\"isInitialized\" class=\"ynkap-welcome\">\n <div class=\"ynkap-logo\">\n <img src=\"https://placeholder-for-ynkap-logo.svg\" alt=\"Y-Nkap\" />\n </div>\n <h2 class=\"ynkap-title\">Y-Nkap - La passerelle de paiement sécurisée</h2>\n <p class=\"ynkap-description\">\n Intégrez facilement les paiements en ligne dans votre application avec Y-Nkap.\n Utilisez les composants spécifiques pour accéder aux fonctionnalités :\n </p>\n <ul class=\"ynkap-components-list\">\n <li><code>&lt;lib-ynkap-payment&gt;</code> - Module de paiement</li>\n <li><code>&lt;lib-configuration&gt;</code> - Configuration de l'API</li>\n </ul>\n <div class=\"ynkap-version\">Version: {{version}}</div>\n </div>\n </div>\n `,\n styles: [`\n .ynkap-container {\n font-family: 'Roboto', 'Segoe UI', Arial, sans-serif;\n padding: 20px;\n border-radius: 8px;\n background-color: #f9f9f9;\n color: #333;\n max-width: 600px;\n margin: 0 auto;\n }\n \n .ynkap-error-message {\n background-color: #ffeaea;\n border-left: 4px solid #f44336;\n padding: 12px 15px;\n border-radius: 4px;\n color: #d32f2f;\n font-size: 0.9rem;\n }\n \n .ynkap-welcome {\n text-align: center;\n }\n \n .ynkap-logo {\n margin: 0 auto 20px;\n width: 80px;\n height: 80px;\n background-color: #007bff;\n border-radius: 50%;\n display: flex;\n align-items: center;\n justify-content: center;\n color: white;\n font-weight: bold;\n font-size: 1.5rem;\n }\n \n .ynkap-title {\n font-size: 1.5rem;\n color: #2c3e50;\n margin-bottom: 15px;\n }\n \n .ynkap-description {\n font-size: 1rem;\n line-height: 1.5;\n color: #666;\n margin-bottom: 20px;\n }\n \n .ynkap-components-list {\n text-align: left;\n width: max-content;\n margin: 0 auto 20px;\n padding-left: 20px;\n }\n \n .ynkap-components-list li {\n margin-bottom: 8px;\n }\n \n .ynkap-components-list code {\n background-color: #e9ecef;\n padding: 2px 6px;\n border-radius: 4px;\n color: #007bff;\n font-family: monospace;\n }\n \n .ynkap-version {\n font-size: 0.8rem;\n color: #999;\n margin-top: 20px;\n }\n `]\n})\nexport class YnkapComponent implements OnInit {\n /**\n * Version actuelle du module Y-Nkap\n */\n version = '1.0.0-dev';\n \n /**\n * Indique si le module est correctement initialisé\n */\n isInitialized = false;\n \n /**\n * Configuration optionnelle fournie directement au composant\n */\n @Input() config?: ApiConfig;\n \n constructor(\n private configService: ConfigurationService,\n @Optional() @Inject('Y_NKAP_CONFIG') private moduleConfig?: ApiConfig\n ) {}\n \n ngOnInit(): void {\n // Vérifie si le service est déjà initialisé\n this.isInitialized = this.configService.isInitialized();\n \n // Si non initialisé mais config disponible, initialiser\n if (!this.isInitialized) {\n // Priorité à la config fournie par input, puis celle du module\n const config = this.config || this.moduleConfig;\n if (config) {\n this.isInitialized = this.configService.initialize(config);\n }\n }\n }\n \n /**\n * Initialise manuellement le module Y-Nkap avec une configuration\n * @param config Configuration de l'API Y-Nkap\n * @returns true si l'initialisation a réussi, false sinon\n */\n initialize(config: ApiConfig): boolean {\n this.isInitialized = this.configService.initialize(config);\n return this.isInitialized;\n }\n}\n","/**\n * Interface for Y-Nkap API Error\n */\nexport interface YnkapError {\n /**\n * Error code\n */\n code: string;\n \n /**\n * Human-readable error message\n */\n message: string;\n \n /**\n * Timestamp when the error occurred\n */\n timestamp: Date;\n \n /**\n * Detailed information about the error\n */\n details?: any;\n \n /**\n * Original error object if available\n */\n originalError?: any;\n \n /**\n * HTTP status code associated with the error\n */\n httpStatus?: number;\n \n /**\n * Indicates if the operation can be retried\n */\n retryable?: boolean;\n \n /**\n * Suggested wait time in milliseconds before retrying\n */\n retryAfterMs?: number;\n}\n\n/**\n * Enum for error categories\n */\nexport enum ErrorCategory {\n NETWORK = 'NETWORK',\n AUTHENTICATION = 'AUTHENTICATION',\n AUTHORIZATION = 'AUTHORIZATION',\n VALIDATION = 'VALIDATION',\n PAYMENT_PROCESSING = 'PAYMENT_PROCESSING',\n SERVER = 'SERVER',\n UNKNOWN = 'UNKNOWN'\n}\n\n/**\n * Interface for error retry options\n */\nexport interface RetryOptions {\n /**\n * Maximum number of retry attempts\n */\n maxRetries: number;\n \n /**\n * Base time in milliseconds to wait between retries (will be used with exponential backoff)\n */\n baseDelayMs: number;\n \n /**\n * Maximum time in milliseconds to wait between retries\n */\n maxDelayMs: number;\n \n /**\n * List of error codes or categories that should not be retried\n */\n nonRetryableErrors?: string[];\n \n /**\n * Callback function to execute before each retry attempt\n */\n onRetry?: (error: YnkapError, attemptNumber: number) => void;\n}\n\nexport class YnkapError extends Error {\n constructor(\n public code: string,\n public override message: string,\n public details?: any\n ) {\n super(message);\n this.name = 'YnkapError';\n Object.setPrototypeOf(this, YnkapError.prototype);\n }\n}\n","import { Injectable } from '@angular/core';\nimport { BehaviorSubject, Observable, throwError, timer } from 'rxjs';\nimport { mergeMap } from 'rxjs/operators';\nimport { ErrorCategory, RetryOptions, YnkapError } from './models/error.model';\n\n@Injectable({\n providedIn: 'root'\n})\nexport class ErrorHandlingService {\n private errorSubject = new BehaviorSubject<YnkapError | null>(null);\n \n // Default retry options\n private defaultRetryOptions: RetryOptions = {\n maxRetries: 3,\n baseDelayMs: 1000,\n maxDelayMs: 10000,\n nonRetryableErrors: [\n ErrorCategory.AUTHENTICATION,\n ErrorCategory.AUTHORIZATION,\n ErrorCategory.VALIDATION\n ]\n };\n\n constructor() {}\n\n handleError(error: any): void {\n let ynkapError: YnkapError;\n\n if (error instanceof YnkapError) {\n ynkapError = error;\n } else {\n ynkapError = new YnkapError(\n error.code || 'UNKNOWN_ERROR',\n error.message || 'Une erreur inattendue est survenue',\n error.details\n );\n }\n\n this.errorSubject.next(ynkapError);\n console.error('Y-Nkap Error:', ynkapError);\n }\n\n getError(): Observable<YnkapError | null> {\n return this.errorSubject.asObservable();\n }\n\n clearError(): void {\n this.errorSubject.next(null);\n }\n\n /**\n * Apply retry logic with exponential backoff for failed operations\n * @param error The error that occurred\n * @param retryFn Function to retry\n * @param options Retry options\n * @returns An observable that will retry the operation\n */\n retryWithBackoff<T>(error: YnkapError, retryFn: () => Observable<T>, options?: Partial<RetryOptions>): Observable<T> {\n const retryOpts: RetryOptions = { ...this.defaultRetryOptions, ...options };\n \n // Check if error is retryable\n if (!error.retryable) {\n return throwError(() => error);\n }\n \n // Check if the error type is in the non-retryable list\n if (retryOpts.nonRetryableErrors?.includes(error.code)) {\n return throwError(() => error);\n }\n \n // Check if we've exceeded max retries\n const attemptNumber = (error.details?.attemptNumber || 0) + 1;\n if (attemptNumber > retryOpts.maxRetries) {\n return throwError(() => error);\n }\n \n // Calculate backoff delay with exponential increase\n const delayMs = Math.min(\n retryOpts.baseDelayMs * Math.pow(2, attemptNumber - 1),\n retryOpts.maxDelayMs\n );\n \n // Call the onRetry callback if provided\n if (retryOpts.onRetry) {\n retryOpts.onRetry(error, attemptNumber);\n }\n \n // Update attempt number in error details\n error.details = { ...error.details, attemptNumber };\n \n // Wait for the delay then retry\n return timer(delayMs).pipe(\n mergeMap(() => retryFn())\n );\n }\n\n /**\n * Convert any error to a standardized YnkapError\n * @param error The error to convert\n * @param context Optional context information\n * @returns A standardized YnkapError\n */\n private convertToYnkapError(error: any, context?: string): YnkapError {\n const timestamp = new Date();\n // HTTP errors\n if (error && error.status) {\n const httpStatus = error.status;\n const errorBody = error.error || {};\n let category: ErrorCategory;\n let retryable = false;\n switch (true) {\n case httpStatus === 401:\n category = ErrorCategory.AUTHENTICATION;\n retryable = false;\n break;\n case httpStatus === 403:\n category = ErrorCategory.AUTHORIZATION;\n retryable = false;\n break;\n case httpStatus >= 400 && httpStatus < 500:\n category = ErrorCategory.VALIDATION;\n retryable = false;\n break;\n case httpStatus >= 500:\n category = ErrorCategory.SERVER;\n retryable = true;\n break;\n default:\n category = ErrorCategory.UNKNOWN;\n retryable = true;\n }\n return new YnkapError(\n errorBody.code || category,\n errorBody.message || `Error ${httpStatus}: ${error.statusText || 'Unknown HTTP error'}`,\n {\n details: errorBody.details || {},\n originalError: error,\n httpStatus,\n retryable,\n timestamp\n }\n );\n } \n // Network errors\n else if (error instanceof TypeError && error.message.includes('network')) {\n return new YnkapError(\n ErrorCategory.NETWORK,\n 'Network error: Could not connect to Y-Nkap API',\n {\n originalError: error,\n retryable: true,\n retryAfterMs: 3000,\n timestamp\n }\n );\n } \n // Other errors\n else {\n return new YnkapError(\n ErrorCategory.UNKNOWN,\n error?.message || 'An unknown error occurred',\n {\n originalError: error,\n retryable: true,\n details: { context },\n timestamp\n }\n );\n }\n }\n}\n","import { Injectable } from '@angular/core';\r\nimport { BehaviorSubject, Observable } from 'rxjs';\r\n\r\n@Injectable({\r\n providedIn: 'root'\r\n})\r\nexport class AuthService {\r\n private apiKeySubject = new BehaviorSubject<string | null>(null);\r\n private isAuthenticatedSubject = new BehaviorSubject<boolean>(false);\r\n\r\n constructor() {\r\n // Vérifier si une clé API existe dans le localStorage\r\n const savedApiKey = localStorage.getItem('ynkap_api_key');\r\n if (savedApiKey) {\r\n this.setApiKey(savedApiKey);\r\n }\r\n }\r\n\r\n /**\r\n * Définit la clé API Y-Nkap\r\n * @param apiKey La clé API à utiliser\r\n */\r\n setApiKey(apiKey: string): void {\r\n if (!this.validateApiKey(apiKey)) {\r\n throw new Error('Clé API invalide');\r\n }\r\n localStorage.setItem('ynkap_api_key', apiKey);\r\n this.apiKeySubject.next(apiKey);\r\n this.isAuthenticatedSubject.next(true);\r\n }\r\n\r\n /**\r\n * Récupère la clé API actuelle\r\n * @returns Observable de la clé API\r\n */\r\n getApiKey(): Observable<string | null> {\r\n return this.apiKeySubject.asObservable();\r\n }\r\n\r\n /**\r\n * Vérifie si l'utilisateur est authentifié\r\n * @returns Observable de l'état d'authentification\r\n */\r\n isAuthenticated(): Observable<boolean> {\r\n return this.isAuthenticatedSubject.asObservable();\r\n }\r\n\r\n /**\r\n * Déconnecte l'utilisateur\r\n */\r\n logout(): void {\r\n localStorage.removeItem('ynkap_api_key');\r\n this.apiKeySubject.next(null);\r\n this.isAuthenticatedSubject.next(false);\r\n }\r\n\r\n /**\r\n * Valide le format de la clé API\r\n * @param apiKey La clé API à valider\r\n * @returns true si la clé est valide\r\n */\r\n private validateApiKey(apiKey: string): boolean {\r\n // Format attendu : ykap_xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\r\n const apiKeyRegex = /^ynkap_[a-zA-Z0-9]{32}$/;\r\n return apiKeyRegex.test(apiKey);\r\n }\r\n} ","import { Injectable } from '@angular/core';\nimport { Observable, timer, throwError, switchMap, tap } from 'rxjs';\nimport { YnkapError, RetryOptions, ErrorCategory } from '../../error-handling/models/error.model';\n\nexport interface RetryState {\n attemptNumber: number;\n maxRetries: number;\n nextRetryDelayMs: number;\n canRetry: boolean;\n lastError?: YnkapError;\n transactionId?: string;\n}\n\n@Injectable({\n providedIn: 'root'\n})\nexport class RetryService {\n private retryStates = new Map<string, RetryState>();\n \n private defaultRetryOptions: RetryOptions = {\n maxRetries: 3,\n baseDelayMs: 1000,\n maxDelayMs: 30000,\n nonRetryableErrors: [\n ErrorCategory.AUTHENTICATION,\n ErrorCategory.AUTHORIZATION,\n ErrorCategory.VALIDATION\n ]\n };\n\n /**\n * Initialize retry state for a transaction\n */\n initializeRetryState(transactionId: string, options?: Partial<RetryOptions>): RetryState {\n const retryOptions = { ...this.defaultRetryOptions, ...options };\n \n const state: RetryState = {\n attemptNumber: 0,\n maxRetries: retryOptions.maxRetries,\n nextRetryDelayMs: retryOptions.baseDelayMs,\n canRetry: true,\n transactionId\n };\n \n this.retryStates.set(transactionId, state);\n return state;\n }\n\n /**\n * Get retry state for a transaction\n */\n getRetryState(transactionId: string): RetryState | null {\n return this.retryStates.get(transactionId) || null;\n }\n\n /**\n * Check if an error is retryable\n */\n isRetryable(error: YnkapError, options?: Partial<RetryOptions>): boolean {\n const retryOptions = { ...this.defaultRetryOptions, ...options };\n\n // Check if error explicitly says it's not retryable\n if (error.retryable === false) {\n return false;\n }\n\n // Check if error category is in non-retryable list\n if (retryOptions.nonRetryableErrors?.includes(error.code as ErrorCategory)) {\n return false;\n }\n\n // Network errors are usually retryable\n if (error.code === ErrorCategory.NETWORK) {\n return true;\n }\n\n // Server errors (5xx) are retryable\n if (error.httpStatus && error.httpStatus >= 500) {\n return true;\n }\n\n // Payment processing errors might be retryable\n if (error.code === ErrorCategory.PAYMENT_PROCESSING) {\n return true;\n }\n\n // Default to retryable if not explicitly set to false\n return error.retryable === undefined || error.retryable === true;\n }\n\n /**\n * Calculate next retry delay with exponential backoff\n */\n calculateRetryDelay(attemptNumber: number, options?: Partial<RetryOptions>): number {\n const retryOptions = { ...this.defaultRetryOptions, ...options };\n \n // Exponential backoff: baseDelay * 2^(attempt-1)\n const delay = retryOptions.baseDelayMs * Math.pow(2, attemptNumber - 1);\n \n // Add some jitter to avoid thundering herd\n const jitter = Math.random() * 0.1 * delay;\n \n return Math.min(delay + jitter, retryOptions.maxDelayMs);\n }\n\n /**\n * Update retry state after an attempt\n */\n updateRetryState(transactionId: string, error: YnkapError): RetryState | null {\n const state = this.retryStates.get(transactionId);\n if (!state) return null;\n \n state.attemptNumber++;\n state.lastError = error;\n state.canRetry = state.attemptNumber < state.maxRetries && this.isRetryable(error);\n \n if (state.canRetry) {\n state.nextRetryDelayMs = this.calculateRetryDelay(state.attemptNumber);\n }\n \n return state;\n }\n\n /**\n * Execute retry with proper delay and state management\n */\n executeRetry<T>(\n transactionId: string, \n retryFn: () => Observable<T>,\n options?: Partial<RetryOptions>\n ): Observable<T> {\n const state = this.getRetryState(transactionId);\n \n if (!state || !state.canRetry) {\n return throwError(() => state?.lastError || new YnkapError(\n 'RETRY_EXHAUSTED',\n 'Maximum retry attempts reached'\n ));\n }\n \n const retryOptions = { ...this.defaultRetryOptions, ...options };\n \n // Call onRetry callback if provided\n if (retryOptions.onRetry && state.lastError) {\n retryOptions.onRetry(state.lastError, state.attemptNumber + 1);\n }\n \n // Wait for the calculated delay, then execute retry\n return timer(state.nextRetryDelayMs).pipe(\n switchMap(() => retryFn()),\n tap({\n error: (error) => {\n this.updateRetryState(transactionId, error);\n }\n })\n );\n }\n\n /**\n * Clear retry state for a transaction\n */\n clearRetryState(transactionId: string): void {\n this.retryStates.delete(transactionId);\n }\n\n /**\n * Get human-readable retry information\n */\n getRetryInfo(transactionId: string): string {\n const state = this.getRetryState(transactionId);\n if (!state) return '';\n \n if (!state.canRetry) {\n return 'Nombre maximum de tentatives atteint';\n }\n \n const remainingAttempts = state.maxRetries - state.attemptNumber;\n const nextRetrySeconds = Math.ceil(state.nextRetryDelayMs / 1000);\n \n return `${remainingAttempts} tentative(s) restante(s). Prochaine tentative dans ${nextRetrySeconds}s`;\n }\n\n /**\n * Check if automatic retry should be attempted\n */\n shouldAutoRetry(error: YnkapError): boolean {\n // Auto-retry only for network errors and server errors\n return error.code === ErrorCategory.NETWORK || \n (error.httpStatus !== undefined && error.httpStatus >= 500);\n }\n}\n","\n/**\n * Logos en base64 générés automatiquement\n * Générés le: 2025-06-27T14:26:54.389Z\n */\n\nexport const LOGOS_BASE64 = {\n ynkap: '',\n mtn: '