@aakash58/chatbot 1.1.15 → 1.1.17

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 (167) hide show
  1. package/esm2022/aakash58-chatbot.mjs +5 -0
  2. package/esm2022/lib/app/chat/chat-ui-state.service.mjs +170 -0
  3. package/esm2022/lib/app/chat/chat.service.mjs +445 -0
  4. package/esm2022/lib/app/chat/components/chat-button/chat-button.component.mjs +50 -0
  5. package/esm2022/lib/app/chat/components/chat-footer/chat-footer.component.mjs +12 -0
  6. package/esm2022/lib/app/chat/components/chat-header/chat-header.component.mjs +66 -0
  7. package/esm2022/lib/app/chat/components/chat-history-sidebar/chat-history-sidebar.component.mjs +186 -0
  8. package/esm2022/lib/app/chat/components/chat-window/chat-window.component.mjs +312 -0
  9. package/esm2022/lib/app/chat/components/message-input/message-input.component.mjs +36 -0
  10. package/esm2022/lib/app/chat/components/message-list/message-list.component.mjs +115 -0
  11. package/esm2022/lib/app/chat/model/chat-history.model.mjs +2 -0
  12. package/esm2022/lib/app/chat/model/chat-request.model.mjs +2 -0
  13. package/esm2022/lib/app/chat/model/chat-response.model.mjs +2 -0
  14. package/esm2022/lib/app/chat/model/chat-session.model.mjs +2 -0
  15. package/esm2022/lib/app/chat/model/chat-stream-message.model.mjs +2 -0
  16. package/esm2022/lib/app/chat/model/chat-stream-response.model.mjs +2 -0
  17. package/esm2022/lib/app/chat/services/chat-api.service.mjs +61 -0
  18. package/esm2022/lib/app/chat/services/chat-audio.service.mjs +50 -0
  19. package/esm2022/lib/app/chat/services/chat-history.service.mjs +252 -0
  20. package/esm2022/lib/app/login/login-form.component.mjs +46 -0
  21. package/esm2022/lib/app/personalization/personalization-dialog.component.mjs +194 -0
  22. package/esm2022/lib/app/personalization/personalization.service.mjs +149 -0
  23. package/esm2022/lib/app/personalization/sections/account/account-section.component.mjs +122 -0
  24. package/esm2022/lib/app/personalization/sections/preferences/color-picker-dialog.component.mjs +86 -0
  25. package/esm2022/lib/app/personalization/sections/preferences/preferences-section.component.mjs +115 -0
  26. package/esm2022/lib/app/personalization/sections/profile/profile-section.component.mjs +29 -0
  27. package/esm2022/lib/app/personalization/sections/settings/setting-section.component.mjs +30 -0
  28. package/esm2022/lib/app/personalization/sections/terms/terms-section.component.mjs +12 -0
  29. package/esm2022/lib/constant/doohbot-constant.mjs +28 -0
  30. package/esm2022/lib/constant/html-entities.mjs +9 -0
  31. package/esm2022/lib/constant/utf8.mjs +10 -0
  32. package/esm2022/lib/core/app-const.mjs +61 -0
  33. package/esm2022/lib/core/auth/account-api.service.mjs +40 -0
  34. package/esm2022/lib/core/auth/auth.service.mjs +391 -0
  35. package/esm2022/lib/core/auth/models/auth-result.model.mjs +3 -0
  36. package/esm2022/lib/core/auth/models/federated-login-request.model.mjs +6 -0
  37. package/esm2022/lib/core/auth/models/login-request.model.mjs +6 -0
  38. package/esm2022/lib/core/auth/storage.service.mjs +110 -0
  39. package/esm2022/lib/core/directives/draggable/draggable-dialog.directive.mjs +112 -0
  40. package/esm2022/lib/core/directives/fullscreen/fullscreen.directive.mjs +55 -0
  41. package/esm2022/lib/core/directives/resizable/resizable-dialog.directive.mjs +179 -0
  42. package/esm2022/lib/core/environments/environment.mjs +15 -0
  43. package/esm2022/lib/core/environments/environment.prod.mjs +15 -0
  44. package/esm2022/lib/core/helpers/crypto-helper.service.mjs +52 -0
  45. package/esm2022/lib/core/http/http-stream.service.mjs +97 -0
  46. package/esm2022/lib/core/http/http.service.mjs +103 -0
  47. package/esm2022/lib/core/interceptors/auth.interceptor.mjs +96 -0
  48. package/esm2022/lib/core/interceptors/license.interceptor.mjs +44 -0
  49. package/esm2022/lib/core/models/api-config.model.mjs +18 -0
  50. package/esm2022/lib/core/models/api-request.model.mjs +2 -0
  51. package/esm2022/lib/core/models/api-response.model.mjs +8 -0
  52. package/esm2022/lib/core/models/doohbot-config.model.mjs +18 -0
  53. package/esm2022/lib/core/models/license.model.mjs +2 -0
  54. package/esm2022/lib/core/models/message.mjs +2 -0
  55. package/esm2022/lib/core/models/theme-config.model.mjs +2 -0
  56. package/esm2022/lib/core/services/core-config.service.mjs +52 -0
  57. package/esm2022/lib/core/services/license.service.mjs +145 -0
  58. package/esm2022/lib/core/services/markdown.service.mjs +64 -0
  59. package/esm2022/lib/core/services/theme.service.mjs +248 -0
  60. package/esm2022/lib/core/types/auth-mode.type.mjs +2 -0
  61. package/esm2022/lib/core/types/auth-status.type.mjs +5 -0
  62. package/esm2022/lib/core/types/chat-stream.type.mjs +2 -0
  63. package/esm2022/lib/core/types/message-role.type.mjs +2 -0
  64. package/esm2022/lib/core/types/prompt-mode.type.mjs +5 -0
  65. package/esm2022/lib/core/types/snackbar-error.type.mjs +5 -0
  66. package/esm2022/lib/core/types/tenant-resolution-strategy.type.mjs +2 -0
  67. package/esm2022/lib/core/utils/logger.service.mjs +42 -0
  68. package/esm2022/lib/doohbot-input.mjs +20 -0
  69. package/esm2022/lib/doohbot.component.mjs +444 -0
  70. package/esm2022/lib/predefined_messages.mjs +15 -0
  71. package/esm2022/lib/shared/chips/chips.component.mjs +28 -0
  72. package/esm2022/lib/shared/dialog/dialog.component.mjs +36 -0
  73. package/esm2022/lib/shared/dialog/dialog.service.mjs +64 -0
  74. package/esm2022/lib/shared/dialog/dialog.utils.mjs +85 -0
  75. package/esm2022/lib/shared/dropdown-menu/dropdown-menu.component.mjs +29 -0
  76. package/esm2022/lib/shared/input-dialog/input-dialog.component.mjs +38 -0
  77. package/esm2022/lib/shared/menu-item/menu-item.component.mjs +24 -0
  78. package/esm2022/lib/shared/pipes/simple-markdown.pipe.mjs +27 -0
  79. package/esm2022/lib/shared/snackbar/snackbar.component.mjs +43 -0
  80. package/esm2022/lib/shared/snackbar/snackbar.service.mjs +46 -0
  81. package/esm2022/lib/shared/snackbar/snackbar.utils.mjs +43 -0
  82. package/esm2022/public-api.mjs +37 -0
  83. package/fesm2022/aakash58-chatbot.mjs +1114 -867
  84. package/fesm2022/aakash58-chatbot.mjs.map +1 -1
  85. package/index.d.ts +3 -377
  86. package/lib/app/chat/chat-ui-state.service.d.ts +96 -0
  87. package/lib/app/chat/chat.service.d.ts +88 -0
  88. package/lib/app/chat/components/chat-button/chat-button.component.d.ts +16 -0
  89. package/lib/app/chat/components/chat-footer/chat-footer.component.d.ts +5 -0
  90. package/lib/app/chat/components/chat-header/chat-header.component.d.ts +24 -0
  91. package/lib/app/chat/components/chat-history-sidebar/chat-history-sidebar.component.d.ts +49 -0
  92. package/lib/app/chat/components/chat-window/chat-window.component.d.ts +107 -0
  93. package/lib/app/chat/components/message-input/message-input.component.d.ts +12 -0
  94. package/lib/app/chat/components/message-list/message-list.component.d.ts +40 -0
  95. package/lib/app/chat/model/chat-history.model.d.ts +51 -0
  96. package/lib/app/chat/model/chat-request.model.d.ts +10 -0
  97. package/lib/app/chat/model/chat-response.model.d.ts +9 -0
  98. package/lib/app/chat/model/chat-session.model.d.ts +12 -0
  99. package/lib/app/chat/model/chat-stream-message.model.d.ts +5 -0
  100. package/lib/app/chat/model/chat-stream-response.model.d.ts +10 -0
  101. package/lib/app/chat/services/chat-api.service.d.ts +30 -0
  102. package/lib/app/chat/services/chat-audio.service.d.ts +19 -0
  103. package/lib/app/chat/services/chat-history.service.d.ts +53 -0
  104. package/lib/app/login/login-form.component.d.ts +20 -0
  105. package/lib/app/personalization/personalization-dialog.component.d.ts +53 -0
  106. package/lib/app/personalization/personalization.service.d.ts +66 -0
  107. package/lib/app/personalization/sections/account/account-section.component.d.ts +30 -0
  108. package/lib/app/personalization/sections/preferences/color-picker-dialog.component.d.ts +17 -0
  109. package/lib/app/personalization/sections/preferences/preferences-section.component.d.ts +27 -0
  110. package/lib/app/personalization/sections/profile/profile-section.component.d.ts +17 -0
  111. package/lib/app/personalization/sections/settings/setting-section.component.d.ts +10 -0
  112. package/lib/app/personalization/sections/terms/terms-section.component.d.ts +5 -0
  113. package/lib/constant/doohbot-constant.d.ts +12 -0
  114. package/lib/constant/html-entities.d.ts +8 -0
  115. package/lib/constant/utf8.d.ts +9 -0
  116. package/lib/core/app-const.d.ts +11 -0
  117. package/lib/core/auth/account-api.service.d.ts +20 -0
  118. package/lib/core/auth/auth.service.d.ts +90 -0
  119. package/lib/core/auth/models/auth-result.model.d.ts +4 -0
  120. package/lib/core/auth/models/federated-login-request.model.d.ts +5 -0
  121. package/lib/core/auth/models/login-request.model.d.ts +6 -0
  122. package/lib/core/auth/storage.service.d.ts +21 -0
  123. package/lib/core/directives/draggable/draggable-dialog.directive.d.ts +23 -0
  124. package/lib/core/directives/fullscreen/fullscreen.directive.d.ts +14 -0
  125. package/lib/core/directives/resizable/resizable-dialog.directive.d.ts +30 -0
  126. package/lib/core/environments/environment.d.ts +7 -0
  127. package/lib/core/environments/environment.prod.d.ts +7 -0
  128. package/lib/core/helpers/crypto-helper.service.d.ts +12 -0
  129. package/lib/core/http/http-stream.service.d.ts +18 -0
  130. package/lib/core/http/http.service.d.ts +20 -0
  131. package/lib/core/interceptors/auth.interceptor.d.ts +18 -0
  132. package/lib/core/interceptors/license.interceptor.d.ts +11 -0
  133. package/lib/core/models/api-config.model.d.ts +58 -0
  134. package/lib/core/models/api-request.model.d.ts +77 -0
  135. package/lib/core/models/api-response.model.d.ts +6 -0
  136. package/lib/core/models/doohbot-config.model.d.ts +81 -0
  137. package/lib/core/models/license.model.d.ts +23 -0
  138. package/lib/core/models/message.d.ts +16 -0
  139. package/lib/core/models/theme-config.model.d.ts +28 -0
  140. package/lib/core/services/core-config.service.d.ts +23 -0
  141. package/lib/core/services/license.service.d.ts +33 -0
  142. package/lib/core/services/markdown.service.d.ts +8 -0
  143. package/lib/core/services/theme.service.d.ts +40 -0
  144. package/lib/core/types/auth-mode.type.d.ts +4 -0
  145. package/lib/core/types/auth-status.type.d.ts +4 -0
  146. package/lib/core/types/chat-stream.type.d.ts +4 -0
  147. package/lib/core/types/message-role.type.d.ts +4 -0
  148. package/lib/core/types/prompt-mode.type.d.ts +4 -0
  149. package/lib/core/types/snackbar-error.type.d.ts +4 -0
  150. package/lib/core/types/tenant-resolution-strategy.type.d.ts +4 -0
  151. package/lib/core/utils/logger.service.d.ts +11 -0
  152. package/lib/doohbot-input.d.ts +19 -0
  153. package/lib/doohbot.component.d.ts +108 -0
  154. package/lib/predefined_messages.d.ts +2 -0
  155. package/lib/shared/chips/chips.component.d.ts +10 -0
  156. package/lib/shared/dialog/dialog.component.d.ts +19 -0
  157. package/lib/shared/dialog/dialog.service.d.ts +29 -0
  158. package/lib/shared/dialog/dialog.utils.d.ts +41 -0
  159. package/lib/shared/dropdown-menu/dropdown-menu.component.d.ts +11 -0
  160. package/lib/shared/input-dialog/input-dialog.component.d.ts +20 -0
  161. package/lib/shared/menu-item/menu-item.component.d.ts +9 -0
  162. package/lib/shared/pipes/simple-markdown.pipe.d.ts +10 -0
  163. package/lib/shared/snackbar/snackbar.component.d.ts +14 -0
  164. package/lib/shared/snackbar/snackbar.service.d.ts +19 -0
  165. package/lib/shared/snackbar/snackbar.utils.d.ts +33 -0
  166. package/package.json +4 -2
  167. package/public-api.d.ts +11 -0
@@ -0,0 +1,391 @@
1
+ import { inject, Injectable, signal } from '@angular/core';
2
+ import { BehaviorSubject, map, catchError, of, tap, retry, throwError, firstValueFrom, } from 'rxjs';
3
+ import { JwtHelperService } from '@auth0/angular-jwt';
4
+ import logger from '../utils/logger.service';
5
+ import { SnackbarUtils } from './../../shared/snackbar/snackbar.utils';
6
+ import { DoohbotConst } from '../../constant/doohbot-constant';
7
+ import { DOOHBOT_ADVANCED_CONFIG } from '../models/doohbot-config.model';
8
+ import * as i0 from "@angular/core";
9
+ import * as i1 from "./account-api.service";
10
+ import * as i2 from "../helpers/crypto-helper.service";
11
+ import * as i3 from "./storage.service";
12
+ import * as i4 from "@angular/router";
13
+ import * as i5 from "../services/core-config.service";
14
+ export class AuthService {
15
+ get apiUrl() {
16
+ return this.coreConfig.apiUrl;
17
+ }
18
+ get storageKey() {
19
+ return this.coreConfig.storageKey;
20
+ }
21
+ get secretKey() {
22
+ return this.coreConfig.secretKey;
23
+ }
24
+ get companyCode() {
25
+ return this.coreConfig.companyCode;
26
+ }
27
+ constructor(accountService, cryptoHelper, tokenStorage, router, coreConfig) {
28
+ this.accountService = accountService;
29
+ this.cryptoHelper = cryptoHelper;
30
+ this.tokenStorage = tokenStorage;
31
+ this.router = router;
32
+ this.coreConfig = coreConfig;
33
+ this.jwtHelper = new JwtHelperService();
34
+ this.advancedConfig = inject(DOOHBOT_ADVANCED_CONFIG, { optional: true });
35
+ // Reactive Authentication State
36
+ this.authStatusSubject = new BehaviorSubject('unauthenticated');
37
+ this.authStatus$ = this.authStatusSubject.asObservable();
38
+ // UI State Signals (Merged from ChatAuthService)
39
+ this.isLoggingIn = signal(false);
40
+ this.authError = signal(null);
41
+ this.authSuccess = signal(null);
42
+ // Properties are now derived from coreConfig
43
+ }
44
+ /**
45
+ * Updates the service configuration from the component input
46
+ */
47
+ updateConfig(config) {
48
+ this.coreConfig.updateConfig(config);
49
+ if (config.secretKey) {
50
+ this.cryptoHelper.setSecretKey(config.secretKey);
51
+ }
52
+ }
53
+ getLoggedInUserName() {
54
+ const username = this.tokenStorage.get('username');
55
+ return username || '';
56
+ }
57
+ getLoggedInUserImageUrl() {
58
+ return DoohbotConst.userAvatar;
59
+ }
60
+ getLoggedInUserRole() {
61
+ const roles = this.tokenStorage.get('roles');
62
+ // If roles is an array, return the first role, otherwise return the string
63
+ if (Array.isArray(roles) && roles.length > 0) {
64
+ return roles[0];
65
+ }
66
+ return roles || 'User';
67
+ }
68
+ // Add getters for backward compatibility if needed, but preferably use authStatus$
69
+ get authStatus() {
70
+ return this.authStatusSubject.value;
71
+ }
72
+ /**
73
+ * Resolves a federated token from browser storage
74
+ */
75
+ resolveSilentToken() {
76
+ const config = this.coreConfig.config$();
77
+ const commonKeys = ['access_token', 'id_token', 'token', 'authToken'];
78
+ // 1. Try explicit authToken from config (Highest Priority)
79
+ if (config.authToken) {
80
+ logger.info(`[Auth] Using explicit authToken from input`);
81
+ return config.authToken;
82
+ }
83
+ // 2. Try configured storage key from advanced config
84
+ if (this.advancedConfig?.autoConfigFromStorage) {
85
+ const key = this.advancedConfig.autoConfigFromStorage;
86
+ const token = localStorage.getItem(key) || sessionStorage.getItem(key);
87
+ if (token) {
88
+ logger.debug(`[Auth] Resolved token from configured key: ${key}`);
89
+ return token;
90
+ }
91
+ }
92
+ // 3. Try common fallback keys
93
+ for (const key of commonKeys) {
94
+ const token = localStorage.getItem(key) || sessionStorage.getItem(key);
95
+ if (token) {
96
+ logger.debug(`[Auth] Resolved token from fallback key: ${key}`);
97
+ return token;
98
+ }
99
+ }
100
+ return null;
101
+ }
102
+ /**
103
+ * Authenticate user with federated credentials
104
+ */
105
+ async federatedLogin(credentials, silent = true) {
106
+ logger.log('>>>>>>>Attempting federated login...');
107
+ // Resolve token if not provided
108
+ if (!credentials.accessToken) {
109
+ credentials.accessToken = this.resolveSilentToken() || '';
110
+ }
111
+ // If still no token, exit gracefully
112
+ if (!credentials.accessToken) {
113
+ logger.log('[Auth] No federated token found in storage.');
114
+ return false;
115
+ }
116
+ this.isLoggingIn.set(true);
117
+ if (!silent) {
118
+ this.authError.set(null);
119
+ this.authSuccess.set(null);
120
+ }
121
+ try {
122
+ const response = await firstValueFrom(this.accountService.federateLogin(credentials));
123
+ logger.log('>>>>>>>>>>>Federated Login response<<<<<<<<<<:', response);
124
+ if (response && response.success && response.data) {
125
+ // Map federated result to something AuthService can handle
126
+ // Federated result: { access_token: string, remember_me: boolean }
127
+ const authResult = {
128
+ access_token: response.data.access_token,
129
+ refresh_token: credentials.rememberMe ? response.data.refresh_token : '',
130
+ };
131
+ this.setSession(authResult, credentials.rememberMe || false);
132
+ logger.log('Federated login successful, authenticated state set to true');
133
+ if (!silent) {
134
+ SnackbarUtils.success('Login successful!');
135
+ this.authSuccess.set('Login successful!');
136
+ }
137
+ this.isLoggingIn.set(false);
138
+ return true;
139
+ }
140
+ else {
141
+ if (!silent) {
142
+ this.authError.set('Federated login failed.');
143
+ }
144
+ this.isLoggingIn.set(false);
145
+ return false;
146
+ }
147
+ }
148
+ catch (error) {
149
+ logger.error('Federated login failed:', error);
150
+ if (!silent) {
151
+ this.authError.set('An error occurred during federated login.');
152
+ }
153
+ this.isLoggingIn.set(false);
154
+ return false;
155
+ }
156
+ }
157
+ /**
158
+ * Authenticate user with credentials
159
+ */
160
+ async login(credentials, silent = false) {
161
+ this.isLoggingIn.set(true);
162
+ if (!silent) {
163
+ this.authError.set(null); //TODO
164
+ this.authSuccess.set(null);
165
+ }
166
+ try {
167
+ const response = await firstValueFrom(this.accountService.login(credentials));
168
+ logger.log('>>>>>>>>>>>Login response<<<<<<<<<<:', response);
169
+ if (response && response.success && response.data) {
170
+ this.setSession(response.data, credentials.rememberMe);
171
+ logger.log('Login successful, authenticated state set to true');
172
+ if (!silent) {
173
+ this.authSuccess.set('Login successful!');
174
+ }
175
+ this.isLoggingIn.set(false);
176
+ return true;
177
+ }
178
+ else {
179
+ if (!silent) {
180
+ SnackbarUtils.error('Login failed. Please check your credentials.');
181
+ this.authError.set('Login failed. Please check your credentials.');
182
+ }
183
+ this.isLoggingIn.set(false);
184
+ return false;
185
+ }
186
+ }
187
+ catch (error) {
188
+ logger.error('Login failed:', error);
189
+ if (!silent) {
190
+ this.authError.set('An error occurred during login.');
191
+ }
192
+ this.isLoggingIn.set(false);
193
+ return false;
194
+ }
195
+ }
196
+ /*
197
+ * Logout logic enhanced with UI signals
198
+ */
199
+ async logout(refreshToken) {
200
+ const token = refreshToken || this.getRefreshToken() || '';
201
+ try {
202
+ // If no token, just clear local state
203
+ if (!token) {
204
+ this.clearAuth();
205
+ this.authSuccess.set('Logged out successfully!');
206
+ setTimeout(() => this.authSuccess.set(null), 3000);
207
+ return true;
208
+ }
209
+ const response = await firstValueFrom(this.accountService.logout(token));
210
+ if (response && response.success) {
211
+ this.clearAuth();
212
+ this.authSuccess.set('Logged out successfully!');
213
+ setTimeout(() => this.authSuccess.set(null), 3000);
214
+ return true;
215
+ }
216
+ else {
217
+ this.clearAuth();
218
+ return false;
219
+ }
220
+ }
221
+ catch (error) {
222
+ logger.error('Logout failed:', error);
223
+ SnackbarUtils.error('An error occurred during logout.');
224
+ // Force clear auth on error to ensure user isn't stuck
225
+ this.clearAuth();
226
+ return false;
227
+ }
228
+ }
229
+ // Utility
230
+ setSession(session, rememberMe = true) {
231
+ const user = this.jwtHelper.decodeToken(session.access_token);
232
+ // Set storage type based on rememberMe preference
233
+ this.tokenStorage.setRememberMe(rememberMe);
234
+ this.tokenStorage.set('access_token', session.access_token);
235
+ this.tokenStorage.set('refresh_token', session.refresh_token);
236
+ this.tokenStorage.set('user_id', user?.user_id);
237
+ this.tokenStorage.set('username', user?.username);
238
+ this.tokenStorage.set('roles', user?.roles);
239
+ this.tokenStorage.set('permissions', user?.permissions);
240
+ this.tokenStorage.set('organization_id', user?.organization_id);
241
+ this.tokenStorage.set('company_id', user?.company_id);
242
+ this.tokenStorage.set('office_id', user?.office_id);
243
+ // Notify subscribers
244
+ this.authStatusSubject.next('authenticated');
245
+ }
246
+ getOrganizationId() {
247
+ logger.debug('>>>>>>>>>>getOrganizationId<<<<<<<:', this.tokenStorage.get('organization_id'));
248
+ return this.tokenStorage.get('organization_id');
249
+ }
250
+ getRefreshToken() {
251
+ logger.debug('>>>>>>>>>>getRefreshToken<<<<<<<:', this.tokenStorage.get('refresh_token'));
252
+ return this.tokenStorage.get('refresh_token');
253
+ }
254
+ refreshToken() {
255
+ const refreshToken = this.getRefreshToken();
256
+ if (!refreshToken) {
257
+ return throwError(() => new Error('No refresh token available'));
258
+ }
259
+ return this.accountService.refreshLogin({ refresh_token: refreshToken }).pipe(retry(3), // Retry 3 times on failure
260
+ tap((response) => {
261
+ if (response && response.success && response.data) {
262
+ this.setSession(response.data);
263
+ }
264
+ }), map((response) => {
265
+ if (response && response.success && response.data) {
266
+ return response.data;
267
+ }
268
+ else {
269
+ throw new Error('Token refresh failed');
270
+ }
271
+ }), catchError((err) => {
272
+ logger.error('[Auth] Token refresh failed after retries:', err);
273
+ this.clearAuth();
274
+ SnackbarUtils.error('Session Expired. Please login.');
275
+ return throwError(() => err);
276
+ }));
277
+ }
278
+ isAuthenticated() {
279
+ const token = this.tokenStorage.get('access_token');
280
+ return !!token && !this.isTokenExpired();
281
+ }
282
+ isTokenExpired() {
283
+ let token = (this.tokenStorage.get('access_token') ?? '');
284
+ if (token) {
285
+ return this.jwtHelper.isTokenExpired(token);
286
+ }
287
+ else {
288
+ return true;
289
+ }
290
+ }
291
+ /**
292
+ * Validate stored token on app startup
293
+ */
294
+ validateStoredToken() {
295
+ const accessToken = this.tokenStorage.get('access_token');
296
+ //! No access token
297
+ if (!accessToken) {
298
+ //TODO: FORCE AUTHENTICATED FOR DEV
299
+ // this.authStatusSubject.next('authenticated');
300
+ // return of('authenticated');
301
+ //! For production ( Backend Validation )
302
+ this.authStatusSubject.next('unauthenticated');
303
+ return of('unauthenticated');
304
+ }
305
+ // Access token valid
306
+ if (!this.isTokenExpired()) {
307
+ this.authStatusSubject.next('authenticated');
308
+ return of('authenticated');
309
+ }
310
+ // Access Token expired -> try refresh
311
+ const refreshToken = this.getRefreshToken();
312
+ if (!refreshToken) {
313
+ this.clearAuth();
314
+ this.authStatusSubject.next('unauthenticated');
315
+ return of('unauthenticated');
316
+ }
317
+ //! Access expired but refresh token exists → attempt refresh
318
+ this.authStatusSubject.next('expired');
319
+ return this.refreshToken().pipe(map(() => {
320
+ this.authStatusSubject.next('authenticated');
321
+ return 'authenticated';
322
+ }), catchError((err) => {
323
+ logger.error('[Auth] Token refresh failed:', err);
324
+ this.clearAuth();
325
+ this.authStatusSubject.next('unauthenticated');
326
+ return of('unauthenticated');
327
+ }));
328
+ }
329
+ getUserId() {
330
+ return this.tokenStorage.get('user_id') || '';
331
+ }
332
+ getLicensingContext() {
333
+ const roles = this.tokenStorage.get('roles');
334
+ // Prefer company_id (specific tenant) over organization_id (system group)
335
+ const tenantId = this.tokenStorage.get('company_id') || this.getOrganizationId() || '';
336
+ return {
337
+ tenantId: tenantId,
338
+ principalId: this.getUserId() || '',
339
+ roles: Array.isArray(roles) ? roles : roles ? [roles] : [],
340
+ };
341
+ }
342
+ get licenseKey() {
343
+ return this.tokenStorage.get('license_key') || this.coreConfig.licenseKey || '';
344
+ }
345
+ get isFederated() {
346
+ return this.licenseKey.startsWith('doohbot_lic_');
347
+ }
348
+ setLicenseKey(key) {
349
+ this.tokenStorage.set('license_key', key);
350
+ }
351
+ //! clear local storage
352
+ /**
353
+ * Clear all authentication related data from local storage, session storage, cache, and tokens.
354
+ * This method is called when the user logs out.
355
+ */
356
+ clearAuth() {
357
+ const userId = this.getUserId();
358
+ logger.log('[Auth] Clearing auth:', { userId });
359
+ this.clearSession();
360
+ this.clearStorage();
361
+ this.clearCache();
362
+ this.clearLocalStorage();
363
+ // ENSURE authStatus$ emits unauthenticated
364
+ this.authStatusSubject.next('unauthenticated');
365
+ logger.log('[Auth] Auth cleared successfully');
366
+ }
367
+ clearSession() {
368
+ sessionStorage.clear();
369
+ }
370
+ clearTabInfo() {
371
+ localStorage.removeItem('tabs');
372
+ }
373
+ clearStorage() {
374
+ this.tokenStorage.clear();
375
+ }
376
+ clearLocalStorage() {
377
+ localStorage.clear();
378
+ }
379
+ clearCache() {
380
+ // this.cacheService.clear();
381
+ }
382
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AuthService, deps: [{ token: i1.AccountService }, { token: i2.CryptoHelperService }, { token: i3.StorageService }, { token: i4.Router }, { token: i5.CoreConfigService }], target: i0.ɵɵFactoryTarget.Injectable }); }
383
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AuthService, providedIn: 'root' }); }
384
+ }
385
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AuthService, decorators: [{
386
+ type: Injectable,
387
+ args: [{
388
+ providedIn: 'root',
389
+ }]
390
+ }], ctorParameters: () => [{ type: i1.AccountService }, { type: i2.CryptoHelperService }, { type: i3.StorageService }, { type: i4.Router }, { type: i5.CoreConfigService }] });
391
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.service.js","sourceRoot":"","sources":["../../../../../../projects/doohbot/src/lib/core/auth/auth.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable, signal } from '@angular/core';\r\nimport { CoreConfigService } from '../services/core-config.service';\r\nimport { DoohbotInput } from '../../doohbot-input';\r\nimport {\r\n  BehaviorSubject,\r\n  map,\r\n  Observable,\r\n  catchError,\r\n  of,\r\n  tap,\r\n  retry,\r\n  throwError,\r\n  firstValueFrom,\r\n} from 'rxjs';\r\nimport { AuthResult } from './models/auth-result.model';\r\nimport { FederatedLoginRequest } from './models/federated-login-request.model';\r\nimport { LoginRequest } from './models/login-request.model';\r\nimport { JwtHelperService } from '@auth0/angular-jwt';\r\nimport { CryptoHelperService } from '../helpers/crypto-helper.service';\r\nimport { StorageService } from './storage.service';\r\nimport logger from '../utils/logger.service';\r\nimport { AuthStatus } from '../types/auth-status.type';\r\nimport { AccountService } from './account-api.service';\r\nimport { SnackbarUtils } from './../../shared/snackbar/snackbar.utils';\r\nimport { DoohbotConst } from '../../constant/doohbot-constant';\r\nimport { Router } from '@angular/router';\r\nimport { LicensingContext } from '../models/license.model';\r\nimport { DOOHBOT_ADVANCED_CONFIG } from '../models/doohbot-config.model';\r\n\r\n//! inherit Window storageEventAdded for duplicate tab detection\r\ndeclare global {\r\n  interface Window {\r\n    storageEventAdded?: boolean;\r\n    tabCLoseEventAdded?: boolean;\r\n    abp?: any;\r\n    AppPreBootstrap?: any;\r\n    AppConsts?: any;\r\n  }\r\n}\r\n\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class AuthService {\r\n  get apiUrl(): string {\r\n    return this.coreConfig.apiUrl;\r\n  }\r\n\r\n  get storageKey(): string {\r\n    return this.coreConfig.storageKey;\r\n  }\r\n\r\n  get secretKey(): string {\r\n    return this.coreConfig.secretKey;\r\n  }\r\n\r\n  get companyCode(): string {\r\n    return this.coreConfig.companyCode;\r\n  }\r\n\r\n  jwtHelper: JwtHelperService = new JwtHelperService();\r\n  private advancedConfig = inject(DOOHBOT_ADVANCED_CONFIG, { optional: true });\r\n\r\n  // Reactive Authentication State\r\n  private authStatusSubject = new BehaviorSubject<AuthStatus>('unauthenticated');\r\n  public authStatus$ = this.authStatusSubject.asObservable();\r\n\r\n  // UI State Signals (Merged from ChatAuthService)\r\n  public isLoggingIn = signal<boolean>(false);\r\n  public authError = signal<string | null>(null);\r\n  public authSuccess = signal<string | null>(null);\r\n\r\n  constructor(\r\n    private accountService: AccountService,\r\n    private cryptoHelper: CryptoHelperService,\r\n    private tokenStorage: StorageService,\r\n    private router: Router,\r\n    private coreConfig: CoreConfigService,\r\n  ) {\r\n    // Properties are now derived from coreConfig\r\n  }\r\n\r\n  /**\r\n   * Updates the service configuration from the component input\r\n   */\r\n  public updateConfig(config: DoohbotInput): void {\r\n    this.coreConfig.updateConfig(config);\r\n    if (config.secretKey) {\r\n      this.cryptoHelper.setSecretKey(config.secretKey);\r\n    }\r\n  }\r\n\r\n  getLoggedInUserName(): string {\r\n    const username = this.tokenStorage.get('username');\r\n    return username || '';\r\n  }\r\n\r\n  getLoggedInUserImageUrl(): string | null {\r\n    return DoohbotConst.userAvatar;\r\n  }\r\n\r\n  getLoggedInUserRole(): string {\r\n    const roles = this.tokenStorage.get('roles');\r\n    // If roles is an array, return the first role, otherwise return the string\r\n    if (Array.isArray(roles) && roles.length > 0) {\r\n      return roles[0];\r\n    }\r\n    return roles || 'User';\r\n  }\r\n\r\n  // Add getters for backward compatibility if needed, but preferably use authStatus$\r\n  public get authStatus(): AuthStatus {\r\n    return this.authStatusSubject.value;\r\n  }\r\n\r\n  /**\r\n   * Resolves a federated token from browser storage\r\n   */\r\n  private resolveSilentToken(): string | null {\r\n    const config = this.coreConfig.config$();\r\n    const commonKeys = ['access_token', 'id_token', 'token', 'authToken'];\r\n\r\n    // 1. Try explicit authToken from config (Highest Priority)\r\n    if (config.authToken) {\r\n      logger.info(`[Auth] Using explicit authToken from input`);\r\n      return config.authToken;\r\n    }\r\n\r\n    // 2. Try configured storage key from advanced config\r\n    if (this.advancedConfig?.autoConfigFromStorage) {\r\n      const key = this.advancedConfig.autoConfigFromStorage;\r\n      const token = localStorage.getItem(key) || sessionStorage.getItem(key);\r\n      if (token) {\r\n        logger.debug(`[Auth] Resolved token from configured key: ${key}`);\r\n        return token;\r\n      }\r\n    }\r\n\r\n    // 3. Try common fallback keys\r\n    for (const key of commonKeys) {\r\n      const token = localStorage.getItem(key) || sessionStorage.getItem(key);\r\n      if (token) {\r\n        logger.debug(`[Auth] Resolved token from fallback key: ${key}`);\r\n        return token;\r\n      }\r\n    }\r\n\r\n    return null;\r\n  }\r\n\r\n  /**\r\n   * Authenticate user with federated credentials\r\n   */\r\n  async federatedLogin(\r\n    credentials: Partial<FederatedLoginRequest>,\r\n    silent: boolean = true,\r\n  ): Promise<boolean> {\r\n    logger.log('>>>>>>>Attempting federated login...');\r\n\r\n    // Resolve token if not provided\r\n    if (!credentials.accessToken) {\r\n      credentials.accessToken = this.resolveSilentToken() || '';\r\n    }\r\n\r\n    // If still no token, exit gracefully\r\n    if (!credentials.accessToken) {\r\n      logger.log('[Auth] No federated token found in storage.');\r\n      return false;\r\n    }\r\n\r\n    this.isLoggingIn.set(true);\r\n    if (!silent) {\r\n      this.authError.set(null);\r\n      this.authSuccess.set(null);\r\n    }\r\n\r\n    try {\r\n      const response = await firstValueFrom(\r\n        this.accountService.federateLogin(credentials as FederatedLoginRequest),\r\n      );\r\n      logger.log('>>>>>>>>>>>Federated Login response<<<<<<<<<<:', response);\r\n      if (response && response.success && response.data) {\r\n        // Map federated result to something AuthService can handle\r\n        // Federated result: { access_token: string, remember_me: boolean }\r\n        const authResult = {\r\n          access_token: response.data.access_token,\r\n          refresh_token: credentials.rememberMe ? response.data.refresh_token : '',\r\n        };\r\n\r\n        this.setSession(authResult, credentials.rememberMe || false);\r\n\r\n        logger.log('Federated login successful, authenticated state set to true');\r\n        if (!silent) {\r\n          SnackbarUtils.success('Login successful!');\r\n          this.authSuccess.set('Login successful!');\r\n        }\r\n        this.isLoggingIn.set(false);\r\n        return true;\r\n      } else {\r\n        if (!silent) {\r\n          this.authError.set('Federated login failed.');\r\n        }\r\n        this.isLoggingIn.set(false);\r\n        return false;\r\n      }\r\n    } catch (error) {\r\n      logger.error('Federated login failed:', error);\r\n      if (!silent) {\r\n        this.authError.set('An error occurred during federated login.');\r\n      }\r\n      this.isLoggingIn.set(false);\r\n      return false;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Authenticate user with credentials\r\n   */\r\n  async login(credentials: LoginRequest, silent: boolean = false): Promise<boolean> {\r\n    this.isLoggingIn.set(true);\r\n    if (!silent) {\r\n      this.authError.set(null); //TODO\r\n      this.authSuccess.set(null);\r\n    }\r\n\r\n    try {\r\n      const response = await firstValueFrom(this.accountService.login(credentials));\r\n      logger.log('>>>>>>>>>>>Login response<<<<<<<<<<:', response);\r\n      if (response && response.success && response.data) {\r\n        this.setSession(response.data, credentials.rememberMe);\r\n\r\n        logger.log('Login successful, authenticated state set to true');\r\n        if (!silent) {\r\n          this.authSuccess.set('Login successful!');\r\n        }\r\n        this.isLoggingIn.set(false);\r\n        return true;\r\n      } else {\r\n        if (!silent) {\r\n          SnackbarUtils.error('Login failed. Please check your credentials.');\r\n          this.authError.set('Login failed. Please check your credentials.');\r\n        }\r\n        this.isLoggingIn.set(false);\r\n        return false;\r\n      }\r\n    } catch (error) {\r\n      logger.error('Login failed:', error);\r\n      if (!silent) {\r\n        this.authError.set('An error occurred during login.');\r\n      }\r\n      this.isLoggingIn.set(false);\r\n      return false;\r\n    }\r\n  }\r\n\r\n  /*\r\n   * Logout logic enhanced with UI signals\r\n   */\r\n  async logout(refreshToken?: string): Promise<boolean> {\r\n    const token = refreshToken || this.getRefreshToken() || '';\r\n\r\n    try {\r\n      // If no token, just clear local state\r\n      if (!token) {\r\n        this.clearAuth();\r\n        this.authSuccess.set('Logged out successfully!');\r\n        setTimeout(() => this.authSuccess.set(null), 3000);\r\n        return true;\r\n      }\r\n\r\n      const response = await firstValueFrom(this.accountService.logout(token));\r\n\r\n      if (response && response.success) {\r\n        this.clearAuth();\r\n        this.authSuccess.set('Logged out successfully!');\r\n        setTimeout(() => this.authSuccess.set(null), 3000);\r\n        return true;\r\n      } else {\r\n        this.clearAuth();\r\n        return false;\r\n      }\r\n    } catch (error) {\r\n      logger.error('Logout failed:', error);\r\n      SnackbarUtils.error('An error occurred during logout.');\r\n      // Force clear auth on error to ensure user isn't stuck\r\n      this.clearAuth();\r\n      return false;\r\n    }\r\n  }\r\n\r\n  // Utility\r\n  setSession(session: AuthResult, rememberMe: boolean = true) {\r\n    const user = this.jwtHelper.decodeToken(session.access_token);\r\n\r\n    // Set storage type based on rememberMe preference\r\n    this.tokenStorage.setRememberMe(rememberMe);\r\n\r\n    this.tokenStorage.set('access_token', session.access_token);\r\n    this.tokenStorage.set('refresh_token', session.refresh_token);\r\n    this.tokenStorage.set('user_id', user?.user_id);\r\n    this.tokenStorage.set('username', user?.username);\r\n    this.tokenStorage.set('roles', user?.roles);\r\n    this.tokenStorage.set('permissions', user?.permissions);\r\n    this.tokenStorage.set('organization_id', user?.organization_id);\r\n    this.tokenStorage.set('company_id', user?.company_id);\r\n    this.tokenStorage.set('office_id', user?.office_id);\r\n\r\n    // Notify subscribers\r\n    this.authStatusSubject.next('authenticated');\r\n  }\r\n\r\n  getOrganizationId(): string | null {\r\n    logger.debug('>>>>>>>>>>getOrganizationId<<<<<<<:', this.tokenStorage.get('organization_id'));\r\n    return this.tokenStorage.get('organization_id');\r\n  }\r\n\r\n  getRefreshToken(): string | null {\r\n    logger.debug('>>>>>>>>>>getRefreshToken<<<<<<<:', this.tokenStorage.get('refresh_token'));\r\n    return this.tokenStorage.get('refresh_token');\r\n  }\r\n\r\n  refreshToken(): Observable<AuthResult> {\r\n    const refreshToken = this.getRefreshToken();\r\n    if (!refreshToken) {\r\n      return throwError(() => new Error('No refresh token available'));\r\n    }\r\n\r\n    return this.accountService.refreshLogin({ refresh_token: refreshToken }).pipe(\r\n      retry(3), // Retry 3 times on failure\r\n      tap((response: any) => {\r\n        if (response && response.success && response.data) {\r\n          this.setSession(response.data);\r\n        }\r\n      }),\r\n      map((response: any) => {\r\n        if (response && response.success && response.data) {\r\n          return response.data as AuthResult;\r\n        } else {\r\n          throw new Error('Token refresh failed');\r\n        }\r\n      }),\r\n      catchError((err) => {\r\n        logger.error('[Auth] Token refresh failed after retries:', err);\r\n        this.clearAuth();\r\n        SnackbarUtils.error('Session Expired. Please login.');\r\n        return throwError(() => err);\r\n      }),\r\n    );\r\n  }\r\n\r\n  isAuthenticated(): boolean {\r\n    const token = this.tokenStorage.get('access_token');\r\n    return !!token && !this.isTokenExpired();\r\n  }\r\n\r\n  isTokenExpired(): boolean {\r\n    let token = (this.tokenStorage.get('access_token') ?? '') as string;\r\n    if (token) {\r\n      return this.jwtHelper.isTokenExpired(token);\r\n    } else {\r\n      return true;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Validate stored token on app startup\r\n   */\r\n  validateStoredToken(): Observable<AuthStatus> {\r\n    const accessToken = this.tokenStorage.get('access_token');\r\n\r\n    //! No access token\r\n    if (!accessToken) {\r\n      //TODO: FORCE AUTHENTICATED FOR DEV\r\n      // this.authStatusSubject.next('authenticated');\r\n      // return of('authenticated');\r\n\r\n      //! For production ( Backend Validation )\r\n      this.authStatusSubject.next('unauthenticated');\r\n      return of('unauthenticated');\r\n    }\r\n\r\n    // Access token valid\r\n    if (!this.isTokenExpired()) {\r\n      this.authStatusSubject.next('authenticated');\r\n      return of('authenticated');\r\n    }\r\n\r\n    // Access Token expired -> try refresh\r\n    const refreshToken = this.getRefreshToken();\r\n    if (!refreshToken) {\r\n      this.clearAuth();\r\n      this.authStatusSubject.next('unauthenticated');\r\n      return of('unauthenticated');\r\n    }\r\n\r\n    //! Access expired but refresh token exists → attempt refresh\r\n    this.authStatusSubject.next('expired');\r\n    return this.refreshToken().pipe(\r\n      map(() => {\r\n        this.authStatusSubject.next('authenticated');\r\n        return 'authenticated' as AuthStatus;\r\n      }),\r\n      catchError((err) => {\r\n        logger.error('[Auth] Token refresh failed:', err);\r\n        this.clearAuth();\r\n        this.authStatusSubject.next('unauthenticated');\r\n        return of('unauthenticated' as AuthStatus);\r\n      }),\r\n    );\r\n  }\r\n\r\n  getUserId() {\r\n    return this.tokenStorage.get('user_id') || '';\r\n  }\r\n\r\n  getLicensingContext(): LicensingContext {\r\n    const roles = this.tokenStorage.get('roles');\r\n    // Prefer company_id (specific tenant) over organization_id (system group)\r\n    const tenantId = this.tokenStorage.get('company_id') || this.getOrganizationId() || '';\r\n\r\n    return {\r\n      tenantId: tenantId,\r\n      principalId: this.getUserId() || '',\r\n      roles: Array.isArray(roles) ? roles : roles ? [roles] : [],\r\n    };\r\n  }\r\n\r\n  get licenseKey(): string {\r\n    return this.tokenStorage.get('license_key') || this.coreConfig.licenseKey || '';\r\n  }\r\n\r\n  get isFederated(): boolean {\r\n    return this.licenseKey.startsWith('doohbot_lic_');\r\n  }\r\n\r\n  setLicenseKey(key: string) {\r\n    this.tokenStorage.set('license_key', key);\r\n  }\r\n\r\n  //! clear local storage\r\n  /**\r\n   * Clear all authentication related data from local storage, session storage, cache, and tokens.\r\n   * This method is called when the user logs out.\r\n   */\r\n  clearAuth() {\r\n    const userId = this.getUserId();\r\n    logger.log('[Auth] Clearing auth:', { userId });\r\n\r\n    this.clearSession();\r\n    this.clearStorage();\r\n    this.clearCache();\r\n    this.clearLocalStorage();\r\n\r\n    // ENSURE authStatus$ emits unauthenticated\r\n    this.authStatusSubject.next('unauthenticated');\r\n\r\n    logger.log('[Auth] Auth cleared successfully');\r\n  }\r\n\r\n  clearSession() {\r\n    sessionStorage.clear();\r\n  }\r\n\r\n  clearTabInfo() {\r\n    localStorage.removeItem('tabs');\r\n  }\r\n\r\n  clearStorage() {\r\n    this.tokenStorage.clear();\r\n  }\r\n  clearLocalStorage() {\r\n    localStorage.clear();\r\n  }\r\n\r\n  clearCache() {\r\n    // this.cacheService.clear();\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,3 @@
1
+ export class AuthResult {
2
+ }
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0aC1yZXN1bHQubW9kZWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9kb29oYm90L3NyYy9saWIvY29yZS9hdXRoL21vZGVscy9hdXRoLXJlc3VsdC5tb2RlbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLE9BQU8sVUFBVTtDQUd0QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBjbGFzcyBBdXRoUmVzdWx0IHtcclxuICBhY2Nlc3NfdG9rZW4hOiBzdHJpbmc7XHJcbiAgcmVmcmVzaF90b2tlbiE6IHN0cmluZztcclxufVxyXG4iXX0=
@@ -0,0 +1,6 @@
1
+ export class FederatedLoginRequest {
2
+ constructor() {
3
+ this.rememberMe = false;
4
+ }
5
+ }
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmVkZXJhdGVkLWxvZ2luLXJlcXVlc3QubW9kZWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9kb29oYm90L3NyYy9saWIvY29yZS9hdXRoL21vZGVscy9mZWRlcmF0ZWQtbG9naW4tcmVxdWVzdC5tb2RlbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLE9BQU8scUJBQXFCO0lBQWxDO1FBRUUsZUFBVSxHQUFhLEtBQUssQ0FBQztJQUUvQixDQUFDO0NBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY2xhc3MgRmVkZXJhdGVkTG9naW5SZXF1ZXN0IHtcclxuICBhY2Nlc3NUb2tlbiE6IHN0cmluZztcclxuICByZW1lbWJlck1lPzogYm9vbGVhbiA9IGZhbHNlO1xyXG4gIGxpY2Vuc2U/OiBzdHJpbmc7XHJcbn1cclxuIl19
@@ -0,0 +1,6 @@
1
+ export class LoginRequest {
2
+ constructor() {
3
+ this.rememberMe = false;
4
+ }
5
+ }
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9naW4tcmVxdWVzdC5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2Rvb2hib3Qvc3JjL2xpYi9jb3JlL2F1dGgvbW9kZWxzL2xvZ2luLXJlcXVlc3QubW9kZWwudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsTUFBTSxPQUFPLFlBQVk7SUFBekI7UUFHRSxlQUFVLEdBQWEsS0FBSyxDQUFDO0lBRS9CLENBQUM7Q0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBjbGFzcyBMb2dpblJlcXVlc3Qge1xyXG4gIHVzZXJuYW1lITogc3RyaW5nO1xyXG4gIHBhc3N3b3JkITogc3RyaW5nO1xyXG4gIHJlbWVtYmVyTWU/OiBib29sZWFuID0gZmFsc2U7XHJcbiAgbGljZW5zZT86IHN0cmluZztcclxufVxyXG4iXX0=
@@ -0,0 +1,110 @@
1
+ import { Injectable } from '@angular/core';
2
+ import * as CryptoJS from 'crypto-js';
3
+ import Logger from '../utils/logger.service';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "../services/core-config.service";
6
+ export class StorageService {
7
+ get storageKey() {
8
+ return this.coreConfig.storageKey;
9
+ }
10
+ get secretKey() {
11
+ return this.coreConfig.secretKey;
12
+ }
13
+ constructor(coreConfig) {
14
+ this.coreConfig = coreConfig;
15
+ // Determine initial storage
16
+ // If session storage has data, use it (implies remember me was false)
17
+ // Otherwise default/fallback to local storage
18
+ if (sessionStorage.getItem(this.storageKey)) {
19
+ this.storage = sessionStorage;
20
+ }
21
+ else {
22
+ this.storage = localStorage;
23
+ }
24
+ }
25
+ setRememberMe(rememberMe) {
26
+ const currentData = this.storage.getItem(this.storageKey);
27
+ const targetStorage = rememberMe ? localStorage : sessionStorage;
28
+ // If storage type is changing, move data
29
+ if (this.storage !== targetStorage) {
30
+ if (currentData) {
31
+ targetStorage.setItem(this.storageKey, currentData);
32
+ this.storage.removeItem(this.storageKey);
33
+ }
34
+ this.storage = targetStorage;
35
+ }
36
+ }
37
+ set(key, value) {
38
+ const data = this.storage.getItem(this.storageKey);
39
+ let dataJson = {};
40
+ if (data) {
41
+ const dataDec = this.decrypt(data);
42
+ if (dataDec) {
43
+ dataJson = JSON.parse(dataDec);
44
+ }
45
+ }
46
+ dataJson = Object.assign(dataJson, { [key]: value });
47
+ this.storage.setItem(this.storageKey, this.encrypt(JSON.stringify(dataJson)));
48
+ }
49
+ get(key) {
50
+ const data = this.storage.getItem(this.storageKey);
51
+ if (data) {
52
+ const dataDec = this.decrypt(data);
53
+ if (dataDec) {
54
+ try {
55
+ const dataJson = JSON.parse(dataDec);
56
+ return dataJson[key] || null;
57
+ }
58
+ catch (e) {
59
+ Logger.error('[TokenStorage] Failed to parse decrypted data, clearing corrupt storage:', e);
60
+ this.clear();
61
+ return null;
62
+ }
63
+ }
64
+ else {
65
+ // Decrypt failed, auto-clear corrupt storage
66
+ Logger.error('[TokenStorage] Decrypt failed, clearing corrupt storage');
67
+ this.clear();
68
+ }
69
+ }
70
+ return null;
71
+ }
72
+ /**
73
+ * Check if storage is valid (can be decrypted) without side effects
74
+ */
75
+ isValid() {
76
+ const data = this.storage.getItem(this.storageKey);
77
+ if (!data)
78
+ return false;
79
+ const decrypted = this.decrypt(data);
80
+ return decrypted !== null && decrypted !== '';
81
+ }
82
+ clear() {
83
+ // Clear both to be safe
84
+ localStorage.removeItem(this.storageKey);
85
+ sessionStorage.removeItem(this.storageKey);
86
+ }
87
+ encrypt(value) {
88
+ return CryptoJS.AES.encrypt(value, this.secretKey).toString();
89
+ }
90
+ decrypt(value) {
91
+ try {
92
+ const bytes = CryptoJS.AES.decrypt(value, this.secretKey);
93
+ const decrypted = bytes.toString(CryptoJS.enc.Utf8);
94
+ return decrypted || null;
95
+ }
96
+ catch (e) {
97
+ Logger.error('[TokenStorage] Decryption error:', e);
98
+ return null;
99
+ }
100
+ }
101
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: StorageService, deps: [{ token: i1.CoreConfigService }], target: i0.ɵɵFactoryTarget.Injectable }); }
102
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: StorageService, providedIn: 'root' }); }
103
+ }
104
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: StorageService, decorators: [{
105
+ type: Injectable,
106
+ args: [{
107
+ providedIn: 'root',
108
+ }]
109
+ }], ctorParameters: () => [{ type: i1.CoreConfigService }] });
110
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"storage.service.js","sourceRoot":"","sources":["../../../../../../projects/doohbot/src/lib/core/auth/storage.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAC3C,OAAO,KAAK,QAAQ,MAAM,WAAW,CAAC;AAEtC,OAAO,MAAM,MAAM,yBAAyB,CAAC;;;AAK7C,MAAM,OAAO,cAAc;IACzB,IAAI,UAAU;QACZ,OAAO,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC;IACpC,CAAC;IAED,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC;IACnC,CAAC;IAID,YAAoB,UAA6B;QAA7B,eAAU,GAAV,UAAU,CAAmB;QAC/C,4BAA4B;QAC5B,sEAAsE;QACtE,8CAA8C;QAC9C,IAAI,cAAc,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE;YAC3C,IAAI,CAAC,OAAO,GAAG,cAAc,CAAC;SAC/B;aAAM;YACL,IAAI,CAAC,OAAO,GAAG,YAAY,CAAC;SAC7B;IACH,CAAC;IAED,aAAa,CAAC,UAAmB;QAC/B,MAAM,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAC1D,MAAM,aAAa,GAAG,UAAU,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,cAAc,CAAC;QAEjE,yCAAyC;QACzC,IAAI,IAAI,CAAC,OAAO,KAAK,aAAa,EAAE;YAClC,IAAI,WAAW,EAAE;gBACf,aAAa,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,EAAE,WAAW,CAAC,CAAC;gBACpD,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;aAC1C;YACD,IAAI,CAAC,OAAO,GAAG,aAAa,CAAC;SAC9B;IACH,CAAC;IAED,GAAG,CAAC,GAAW,EAAE,KAAU;QACzB,MAAM,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACnD,IAAI,QAAQ,GAAQ,EAAE,CAAC;QACvB,IAAI,IAAI,EAAE;YACR,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;YACnC,IAAI,OAAO,EAAE;gBACX,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;aAChC;SACF;QACD,QAAQ,GAAG,MAAM,CAAC,MAAM,CAAC,QAAQ,EAAE,EAAE,CAAC,GAAG,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;QACrD,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;IAChF,CAAC;IAED,GAAG,CAAC,GAAW;QACb,MAAM,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACnD,IAAI,IAAI,EAAE;YACR,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;YACnC,IAAI,OAAO,EAAE;gBACX,IAAI;oBACF,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;oBACrC,OAAO,QAAS,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC;iBAC/B;gBAAC,OAAO,CAAC,EAAE;oBACV,MAAM,CAAC,KAAK,CACV,0EAA0E,EAC1E,CAAC,CACF,CAAC;oBACF,IAAI,CAAC,KAAK,EAAE,CAAC;oBACb,OAAO,IAAI,CAAC;iBACb;aACF;iBAAM;gBACL,6CAA6C;gBAC7C,MAAM,CAAC,KAAK,CAAC,yDAAyD,CAAC,CAAC;gBACxE,IAAI,CAAC,KAAK,EAAE,CAAC;aACd;SACF;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;OAEG;IACH,OAAO;QACL,MAAM,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACnD,IAAI,CAAC,IAAI;YAAE,OAAO,KAAK,CAAC;QACxB,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QACrC,OAAO,SAAS,KAAK,IAAI,IAAI,SAAS,KAAK,EAAE,CAAC;IAChD,CAAC;IAED,KAAK;QACH,wBAAwB;QACxB,YAAY,CAAC,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACzC,cAAc,CAAC,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC7C,CAAC;IAEO,OAAO,CAAC,KAAa;QAC3B,OAAO,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,KAAK,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC,QAAQ,EAAE,CAAC;IAChE,CAAC;IAEO,OAAO,CAAC,KAAa;QAC3B,IAAI;YACF,MAAM,KAAK,GAAG,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,KAAK,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;YAC1D,MAAM,SAAS,GAAG,KAAK,CAAC,QAAQ,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YACpD,OAAO,SAAS,IAAI,IAAI,CAAC;SAC1B;QAAC,OAAO,CAAC,EAAE;YACV,MAAM,CAAC,KAAK,CAAC,kCAAkC,EAAE,CAAC,CAAC,CAAC;YACpD,OAAO,IAAI,CAAC;SACb;IACH,CAAC;+GAvGU,cAAc;mHAAd,cAAc,cAFb,MAAM;;4FAEP,cAAc;kBAH1B,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport * as CryptoJS from 'crypto-js';\r\nimport { CoreConfigService } from '../services/core-config.service';\r\nimport Logger from '../utils/logger.service';\r\n\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class StorageService {\r\n  get storageKey(): string {\r\n    return this.coreConfig.storageKey;\r\n  }\r\n\r\n  get secretKey(): string {\r\n    return this.coreConfig.secretKey;\r\n  }\r\n\r\n  private storage: Storage;\r\n\r\n  constructor(private coreConfig: CoreConfigService) {\r\n    // Determine initial storage\r\n    // If session storage has data, use it (implies remember me was false)\r\n    // Otherwise default/fallback to local storage\r\n    if (sessionStorage.getItem(this.storageKey)) {\r\n      this.storage = sessionStorage;\r\n    } else {\r\n      this.storage = localStorage;\r\n    }\r\n  }\r\n\r\n  setRememberMe(rememberMe: boolean): void {\r\n    const currentData = this.storage.getItem(this.storageKey);\r\n    const targetStorage = rememberMe ? localStorage : sessionStorage;\r\n\r\n    // If storage type is changing, move data\r\n    if (this.storage !== targetStorage) {\r\n      if (currentData) {\r\n        targetStorage.setItem(this.storageKey, currentData);\r\n        this.storage.removeItem(this.storageKey);\r\n      }\r\n      this.storage = targetStorage;\r\n    }\r\n  }\r\n\r\n  set(key: string, value: any): void {\r\n    const data = this.storage.getItem(this.storageKey);\r\n    let dataJson: any = {};\r\n    if (data) {\r\n      const dataDec = this.decrypt(data);\r\n      if (dataDec) {\r\n        dataJson = JSON.parse(dataDec);\r\n      }\r\n    }\r\n    dataJson = Object.assign(dataJson, { [key]: value });\r\n    this.storage.setItem(this.storageKey, this.encrypt(JSON.stringify(dataJson)));\r\n  }\r\n\r\n  get(key: string): any {\r\n    const data = this.storage.getItem(this.storageKey);\r\n    if (data) {\r\n      const dataDec = this.decrypt(data);\r\n      if (dataDec) {\r\n        try {\r\n          const dataJson = JSON.parse(dataDec);\r\n          return dataJson![key] || null;\r\n        } catch (e) {\r\n          Logger.error(\r\n            '[TokenStorage] Failed to parse decrypted data, clearing corrupt storage:',\r\n            e,\r\n          );\r\n          this.clear();\r\n          return null;\r\n        }\r\n      } else {\r\n        // Decrypt failed, auto-clear corrupt storage\r\n        Logger.error('[TokenStorage] Decrypt failed, clearing corrupt storage');\r\n        this.clear();\r\n      }\r\n    }\r\n    return null;\r\n  }\r\n\r\n  /**\r\n   * Check if storage is valid (can be decrypted) without side effects\r\n   */\r\n  isValid(): boolean {\r\n    const data = this.storage.getItem(this.storageKey);\r\n    if (!data) return false;\r\n    const decrypted = this.decrypt(data);\r\n    return decrypted !== null && decrypted !== '';\r\n  }\r\n\r\n  clear(): void {\r\n    // Clear both to be safe\r\n    localStorage.removeItem(this.storageKey);\r\n    sessionStorage.removeItem(this.storageKey);\r\n  }\r\n\r\n  private encrypt(value: string): string {\r\n    return CryptoJS.AES.encrypt(value, this.secretKey).toString();\r\n  }\r\n\r\n  private decrypt(value: string): string | null {\r\n    try {\r\n      const bytes = CryptoJS.AES.decrypt(value, this.secretKey);\r\n      const decrypted = bytes.toString(CryptoJS.enc.Utf8);\r\n      return decrypted || null;\r\n    } catch (e) {\r\n      Logger.error('[TokenStorage] Decryption error:', e);\r\n      return null;\r\n    }\r\n  }\r\n}\r\n"]}