primus-angular-auth 1.0.0

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.
@@ -0,0 +1,23 @@
1
+ import { Router } from '@angular/router';
2
+ import { inject } from '@angular/core';
3
+ import { PrimusAuthService } from './primus-auth.service';
4
+ import { map, filter, switchMap, take } from 'rxjs/operators';
5
+ import { PRIMUS_AUTH_CONFIG } from './primus-auth.tokens';
6
+ export const primusAuthGuard = (route, state) => {
7
+ const auth = inject(PrimusAuthService);
8
+ const router = inject(Router);
9
+ const config = inject(PRIMUS_AUTH_CONFIG);
10
+ // Wait for initialization to complete, then check authentication status
11
+ return auth.initialized$.pipe(filter((isInit) => isInit), // Wait until true
12
+ take(1), switchMap(() => auth.isAuthenticated$), take(1), map((isAuthenticated) => {
13
+ if (isAuthenticated) {
14
+ return true;
15
+ }
16
+ // Redirect to login
17
+ const loginPath = config.loginPath || '/login';
18
+ return router.createUrlTree([loginPath], {
19
+ queryParams: { returnUrl: state.url }
20
+ });
21
+ }));
22
+ };
23
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJpbXVzLWF1dGguZ3VhcmQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9hbmd1bGFyL3ByaW11cy1hdXRoL3NyYy9saWIvcHJpbXVzLWF1dGguZ3VhcmQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFpQixNQUFNLEVBQXdELE1BQU0saUJBQWlCLENBQUM7QUFDOUcsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN2QyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUMxRCxPQUFPLEVBQUUsR0FBRyxFQUFFLE1BQU0sRUFBRSxTQUFTLEVBQUUsSUFBSSxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDOUQsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sc0JBQXNCLENBQUM7QUFFMUQsTUFBTSxDQUFDLE1BQU0sZUFBZSxHQUFrQixDQUFDLEtBQTZCLEVBQUUsS0FBMEIsRUFBRSxFQUFFO0lBQ3hHLE1BQU0sSUFBSSxHQUFHLE1BQU0sQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDO0lBQ3ZDLE1BQU0sTUFBTSxHQUFHLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUM5QixNQUFNLE1BQU0sR0FBRyxNQUFNLENBQUMsa0JBQWtCLENBQUMsQ0FBQztJQUUxQyx3RUFBd0U7SUFDeEUsT0FBTyxJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksQ0FDekIsTUFBTSxDQUFDLENBQUMsTUFBZSxFQUFFLEVBQUUsQ0FBQyxNQUFNLENBQUMsRUFBRSxrQkFBa0I7SUFDdkQsSUFBSSxDQUFDLENBQUMsQ0FBQyxFQUNQLFNBQVMsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsRUFDdEMsSUFBSSxDQUFDLENBQUMsQ0FBQyxFQUNQLEdBQUcsQ0FBQyxDQUFDLGVBQXdCLEVBQUUsRUFBRTtRQUM3QixJQUFJLGVBQWUsRUFBRTtZQUNqQixPQUFPLElBQUksQ0FBQztTQUNmO1FBRUQsb0JBQW9CO1FBQ3BCLE1BQU0sU0FBUyxHQUFHLE1BQU0sQ0FBQyxTQUFTLElBQUksUUFBUSxDQUFDO1FBQy9DLE9BQU8sTUFBTSxDQUFDLGFBQWEsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxFQUFFO1lBQ3JDLFdBQVcsRUFBRSxFQUFFLFNBQVMsRUFBRSxLQUFLLENBQUMsR0FBRyxFQUFFO1NBQ3hDLENBQUMsQ0FBQztJQUNQLENBQUMsQ0FBQyxDQUNMLENBQUM7QUFDTixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDYW5BY3RpdmF0ZUZuLCBSb3V0ZXIsIFVybFRyZWUsIEFjdGl2YXRlZFJvdXRlU25hcHNob3QsIFJvdXRlclN0YXRlU25hcHNob3QgfSBmcm9tICdAYW5ndWxhci9yb3V0ZXInO1xyXG5pbXBvcnQgeyBpbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgUHJpbXVzQXV0aFNlcnZpY2UgfSBmcm9tICcuL3ByaW11cy1hdXRoLnNlcnZpY2UnO1xyXG5pbXBvcnQgeyBtYXAsIGZpbHRlciwgc3dpdGNoTWFwLCB0YWtlIH0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xyXG5pbXBvcnQgeyBQUklNVVNfQVVUSF9DT05GSUcgfSBmcm9tICcuL3ByaW11cy1hdXRoLnRva2Vucyc7XHJcblxyXG5leHBvcnQgY29uc3QgcHJpbXVzQXV0aEd1YXJkOiBDYW5BY3RpdmF0ZUZuID0gKHJvdXRlOiBBY3RpdmF0ZWRSb3V0ZVNuYXBzaG90LCBzdGF0ZTogUm91dGVyU3RhdGVTbmFwc2hvdCkgPT4ge1xyXG4gICAgY29uc3QgYXV0aCA9IGluamVjdChQcmltdXNBdXRoU2VydmljZSk7XHJcbiAgICBjb25zdCByb3V0ZXIgPSBpbmplY3QoUm91dGVyKTtcclxuICAgIGNvbnN0IGNvbmZpZyA9IGluamVjdChQUklNVVNfQVVUSF9DT05GSUcpO1xyXG5cclxuICAgIC8vIFdhaXQgZm9yIGluaXRpYWxpemF0aW9uIHRvIGNvbXBsZXRlLCB0aGVuIGNoZWNrIGF1dGhlbnRpY2F0aW9uIHN0YXR1c1xyXG4gICAgcmV0dXJuIGF1dGguaW5pdGlhbGl6ZWQkLnBpcGUoXHJcbiAgICAgICAgZmlsdGVyKChpc0luaXQ6IGJvb2xlYW4pID0+IGlzSW5pdCksIC8vIFdhaXQgdW50aWwgdHJ1ZVxyXG4gICAgICAgIHRha2UoMSksXHJcbiAgICAgICAgc3dpdGNoTWFwKCgpID0+IGF1dGguaXNBdXRoZW50aWNhdGVkJCksXHJcbiAgICAgICAgdGFrZSgxKSxcclxuICAgICAgICBtYXAoKGlzQXV0aGVudGljYXRlZDogYm9vbGVhbikgPT4ge1xyXG4gICAgICAgICAgICBpZiAoaXNBdXRoZW50aWNhdGVkKSB7XHJcbiAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTtcclxuICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgLy8gUmVkaXJlY3QgdG8gbG9naW5cclxuICAgICAgICAgICAgY29uc3QgbG9naW5QYXRoID0gY29uZmlnLmxvZ2luUGF0aCB8fCAnL2xvZ2luJztcclxuICAgICAgICAgICAgcmV0dXJuIHJvdXRlci5jcmVhdGVVcmxUcmVlKFtsb2dpblBhdGhdLCB7XHJcbiAgICAgICAgICAgICAgICBxdWVyeVBhcmFtczogeyByZXR1cm5Vcmw6IHN0YXRlLnVybCB9XHJcbiAgICAgICAgICAgIH0pO1xyXG4gICAgICAgIH0pXHJcbiAgICApO1xyXG59O1xyXG4iXX0=
@@ -0,0 +1,45 @@
1
+ import { inject } from '@angular/core';
2
+ import { catchError, throwError } from 'rxjs';
3
+ import { PrimusAuthService } from './primus-auth.service';
4
+ export const primusAuthInterceptor = (req, next) => {
5
+ const authService = inject(PrimusAuthService);
6
+ // 1. Add CSRF Header to non-GET requests
7
+ // Double-Submit Cookie Pattern: Read XSRF-TOKEN cookie and mirror in header.
8
+ let request = req;
9
+ if (req.method !== 'GET' && req.method !== 'HEAD' && req.method !== 'OPTIONS') {
10
+ const token = getCookie('XSRF-TOKEN') || '';
11
+ request = req.clone({
12
+ setHeaders: {
13
+ 'X-Primus-CSRF': token
14
+ },
15
+ withCredentials: true // Ensure cookies are sent
16
+ });
17
+ }
18
+ else {
19
+ // Determine if we need credentials for GETs?
20
+ // Usually yes for API calls to our backend.
21
+ // We assume calls to same origin or API URL need creds.
22
+ request = req.clone({
23
+ withCredentials: true
24
+ });
25
+ }
26
+ return next(request).pipe(catchError((error) => {
27
+ // 2. Handle 401 Unauthorized
28
+ if (error.status === 401) {
29
+ authService.handleUnauthorized();
30
+ }
31
+ return throwError(() => error);
32
+ }));
33
+ };
34
+ function getCookie(name) {
35
+ try {
36
+ if (typeof document === 'undefined')
37
+ return null;
38
+ const matches = document.cookie.match(new RegExp('(?:^|; )' + name.replace(/([\.$?*|{}\(\)\[\]\\\/\+^])/g, '\\$1') + '=([^;]*)'));
39
+ return matches ? decodeURIComponent(matches[1]) : null;
40
+ }
41
+ catch {
42
+ return null;
43
+ }
44
+ }
45
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJpbXVzLWF1dGguaW50ZXJjZXB0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9hbmd1bGFyL3ByaW11cy1hdXRoL3NyYy9saWIvcHJpbXVzLWF1dGguaW50ZXJjZXB0b3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN2QyxPQUFPLEVBQUUsVUFBVSxFQUFFLFVBQVUsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUM5QyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUUxRCxNQUFNLENBQUMsTUFBTSxxQkFBcUIsR0FBc0IsQ0FBQyxHQUFHLEVBQUUsSUFBSSxFQUFFLEVBQUU7SUFDbEUsTUFBTSxXQUFXLEdBQUcsTUFBTSxDQUFDLGlCQUFpQixDQUFDLENBQUM7SUFFMUMseUNBQXlDO0lBQ3pDLDZFQUE2RTtJQUM3RSxJQUFJLE9BQU8sR0FBRyxHQUFHLENBQUM7SUFDbEIsSUFBSSxHQUFHLENBQUMsTUFBTSxLQUFLLEtBQUssSUFBSSxHQUFHLENBQUMsTUFBTSxLQUFLLE1BQU0sSUFBSSxHQUFHLENBQUMsTUFBTSxLQUFLLFNBQVMsRUFBRTtRQUMzRSxNQUFNLEtBQUssR0FBRyxTQUFTLENBQUMsWUFBWSxDQUFDLElBQUksRUFBRSxDQUFDO1FBQzVDLE9BQU8sR0FBRyxHQUFHLENBQUMsS0FBSyxDQUFDO1lBQ2hCLFVBQVUsRUFBRTtnQkFDUixlQUFlLEVBQUUsS0FBSzthQUN6QjtZQUNELGVBQWUsRUFBRSxJQUFJLENBQUMsMEJBQTBCO1NBQ25ELENBQUMsQ0FBQztLQUNOO1NBQU07UUFDSCw2Q0FBNkM7UUFDN0MsNENBQTRDO1FBQzVDLHdEQUF3RDtRQUN4RCxPQUFPLEdBQUcsR0FBRyxDQUFDLEtBQUssQ0FBQztZQUNoQixlQUFlLEVBQUUsSUFBSTtTQUN4QixDQUFDLENBQUM7S0FDTjtJQUVELE9BQU8sSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDLElBQUksQ0FDckIsVUFBVSxDQUFDLENBQUMsS0FBd0IsRUFBRSxFQUFFO1FBQ3BDLDZCQUE2QjtRQUM3QixJQUFJLEtBQUssQ0FBQyxNQUFNLEtBQUssR0FBRyxFQUFFO1lBQ3RCLFdBQVcsQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1NBQ3BDO1FBQ0QsT0FBTyxVQUFVLENBQUMsR0FBRyxFQUFFLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDbkMsQ0FBQyxDQUFDLENBQ0wsQ0FBQztBQUNOLENBQUMsQ0FBQztBQUVGLFNBQVMsU0FBUyxDQUFDLElBQVk7SUFDM0IsSUFBSTtRQUNBLElBQUksT0FBTyxRQUFRLEtBQUssV0FBVztZQUFFLE9BQU8sSUFBSSxDQUFDO1FBQ2pELE1BQU0sT0FBTyxHQUFHLFFBQVEsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLElBQUksTUFBTSxDQUFDLFVBQVUsR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLDhCQUE4QixFQUFFLE1BQU0sQ0FBQyxHQUFHLFVBQVUsQ0FBQyxDQUFDLENBQUM7UUFDbEksT0FBTyxPQUFPLENBQUMsQ0FBQyxDQUFDLGtCQUFrQixDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUM7S0FDMUQ7SUFBQyxNQUFNO1FBQ0osT0FBTyxJQUFJLENBQUM7S0FDZjtBQUNMLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBIdHRwSW50ZXJjZXB0b3JGbiwgSHR0cEVycm9yUmVzcG9uc2UgfSBmcm9tICdAYW5ndWxhci9jb21tb24vaHR0cCc7XHJcbmltcG9ydCB7IGluamVjdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBjYXRjaEVycm9yLCB0aHJvd0Vycm9yIH0gZnJvbSAncnhqcyc7XHJcbmltcG9ydCB7IFByaW11c0F1dGhTZXJ2aWNlIH0gZnJvbSAnLi9wcmltdXMtYXV0aC5zZXJ2aWNlJztcclxuXHJcbmV4cG9ydCBjb25zdCBwcmltdXNBdXRoSW50ZXJjZXB0b3I6IEh0dHBJbnRlcmNlcHRvckZuID0gKHJlcSwgbmV4dCkgPT4ge1xyXG4gICAgY29uc3QgYXV0aFNlcnZpY2UgPSBpbmplY3QoUHJpbXVzQXV0aFNlcnZpY2UpO1xyXG5cclxuICAgICAgICAvLyAxLiBBZGQgQ1NSRiBIZWFkZXIgdG8gbm9uLUdFVCByZXF1ZXN0c1xyXG4gICAgICAgIC8vIERvdWJsZS1TdWJtaXQgQ29va2llIFBhdHRlcm46IFJlYWQgWFNSRi1UT0tFTiBjb29raWUgYW5kIG1pcnJvciBpbiBoZWFkZXIuXHJcbiAgICAgICAgbGV0IHJlcXVlc3QgPSByZXE7XHJcbiAgICAgICAgaWYgKHJlcS5tZXRob2QgIT09ICdHRVQnICYmIHJlcS5tZXRob2QgIT09ICdIRUFEJyAmJiByZXEubWV0aG9kICE9PSAnT1BUSU9OUycpIHtcclxuICAgICAgICAgICAgY29uc3QgdG9rZW4gPSBnZXRDb29raWUoJ1hTUkYtVE9LRU4nKSB8fCAnJztcclxuICAgICAgICAgICAgcmVxdWVzdCA9IHJlcS5jbG9uZSh7XHJcbiAgICAgICAgICAgICAgICBzZXRIZWFkZXJzOiB7XHJcbiAgICAgICAgICAgICAgICAgICAgJ1gtUHJpbXVzLUNTUkYnOiB0b2tlblxyXG4gICAgICAgICAgICAgICAgfSxcclxuICAgICAgICAgICAgICAgIHdpdGhDcmVkZW50aWFsczogdHJ1ZSAvLyBFbnN1cmUgY29va2llcyBhcmUgc2VudFxyXG4gICAgICAgICAgICB9KTtcclxuICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICAvLyBEZXRlcm1pbmUgaWYgd2UgbmVlZCBjcmVkZW50aWFscyBmb3IgR0VUcz9cclxuICAgICAgICAgICAgLy8gVXN1YWxseSB5ZXMgZm9yIEFQSSBjYWxscyB0byBvdXIgYmFja2VuZC5cclxuICAgICAgICAgICAgLy8gV2UgYXNzdW1lIGNhbGxzIHRvIHNhbWUgb3JpZ2luIG9yIEFQSSBVUkwgbmVlZCBjcmVkcy5cclxuICAgICAgICAgICAgcmVxdWVzdCA9IHJlcS5jbG9uZSh7XHJcbiAgICAgICAgICAgICAgICB3aXRoQ3JlZGVudGlhbHM6IHRydWVcclxuICAgICAgICAgICAgfSk7XHJcbiAgICAgICAgfVxyXG4gICAgXHJcbiAgICAgICAgcmV0dXJuIG5leHQocmVxdWVzdCkucGlwZShcclxuICAgICAgICAgICAgY2F0Y2hFcnJvcigoZXJyb3I6IEh0dHBFcnJvclJlc3BvbnNlKSA9PiB7XHJcbiAgICAgICAgICAgICAgICAvLyAyLiBIYW5kbGUgNDAxIFVuYXV0aG9yaXplZFxyXG4gICAgICAgICAgICAgICAgaWYgKGVycm9yLnN0YXR1cyA9PT0gNDAxKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgYXV0aFNlcnZpY2UuaGFuZGxlVW5hdXRob3JpemVkKCk7XHJcbiAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICByZXR1cm4gdGhyb3dFcnJvcigoKSA9PiBlcnJvcik7XHJcbiAgICAgICAgICAgIH0pXHJcbiAgICAgICAgKTtcclxuICAgIH07XHJcbiAgICBcclxuICAgIGZ1bmN0aW9uIGdldENvb2tpZShuYW1lOiBzdHJpbmcpOiBzdHJpbmcgfCBudWxsIHtcclxuICAgICAgICB0cnkge1xyXG4gICAgICAgICAgICBpZiAodHlwZW9mIGRvY3VtZW50ID09PSAndW5kZWZpbmVkJykgcmV0dXJuIG51bGw7XHJcbiAgICAgICAgICAgIGNvbnN0IG1hdGNoZXMgPSBkb2N1bWVudC5jb29raWUubWF0Y2gobmV3IFJlZ0V4cCgnKD86Xnw7ICknICsgbmFtZS5yZXBsYWNlKC8oW1xcLiQ/Knx7fVxcKFxcKVxcW1xcXVxcXFxcXC9cXCteXSkvZywgJ1xcXFwkMScpICsgJz0oW147XSopJykpO1xyXG4gICAgICAgICAgICByZXR1cm4gbWF0Y2hlcyA/IGRlY29kZVVSSUNvbXBvbmVudChtYXRjaGVzWzFdKSA6IG51bGw7XHJcbiAgICAgICAgfSBjYXRjaCB7XHJcbiAgICAgICAgICAgIHJldHVybiBudWxsO1xyXG4gICAgICAgIH1cclxuICAgIH1cclxuIl19
@@ -0,0 +1,29 @@
1
+ import { makeEnvironmentProviders, APP_INITIALIZER, PLATFORM_ID } from '@angular/core';
2
+ import { provideHttpClient, withInterceptors } from '@angular/common/http';
3
+ import { PRIMUS_AUTH_CONFIG } from './primus-auth.tokens';
4
+ import { PrimusAuthService } from './primus-auth.service';
5
+ import { primusAuthInterceptor } from './primus-auth.interceptor';
6
+ export function providePrimusAuth(config) {
7
+ return makeEnvironmentProviders([
8
+ {
9
+ provide: PRIMUS_AUTH_CONFIG,
10
+ useValue: config
11
+ },
12
+ PrimusAuthService,
13
+ provideHttpClient(withInterceptors([primusAuthInterceptor])),
14
+ {
15
+ provide: APP_INITIALIZER,
16
+ useFactory: (auth, platform) => () => {
17
+ // We trigger checkSession in constructor, but we can also do it here if we want to await it.
18
+ // However, awaiting APP_INITIALIZER blocks the whole app render.
19
+ // Better to let it run in background and use the Guard to block specific routes.
20
+ // So we just ensure service is instantiated.
21
+ // Actually, checkSession is already called in constructor.
22
+ // This APP_INITIALIZER forces the service to be created immediately on app start.
23
+ },
24
+ deps: [PrimusAuthService, PLATFORM_ID],
25
+ multi: true
26
+ }
27
+ ]);
28
+ }
29
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJpbXVzLWF1dGgucHJvdmlkZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9hbmd1bGFyL3ByaW11cy1hdXRoL3NyYy9saWIvcHJpbXVzLWF1dGgucHJvdmlkZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUF3Qix3QkFBd0IsRUFBWSxlQUFlLEVBQUUsV0FBVyxFQUFVLE1BQU0sZUFBZSxDQUFDO0FBQy9ILE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLHNCQUFzQixDQUFDO0FBRTNFLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLHNCQUFzQixDQUFDO0FBQzFELE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBQzFELE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBR2xFLE1BQU0sVUFBVSxpQkFBaUIsQ0FBQyxNQUF3QjtJQUN0RCxPQUFPLHdCQUF3QixDQUFDO1FBQzVCO1lBQ0ksT0FBTyxFQUFFLGtCQUFrQjtZQUMzQixRQUFRLEVBQUUsTUFBTTtTQUNuQjtRQUNELGlCQUFpQjtRQUNqQixpQkFBaUIsQ0FDYixnQkFBZ0IsQ0FBQyxDQUFDLHFCQUFxQixDQUFDLENBQUMsQ0FDNUM7UUFDRDtZQUNJLE9BQU8sRUFBRSxlQUFlO1lBQ3hCLFVBQVUsRUFBRSxDQUFDLElBQXVCLEVBQUUsUUFBZ0IsRUFBRSxFQUFFLENBQUMsR0FBRyxFQUFFO2dCQUM1RCw2RkFBNkY7Z0JBQzdGLGlFQUFpRTtnQkFDakUsaUZBQWlGO2dCQUNqRiw2Q0FBNkM7Z0JBRTdDLDJEQUEyRDtnQkFDM0Qsa0ZBQWtGO1lBQ3RGLENBQUM7WUFDRCxJQUFJLEVBQUUsQ0FBQyxpQkFBaUIsRUFBRSxXQUFXLENBQUM7WUFDdEMsS0FBSyxFQUFFLElBQUk7U0FDZDtLQUNKLENBQUMsQ0FBQztBQUNQLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBFbnZpcm9ubWVudFByb3ZpZGVycywgbWFrZUVudmlyb25tZW50UHJvdmlkZXJzLCBQcm92aWRlciwgQVBQX0lOSVRJQUxJWkVSLCBQTEFURk9STV9JRCwgaW5qZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcbmltcG9ydCB7IHByb3ZpZGVIdHRwQ2xpZW50LCB3aXRoSW50ZXJjZXB0b3JzIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uL2h0dHAnO1xyXG5pbXBvcnQgeyBQcmltdXNBdXRoQ29uZmlnIH0gZnJvbSAnLi9wcmltdXMtYXV0aC50eXBlcyc7XHJcbmltcG9ydCB7IFBSSU1VU19BVVRIX0NPTkZJRyB9IGZyb20gJy4vcHJpbXVzLWF1dGgudG9rZW5zJztcclxuaW1wb3J0IHsgUHJpbXVzQXV0aFNlcnZpY2UgfSBmcm9tICcuL3ByaW11cy1hdXRoLnNlcnZpY2UnO1xyXG5pbXBvcnQgeyBwcmltdXNBdXRoSW50ZXJjZXB0b3IgfSBmcm9tICcuL3ByaW11cy1hdXRoLmludGVyY2VwdG9yJztcclxuaW1wb3J0IHsgaXNQbGF0Zm9ybUJyb3dzZXIgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xyXG5cclxuZXhwb3J0IGZ1bmN0aW9uIHByb3ZpZGVQcmltdXNBdXRoKGNvbmZpZzogUHJpbXVzQXV0aENvbmZpZyk6IEVudmlyb25tZW50UHJvdmlkZXJzIHtcclxuICAgIHJldHVybiBtYWtlRW52aXJvbm1lbnRQcm92aWRlcnMoW1xyXG4gICAgICAgIHtcclxuICAgICAgICAgICAgcHJvdmlkZTogUFJJTVVTX0FVVEhfQ09ORklHLFxyXG4gICAgICAgICAgICB1c2VWYWx1ZTogY29uZmlnXHJcbiAgICAgICAgfSxcclxuICAgICAgICBQcmltdXNBdXRoU2VydmljZSxcclxuICAgICAgICBwcm92aWRlSHR0cENsaWVudChcclxuICAgICAgICAgICAgd2l0aEludGVyY2VwdG9ycyhbcHJpbXVzQXV0aEludGVyY2VwdG9yXSlcclxuICAgICAgICApLFxyXG4gICAgICAgIHtcclxuICAgICAgICAgICAgcHJvdmlkZTogQVBQX0lOSVRJQUxJWkVSLFxyXG4gICAgICAgICAgICB1c2VGYWN0b3J5OiAoYXV0aDogUHJpbXVzQXV0aFNlcnZpY2UsIHBsYXRmb3JtOiBPYmplY3QpID0+ICgpID0+IHtcclxuICAgICAgICAgICAgICAgIC8vIFdlIHRyaWdnZXIgY2hlY2tTZXNzaW9uIGluIGNvbnN0cnVjdG9yLCBidXQgd2UgY2FuIGFsc28gZG8gaXQgaGVyZSBpZiB3ZSB3YW50IHRvIGF3YWl0IGl0LlxyXG4gICAgICAgICAgICAgICAgLy8gSG93ZXZlciwgYXdhaXRpbmcgQVBQX0lOSVRJQUxJWkVSIGJsb2NrcyB0aGUgd2hvbGUgYXBwIHJlbmRlci5cclxuICAgICAgICAgICAgICAgIC8vIEJldHRlciB0byBsZXQgaXQgcnVuIGluIGJhY2tncm91bmQgYW5kIHVzZSB0aGUgR3VhcmQgdG8gYmxvY2sgc3BlY2lmaWMgcm91dGVzLlxyXG4gICAgICAgICAgICAgICAgLy8gU28gd2UganVzdCBlbnN1cmUgc2VydmljZSBpcyBpbnN0YW50aWF0ZWQuXHJcblxyXG4gICAgICAgICAgICAgICAgLy8gQWN0dWFsbHksIGNoZWNrU2Vzc2lvbiBpcyBhbHJlYWR5IGNhbGxlZCBpbiBjb25zdHJ1Y3Rvci5cclxuICAgICAgICAgICAgICAgIC8vIFRoaXMgQVBQX0lOSVRJQUxJWkVSIGZvcmNlcyB0aGUgc2VydmljZSB0byBiZSBjcmVhdGVkIGltbWVkaWF0ZWx5IG9uIGFwcCBzdGFydC5cclxuICAgICAgICAgICAgfSxcclxuICAgICAgICAgICAgZGVwczogW1ByaW11c0F1dGhTZXJ2aWNlLCBQTEFURk9STV9JRF0sXHJcbiAgICAgICAgICAgIG11bHRpOiB0cnVlXHJcbiAgICAgICAgfVxyXG4gICAgXSk7XHJcbn1cclxuIl19
@@ -0,0 +1,161 @@
1
+ import { Injectable, Inject, signal, computed, PLATFORM_ID } from '@angular/core';
2
+ import { isPlatformBrowser } from '@angular/common';
3
+ import { catchError, map, tap } from 'rxjs/operators';
4
+ import { of, BehaviorSubject, interval } from 'rxjs';
5
+ import { PRIMUS_AUTH_CONFIG } from './primus-auth.tokens';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "@angular/common/http";
8
+ import * as i2 from "@angular/router";
9
+ import * as i3 from "@angular/common";
10
+ export class PrimusAuthService {
11
+ http;
12
+ router;
13
+ location;
14
+ config;
15
+ platformId;
16
+ // Reactive State (Signals)
17
+ _user = signal(null);
18
+ _isLoading = signal(true);
19
+ user = this._user.asReadonly();
20
+ isLoading = this._isLoading.asReadonly();
21
+ isAuthenticated = computed(() => !!this._user());
22
+ // Legacy Observable support
23
+ user$ = new BehaviorSubject(null);
24
+ isAuthenticated$ = this.user$.pipe(map(u => !!u));
25
+ initialized$ = new BehaviorSubject(false);
26
+ heartbeatSub = null;
27
+ HEARTBEAT_INTERVAL = 5 * 60 * 1000; // 5 minutes
28
+ constructor(http, router, location, config, platformId) {
29
+ this.http = http;
30
+ this.router = router;
31
+ this.location = location;
32
+ this.config = config;
33
+ this.platformId = platformId;
34
+ this.checkSession();
35
+ }
36
+ ngOnDestroy() {
37
+ this.stopHeartbeat();
38
+ }
39
+ /**
40
+ * Checks /api/auth/me to see if user is logged in.
41
+ * Should be called on app startup.
42
+ */
43
+ checkSession() {
44
+ if (!isPlatformBrowser(this.platformId)) {
45
+ this._isLoading.set(false);
46
+ this.initialized$.next(true);
47
+ return;
48
+ }
49
+ this._isLoading.set(true);
50
+ this.fetchUser().subscribe(() => {
51
+ this._isLoading.set(false);
52
+ this.initialized$.next(true);
53
+ });
54
+ }
55
+ fetchUser() {
56
+ const endpoint = `${this.config.apiUrl}/api/auth/me`;
57
+ return this.http.get(endpoint, { withCredentials: true }).pipe(tap(user => this.updateState(user)), catchError(() => {
58
+ this.updateState(null);
59
+ return of(null);
60
+ }));
61
+ }
62
+ /**
63
+ * Performs a local login (username/password) via the Backend Broker.
64
+ * The backend must accept a POST to /api/auth/login and set a cookie.
65
+ */
66
+ login(credentials) {
67
+ const endpoint = `${this.config.apiUrl}/api/auth/login`;
68
+ // We expect the backend to return the User object on success (or we fetch it)
69
+ // If the backend returns just 200 OK, we might need to chain a fetchUser()
70
+ return this.http.post(endpoint, credentials, { withCredentials: true }).pipe(tap(user => {
71
+ // If the backend returns the user directly, great.
72
+ // If it returns a generic success, we might get 'null' here if typing doesn't match.
73
+ // For safety, if we get a success response, we can also force-fetch the user to be sure.
74
+ if (user && user.email) {
75
+ this.updateState(user);
76
+ }
77
+ else {
78
+ // Fallback: Fetch user info if response was just { success: true }
79
+ this.fetchUser().subscribe();
80
+ }
81
+ }));
82
+ }
83
+ /**
84
+ * Redirects the browser to the backend login endpoint for the provider.
85
+ * e.g. /api/auth/azure, /api/auth/auth0
86
+ */
87
+ loginWithProvider(provider, redirectUrl) {
88
+ const returnUrl = redirectUrl || this.router.url;
89
+ // We can pass returnUrl if backend supports it, usually configured globally
90
+ // or via a query param handled by the Resolution logic.
91
+ // For now we assume standard flow or basic redirect.
92
+ const target = `${this.config.apiUrl}/api/auth/${provider}`;
93
+ window.location.href = target;
94
+ }
95
+ logout() {
96
+ const endpoint = `${this.config.apiUrl}/api/auth/logout`;
97
+ this.http.post(endpoint, {}, {
98
+ withCredentials: true
99
+ // Interceptor will add CSRF header automatically
100
+ }).subscribe({
101
+ next: () => {
102
+ this.updateState(null);
103
+ const redirect = this.config.logoutRedirect || '/login';
104
+ this.router.navigateByUrl(redirect);
105
+ },
106
+ error: () => {
107
+ // Force logout state even if API failed
108
+ this.updateState(null);
109
+ const redirect = this.config.logoutRedirect || '/login';
110
+ this.router.navigateByUrl(redirect);
111
+ }
112
+ });
113
+ }
114
+ updateState(user) {
115
+ this._user.set(user);
116
+ this.user$.next(user);
117
+ // Auto-sync signal to loading state
118
+ if (user) {
119
+ this._isLoading.set(false);
120
+ this.startHeartbeat();
121
+ }
122
+ else {
123
+ this.stopHeartbeat();
124
+ }
125
+ }
126
+ // Called by Interceptor on 401
127
+ handleUnauthorized() {
128
+ this.updateState(null);
129
+ }
130
+ startHeartbeat() {
131
+ if (!isPlatformBrowser(this.platformId))
132
+ return;
133
+ if (this.heartbeatSub)
134
+ return; // Already running
135
+ this.heartbeatSub = interval(this.HEARTBEAT_INTERVAL).subscribe(() => {
136
+ // Quietly check session
137
+ this.fetchUser().subscribe();
138
+ });
139
+ }
140
+ stopHeartbeat() {
141
+ if (this.heartbeatSub) {
142
+ this.heartbeatSub.unsubscribe();
143
+ this.heartbeatSub = null;
144
+ }
145
+ }
146
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PrimusAuthService, deps: [{ token: i1.HttpClient }, { token: i2.Router }, { token: i3.Location }, { token: PRIMUS_AUTH_CONFIG }, { token: PLATFORM_ID }], target: i0.ɵɵFactoryTarget.Injectable });
147
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PrimusAuthService, providedIn: 'root' });
148
+ }
149
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PrimusAuthService, decorators: [{
150
+ type: Injectable,
151
+ args: [{
152
+ providedIn: 'root'
153
+ }]
154
+ }], ctorParameters: () => [{ type: i1.HttpClient }, { type: i2.Router }, { type: i3.Location }, { type: undefined, decorators: [{
155
+ type: Inject,
156
+ args: [PRIMUS_AUTH_CONFIG]
157
+ }] }, { type: Object, decorators: [{
158
+ type: Inject,
159
+ args: [PLATFORM_ID]
160
+ }] }] });
161
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"primus-auth.service.js","sourceRoot":"","sources":["../../../../angular/primus-auth/src/lib/primus-auth.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Inject, signal, computed, PLATFORM_ID, OnDestroy } from '@angular/core';\r\nimport { HttpClient } from '@angular/common/http';\r\nimport { Router } from '@angular/router';\r\nimport { isPlatformBrowser, Location } from '@angular/common';\r\nimport { catchError, map, tap } from 'rxjs/operators';\r\nimport { Observable, of, BehaviorSubject, Subscription, interval } from 'rxjs';\r\nimport { PrimusAuthConfig, PrimusUser } from './primus-auth.types';\r\nimport { PRIMUS_AUTH_CONFIG } from './primus-auth.tokens';\r\n\r\n@Injectable({\r\n    providedIn: 'root'\r\n})\r\nexport class PrimusAuthService implements OnDestroy {\r\n    // Reactive State (Signals)\r\n    private _user = signal<PrimusUser | null>(null);\r\n    private _isLoading = signal<boolean>(true);\r\n\r\n    public readonly user = this._user.asReadonly();\r\n    public readonly isLoading = this._isLoading.asReadonly();\r\n    public readonly isAuthenticated = computed(() => !!this._user());\r\n\r\n    // Legacy Observable support\r\n    public readonly user$ = new BehaviorSubject<PrimusUser | null>(null);\r\n    public readonly isAuthenticated$ = this.user$.pipe(map(u => !!u));\r\n    public readonly initialized$ = new BehaviorSubject<boolean>(false);\r\n\r\n    private heartbeatSub: Subscription | null = null;\r\n    private readonly HEARTBEAT_INTERVAL = 5 * 60 * 1000; // 5 minutes\r\n\r\n    constructor(\r\n        private http: HttpClient,\r\n        private router: Router,\r\n        private location: Location,\r\n        @Inject(PRIMUS_AUTH_CONFIG) private config: PrimusAuthConfig,\r\n        @Inject(PLATFORM_ID) private platformId: Object\r\n    ) {\r\n        this.checkSession();\r\n    }\r\n\r\n    public ngOnDestroy(): void {\r\n        this.stopHeartbeat();\r\n    }\r\n\r\n    /**\r\n     * Checks /api/auth/me to see if user is logged in.\r\n     * Should be called on app startup.\r\n     */\r\n    public checkSession(): void {\r\n        if (!isPlatformBrowser(this.platformId)) {\r\n            this._isLoading.set(false);\r\n            this.initialized$.next(true);\r\n            return;\r\n        }\r\n\r\n        this._isLoading.set(true);\r\n        this.fetchUser().subscribe(() => {\r\n            this._isLoading.set(false);\r\n            this.initialized$.next(true);\r\n        });\r\n    }\r\n\r\n    private fetchUser(): Observable<PrimusUser | null> {\r\n        const endpoint = `${this.config.apiUrl}/api/auth/me`;\r\n        return this.http.get<PrimusUser>(endpoint, { withCredentials: true }).pipe(\r\n            tap(user => this.updateState(user)),\r\n            catchError(() => {\r\n                this.updateState(null);\r\n                return of(null);\r\n            })\r\n        );\r\n    }\r\n\r\n    /**\r\n     * Performs a local login (username/password) via the Backend Broker.\r\n     * The backend must accept a POST to /api/auth/login and set a cookie.\r\n     */\r\n    public login(credentials: Record<string, any>): Observable<PrimusUser | null> {\r\n        const endpoint = `${this.config.apiUrl}/api/auth/login`;\r\n        // We expect the backend to return the User object on success (or we fetch it)\r\n        // If the backend returns just 200 OK, we might need to chain a fetchUser()\r\n        return this.http.post<PrimusUser>(endpoint, credentials, { withCredentials: true }).pipe(\r\n            tap(user => {\r\n                // If the backend returns the user directly, great.\r\n                // If it returns a generic success, we might get 'null' here if typing doesn't match.\r\n                // For safety, if we get a success response, we can also force-fetch the user to be sure.\r\n                if (user && user.email) {\r\n                    this.updateState(user);\r\n                } else {\r\n                    // Fallback: Fetch user info if response was just { success: true }\r\n                    this.fetchUser().subscribe();\r\n                }\r\n            })\r\n        );\r\n    }\r\n\r\n    /**\r\n     * Redirects the browser to the backend login endpoint for the provider.\r\n     * e.g. /api/auth/azure, /api/auth/auth0\r\n     */\r\n    public loginWithProvider(provider: string, redirectUrl?: string): void {\r\n        const returnUrl = redirectUrl || this.router.url;\r\n        // We can pass returnUrl if backend supports it, usually configured globally\r\n        // or via a query param handled by the Resolution logic.\r\n        // For now we assume standard flow or basic redirect.\r\n\r\n        const target = `${this.config.apiUrl}/api/auth/${provider}`;\r\n        window.location.href = target;\r\n    }\r\n\r\n    public logout(): void {\r\n        const endpoint = `${this.config.apiUrl}/api/auth/logout`;\r\n        this.http.post(endpoint, {}, {\r\n            withCredentials: true\r\n            // Interceptor will add CSRF header automatically\r\n        }).subscribe({\r\n            next: () => {\r\n                this.updateState(null);\r\n                const redirect = this.config.logoutRedirect || '/login';\r\n                this.router.navigateByUrl(redirect);\r\n            },\r\n            error: () => {\r\n                // Force logout state even if API failed\r\n                this.updateState(null);\r\n                const redirect = this.config.logoutRedirect || '/login';\r\n                this.router.navigateByUrl(redirect);\r\n            }\r\n        });\r\n    }\r\n\r\n    private updateState(user: PrimusUser | null) {\r\n        this._user.set(user);\r\n        this.user$.next(user);\r\n\r\n        // Auto-sync signal to loading state\r\n        if (user) {\r\n            this._isLoading.set(false);\r\n            this.startHeartbeat();\r\n        } else {\r\n            this.stopHeartbeat();\r\n        }\r\n    }\r\n\r\n    // Called by Interceptor on 401\r\n    public handleUnauthorized(): void {\r\n        this.updateState(null);\r\n    }\r\n\r\n    private startHeartbeat(): void {\r\n        if (!isPlatformBrowser(this.platformId)) return;\r\n        if (this.heartbeatSub) return; // Already running\r\n\r\n        this.heartbeatSub = interval(this.HEARTBEAT_INTERVAL).subscribe(() => {\r\n            // Quietly check session\r\n            this.fetchUser().subscribe();\r\n        });\r\n    }\r\n\r\n    private stopHeartbeat(): void {\r\n        if (this.heartbeatSub) {\r\n            this.heartbeatSub.unsubscribe();\r\n            this.heartbeatSub = null;\r\n        }\r\n    }\r\n}\r\n"]}
@@ -0,0 +1,3 @@
1
+ import { InjectionToken } from '@angular/core';
2
+ export const PRIMUS_AUTH_CONFIG = new InjectionToken('PRIMUS_AUTH_CONFIG');
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJpbXVzLWF1dGgudG9rZW5zLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vYW5ndWxhci9wcmltdXMtYXV0aC9zcmMvbGliL3ByaW11cy1hdXRoLnRva2Vucy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRy9DLE1BQU0sQ0FBQyxNQUFNLGtCQUFrQixHQUFHLElBQUksY0FBYyxDQUFtQixvQkFBb0IsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0aW9uVG9rZW4gfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgUHJpbXVzQXV0aENvbmZpZyB9IGZyb20gJy4vcHJpbXVzLWF1dGgudHlwZXMnO1xyXG5cclxuZXhwb3J0IGNvbnN0IFBSSU1VU19BVVRIX0NPTkZJRyA9IG5ldyBJbmplY3Rpb25Ub2tlbjxQcmltdXNBdXRoQ29uZmlnPignUFJJTVVTX0FVVEhfQ09ORklHJyk7XHJcbiJdfQ==
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJpbXVzLWF1dGgudHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9hbmd1bGFyL3ByaW11cy1hdXRoL3NyYy9saWIvcHJpbXVzLWF1dGgudHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBpbnRlcmZhY2UgUHJpbXVzQXV0aENvbmZpZyB7XHJcbiAgICBhcGlVcmw6IHN0cmluZztcclxuICAgIGxvZ2luUGF0aD86IHN0cmluZzsgLy8gRGVmYXVsdDogL2xvZ2luXHJcbiAgICBsb2dvdXRSZWRpcmVjdD86IHN0cmluZzsgLy8gRGVmYXVsdDogL2xvZ2luXHJcbn1cclxuXHJcbmV4cG9ydCBpbnRlcmZhY2UgUHJpbXVzVXNlciB7XHJcbiAgICBpZDogc3RyaW5nO1xyXG4gICAgZW1haWw6IHN0cmluZztcclxuICAgIG5hbWU/OiBzdHJpbmc7XHJcbiAgICByb2xlPzogc3RyaW5nO1xyXG4gICAgcHJvdmlkZXI/OiBzdHJpbmc7XHJcbiAgICBwaWN0dXJlPzogc3RyaW5nO1xyXG4gICAgW2tleTogc3RyaW5nXTogYW55O1xyXG59XHJcblxyXG5leHBvcnQgaW50ZXJmYWNlIEF1dGhTdGF0ZSB7XHJcbiAgICBpc0F1dGhlbnRpY2F0ZWQ6IGJvb2xlYW47XHJcbiAgICB1c2VyOiBQcmltdXNVc2VyIHwgbnVsbDtcclxuICAgIGlzTG9hZGluZzogYm9vbGVhbjtcclxufVxyXG4iXX0=
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ export * from './public-api';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJpbXVzLWFuZ3VsYXItYXV0aC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL2FuZ3VsYXIvcHJpbXVzLWF1dGgvc3JjL3ByaW11cy1hbmd1bGFyLWF1dGgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLGNBQWMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9wdWJsaWMtYXBpJztcbiJdfQ==
@@ -0,0 +1,9 @@
1
+ /*
2
+ * Public API Surface of primus-angular-auth
3
+ */
4
+ export * from './lib/primus-auth.service';
5
+ export * from './lib/primus-auth.interceptor';
6
+ export * from './lib/primus-auth.guard';
7
+ export * from './lib/primus-auth.provider';
8
+ export * from './lib/primus-auth.types';
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL2FuZ3VsYXIvcHJpbXVzLWF1dGgvc3JjL3B1YmxpYy1hcGkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLDJCQUEyQixDQUFDO0FBQzFDLGNBQWMsK0JBQStCLENBQUM7QUFDOUMsY0FBYyx5QkFBeUIsQ0FBQztBQUN4QyxjQUFjLDRCQUE0QixDQUFDO0FBQzNDLGNBQWMseUJBQXlCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxyXG4gKiBQdWJsaWMgQVBJIFN1cmZhY2Ugb2YgcHJpbXVzLWFuZ3VsYXItYXV0aFxyXG4gKi9cclxuXHJcbmV4cG9ydCAqIGZyb20gJy4vbGliL3ByaW11cy1hdXRoLnNlcnZpY2UnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9wcmltdXMtYXV0aC5pbnRlcmNlcHRvcic7XHJcbmV4cG9ydCAqIGZyb20gJy4vbGliL3ByaW11cy1hdXRoLmd1YXJkJztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvcHJpbXVzLWF1dGgucHJvdmlkZXInO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9wcmltdXMtYXV0aC50eXBlcyc7XHJcbiJdfQ==
@@ -0,0 +1,259 @@
1
+ import * as i0 from '@angular/core';
2
+ import { InjectionToken, signal, computed, PLATFORM_ID, Inject, Injectable, inject, makeEnvironmentProviders, APP_INITIALIZER } from '@angular/core';
3
+ import * as i3 from '@angular/common';
4
+ import { isPlatformBrowser } from '@angular/common';
5
+ import { map, tap, catchError, filter, take, switchMap } from 'rxjs/operators';
6
+ import { BehaviorSubject, of, interval, catchError as catchError$1, throwError } from 'rxjs';
7
+ import * as i1 from '@angular/common/http';
8
+ import { provideHttpClient, withInterceptors } from '@angular/common/http';
9
+ import * as i2 from '@angular/router';
10
+ import { Router } from '@angular/router';
11
+
12
+ const PRIMUS_AUTH_CONFIG = new InjectionToken('PRIMUS_AUTH_CONFIG');
13
+
14
+ class PrimusAuthService {
15
+ http;
16
+ router;
17
+ location;
18
+ config;
19
+ platformId;
20
+ // Reactive State (Signals)
21
+ _user = signal(null);
22
+ _isLoading = signal(true);
23
+ user = this._user.asReadonly();
24
+ isLoading = this._isLoading.asReadonly();
25
+ isAuthenticated = computed(() => !!this._user());
26
+ // Legacy Observable support
27
+ user$ = new BehaviorSubject(null);
28
+ isAuthenticated$ = this.user$.pipe(map(u => !!u));
29
+ initialized$ = new BehaviorSubject(false);
30
+ heartbeatSub = null;
31
+ HEARTBEAT_INTERVAL = 5 * 60 * 1000; // 5 minutes
32
+ constructor(http, router, location, config, platformId) {
33
+ this.http = http;
34
+ this.router = router;
35
+ this.location = location;
36
+ this.config = config;
37
+ this.platformId = platformId;
38
+ this.checkSession();
39
+ }
40
+ ngOnDestroy() {
41
+ this.stopHeartbeat();
42
+ }
43
+ /**
44
+ * Checks /api/auth/me to see if user is logged in.
45
+ * Should be called on app startup.
46
+ */
47
+ checkSession() {
48
+ if (!isPlatformBrowser(this.platformId)) {
49
+ this._isLoading.set(false);
50
+ this.initialized$.next(true);
51
+ return;
52
+ }
53
+ this._isLoading.set(true);
54
+ this.fetchUser().subscribe(() => {
55
+ this._isLoading.set(false);
56
+ this.initialized$.next(true);
57
+ });
58
+ }
59
+ fetchUser() {
60
+ const endpoint = `${this.config.apiUrl}/api/auth/me`;
61
+ return this.http.get(endpoint, { withCredentials: true }).pipe(tap(user => this.updateState(user)), catchError(() => {
62
+ this.updateState(null);
63
+ return of(null);
64
+ }));
65
+ }
66
+ /**
67
+ * Performs a local login (username/password) via the Backend Broker.
68
+ * The backend must accept a POST to /api/auth/login and set a cookie.
69
+ */
70
+ login(credentials) {
71
+ const endpoint = `${this.config.apiUrl}/api/auth/login`;
72
+ // We expect the backend to return the User object on success (or we fetch it)
73
+ // If the backend returns just 200 OK, we might need to chain a fetchUser()
74
+ return this.http.post(endpoint, credentials, { withCredentials: true }).pipe(tap(user => {
75
+ // If the backend returns the user directly, great.
76
+ // If it returns a generic success, we might get 'null' here if typing doesn't match.
77
+ // For safety, if we get a success response, we can also force-fetch the user to be sure.
78
+ if (user && user.email) {
79
+ this.updateState(user);
80
+ }
81
+ else {
82
+ // Fallback: Fetch user info if response was just { success: true }
83
+ this.fetchUser().subscribe();
84
+ }
85
+ }));
86
+ }
87
+ /**
88
+ * Redirects the browser to the backend login endpoint for the provider.
89
+ * e.g. /api/auth/azure, /api/auth/auth0
90
+ */
91
+ loginWithProvider(provider, redirectUrl) {
92
+ const returnUrl = redirectUrl || this.router.url;
93
+ // We can pass returnUrl if backend supports it, usually configured globally
94
+ // or via a query param handled by the Resolution logic.
95
+ // For now we assume standard flow or basic redirect.
96
+ const target = `${this.config.apiUrl}/api/auth/${provider}`;
97
+ window.location.href = target;
98
+ }
99
+ logout() {
100
+ const endpoint = `${this.config.apiUrl}/api/auth/logout`;
101
+ this.http.post(endpoint, {}, {
102
+ withCredentials: true
103
+ // Interceptor will add CSRF header automatically
104
+ }).subscribe({
105
+ next: () => {
106
+ this.updateState(null);
107
+ const redirect = this.config.logoutRedirect || '/login';
108
+ this.router.navigateByUrl(redirect);
109
+ },
110
+ error: () => {
111
+ // Force logout state even if API failed
112
+ this.updateState(null);
113
+ const redirect = this.config.logoutRedirect || '/login';
114
+ this.router.navigateByUrl(redirect);
115
+ }
116
+ });
117
+ }
118
+ updateState(user) {
119
+ this._user.set(user);
120
+ this.user$.next(user);
121
+ // Auto-sync signal to loading state
122
+ if (user) {
123
+ this._isLoading.set(false);
124
+ this.startHeartbeat();
125
+ }
126
+ else {
127
+ this.stopHeartbeat();
128
+ }
129
+ }
130
+ // Called by Interceptor on 401
131
+ handleUnauthorized() {
132
+ this.updateState(null);
133
+ }
134
+ startHeartbeat() {
135
+ if (!isPlatformBrowser(this.platformId))
136
+ return;
137
+ if (this.heartbeatSub)
138
+ return; // Already running
139
+ this.heartbeatSub = interval(this.HEARTBEAT_INTERVAL).subscribe(() => {
140
+ // Quietly check session
141
+ this.fetchUser().subscribe();
142
+ });
143
+ }
144
+ stopHeartbeat() {
145
+ if (this.heartbeatSub) {
146
+ this.heartbeatSub.unsubscribe();
147
+ this.heartbeatSub = null;
148
+ }
149
+ }
150
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PrimusAuthService, deps: [{ token: i1.HttpClient }, { token: i2.Router }, { token: i3.Location }, { token: PRIMUS_AUTH_CONFIG }, { token: PLATFORM_ID }], target: i0.ɵɵFactoryTarget.Injectable });
151
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PrimusAuthService, providedIn: 'root' });
152
+ }
153
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PrimusAuthService, decorators: [{
154
+ type: Injectable,
155
+ args: [{
156
+ providedIn: 'root'
157
+ }]
158
+ }], ctorParameters: () => [{ type: i1.HttpClient }, { type: i2.Router }, { type: i3.Location }, { type: undefined, decorators: [{
159
+ type: Inject,
160
+ args: [PRIMUS_AUTH_CONFIG]
161
+ }] }, { type: Object, decorators: [{
162
+ type: Inject,
163
+ args: [PLATFORM_ID]
164
+ }] }] });
165
+
166
+ const primusAuthInterceptor = (req, next) => {
167
+ const authService = inject(PrimusAuthService);
168
+ // 1. Add CSRF Header to non-GET requests
169
+ // Double-Submit Cookie Pattern: Read XSRF-TOKEN cookie and mirror in header.
170
+ let request = req;
171
+ if (req.method !== 'GET' && req.method !== 'HEAD' && req.method !== 'OPTIONS') {
172
+ const token = getCookie('XSRF-TOKEN') || '';
173
+ request = req.clone({
174
+ setHeaders: {
175
+ 'X-Primus-CSRF': token
176
+ },
177
+ withCredentials: true // Ensure cookies are sent
178
+ });
179
+ }
180
+ else {
181
+ // Determine if we need credentials for GETs?
182
+ // Usually yes for API calls to our backend.
183
+ // We assume calls to same origin or API URL need creds.
184
+ request = req.clone({
185
+ withCredentials: true
186
+ });
187
+ }
188
+ return next(request).pipe(catchError$1((error) => {
189
+ // 2. Handle 401 Unauthorized
190
+ if (error.status === 401) {
191
+ authService.handleUnauthorized();
192
+ }
193
+ return throwError(() => error);
194
+ }));
195
+ };
196
+ function getCookie(name) {
197
+ try {
198
+ if (typeof document === 'undefined')
199
+ return null;
200
+ const matches = document.cookie.match(new RegExp('(?:^|; )' + name.replace(/([\.$?*|{}\(\)\[\]\\\/\+^])/g, '\\$1') + '=([^;]*)'));
201
+ return matches ? decodeURIComponent(matches[1]) : null;
202
+ }
203
+ catch {
204
+ return null;
205
+ }
206
+ }
207
+
208
+ const primusAuthGuard = (route, state) => {
209
+ const auth = inject(PrimusAuthService);
210
+ const router = inject(Router);
211
+ const config = inject(PRIMUS_AUTH_CONFIG);
212
+ // Wait for initialization to complete, then check authentication status
213
+ return auth.initialized$.pipe(filter((isInit) => isInit), // Wait until true
214
+ take(1), switchMap(() => auth.isAuthenticated$), take(1), map((isAuthenticated) => {
215
+ if (isAuthenticated) {
216
+ return true;
217
+ }
218
+ // Redirect to login
219
+ const loginPath = config.loginPath || '/login';
220
+ return router.createUrlTree([loginPath], {
221
+ queryParams: { returnUrl: state.url }
222
+ });
223
+ }));
224
+ };
225
+
226
+ function providePrimusAuth(config) {
227
+ return makeEnvironmentProviders([
228
+ {
229
+ provide: PRIMUS_AUTH_CONFIG,
230
+ useValue: config
231
+ },
232
+ PrimusAuthService,
233
+ provideHttpClient(withInterceptors([primusAuthInterceptor])),
234
+ {
235
+ provide: APP_INITIALIZER,
236
+ useFactory: (auth, platform) => () => {
237
+ // We trigger checkSession in constructor, but we can also do it here if we want to await it.
238
+ // However, awaiting APP_INITIALIZER blocks the whole app render.
239
+ // Better to let it run in background and use the Guard to block specific routes.
240
+ // So we just ensure service is instantiated.
241
+ // Actually, checkSession is already called in constructor.
242
+ // This APP_INITIALIZER forces the service to be created immediately on app start.
243
+ },
244
+ deps: [PrimusAuthService, PLATFORM_ID],
245
+ multi: true
246
+ }
247
+ ]);
248
+ }
249
+
250
+ /*
251
+ * Public API Surface of primus-angular-auth
252
+ */
253
+
254
+ /**
255
+ * Generated bundle index. Do not edit.
256
+ */
257
+
258
+ export { PrimusAuthService, primusAuthGuard, primusAuthInterceptor, providePrimusAuth };
259
+ //# sourceMappingURL=primus-angular-auth.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"primus-angular-auth.mjs","sources":["../../../angular/primus-auth/src/lib/primus-auth.tokens.ts","../../../angular/primus-auth/src/lib/primus-auth.service.ts","../../../angular/primus-auth/src/lib/primus-auth.interceptor.ts","../../../angular/primus-auth/src/lib/primus-auth.guard.ts","../../../angular/primus-auth/src/lib/primus-auth.provider.ts","../../../angular/primus-auth/src/public-api.ts","../../../angular/primus-auth/src/primus-angular-auth.ts"],"sourcesContent":["import { InjectionToken } from '@angular/core';\r\nimport { PrimusAuthConfig } from './primus-auth.types';\r\n\r\nexport const PRIMUS_AUTH_CONFIG = new InjectionToken<PrimusAuthConfig>('PRIMUS_AUTH_CONFIG');\r\n","import { Injectable, Inject, signal, computed, PLATFORM_ID, OnDestroy } from '@angular/core';\r\nimport { HttpClient } from '@angular/common/http';\r\nimport { Router } from '@angular/router';\r\nimport { isPlatformBrowser, Location } from '@angular/common';\r\nimport { catchError, map, tap } from 'rxjs/operators';\r\nimport { Observable, of, BehaviorSubject, Subscription, interval } from 'rxjs';\r\nimport { PrimusAuthConfig, PrimusUser } from './primus-auth.types';\r\nimport { PRIMUS_AUTH_CONFIG } from './primus-auth.tokens';\r\n\r\n@Injectable({\r\n providedIn: 'root'\r\n})\r\nexport class PrimusAuthService implements OnDestroy {\r\n // Reactive State (Signals)\r\n private _user = signal<PrimusUser | null>(null);\r\n private _isLoading = signal<boolean>(true);\r\n\r\n public readonly user = this._user.asReadonly();\r\n public readonly isLoading = this._isLoading.asReadonly();\r\n public readonly isAuthenticated = computed(() => !!this._user());\r\n\r\n // Legacy Observable support\r\n public readonly user$ = new BehaviorSubject<PrimusUser | null>(null);\r\n public readonly isAuthenticated$ = this.user$.pipe(map(u => !!u));\r\n public readonly initialized$ = new BehaviorSubject<boolean>(false);\r\n\r\n private heartbeatSub: Subscription | null = null;\r\n private readonly HEARTBEAT_INTERVAL = 5 * 60 * 1000; // 5 minutes\r\n\r\n constructor(\r\n private http: HttpClient,\r\n private router: Router,\r\n private location: Location,\r\n @Inject(PRIMUS_AUTH_CONFIG) private config: PrimusAuthConfig,\r\n @Inject(PLATFORM_ID) private platformId: Object\r\n ) {\r\n this.checkSession();\r\n }\r\n\r\n public ngOnDestroy(): void {\r\n this.stopHeartbeat();\r\n }\r\n\r\n /**\r\n * Checks /api/auth/me to see if user is logged in.\r\n * Should be called on app startup.\r\n */\r\n public checkSession(): void {\r\n if (!isPlatformBrowser(this.platformId)) {\r\n this._isLoading.set(false);\r\n this.initialized$.next(true);\r\n return;\r\n }\r\n\r\n this._isLoading.set(true);\r\n this.fetchUser().subscribe(() => {\r\n this._isLoading.set(false);\r\n this.initialized$.next(true);\r\n });\r\n }\r\n\r\n private fetchUser(): Observable<PrimusUser | null> {\r\n const endpoint = `${this.config.apiUrl}/api/auth/me`;\r\n return this.http.get<PrimusUser>(endpoint, { withCredentials: true }).pipe(\r\n tap(user => this.updateState(user)),\r\n catchError(() => {\r\n this.updateState(null);\r\n return of(null);\r\n })\r\n );\r\n }\r\n\r\n /**\r\n * Performs a local login (username/password) via the Backend Broker.\r\n * The backend must accept a POST to /api/auth/login and set a cookie.\r\n */\r\n public login(credentials: Record<string, any>): Observable<PrimusUser | null> {\r\n const endpoint = `${this.config.apiUrl}/api/auth/login`;\r\n // We expect the backend to return the User object on success (or we fetch it)\r\n // If the backend returns just 200 OK, we might need to chain a fetchUser()\r\n return this.http.post<PrimusUser>(endpoint, credentials, { withCredentials: true }).pipe(\r\n tap(user => {\r\n // If the backend returns the user directly, great.\r\n // If it returns a generic success, we might get 'null' here if typing doesn't match.\r\n // For safety, if we get a success response, we can also force-fetch the user to be sure.\r\n if (user && user.email) {\r\n this.updateState(user);\r\n } else {\r\n // Fallback: Fetch user info if response was just { success: true }\r\n this.fetchUser().subscribe();\r\n }\r\n })\r\n );\r\n }\r\n\r\n /**\r\n * Redirects the browser to the backend login endpoint for the provider.\r\n * e.g. /api/auth/azure, /api/auth/auth0\r\n */\r\n public loginWithProvider(provider: string, redirectUrl?: string): void {\r\n const returnUrl = redirectUrl || this.router.url;\r\n // We can pass returnUrl if backend supports it, usually configured globally\r\n // or via a query param handled by the Resolution logic.\r\n // For now we assume standard flow or basic redirect.\r\n\r\n const target = `${this.config.apiUrl}/api/auth/${provider}`;\r\n window.location.href = target;\r\n }\r\n\r\n public logout(): void {\r\n const endpoint = `${this.config.apiUrl}/api/auth/logout`;\r\n this.http.post(endpoint, {}, {\r\n withCredentials: true\r\n // Interceptor will add CSRF header automatically\r\n }).subscribe({\r\n next: () => {\r\n this.updateState(null);\r\n const redirect = this.config.logoutRedirect || '/login';\r\n this.router.navigateByUrl(redirect);\r\n },\r\n error: () => {\r\n // Force logout state even if API failed\r\n this.updateState(null);\r\n const redirect = this.config.logoutRedirect || '/login';\r\n this.router.navigateByUrl(redirect);\r\n }\r\n });\r\n }\r\n\r\n private updateState(user: PrimusUser | null) {\r\n this._user.set(user);\r\n this.user$.next(user);\r\n\r\n // Auto-sync signal to loading state\r\n if (user) {\r\n this._isLoading.set(false);\r\n this.startHeartbeat();\r\n } else {\r\n this.stopHeartbeat();\r\n }\r\n }\r\n\r\n // Called by Interceptor on 401\r\n public handleUnauthorized(): void {\r\n this.updateState(null);\r\n }\r\n\r\n private startHeartbeat(): void {\r\n if (!isPlatformBrowser(this.platformId)) return;\r\n if (this.heartbeatSub) return; // Already running\r\n\r\n this.heartbeatSub = interval(this.HEARTBEAT_INTERVAL).subscribe(() => {\r\n // Quietly check session\r\n this.fetchUser().subscribe();\r\n });\r\n }\r\n\r\n private stopHeartbeat(): void {\r\n if (this.heartbeatSub) {\r\n this.heartbeatSub.unsubscribe();\r\n this.heartbeatSub = null;\r\n }\r\n }\r\n}\r\n","import { HttpInterceptorFn, HttpErrorResponse } from '@angular/common/http';\r\nimport { inject } from '@angular/core';\r\nimport { catchError, throwError } from 'rxjs';\r\nimport { PrimusAuthService } from './primus-auth.service';\r\n\r\nexport const primusAuthInterceptor: HttpInterceptorFn = (req, next) => {\r\n const authService = inject(PrimusAuthService);\r\n\r\n // 1. Add CSRF Header to non-GET requests\r\n // Double-Submit Cookie Pattern: Read XSRF-TOKEN cookie and mirror in header.\r\n let request = req;\r\n if (req.method !== 'GET' && req.method !== 'HEAD' && req.method !== 'OPTIONS') {\r\n const token = getCookie('XSRF-TOKEN') || '';\r\n request = req.clone({\r\n setHeaders: {\r\n 'X-Primus-CSRF': token\r\n },\r\n withCredentials: true // Ensure cookies are sent\r\n });\r\n } else {\r\n // Determine if we need credentials for GETs?\r\n // Usually yes for API calls to our backend.\r\n // We assume calls to same origin or API URL need creds.\r\n request = req.clone({\r\n withCredentials: true\r\n });\r\n }\r\n \r\n return next(request).pipe(\r\n catchError((error: HttpErrorResponse) => {\r\n // 2. Handle 401 Unauthorized\r\n if (error.status === 401) {\r\n authService.handleUnauthorized();\r\n }\r\n return throwError(() => error);\r\n })\r\n );\r\n };\r\n \r\n function getCookie(name: string): string | null {\r\n try {\r\n if (typeof document === 'undefined') return null;\r\n const matches = document.cookie.match(new RegExp('(?:^|; )' + name.replace(/([\\.$?*|{}\\(\\)\\[\\]\\\\\\/\\+^])/g, '\\\\$1') + '=([^;]*)'));\r\n return matches ? decodeURIComponent(matches[1]) : null;\r\n } catch {\r\n return null;\r\n }\r\n }\r\n","import { CanActivateFn, Router, UrlTree, ActivatedRouteSnapshot, RouterStateSnapshot } from '@angular/router';\r\nimport { inject } from '@angular/core';\r\nimport { PrimusAuthService } from './primus-auth.service';\r\nimport { map, filter, switchMap, take } from 'rxjs/operators';\r\nimport { PRIMUS_AUTH_CONFIG } from './primus-auth.tokens';\r\n\r\nexport const primusAuthGuard: CanActivateFn = (route: ActivatedRouteSnapshot, state: RouterStateSnapshot) => {\r\n const auth = inject(PrimusAuthService);\r\n const router = inject(Router);\r\n const config = inject(PRIMUS_AUTH_CONFIG);\r\n\r\n // Wait for initialization to complete, then check authentication status\r\n return auth.initialized$.pipe(\r\n filter((isInit: boolean) => isInit), // Wait until true\r\n take(1),\r\n switchMap(() => auth.isAuthenticated$),\r\n take(1),\r\n map((isAuthenticated: boolean) => {\r\n if (isAuthenticated) {\r\n return true;\r\n }\r\n\r\n // Redirect to login\r\n const loginPath = config.loginPath || '/login';\r\n return router.createUrlTree([loginPath], {\r\n queryParams: { returnUrl: state.url }\r\n });\r\n })\r\n );\r\n};\r\n","import { EnvironmentProviders, makeEnvironmentProviders, Provider, APP_INITIALIZER, PLATFORM_ID, inject } from '@angular/core';\r\nimport { provideHttpClient, withInterceptors } from '@angular/common/http';\r\nimport { PrimusAuthConfig } from './primus-auth.types';\r\nimport { PRIMUS_AUTH_CONFIG } from './primus-auth.tokens';\r\nimport { PrimusAuthService } from './primus-auth.service';\r\nimport { primusAuthInterceptor } from './primus-auth.interceptor';\r\nimport { isPlatformBrowser } from '@angular/common';\r\n\r\nexport function providePrimusAuth(config: PrimusAuthConfig): EnvironmentProviders {\r\n return makeEnvironmentProviders([\r\n {\r\n provide: PRIMUS_AUTH_CONFIG,\r\n useValue: config\r\n },\r\n PrimusAuthService,\r\n provideHttpClient(\r\n withInterceptors([primusAuthInterceptor])\r\n ),\r\n {\r\n provide: APP_INITIALIZER,\r\n useFactory: (auth: PrimusAuthService, platform: Object) => () => {\r\n // We trigger checkSession in constructor, but we can also do it here if we want to await it.\r\n // However, awaiting APP_INITIALIZER blocks the whole app render.\r\n // Better to let it run in background and use the Guard to block specific routes.\r\n // So we just ensure service is instantiated.\r\n\r\n // Actually, checkSession is already called in constructor.\r\n // This APP_INITIALIZER forces the service to be created immediately on app start.\r\n },\r\n deps: [PrimusAuthService, PLATFORM_ID],\r\n multi: true\r\n }\r\n ]);\r\n}\r\n","/*\r\n * Public API Surface of primus-angular-auth\r\n */\r\n\r\nexport * from './lib/primus-auth.service';\r\nexport * from './lib/primus-auth.interceptor';\r\nexport * from './lib/primus-auth.guard';\r\nexport * from './lib/primus-auth.provider';\r\nexport * from './lib/primus-auth.types';\r\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":["catchError"],"mappings":"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"}
package/index.d.ts ADDED
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ /// <amd-module name="primus-angular-auth" />
5
+ export * from './public-api';
@@ -0,0 +1,2 @@
1
+ import { CanActivateFn } from '@angular/router';
2
+ export declare const primusAuthGuard: CanActivateFn;
@@ -0,0 +1,2 @@
1
+ import { HttpInterceptorFn } from '@angular/common/http';
2
+ export declare const primusAuthInterceptor: HttpInterceptorFn;
@@ -0,0 +1,3 @@
1
+ import { EnvironmentProviders } from '@angular/core';
2
+ import { PrimusAuthConfig } from './primus-auth.types';
3
+ export declare function providePrimusAuth(config: PrimusAuthConfig): EnvironmentProviders;
@@ -0,0 +1,49 @@
1
+ import { OnDestroy } from '@angular/core';
2
+ import { HttpClient } from '@angular/common/http';
3
+ import { Router } from '@angular/router';
4
+ import { Location } from '@angular/common';
5
+ import { Observable, BehaviorSubject } from 'rxjs';
6
+ import { PrimusAuthConfig, PrimusUser } from './primus-auth.types';
7
+ import * as i0 from "@angular/core";
8
+ export declare class PrimusAuthService implements OnDestroy {
9
+ private http;
10
+ private router;
11
+ private location;
12
+ private config;
13
+ private platformId;
14
+ private _user;
15
+ private _isLoading;
16
+ readonly user: import("@angular/core").Signal<PrimusUser>;
17
+ readonly isLoading: import("@angular/core").Signal<boolean>;
18
+ readonly isAuthenticated: import("@angular/core").Signal<boolean>;
19
+ readonly user$: BehaviorSubject<PrimusUser>;
20
+ readonly isAuthenticated$: Observable<boolean>;
21
+ readonly initialized$: BehaviorSubject<boolean>;
22
+ private heartbeatSub;
23
+ private readonly HEARTBEAT_INTERVAL;
24
+ constructor(http: HttpClient, router: Router, location: Location, config: PrimusAuthConfig, platformId: Object);
25
+ ngOnDestroy(): void;
26
+ /**
27
+ * Checks /api/auth/me to see if user is logged in.
28
+ * Should be called on app startup.
29
+ */
30
+ checkSession(): void;
31
+ private fetchUser;
32
+ /**
33
+ * Performs a local login (username/password) via the Backend Broker.
34
+ * The backend must accept a POST to /api/auth/login and set a cookie.
35
+ */
36
+ login(credentials: Record<string, any>): Observable<PrimusUser | null>;
37
+ /**
38
+ * Redirects the browser to the backend login endpoint for the provider.
39
+ * e.g. /api/auth/azure, /api/auth/auth0
40
+ */
41
+ loginWithProvider(provider: string, redirectUrl?: string): void;
42
+ logout(): void;
43
+ private updateState;
44
+ handleUnauthorized(): void;
45
+ private startHeartbeat;
46
+ private stopHeartbeat;
47
+ static ɵfac: i0.ɵɵFactoryDeclaration<PrimusAuthService, never>;
48
+ static ɵprov: i0.ɵɵInjectableDeclaration<PrimusAuthService>;
49
+ }
@@ -0,0 +1,3 @@
1
+ import { InjectionToken } from '@angular/core';
2
+ import { PrimusAuthConfig } from './primus-auth.types';
3
+ export declare const PRIMUS_AUTH_CONFIG: InjectionToken<PrimusAuthConfig>;
@@ -0,0 +1,19 @@
1
+ export interface PrimusAuthConfig {
2
+ apiUrl: string;
3
+ loginPath?: string;
4
+ logoutRedirect?: string;
5
+ }
6
+ export interface PrimusUser {
7
+ id: string;
8
+ email: string;
9
+ name?: string;
10
+ role?: string;
11
+ provider?: string;
12
+ picture?: string;
13
+ [key: string]: any;
14
+ }
15
+ export interface AuthState {
16
+ isAuthenticated: boolean;
17
+ user: PrimusUser | null;
18
+ isLoading: boolean;
19
+ }
package/package.json ADDED
@@ -0,0 +1,29 @@
1
+ {
2
+ "name": "primus-angular-auth",
3
+ "version": "1.0.0",
4
+ "description": "Enterprise-grade BFF authentication client for Primus SaaS Framework in Angular.",
5
+ "license": "MIT",
6
+ "peerDependencies": {
7
+ "@angular/common": "^17.0.0 || ^18.0.0 || ^19.0.0 || ^20.0.0 || ^21.0.0",
8
+ "@angular/core": "^17.0.0 || ^18.0.0 || ^19.0.0 || ^20.0.0 || ^21.0.0",
9
+ "@angular/router": "^17.0.0 || ^18.0.0 || ^19.0.0 || ^20.0.0 || ^21.0.0",
10
+ "rxjs": "^7.8.0"
11
+ },
12
+ "dependencies": {
13
+ "tslib": "^2.3.0"
14
+ },
15
+ "sideEffects": false,
16
+ "module": "fesm2022/primus-angular-auth.mjs",
17
+ "typings": "index.d.ts",
18
+ "exports": {
19
+ "./package.json": {
20
+ "default": "./package.json"
21
+ },
22
+ ".": {
23
+ "types": "./index.d.ts",
24
+ "esm2022": "./esm2022/primus-angular-auth.mjs",
25
+ "esm": "./esm2022/primus-angular-auth.mjs",
26
+ "default": "./fesm2022/primus-angular-auth.mjs"
27
+ }
28
+ }
29
+ }
@@ -0,0 +1,5 @@
1
+ export * from './lib/primus-auth.service';
2
+ export * from './lib/primus-auth.interceptor';
3
+ export * from './lib/primus-auth.guard';
4
+ export * from './lib/primus-auth.provider';
5
+ export * from './lib/primus-auth.types';