@umituz/react-native-firebase 2.6.1 → 2.6.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (87) hide show
  1. package/package.json +1 -1
  2. package/src/application/auth/ports/AuthPort_part_aa +150 -0
  3. package/src/application/auth/ports/AuthPort_part_ab +14 -0
  4. package/src/application/auth/use-cases/SignInUseCaseHelpers.ts +0 -0
  5. package/src/application/auth/use-cases/SignInUseCaseMain.ts +0 -0
  6. package/src/application/auth/use-cases/SignInUseCase_part_aa +150 -0
  7. package/src/application/auth/use-cases/SignInUseCase_part_ab +103 -0
  8. package/src/application/auth/use-cases/SignOutUseCaseCleanup.ts +0 -0
  9. package/src/application/auth/use-cases/SignOutUseCaseMain.ts +0 -0
  10. package/src/application/auth/use-cases/SignOutUseCase_part_aa +150 -0
  11. package/src/application/auth/use-cases/SignOutUseCase_part_ab +138 -0
  12. package/src/domains/account-deletion/domain/services/UserValidationHelpers.ts.bak +181 -0
  13. package/src/domains/account-deletion/domain/services/UserValidationHelpers_part_aa +150 -0
  14. package/src/domains/account-deletion/domain/services/UserValidationHelpers_part_ab +31 -0
  15. package/src/domains/account-deletion/domain/services/{UserValidationService.ts → UserValidationService.ts.bak} +1 -10
  16. package/src/domains/account-deletion/domain/services/UserValidationService_part_aa +150 -0
  17. package/src/domains/account-deletion/domain/services/UserValidationService_part_ab +136 -0
  18. package/src/domains/account-deletion/infrastructure/services/AccountDeletionExecutor_part_aa +150 -0
  19. package/src/domains/account-deletion/infrastructure/services/AccountDeletionExecutor_part_ab +80 -0
  20. package/src/domains/account-deletion/infrastructure/services/AccountDeletionReauthHandler_part_aa +150 -0
  21. package/src/domains/account-deletion/infrastructure/services/AccountDeletionReauthHandler_part_ab +24 -0
  22. package/src/domains/account-deletion/infrastructure/services/AccountDeletionRepository_part_aa +150 -0
  23. package/src/domains/account-deletion/infrastructure/services/AccountDeletionRepository_part_ab +116 -0
  24. package/src/domains/account-deletion/infrastructure/services/reauthentication.service_part_aa +150 -0
  25. package/src/domains/account-deletion/infrastructure/services/reauthentication.service_part_ab +10 -0
  26. package/src/domains/auth/infrastructure_part_aa +150 -0
  27. package/src/domains/auth/infrastructure_part_ab +6 -0
  28. package/src/domains/auth/presentation/hooks/GoogleOAuthHelpers.ts +0 -0
  29. package/src/domains/auth/presentation/hooks/GoogleOAuthHookService_part_aa +150 -0
  30. package/src/domains/auth/presentation/hooks/GoogleOAuthHookService_part_ab +97 -0
  31. package/src/domains/auth/presentation/hooks/GoogleOAuthService.ts +0 -0
  32. package/src/domains/firestore/domain/entities/Collection.ts +31 -191
  33. package/src/domains/firestore/domain/entities/Collection.ts.bak +288 -0
  34. package/src/domains/firestore/domain/entities/CollectionFactory.ts +55 -0
  35. package/src/domains/firestore/domain/entities/CollectionHelpers.ts +143 -0
  36. package/src/domains/firestore/domain/entities/CollectionUtils.ts +72 -0
  37. package/src/domains/firestore/domain/entities/CollectionValidation.ts +138 -0
  38. package/src/domains/firestore/domain/entities/Collection_part_aa +150 -0
  39. package/src/domains/firestore/domain/entities/Collection_part_ab +138 -0
  40. package/src/domains/firestore/domain/entities/DocumentHelpers.ts +0 -0
  41. package/src/domains/firestore/domain/entities/DocumentMain.ts +0 -0
  42. package/src/domains/firestore/domain/entities/Document_part_aa +150 -0
  43. package/src/domains/firestore/domain/entities/Document_part_ab +83 -0
  44. package/src/domains/firestore/domain/index.ts +44 -9
  45. package/src/domains/firestore/domain/services/QueryServiceAnalysis_part_aa +150 -0
  46. package/src/domains/firestore/domain/services/QueryServiceAnalysis_part_ab +19 -0
  47. package/src/domains/firestore/domain/services/QueryServiceHelpers_part_aa +150 -0
  48. package/src/domains/firestore/domain/services/QueryServiceHelpers_part_ab +1 -0
  49. package/src/domains/firestore/domain/services/QueryService_part_aa +150 -0
  50. package/src/domains/firestore/domain/services/QueryService_part_ab +32 -0
  51. package/src/domains/firestore/domain/value-objects/QueryOptions.ts.bak +6 -135
  52. package/src/domains/firestore/domain/value-objects/QueryOptionsSerialization_part_aa +150 -0
  53. package/src/domains/firestore/domain/value-objects/QueryOptionsSerialization_part_ab +57 -0
  54. package/src/domains/firestore/domain/value-objects/QueryOptionsValidation_part_aa +150 -0
  55. package/src/domains/firestore/domain/value-objects/QueryOptionsValidation_part_ab +32 -0
  56. package/src/domains/firestore/domain/value-objects/{QueryOptions.ts → QueryOptions_part_aa} +0 -41
  57. package/src/domains/firestore/domain/value-objects/QueryOptions_part_ab +41 -0
  58. package/src/domains/firestore/domain/value-objects/WhereClauseFactory_part_aa +150 -0
  59. package/src/domains/firestore/domain/value-objects/WhereClauseFactory_part_ab +57 -0
  60. package/src/domains/firestore/domain/value-objects/WhereClause_part_aa +150 -0
  61. package/src/domains/firestore/domain/value-objects/WhereClause_part_ab +149 -0
  62. package/src/shared/infrastructure/base/ErrorHandler_part_aa +150 -0
  63. package/src/shared/infrastructure/base/ErrorHandler_part_ab +39 -0
  64. package/src/shared/infrastructure/base/ServiceBase_part_aa +150 -0
  65. package/src/shared/infrastructure/base/ServiceBase_part_ab +70 -0
  66. package/src/shared/infrastructure/config/base/ServiceClientSingleton_part_aa +150 -0
  67. package/src/shared/infrastructure/config/base/ServiceClientSingleton_part_ab +5 -0
  68. /package/src/application/auth/ports/{AuthPort.ts → AuthPort.ts.bak} +0 -0
  69. /package/src/application/auth/use-cases/{SignInUseCase.ts → SignInUseCase.ts.bak} +0 -0
  70. /package/src/application/auth/use-cases/{SignOutUseCase.ts → SignOutUseCase.ts.bak} +0 -0
  71. /package/src/domains/account-deletion/infrastructure/services/{AccountDeletionExecutor.ts → AccountDeletionExecutor.ts.bak} +0 -0
  72. /package/src/domains/account-deletion/infrastructure/services/{AccountDeletionReauthHandler.ts → AccountDeletionReauthHandler.ts.bak} +0 -0
  73. /package/src/domains/account-deletion/infrastructure/services/{AccountDeletionRepository.ts → AccountDeletionRepository.ts.bak} +0 -0
  74. /package/src/domains/account-deletion/infrastructure/services/{reauthentication.service.ts → reauthentication.service.ts.bak} +0 -0
  75. /package/src/domains/auth/{infrastructure.ts → infrastructure.ts.bak} +0 -0
  76. /package/src/domains/auth/presentation/hooks/{GoogleOAuthHookService.ts → GoogleOAuthHookService.ts.bak} +0 -0
  77. /package/src/domains/firestore/domain/entities/{Document.ts → Document.ts.bak} +0 -0
  78. /package/src/domains/firestore/domain/services/{QueryService.ts → QueryService.ts.bak} +0 -0
  79. /package/src/domains/firestore/domain/services/{QueryServiceAnalysis.ts → QueryServiceAnalysis.ts.bak} +0 -0
  80. /package/src/domains/firestore/domain/services/{QueryServiceHelpers.ts → QueryServiceHelpers.ts.bak} +0 -0
  81. /package/src/domains/firestore/domain/value-objects/{QueryOptionsSerialization.ts → QueryOptionsSerialization.ts.bak} +0 -0
  82. /package/src/domains/firestore/domain/value-objects/{QueryOptionsValidation.ts → QueryOptionsValidation.ts.bak} +0 -0
  83. /package/src/domains/firestore/domain/value-objects/{WhereClause.ts → WhereClause.ts.bak} +0 -0
  84. /package/src/domains/firestore/domain/value-objects/{WhereClauseFactory.ts → WhereClauseFactory.ts.bak} +0 -0
  85. /package/src/shared/infrastructure/base/{ErrorHandler.ts → ErrorHandler.ts.bak} +0 -0
  86. /package/src/shared/infrastructure/base/{ServiceBase.ts → ServiceBase.ts.bak} +0 -0
  87. /package/src/shared/infrastructure/config/base/{ServiceClientSingleton.ts → ServiceClientSingleton.ts.bak} +0 -0
@@ -0,0 +1,150 @@
1
+ /**
2
+ * Error Handler Base Class
3
+ * Single Responsibility: Centralized error handling and conversion
4
+ *
5
+ * Eliminates try-catch duplication across 9+ files.
6
+ * Provides standardized error-to-ErrorInfo conversion.
7
+ * Integrates with existing Result pattern.
8
+ *
9
+ * Max lines: 150 (enforced for maintainability)
10
+ */
11
+
12
+ import type { ErrorInfo } from '../../domain/utils';
13
+ import { toErrorInfo, successResult, type Result } from '../../domain/utils';
14
+ import { isCancelledError, isQuotaError, isRetryableError } from '../../domain/utils/error-handlers/error-checkers';
15
+ import { ERROR_MESSAGES } from '../../domain/utils/error-handlers/error-messages';
16
+
17
+ /**
18
+ * Error handler options
19
+ */
20
+ export interface ErrorHandlerOptions {
21
+ /** Default error code if none can be extracted */
22
+ defaultErrorCode?: string;
23
+ /** Custom error message mapper */
24
+ errorMapper?: (error: unknown) => string | null;
25
+ /** Enable detailed logging in development */
26
+ enableDevLogging?: boolean;
27
+ }
28
+
29
+ /**
30
+ * Base class for error handling
31
+ * Provides centralized error conversion and handling
32
+ */
33
+ export class ErrorHandler {
34
+ private readonly options: ErrorHandlerOptions;
35
+
36
+ constructor(options: ErrorHandlerOptions = {}) {
37
+ this.options = {
38
+ defaultErrorCode: 'unknown/error',
39
+ enableDevLogging: __DEV__,
40
+ ...options,
41
+ };
42
+ }
43
+
44
+ /**
45
+ * Wrap an async operation with standardized error handling
46
+ * Eliminates try-catch duplication across services
47
+ */
48
+ async handleAsync<T>(
49
+ operation: () => Promise<T>,
50
+ errorCode: string = this.options.defaultErrorCode || 'unknown/error'
51
+ ): Promise<Result<T>> {
52
+ try {
53
+ const result = await operation();
54
+ return successResult(result);
55
+ } catch (error) {
56
+ if (this.options.enableDevLogging) {
57
+ console.error(`[ErrorHandler] Operation failed: ${errorCode}`, error);
58
+ }
59
+ return { success: false, error: toErrorInfo(error, errorCode) };
60
+ }
61
+ }
62
+
63
+ /**
64
+ * Wrap a synchronous operation with standardized error handling
65
+ */
66
+ handle<T>(
67
+ operation: () => T,
68
+ errorCode: string = this.options.defaultErrorCode || 'unknown/error'
69
+ ): Result<T> {
70
+ try {
71
+ const result = operation();
72
+ return successResult(result);
73
+ } catch (error) {
74
+ if (this.options.enableDevLogging) {
75
+ console.error(`[ErrorHandler] Operation failed: ${errorCode}`, error);
76
+ }
77
+ return { success: false, error: toErrorInfo(error, errorCode) };
78
+ }
79
+ }
80
+
81
+ /**
82
+ * Convert error to ErrorInfo with optional custom code
83
+ */
84
+ toErrorInfo(error: unknown, code?: string): ErrorInfo {
85
+ // Try custom error mapper first
86
+ if (this.options.errorMapper) {
87
+ const customMessage = this.options.errorMapper(error);
88
+ if (customMessage) {
89
+ return {
90
+ code: code || this.options.defaultErrorCode || 'unknown/error',
91
+ message: customMessage,
92
+ };
93
+ }
94
+ }
95
+
96
+ // Use standard conversion
97
+ return toErrorInfo(error, code || this.options.defaultErrorCode);
98
+ }
99
+
100
+ /**
101
+ * Check if error is a cancelled operation
102
+ */
103
+ isCancelledError(error: unknown): boolean {
104
+ return isCancelledError(error);
105
+ }
106
+
107
+ /**
108
+ * Check if error is a quota error
109
+ */
110
+ isQuotaError(error: unknown): boolean {
111
+ return isQuotaError(error);
112
+ }
113
+
114
+ /**
115
+ * Check if error is retryable
116
+ */
117
+ isRetryableError(error: unknown): boolean {
118
+ return isRetryableError(error);
119
+ }
120
+
121
+ /**
122
+ * Get user-friendly error message
123
+ * Maps error codes to user-friendly messages
124
+ */
125
+ getUserMessage(error: ErrorInfo): string {
126
+ // Check if it's a known error category
127
+ if (error.code.startsWith('auth/')) {
128
+ return this.getAuthUserMessage(error.code);
129
+ }
130
+ if (error.code.startsWith('firestore/')) {
131
+ return this.getFirestoreUserMessage(error.code);
132
+ }
133
+ if (error.code.startsWith('storage/')) {
134
+ return ERROR_MESSAGES.STORAGE.GENERIC_ERROR;
135
+ }
136
+
137
+ return error.message || ERROR_MESSAGES.GENERAL.UNKNOWN;
138
+ }
139
+
140
+ /**
141
+ * Get user-friendly auth error message
142
+ */
143
+ private getAuthUserMessage(code: string): string {
144
+ // Map common auth error codes to user-friendly messages
145
+ const authMessages: Record<string, string> = {
146
+ 'auth/user-not-found': ERROR_MESSAGES.AUTH.USER_NOT_FOUND,
147
+ 'auth/wrong-password': ERROR_MESSAGES.AUTH.INVALID_CREDENTIALS,
148
+ 'auth/email-already-in-use': ERROR_MESSAGES.AUTH.EMAIL_ALREADY_IN_USE,
149
+ 'auth/invalid-email': ERROR_MESSAGES.AUTH.INVALID_EMAIL,
150
+ 'auth/weak-password': ERROR_MESSAGES.AUTH.WEAK_PASSWORD,
@@ -0,0 +1,39 @@
1
+ 'auth/too-many-requests': ERROR_MESSAGES.AUTH.TOO_MANY_REQUESTS,
2
+ 'auth/user-disabled': ERROR_MESSAGES.AUTH.USER_DISABLED,
3
+ };
4
+
5
+ return authMessages[code] || ERROR_MESSAGES.AUTH.GENERIC_ERROR;
6
+ }
7
+
8
+ /**
9
+ * Get user-friendly firestore error message
10
+ */
11
+ private getFirestoreUserMessage(code: string): string {
12
+ const firestoreMessages: Record<string, string> = {
13
+ 'firestore/permission-denied': ERROR_MESSAGES.FIRESTORE.PERMISSION_DENIED,
14
+ 'firestore/not-found': ERROR_MESSAGES.FIRESTORE.NOT_FOUND,
15
+ 'firestore/unavailable': ERROR_MESSAGES.FIRESTORE.UNAVAILABLE,
16
+ };
17
+
18
+ return firestoreMessages[code] || ERROR_MESSAGES.FIRESTORE.GENERIC_ERROR;
19
+ }
20
+
21
+ /**
22
+ * Create a failure result from error code
23
+ */
24
+ failureFrom(code: string, message?: string): Result {
25
+ return {
26
+ success: false,
27
+ error: {
28
+ code,
29
+ message: message || ERROR_MESSAGES.GENERAL.UNKNOWN,
30
+ },
31
+ };
32
+ }
33
+ }
34
+
35
+ /**
36
+ * Default error handler instance
37
+ * Can be used across the application
38
+ */
39
+ export const defaultErrorHandler = new ErrorHandler();
@@ -0,0 +1,150 @@
1
+ /**
2
+ * Service Base Class
3
+ * Single Responsibility: Provide common service functionality
4
+ *
5
+ * Base class for all services to eliminate duplication.
6
+ * Integrates error handling and initialization.
7
+ * Reduces service boilerplate by ~50%.
8
+ *
9
+ * Max lines: 150 (enforced for maintainability)
10
+ */
11
+
12
+ import type { Result } from '../../domain/utils';
13
+ import { ErrorHandler, type ErrorHandlerOptions } from './ErrorHandler';
14
+ import { successResult } from '../../domain/utils';
15
+ import type { ErrorInfo } from '../../domain/utils';
16
+
17
+ /**
18
+ * Service state
19
+ */
20
+ interface ServiceState {
21
+ isInitialized: boolean;
22
+ initializationError: string | null;
23
+ }
24
+
25
+ /**
26
+ * Service base options
27
+ */
28
+ export interface ServiceBaseOptions extends ErrorHandlerOptions {
29
+ /** Service name for logging and error messages */
30
+ serviceName: string;
31
+ /** Auto-initialize on first access */
32
+ autoInitialize?: boolean;
33
+ }
34
+
35
+ /**
36
+ * Base class for all services
37
+ * Provides common initialization, error handling, and state management
38
+ *
39
+ * Usage:
40
+ * ```typescript
41
+ * class MyService extends ServiceBase {
42
+ * constructor() {
43
+ * super({ serviceName: 'MyService' });
44
+ * }
45
+ *
46
+ * async myMethod() {
47
+ * return this.handleAsync(async () => {
48
+ * // Your logic here
49
+ * return result;
50
+ * }, 'my-method/failed');
51
+ * }
52
+ * }
53
+ * ```
54
+ */
55
+ export abstract class ServiceBase {
56
+ protected readonly errorHandler: ErrorHandler;
57
+ protected readonly serviceName: string;
58
+ protected readonly autoInitialize: boolean;
59
+ protected state: ServiceState;
60
+ private initInProgress = false;
61
+
62
+ constructor(options: ServiceBaseOptions) {
63
+ this.serviceName = options.serviceName;
64
+ this.autoInitialize = options.autoInitialize ?? false;
65
+ this.errorHandler = new ErrorHandler({
66
+ ...options,
67
+ defaultErrorCode: `${this.serviceName.toLowerCase()}/error`,
68
+ });
69
+ this.state = {
70
+ isInitialized: false,
71
+ initializationError: null,
72
+ };
73
+ }
74
+
75
+ /**
76
+ * Initialize the service
77
+ * Override this method to provide custom initialization logic
78
+ */
79
+ protected async initialize(): Promise<Result<void>> {
80
+ // Override in subclasses
81
+ this.state.isInitialized = true;
82
+ return successResult();
83
+ }
84
+
85
+ /**
86
+ * Ensure service is initialized before operation
87
+ * Automatically initializes if autoInitialize is enabled
88
+ */
89
+ protected async ensureInitialized(): Promise<Result<void>> {
90
+ if (this.state.isInitialized) {
91
+ return successResult();
92
+ }
93
+
94
+ if (this.state.initializationError) {
95
+ return {
96
+ success: false,
97
+ error: {
98
+ code: `${this.serviceName.toLowerCase()}/initialization-failed`,
99
+ message: this.state.initializationError,
100
+ },
101
+ };
102
+ }
103
+
104
+ if (this.initInProgress) {
105
+ return {
106
+ success: false,
107
+ error: {
108
+ code: `${this.serviceName.toLowerCase()}/initialization-in-progress`,
109
+ message: 'Service initialization is in progress',
110
+ },
111
+ };
112
+ }
113
+
114
+ if (!this.autoInitialize) {
115
+ return {
116
+ success: false,
117
+ error: {
118
+ code: `${this.serviceName.toLowerCase()}/not-initialized`,
119
+ message: 'Service is not initialized. Call initialize() first.',
120
+ },
121
+ };
122
+ }
123
+
124
+ this.initInProgress = true;
125
+ try {
126
+ const result = await this.initialize();
127
+ if (!result.success) {
128
+ this.state.initializationError = result.error?.message || 'Initialization failed';
129
+ return result;
130
+ }
131
+ this.state.isInitialized = true;
132
+ return successResult();
133
+ } finally {
134
+ this.initInProgress = false;
135
+ }
136
+ }
137
+
138
+ /**
139
+ * Wrap async operation with error handling and initialization check
140
+ */
141
+ protected async execute<T>(
142
+ operation: () => Promise<T>,
143
+ errorCode?: string
144
+ ): Promise<Result<T>> {
145
+ const initResult = await this.ensureInitialized();
146
+ if (!initResult.success) {
147
+ return {
148
+ success: false,
149
+ error: initResult.error,
150
+ };
@@ -0,0 +1,70 @@
1
+ }
2
+
3
+ return this.errorHandler.handleAsync(operation, errorCode);
4
+ }
5
+
6
+ /**
7
+ * Wrap sync operation with error handling
8
+ */
9
+ protected executeSync<T>(operation: () => T, errorCode?: string): Result<T> {
10
+ return this.errorHandler.handle(operation, errorCode);
11
+ }
12
+
13
+ /**
14
+ * Check if service is initialized
15
+ */
16
+ isInitialized(): boolean {
17
+ return this.state.isInitialized;
18
+ }
19
+
20
+ /**
21
+ * Get initialization error if any
22
+ */
23
+ getInitializationError(): string | null {
24
+ return this.state.initializationError;
25
+ }
26
+
27
+ /**
28
+ * Reset service state
29
+ * Override to add custom cleanup logic
30
+ */
31
+ reset(): void {
32
+ this.state = {
33
+ isInitialized: false,
34
+ initializationError: null,
35
+ };
36
+ this.initInProgress = false;
37
+ }
38
+
39
+ /**
40
+ * Get service name
41
+ */
42
+ getServiceName(): string {
43
+ return this.serviceName;
44
+ }
45
+
46
+ /**
47
+ * Log in development mode
48
+ */
49
+ protected log(message: string, ...args: unknown[]): void {
50
+ if (__DEV__) {
51
+ console.log(`[${this.serviceName}] ${message}`, ...args);
52
+ }
53
+ }
54
+
55
+ /**
56
+ * Log error in development mode
57
+ */
58
+ protected logError(message: string, error?: unknown): void {
59
+ if (__DEV__) {
60
+ console.error(`[${this.serviceName}] ${message}`, error);
61
+ }
62
+ }
63
+
64
+ /**
65
+ * Create a failure result
66
+ */
67
+ protected failure(code: string, message?: string): Result {
68
+ return this.errorHandler.failureFrom(code, message);
69
+ }
70
+ }
@@ -0,0 +1,150 @@
1
+ /**
2
+ * Service Client Singleton Base Class
3
+ *
4
+ * Provides a generic singleton pattern for Firebase service clients.
5
+ * Eliminates code duplication across FirebaseClient, FirestoreClient, FirebaseAuthClient.
6
+ *
7
+ * Features:
8
+ * - Generic singleton pattern
9
+ * - Initialization state management
10
+ * - Error handling and tracking
11
+ * - Automatic cleanup
12
+ *
13
+ * @template TInstance - The service instance type (e.g., Firestore, Auth)
14
+ * @template TConfig - The configuration type (optional)
15
+ */
16
+
17
+ interface ServiceClientState<TInstance> {
18
+ instance: TInstance | null;
19
+ initializationError: string | null;
20
+ isInitialized: boolean;
21
+ }
22
+
23
+ interface ServiceClientOptions<TInstance, TConfig = unknown> {
24
+ serviceName: string;
25
+ initializer?: (config?: TConfig) => TInstance | null;
26
+ autoInitializer?: () => TInstance | null;
27
+ }
28
+
29
+ /**
30
+ * Generic service client singleton base class
31
+ * Provides common initialization, state management, and error handling
32
+ */
33
+ export class ServiceClientSingleton<TInstance, TConfig = unknown> {
34
+ protected state: ServiceClientState<TInstance>;
35
+ private readonly options: ServiceClientOptions<TInstance, TConfig>;
36
+ private initInProgress = false;
37
+
38
+ constructor(options: ServiceClientOptions<TInstance, TConfig>) {
39
+ this.options = options;
40
+ this.state = {
41
+ instance: null,
42
+ initializationError: null,
43
+ isInitialized: false,
44
+ };
45
+ }
46
+
47
+ /**
48
+ * Initialize the service with optional configuration
49
+ */
50
+ initialize(config?: TConfig): TInstance | null {
51
+ if (this.state.isInitialized && this.state.instance) {
52
+ return this.state.instance;
53
+ }
54
+
55
+ if (this.state.initializationError) {
56
+ return null;
57
+ }
58
+
59
+ // Prevent concurrent initialization attempts
60
+ if (this.initInProgress) {
61
+ return null;
62
+ }
63
+
64
+ this.initInProgress = true;
65
+ try {
66
+ const instance = this.options.initializer ? this.options.initializer(config) : null;
67
+ if (instance) {
68
+ this.state.instance = instance;
69
+ this.state.isInitialized = true;
70
+ }
71
+ return instance;
72
+ } catch (error) {
73
+ const errorMessage = error instanceof Error ? error.message : `Failed to initialize ${this.options.serviceName}`;
74
+ this.state.initializationError = errorMessage;
75
+ return null;
76
+ } finally {
77
+ this.initInProgress = false;
78
+ }
79
+ }
80
+
81
+ /**
82
+ * Get the service instance, auto-initializing if needed
83
+ */
84
+ getInstance(autoInit: boolean = false): TInstance | null {
85
+ if (this.state.instance) {
86
+ return this.state.instance;
87
+ }
88
+
89
+ if (this.state.initializationError) {
90
+ return null;
91
+ }
92
+
93
+ // Prevent concurrent auto-initialization attempts
94
+ if (this.initInProgress) {
95
+ return null;
96
+ }
97
+
98
+ if (autoInit && this.options.autoInitializer) {
99
+ this.initInProgress = true;
100
+ try {
101
+ const instance = this.options.autoInitializer();
102
+ if (instance) {
103
+ this.state.instance = instance;
104
+ this.state.isInitialized = true;
105
+ }
106
+ return instance;
107
+ } catch (error) {
108
+ const errorMessage = error instanceof Error ? error.message : `Failed to initialize ${this.options.serviceName}`;
109
+ this.state.initializationError = errorMessage;
110
+ } finally {
111
+ this.initInProgress = false;
112
+ }
113
+ }
114
+
115
+ return null;
116
+ }
117
+
118
+ /**
119
+ * Check if the service is initialized
120
+ */
121
+ isInitialized(): boolean {
122
+ return this.state.isInitialized;
123
+ }
124
+
125
+ /**
126
+ * Get the initialization error if any
127
+ */
128
+ getInitializationError(): string | null {
129
+ return this.state.initializationError;
130
+ }
131
+
132
+ /**
133
+ * Reset the service state
134
+ */
135
+ reset(): void {
136
+ this.state.instance = null;
137
+ this.state.initializationError = null;
138
+ this.state.isInitialized = false;
139
+ this.initInProgress = false;
140
+ }
141
+
142
+ /**
143
+ * Get the current instance without initialization
144
+ */
145
+ protected getCurrentInstance(): TInstance | null {
146
+ return this.state.instance;
147
+ }
148
+
149
+ /**
150
+ * Set initialization error
@@ -0,0 +1,5 @@
1
+ */
2
+ protected setError(error: string): void {
3
+ this.state.initializationError = error;
4
+ }
5
+ }