mirror of
https://github.com/Alfresco/alfresco-content-app.git
synced 2025-05-12 17:04:46 +00:00
[ACA-3131] Add extension loader guard (#1426)
* [ACA-3131] Add extension loader guard * Fix linting issue * Fix cspell issues
This commit is contained in:
parent
5259f840a8
commit
a21aeab12d
@ -65,7 +65,9 @@
|
||||
"submenu",
|
||||
"submenus",
|
||||
"dateitem",
|
||||
"versionable"
|
||||
"versionable",
|
||||
"erroredSpy",
|
||||
"errored"
|
||||
],
|
||||
"dictionaries": ["html", "en-gb", "en_US"]
|
||||
}
|
||||
|
@ -0,0 +1,135 @@
|
||||
/*!
|
||||
* @license
|
||||
* Alfresco Example Content Application
|
||||
*
|
||||
* Copyright (C) 2005 - 2020 Alfresco Software Limited
|
||||
*
|
||||
* This file is part of the Alfresco Example Content Application.
|
||||
* If the software was purchased under a paid Alfresco license, the terms of
|
||||
* the paid license agreement will prevail. Otherwise, the software is
|
||||
* provided under the following open source license terms:
|
||||
*
|
||||
* The Alfresco Example Content Application is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU Lesser General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* The Alfresco Example Content Application is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU Lesser General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Lesser General Public License
|
||||
* along with Alfresco. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
import { ExtensionsDataLoaderGuard } from './extensions-data-loader.guard';
|
||||
import { ActivatedRouteSnapshot } from '@angular/router';
|
||||
import { Subject, throwError } from 'rxjs';
|
||||
|
||||
describe('ExtensionsDataLoaderGuard', () => {
|
||||
let route: ActivatedRouteSnapshot;
|
||||
let emittedSpy;
|
||||
let completedSpy;
|
||||
let erroredSpy;
|
||||
|
||||
describe('canActivate', () => {
|
||||
beforeEach(() => {
|
||||
route = {} as ActivatedRouteSnapshot;
|
||||
emittedSpy = jasmine.createSpy('emitted');
|
||||
completedSpy = jasmine.createSpy('completed');
|
||||
erroredSpy = jasmine.createSpy('errored');
|
||||
});
|
||||
|
||||
it('should emit true and complete if no callback are present', () => {
|
||||
const guard = new ExtensionsDataLoaderGuard([]);
|
||||
|
||||
guard.canActivate(route).subscribe(emittedSpy, erroredSpy, completedSpy);
|
||||
expect(emittedSpy).toHaveBeenCalledWith(true);
|
||||
expect(erroredSpy).not.toHaveBeenCalled();
|
||||
expect(completedSpy).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('should emit true and complete in case of only one callback is present, completed', () => {
|
||||
const subject = new Subject<true>();
|
||||
const guard = new ExtensionsDataLoaderGuard([
|
||||
() => subject.asObservable()
|
||||
]);
|
||||
|
||||
guard.canActivate(route).subscribe(emittedSpy, erroredSpy, completedSpy);
|
||||
|
||||
subject.next(true);
|
||||
expect(emittedSpy).not.toHaveBeenCalled();
|
||||
expect(erroredSpy).not.toHaveBeenCalled();
|
||||
expect(completedSpy).not.toHaveBeenCalled();
|
||||
|
||||
subject.complete();
|
||||
expect(emittedSpy).toHaveBeenCalledWith(true);
|
||||
expect(erroredSpy).not.toHaveBeenCalled();
|
||||
expect(completedSpy).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('should emit true and complete in case of only one callback is present, errored', () => {
|
||||
const guard = new ExtensionsDataLoaderGuard([
|
||||
() => throwError(new Error())
|
||||
]);
|
||||
|
||||
guard.canActivate(route).subscribe(emittedSpy, erroredSpy, completedSpy);
|
||||
expect(emittedSpy).toHaveBeenCalledWith(true);
|
||||
expect(erroredSpy).not.toHaveBeenCalled();
|
||||
expect(completedSpy).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('should NOT complete in case of multiple callbacks are present and not all of them have been completed', () => {
|
||||
const subject1 = new Subject<true>();
|
||||
const subject2 = new Subject<true>();
|
||||
const guard = new ExtensionsDataLoaderGuard([
|
||||
() => subject1.asObservable(),
|
||||
() => subject2.asObservable()
|
||||
]);
|
||||
|
||||
guard.canActivate(route).subscribe(emittedSpy, erroredSpy, completedSpy);
|
||||
|
||||
subject1.next();
|
||||
subject2.next();
|
||||
subject1.complete();
|
||||
expect(emittedSpy).not.toHaveBeenCalled();
|
||||
expect(erroredSpy).not.toHaveBeenCalled();
|
||||
expect(completedSpy).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('should emit true and complete in case of multiple callbacks are present and all of them have been completed', () => {
|
||||
const subject1 = new Subject<true>();
|
||||
const subject2 = new Subject<true>();
|
||||
const guard = new ExtensionsDataLoaderGuard([
|
||||
() => subject1.asObservable(),
|
||||
() => subject2.asObservable()
|
||||
]);
|
||||
|
||||
guard.canActivate(route).subscribe(emittedSpy, erroredSpy, completedSpy);
|
||||
|
||||
subject1.next();
|
||||
subject2.next();
|
||||
subject1.complete();
|
||||
subject2.complete();
|
||||
expect(emittedSpy).toHaveBeenCalledWith(true);
|
||||
expect(erroredSpy).not.toHaveBeenCalled();
|
||||
expect(completedSpy).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('should emit true and complete even if one of the observables are errored, to not block the application loading', () => {
|
||||
const subject1 = new Subject<true>();
|
||||
const guard = new ExtensionsDataLoaderGuard([
|
||||
() => subject1.asObservable(),
|
||||
() => throwError(new Error())
|
||||
]);
|
||||
|
||||
guard.canActivate(route).subscribe(emittedSpy, erroredSpy, completedSpy);
|
||||
|
||||
subject1.next();
|
||||
expect(emittedSpy).toHaveBeenCalledWith(true);
|
||||
expect(erroredSpy).not.toHaveBeenCalled();
|
||||
expect(completedSpy).toHaveBeenCalled();
|
||||
});
|
||||
});
|
||||
});
|
@ -0,0 +1,74 @@
|
||||
/*!
|
||||
* @license
|
||||
* Alfresco Example Content Application
|
||||
*
|
||||
* Copyright (C) 2005 - 2020 Alfresco Software Limited
|
||||
*
|
||||
* This file is part of the Alfresco Example Content Application.
|
||||
* If the software was purchased under a paid Alfresco license, the terms of
|
||||
* the paid license agreement will prevail. Otherwise, the software is
|
||||
* provided under the following open source license terms:
|
||||
*
|
||||
* The Alfresco Example Content Application is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU Lesser General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* The Alfresco Example Content Application is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU Lesser General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Lesser General Public License
|
||||
* along with Alfresco. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
import { Injectable, InjectionToken, Inject } from '@angular/core';
|
||||
import { CanActivate, ActivatedRouteSnapshot } from '@angular/router';
|
||||
import { Observable, forkJoin, of } from 'rxjs';
|
||||
import { map, catchError } from 'rxjs/operators';
|
||||
|
||||
export type ExtensionLoaderCallback = (
|
||||
route: ActivatedRouteSnapshot
|
||||
) => Observable<true>;
|
||||
|
||||
export const EXTENSION_DATA_LOADERS = new InjectionToken<
|
||||
ExtensionLoaderCallback[]
|
||||
>('EXTENSION_DATA_LOADERS', {
|
||||
providedIn: 'root',
|
||||
factory: () => []
|
||||
});
|
||||
|
||||
@Injectable({ providedIn: 'root' })
|
||||
export class ExtensionsDataLoaderGuard implements CanActivate {
|
||||
constructor(
|
||||
@Inject(EXTENSION_DATA_LOADERS)
|
||||
private extensionDataLoaders: ExtensionLoaderCallback[]
|
||||
) {}
|
||||
|
||||
canActivate(route: ActivatedRouteSnapshot): Observable<boolean> {
|
||||
if (!this.extensionDataLoaders.length) {
|
||||
return of(true);
|
||||
}
|
||||
|
||||
const dataLoaderCallbacks = this.extensionDataLoaders.map(callback =>
|
||||
callback(route)
|
||||
);
|
||||
|
||||
// Undocumented forkJoin behaviour/bug:
|
||||
// https://github.com/ReactiveX/rxjs/issues/3246
|
||||
// So all callbacks need to emit before completion, otherwise forkJoin will short circuit
|
||||
return forkJoin(...dataLoaderCallbacks).pipe(
|
||||
map(() => true),
|
||||
catchError(e => {
|
||||
// tslint:disable-next-line
|
||||
console.error(
|
||||
'Some of the extension data loader guards has been errored.'
|
||||
);
|
||||
// tslint:disable-next-line
|
||||
console.error(e);
|
||||
return of(true);
|
||||
})
|
||||
);
|
||||
}
|
||||
}
|
@ -23,6 +23,7 @@
|
||||
* along with Alfresco. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
export * from './lib/adf-extensions/extensions-data-loader.guard';
|
||||
export * from './lib/components/page-layout/page-layout-content.component';
|
||||
export * from './lib/components/page-layout/page-layout-error.component';
|
||||
export * from './lib/components/page-layout/page-layout-header.component';
|
||||
|
@ -33,7 +33,8 @@ import { SearchLibrariesResultsComponent } from './components/search/search-libr
|
||||
import { LoginComponent } from './components/login/login.component';
|
||||
import {
|
||||
AppSharedRuleGuard,
|
||||
GenericErrorComponent
|
||||
GenericErrorComponent,
|
||||
ExtensionsDataLoaderGuard
|
||||
} from '@alfresco/aca-shared';
|
||||
import { AuthGuardEcm } from '@alfresco/adf-core';
|
||||
import { FavoritesComponent } from './components/favorites/favorites.component';
|
||||
@ -76,7 +77,7 @@ export const APP_ROUTES: Routes = [
|
||||
{
|
||||
path: '',
|
||||
component: AppLayoutComponent,
|
||||
canActivate: [AuthGuardEcm],
|
||||
canActivate: [AuthGuardEcm, ExtensionsDataLoaderGuard],
|
||||
children: [
|
||||
{
|
||||
path: '',
|
||||
|
Loading…
x
Reference in New Issue
Block a user