valtech-components 2.0.428 → 2.0.430
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.
- package/esm2022/lib/components/organisms/data-table/data-table.component.mjs +17 -3
- package/esm2022/lib/components/organisms/data-table/types.mjs +1 -1
- package/esm2022/lib/services/auth/auth-state.service.mjs +173 -0
- package/esm2022/lib/services/auth/auth.service.mjs +432 -0
- package/esm2022/lib/services/auth/config.mjs +76 -0
- package/esm2022/lib/services/auth/guards.mjs +194 -0
- package/esm2022/lib/services/auth/index.mjs +70 -0
- package/esm2022/lib/services/auth/interceptor.mjs +98 -0
- package/esm2022/lib/services/auth/storage.service.mjs +138 -0
- package/esm2022/lib/services/auth/sync.service.mjs +146 -0
- package/esm2022/lib/services/auth/token.service.mjs +113 -0
- package/esm2022/lib/services/auth/types.mjs +29 -0
- package/esm2022/public-api.mjs +4 -1
- package/fesm2022/valtech-components.mjs +1465 -8
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/organisms/data-table/types.d.ts +8 -0
- package/lib/services/auth/auth-state.service.d.ts +85 -0
- package/lib/services/auth/auth.service.d.ts +123 -0
- package/lib/services/auth/config.d.ts +38 -0
- package/lib/services/auth/guards.d.ts +123 -0
- package/lib/services/auth/index.d.ts +63 -0
- package/lib/services/auth/interceptor.d.ts +22 -0
- package/lib/services/auth/storage.service.d.ts +48 -0
- package/lib/services/auth/sync.service.d.ts +49 -0
- package/lib/services/auth/token.service.d.ts +51 -0
- package/lib/services/auth/types.d.ts +264 -0
- package/package.json +1 -9
- package/public-api.d.ts +1 -0
|
@@ -0,0 +1,173 @@
|
|
|
1
|
+
import { Injectable, signal, computed } from '@angular/core';
|
|
2
|
+
import { INITIAL_AUTH_STATE, INITIAL_MFA_STATE, } from './types';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
/**
|
|
5
|
+
* Servicio para manejo de estado de autenticación con Angular Signals.
|
|
6
|
+
* Proporciona estado reactivo inmutable.
|
|
7
|
+
*/
|
|
8
|
+
export class AuthStateService {
|
|
9
|
+
constructor() {
|
|
10
|
+
// Estado interno (mutable solo dentro del servicio)
|
|
11
|
+
this._state = signal(INITIAL_AUTH_STATE);
|
|
12
|
+
this._mfaPending = signal(INITIAL_MFA_STATE);
|
|
13
|
+
// =============================================
|
|
14
|
+
// Signals públicos (readonly)
|
|
15
|
+
// =============================================
|
|
16
|
+
/** Estado completo de autenticación */
|
|
17
|
+
this.state = this._state.asReadonly();
|
|
18
|
+
/** Estado de MFA pendiente */
|
|
19
|
+
this.mfaPending = this._mfaPending.asReadonly();
|
|
20
|
+
/** Usuario está autenticado */
|
|
21
|
+
this.isAuthenticated = computed(() => this._state().isAuthenticated);
|
|
22
|
+
/** Estado de carga */
|
|
23
|
+
this.isLoading = computed(() => this._state().isLoading);
|
|
24
|
+
/** Token de acceso */
|
|
25
|
+
this.accessToken = computed(() => this._state().accessToken);
|
|
26
|
+
/** Roles del usuario */
|
|
27
|
+
this.roles = computed(() => this._state().roles);
|
|
28
|
+
/** Permisos del usuario */
|
|
29
|
+
this.permissions = computed(() => this._state().permissions);
|
|
30
|
+
/** Usuario es super admin */
|
|
31
|
+
this.isSuperAdmin = computed(() => this._state().isSuperAdmin);
|
|
32
|
+
/** Error actual */
|
|
33
|
+
this.error = computed(() => this._state().error);
|
|
34
|
+
/** Información del usuario */
|
|
35
|
+
this.user = computed(() => {
|
|
36
|
+
const state = this._state();
|
|
37
|
+
if (!state.isAuthenticated || !state.userId) {
|
|
38
|
+
return null;
|
|
39
|
+
}
|
|
40
|
+
return {
|
|
41
|
+
userId: state.userId,
|
|
42
|
+
email: state.email || '',
|
|
43
|
+
roles: state.roles,
|
|
44
|
+
permissions: state.permissions,
|
|
45
|
+
isSuperAdmin: state.isSuperAdmin,
|
|
46
|
+
};
|
|
47
|
+
});
|
|
48
|
+
}
|
|
49
|
+
// =============================================
|
|
50
|
+
// Métodos de actualización
|
|
51
|
+
// =============================================
|
|
52
|
+
/**
|
|
53
|
+
* Establece el estado de carga.
|
|
54
|
+
*/
|
|
55
|
+
setLoading(isLoading) {
|
|
56
|
+
this._state.update((s) => ({ ...s, isLoading }));
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* Establece el estado de autenticación exitosa.
|
|
60
|
+
*/
|
|
61
|
+
setAuthenticated(data) {
|
|
62
|
+
this._state.set({
|
|
63
|
+
isAuthenticated: true,
|
|
64
|
+
isLoading: false,
|
|
65
|
+
accessToken: data.accessToken,
|
|
66
|
+
refreshToken: data.refreshToken,
|
|
67
|
+
userId: data.userId || null,
|
|
68
|
+
email: data.email || null,
|
|
69
|
+
roles: data.roles,
|
|
70
|
+
permissions: data.permissions,
|
|
71
|
+
isSuperAdmin: data.isSuperAdmin,
|
|
72
|
+
expiresAt: data.expiresAt,
|
|
73
|
+
error: null,
|
|
74
|
+
});
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Actualiza solo el access token (después de refresh).
|
|
78
|
+
*/
|
|
79
|
+
updateAccessToken(accessToken, expiresIn) {
|
|
80
|
+
const expiresAt = Date.now() + expiresIn * 1000;
|
|
81
|
+
this._state.update((s) => ({
|
|
82
|
+
...s,
|
|
83
|
+
accessToken,
|
|
84
|
+
expiresAt,
|
|
85
|
+
}));
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Actualiza los permisos.
|
|
89
|
+
*/
|
|
90
|
+
updatePermissions(roles, permissions, isSuperAdmin) {
|
|
91
|
+
this._state.update((s) => ({
|
|
92
|
+
...s,
|
|
93
|
+
roles,
|
|
94
|
+
permissions,
|
|
95
|
+
isSuperAdmin,
|
|
96
|
+
}));
|
|
97
|
+
}
|
|
98
|
+
/**
|
|
99
|
+
* Establece un error de autenticación.
|
|
100
|
+
*/
|
|
101
|
+
setError(error) {
|
|
102
|
+
this._state.update((s) => ({
|
|
103
|
+
...s,
|
|
104
|
+
error,
|
|
105
|
+
isLoading: false,
|
|
106
|
+
}));
|
|
107
|
+
}
|
|
108
|
+
/**
|
|
109
|
+
* Limpia el error.
|
|
110
|
+
*/
|
|
111
|
+
clearError() {
|
|
112
|
+
this._state.update((s) => ({
|
|
113
|
+
...s,
|
|
114
|
+
error: null,
|
|
115
|
+
}));
|
|
116
|
+
}
|
|
117
|
+
/**
|
|
118
|
+
* Establece estado de MFA pendiente.
|
|
119
|
+
*/
|
|
120
|
+
setMFAPending(mfaState) {
|
|
121
|
+
this._mfaPending.set(mfaState);
|
|
122
|
+
}
|
|
123
|
+
/**
|
|
124
|
+
* Limpia el estado de MFA pendiente.
|
|
125
|
+
*/
|
|
126
|
+
clearMFAPending() {
|
|
127
|
+
this._mfaPending.set(INITIAL_MFA_STATE);
|
|
128
|
+
}
|
|
129
|
+
/**
|
|
130
|
+
* Resetea todo el estado a valores iniciales.
|
|
131
|
+
*/
|
|
132
|
+
reset() {
|
|
133
|
+
this._state.set(INITIAL_AUTH_STATE);
|
|
134
|
+
this._mfaPending.set(INITIAL_MFA_STATE);
|
|
135
|
+
}
|
|
136
|
+
/**
|
|
137
|
+
* Restaura estado desde datos almacenados.
|
|
138
|
+
*/
|
|
139
|
+
restoreFromStorage(stored) {
|
|
140
|
+
if (stored.accessToken) {
|
|
141
|
+
this._state.set({
|
|
142
|
+
isAuthenticated: true,
|
|
143
|
+
isLoading: false,
|
|
144
|
+
accessToken: stored.accessToken,
|
|
145
|
+
refreshToken: stored.refreshToken || null,
|
|
146
|
+
userId: null, // Se extraerá del token
|
|
147
|
+
email: null, // Se extraerá del token
|
|
148
|
+
roles: stored.roles || [],
|
|
149
|
+
permissions: stored.permissions || [],
|
|
150
|
+
isSuperAdmin: stored.isSuperAdmin || false,
|
|
151
|
+
expiresAt: stored.expiresAt || null,
|
|
152
|
+
error: null,
|
|
153
|
+
});
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
/**
|
|
157
|
+
* Actualiza el userId y email (después de parsear el token).
|
|
158
|
+
*/
|
|
159
|
+
updateUserInfo(userId, email) {
|
|
160
|
+
this._state.update((s) => ({
|
|
161
|
+
...s,
|
|
162
|
+
userId,
|
|
163
|
+
email,
|
|
164
|
+
}));
|
|
165
|
+
}
|
|
166
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthStateService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
167
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthStateService, providedIn: 'root' }); }
|
|
168
|
+
}
|
|
169
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthStateService, decorators: [{
|
|
170
|
+
type: Injectable,
|
|
171
|
+
args: [{ providedIn: 'root' }]
|
|
172
|
+
}] });
|
|
173
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth-state.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/auth-state.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, signal, computed } from '@angular/core';\nimport {\n  AuthState,\n  AuthUser,\n  AuthError,\n  MFAPendingState,\n  INITIAL_AUTH_STATE,\n  INITIAL_MFA_STATE,\n  StoredAuthState,\n} from './types';\n\n/**\n * Servicio para manejo de estado de autenticación con Angular Signals.\n * Proporciona estado reactivo inmutable.\n */\n@Injectable({ providedIn: 'root' })\nexport class AuthStateService {\n  // Estado interno (mutable solo dentro del servicio)\n  private _state = signal<AuthState>(INITIAL_AUTH_STATE);\n  private _mfaPending = signal<MFAPendingState>(INITIAL_MFA_STATE);\n\n  // =============================================\n  // Signals públicos (readonly)\n  // =============================================\n\n  /** Estado completo de autenticación */\n  readonly state = this._state.asReadonly();\n\n  /** Estado de MFA pendiente */\n  readonly mfaPending = this._mfaPending.asReadonly();\n\n  /** Usuario está autenticado */\n  readonly isAuthenticated = computed(() => this._state().isAuthenticated);\n\n  /** Estado de carga */\n  readonly isLoading = computed(() => this._state().isLoading);\n\n  /** Token de acceso */\n  readonly accessToken = computed(() => this._state().accessToken);\n\n  /** Roles del usuario */\n  readonly roles = computed(() => this._state().roles);\n\n  /** Permisos del usuario */\n  readonly permissions = computed(() => this._state().permissions);\n\n  /** Usuario es super admin */\n  readonly isSuperAdmin = computed(() => this._state().isSuperAdmin);\n\n  /** Error actual */\n  readonly error = computed(() => this._state().error);\n\n  /** Información del usuario */\n  readonly user = computed<AuthUser | null>(() => {\n    const state = this._state();\n    if (!state.isAuthenticated || !state.userId) {\n      return null;\n    }\n    return {\n      userId: state.userId,\n      email: state.email || '',\n      roles: state.roles,\n      permissions: state.permissions,\n      isSuperAdmin: state.isSuperAdmin,\n    };\n  });\n\n  // =============================================\n  // Métodos de actualización\n  // =============================================\n\n  /**\n   * Establece el estado de carga.\n   */\n  setLoading(isLoading: boolean): void {\n    this._state.update((s) => ({ ...s, isLoading }));\n  }\n\n  /**\n   * Establece el estado de autenticación exitosa.\n   */\n  setAuthenticated(data: {\n    accessToken: string;\n    refreshToken: string;\n    userId?: string;\n    email?: string;\n    roles: string[];\n    permissions: string[];\n    isSuperAdmin: boolean;\n    expiresAt: number;\n  }): void {\n    this._state.set({\n      isAuthenticated: true,\n      isLoading: false,\n      accessToken: data.accessToken,\n      refreshToken: data.refreshToken,\n      userId: data.userId || null,\n      email: data.email || null,\n      roles: data.roles,\n      permissions: data.permissions,\n      isSuperAdmin: data.isSuperAdmin,\n      expiresAt: data.expiresAt,\n      error: null,\n    });\n  }\n\n  /**\n   * Actualiza solo el access token (después de refresh).\n   */\n  updateAccessToken(accessToken: string, expiresIn: number): void {\n    const expiresAt = Date.now() + expiresIn * 1000;\n    this._state.update((s) => ({\n      ...s,\n      accessToken,\n      expiresAt,\n    }));\n  }\n\n  /**\n   * Actualiza los permisos.\n   */\n  updatePermissions(\n    roles: string[],\n    permissions: string[],\n    isSuperAdmin: boolean\n  ): void {\n    this._state.update((s) => ({\n      ...s,\n      roles,\n      permissions,\n      isSuperAdmin,\n    }));\n  }\n\n  /**\n   * Establece un error de autenticación.\n   */\n  setError(error: AuthError): void {\n    this._state.update((s) => ({\n      ...s,\n      error,\n      isLoading: false,\n    }));\n  }\n\n  /**\n   * Limpia el error.\n   */\n  clearError(): void {\n    this._state.update((s) => ({\n      ...s,\n      error: null,\n    }));\n  }\n\n  /**\n   * Establece estado de MFA pendiente.\n   */\n  setMFAPending(mfaState: MFAPendingState): void {\n    this._mfaPending.set(mfaState);\n  }\n\n  /**\n   * Limpia el estado de MFA pendiente.\n   */\n  clearMFAPending(): void {\n    this._mfaPending.set(INITIAL_MFA_STATE);\n  }\n\n  /**\n   * Resetea todo el estado a valores iniciales.\n   */\n  reset(): void {\n    this._state.set(INITIAL_AUTH_STATE);\n    this._mfaPending.set(INITIAL_MFA_STATE);\n  }\n\n  /**\n   * Restaura estado desde datos almacenados.\n   */\n  restoreFromStorage(stored: Partial<StoredAuthState>): void {\n    if (stored.accessToken) {\n      this._state.set({\n        isAuthenticated: true,\n        isLoading: false,\n        accessToken: stored.accessToken,\n        refreshToken: stored.refreshToken || null,\n        userId: null, // Se extraerá del token\n        email: null, // Se extraerá del token\n        roles: stored.roles || [],\n        permissions: stored.permissions || [],\n        isSuperAdmin: stored.isSuperAdmin || false,\n        expiresAt: stored.expiresAt || null,\n        error: null,\n      });\n    }\n  }\n\n  /**\n   * Actualiza el userId y email (después de parsear el token).\n   */\n  updateUserInfo(userId: string, email: string): void {\n    this._state.update((s) => ({\n      ...s,\n      userId,\n      email,\n    }));\n  }\n}\n"]}
|