2016-06-23 12:47:54 -04:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright Google Inc. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2017-08-10 21:08:49 -04:00
|
|
|
import {AnimationBuilder, animate, style, transition, trigger} from '@angular/animations';
|
2017-03-14 18:38:24 -04:00
|
|
|
import {APP_BASE_HREF, PlatformLocation, isPlatformServer} from '@angular/common';
|
2017-07-18 15:45:47 -04:00
|
|
|
import {HttpClient, HttpClientModule} from '@angular/common/http';
|
|
|
|
import {HttpClientTestingModule, HttpTestingController} from '@angular/common/http/testing';
|
2018-05-24 19:04:04 -04:00
|
|
|
import {ApplicationRef, CompilerFactory, Component, HostListener, Inject, Input, NgModule, NgModuleRef, NgZone, PLATFORM_ID, PlatformRef, ViewEncapsulation, destroyPlatform, getPlatform} from '@angular/core';
|
2017-02-20 17:34:15 -05:00
|
|
|
import {TestBed, async, inject} from '@angular/core/testing';
|
2017-02-10 20:00:27 -05:00
|
|
|
import {Http, HttpModule, Response, ResponseOptions, XHRBackend} from '@angular/http';
|
|
|
|
import {MockBackend, MockConnection} from '@angular/http/testing';
|
2017-09-11 03:18:55 -04:00
|
|
|
import {BrowserModule, DOCUMENT, StateKey, Title, TransferState, makeStateKey} from '@angular/platform-browser';
|
2016-06-14 22:49:25 -04:00
|
|
|
import {getDOM} from '@angular/platform-browser/src/dom/dom_adapter';
|
2017-09-11 03:18:55 -04:00
|
|
|
import {BEFORE_APP_SERIALIZED, INITIAL_CONFIG, PlatformState, ServerModule, ServerTransferStateModule, platformDynamicServer, renderModule, renderModuleFactory} from '@angular/platform-server';
|
2018-02-27 17:06:06 -05:00
|
|
|
import {first} from 'rxjs/operators';
|
2016-08-15 16:44:01 -04:00
|
|
|
|
|
|
|
@Component({selector: 'app', template: `Works!`})
|
|
|
|
class MyServerApp {
|
|
|
|
}
|
|
|
|
|
2017-02-10 20:00:27 -05:00
|
|
|
@NgModule({
|
|
|
|
bootstrap: [MyServerApp],
|
|
|
|
declarations: [MyServerApp],
|
|
|
|
imports: [ServerModule],
|
|
|
|
providers: [
|
|
|
|
MockBackend,
|
|
|
|
{provide: XHRBackend, useExisting: MockBackend},
|
|
|
|
]
|
|
|
|
})
|
2016-08-15 16:44:01 -04:00
|
|
|
class ExampleModule {
|
|
|
|
}
|
|
|
|
|
2017-09-04 03:38:42 -04:00
|
|
|
function getTitleRenderHook(doc: any) {
|
|
|
|
return () => {
|
|
|
|
// Set the title as part of the render hook.
|
|
|
|
doc.title = 'RenderHook';
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function exceptionRenderHook() {
|
|
|
|
throw new Error('error');
|
|
|
|
}
|
|
|
|
|
|
|
|
function getMetaRenderHook(doc: any) {
|
|
|
|
return () => {
|
|
|
|
// Add a meta tag before rendering the document.
|
|
|
|
const metaElement = doc.createElement('meta');
|
|
|
|
metaElement.setAttribute('name', 'description');
|
|
|
|
doc.head.appendChild(metaElement);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
bootstrap: [MyServerApp],
|
|
|
|
declarations: [MyServerApp],
|
|
|
|
imports: [BrowserModule.withServerTransition({appId: 'render-hook'}), ServerModule],
|
|
|
|
providers: [
|
|
|
|
{provide: BEFORE_APP_SERIALIZED, useFactory: getTitleRenderHook, multi: true, deps: [DOCUMENT]},
|
|
|
|
]
|
|
|
|
})
|
|
|
|
class RenderHookModule {
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
bootstrap: [MyServerApp],
|
|
|
|
declarations: [MyServerApp],
|
|
|
|
imports: [BrowserModule.withServerTransition({appId: 'render-hook'}), ServerModule],
|
|
|
|
providers: [
|
|
|
|
{provide: BEFORE_APP_SERIALIZED, useFactory: getTitleRenderHook, multi: true, deps: [DOCUMENT]},
|
|
|
|
{provide: BEFORE_APP_SERIALIZED, useValue: exceptionRenderHook, multi: true},
|
|
|
|
{provide: BEFORE_APP_SERIALIZED, useFactory: getMetaRenderHook, multi: true, deps: [DOCUMENT]},
|
|
|
|
]
|
|
|
|
})
|
|
|
|
class MultiRenderHookModule {
|
|
|
|
}
|
|
|
|
|
2017-02-12 12:16:23 -05:00
|
|
|
@Component({selector: 'app', template: `Works too!`})
|
|
|
|
class MyServerApp2 {
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({declarations: [MyServerApp2], imports: [ServerModule], bootstrap: [MyServerApp2]})
|
|
|
|
class ExampleModule2 {
|
|
|
|
}
|
|
|
|
|
2017-03-13 16:22:03 -04:00
|
|
|
@Component({selector: 'app', template: ``})
|
|
|
|
class TitleApp {
|
|
|
|
constructor(private title: Title) {}
|
|
|
|
ngOnInit() { this.title.setTitle('Test App Title'); }
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({declarations: [TitleApp], imports: [ServerModule], bootstrap: [TitleApp]})
|
|
|
|
class TitleAppModule {
|
|
|
|
}
|
|
|
|
|
2017-04-13 14:54:57 -04:00
|
|
|
@Component({selector: 'app', template: '{{text}}<h1 [innerText]="h1"></h1>'})
|
2017-02-12 12:16:23 -05:00
|
|
|
class MyAsyncServerApp {
|
|
|
|
text = '';
|
2017-04-13 14:54:57 -04:00
|
|
|
h1 = '';
|
2017-02-12 12:16:23 -05:00
|
|
|
|
2017-03-14 23:48:01 -04:00
|
|
|
@HostListener('window:scroll')
|
|
|
|
track() { console.error('scroll'); }
|
|
|
|
|
2017-02-12 12:16:23 -05:00
|
|
|
ngOnInit() {
|
2017-04-13 14:54:57 -04:00
|
|
|
Promise.resolve(null).then(() => setTimeout(() => {
|
|
|
|
this.text = 'Works!';
|
|
|
|
this.h1 = 'fine';
|
|
|
|
}, 10));
|
2017-02-12 12:16:23 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-22 19:06:21 -05:00
|
|
|
@NgModule({
|
|
|
|
declarations: [MyAsyncServerApp],
|
|
|
|
imports: [BrowserModule.withServerTransition({appId: 'async-server'}), ServerModule],
|
|
|
|
bootstrap: [MyAsyncServerApp]
|
|
|
|
})
|
2017-02-12 12:16:23 -05:00
|
|
|
class AsyncServerModule {
|
|
|
|
}
|
|
|
|
|
2017-03-14 18:40:55 -04:00
|
|
|
@Component({selector: 'app', template: '<svg><use xlink:href="#clear"></use></svg>'})
|
|
|
|
class SVGComponent {
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
declarations: [SVGComponent],
|
|
|
|
imports: [BrowserModule.withServerTransition({appId: 'svg-server'}), ServerModule],
|
|
|
|
bootstrap: [SVGComponent]
|
|
|
|
})
|
|
|
|
class SVGServerModule {
|
|
|
|
}
|
|
|
|
|
2017-03-13 20:31:03 -04:00
|
|
|
@Component({
|
|
|
|
selector: 'app',
|
|
|
|
template: '<div @myAnimation>{{text}}</div>',
|
|
|
|
animations: [trigger(
|
|
|
|
'myAnimation',
|
|
|
|
[transition('void => *', [style({'opacity': '0'}), animate(500, style({'opacity': '1'}))])])],
|
|
|
|
})
|
|
|
|
class MyAnimationApp {
|
2017-08-10 21:08:49 -04:00
|
|
|
constructor(private builder: AnimationBuilder) {}
|
|
|
|
|
2017-03-13 20:31:03 -04:00
|
|
|
text = 'Works!';
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
declarations: [MyAnimationApp],
|
|
|
|
imports: [BrowserModule.withServerTransition({appId: 'anim-server'}), ServerModule],
|
|
|
|
bootstrap: [MyAnimationApp]
|
|
|
|
})
|
|
|
|
class AnimationServerModule {
|
|
|
|
}
|
|
|
|
|
2018-05-28 03:20:53 -04:00
|
|
|
@Component({
|
|
|
|
selector: 'app',
|
|
|
|
template: `<div>Works!</div>`,
|
|
|
|
styles: ['div {color: blue; } :host { color: red; }']
|
|
|
|
})
|
2017-02-14 14:34:05 -05:00
|
|
|
class MyStylesApp {
|
|
|
|
}
|
|
|
|
|
2017-02-22 19:06:21 -05:00
|
|
|
@NgModule({
|
|
|
|
declarations: [MyStylesApp],
|
|
|
|
imports: [BrowserModule.withServerTransition({appId: 'example-styles'}), ServerModule],
|
|
|
|
bootstrap: [MyStylesApp]
|
|
|
|
})
|
2017-02-14 14:34:05 -05:00
|
|
|
class ExampleStylesModule {
|
|
|
|
}
|
|
|
|
|
2017-02-10 20:00:27 -05:00
|
|
|
@NgModule({
|
|
|
|
bootstrap: [MyServerApp],
|
|
|
|
declarations: [MyServerApp],
|
|
|
|
imports: [HttpModule, ServerModule],
|
|
|
|
providers: [
|
|
|
|
MockBackend,
|
|
|
|
{provide: XHRBackend, useExisting: MockBackend},
|
|
|
|
]
|
|
|
|
})
|
|
|
|
export class HttpBeforeExampleModule {
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
bootstrap: [MyServerApp],
|
|
|
|
declarations: [MyServerApp],
|
|
|
|
imports: [ServerModule, HttpModule],
|
|
|
|
providers: [
|
|
|
|
MockBackend,
|
|
|
|
{provide: XHRBackend, useExisting: MockBackend},
|
|
|
|
]
|
|
|
|
})
|
|
|
|
export class HttpAfterExampleModule {
|
|
|
|
}
|
|
|
|
|
2017-07-18 15:45:47 -04:00
|
|
|
@NgModule({
|
|
|
|
bootstrap: [MyServerApp],
|
|
|
|
declarations: [MyServerApp],
|
|
|
|
imports: [ServerModule, HttpClientModule, HttpClientTestingModule],
|
|
|
|
})
|
2018-03-10 12:14:58 -05:00
|
|
|
export class HttpClientExampleModule {
|
2017-07-18 15:45:47 -04:00
|
|
|
}
|
|
|
|
|
2017-02-13 18:17:40 -05:00
|
|
|
@Component({selector: 'app', template: `<img [src]="'link'">`})
|
|
|
|
class ImageApp {
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({declarations: [ImageApp], imports: [ServerModule], bootstrap: [ImageApp]})
|
|
|
|
class ImageExampleModule {
|
|
|
|
}
|
|
|
|
|
2017-03-14 17:09:24 -04:00
|
|
|
@Component({
|
|
|
|
selector: 'app',
|
|
|
|
template: 'Native works',
|
|
|
|
encapsulation: ViewEncapsulation.Native,
|
|
|
|
styles: [':host { color: red; }']
|
|
|
|
})
|
|
|
|
class NativeEncapsulationApp {
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
declarations: [NativeEncapsulationApp],
|
|
|
|
imports: [BrowserModule.withServerTransition({appId: 'test'}), ServerModule],
|
|
|
|
bootstrap: [NativeEncapsulationApp]
|
|
|
|
})
|
|
|
|
class NativeExampleModule {
|
|
|
|
}
|
|
|
|
|
2017-03-28 16:33:07 -04:00
|
|
|
@Component({selector: 'my-child', template: 'Works!'})
|
|
|
|
class MyChildComponent {
|
|
|
|
@Input() public attr: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Component({selector: 'app', template: '<my-child [attr]="false"></my-child>'})
|
|
|
|
class MyHostComponent {
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
declarations: [MyHostComponent, MyChildComponent],
|
|
|
|
bootstrap: [MyHostComponent],
|
|
|
|
imports: [ServerModule, BrowserModule.withServerTransition({appId: 'false-attributes'})]
|
|
|
|
})
|
|
|
|
class FalseAttributesModule {
|
|
|
|
}
|
|
|
|
|
2017-07-19 16:58:23 -04:00
|
|
|
@Component({selector: 'app', template: '<input [name]="name">'})
|
|
|
|
class MyInputComponent {
|
|
|
|
@Input()
|
|
|
|
name = '';
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
declarations: [MyInputComponent],
|
|
|
|
bootstrap: [MyInputComponent],
|
|
|
|
imports: [ServerModule, BrowserModule.withServerTransition({appId: 'name-attributes'})]
|
|
|
|
})
|
|
|
|
class NameModule {
|
|
|
|
}
|
|
|
|
|
2018-05-24 19:04:04 -04:00
|
|
|
@Component({selector: 'app', template: '<div [innerHTML]="html"></div>'})
|
|
|
|
class HTMLTypesApp {
|
|
|
|
html = '<b>foo</b> bar';
|
|
|
|
constructor(@Inject(DOCUMENT) doc: Document) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
declarations: [HTMLTypesApp],
|
|
|
|
imports: [BrowserModule.withServerTransition({appId: 'inner-html'}), ServerModule],
|
|
|
|
bootstrap: [HTMLTypesApp]
|
|
|
|
})
|
|
|
|
class HTMLTypesModule {
|
|
|
|
}
|
|
|
|
|
2017-09-11 03:18:55 -04:00
|
|
|
const TEST_KEY = makeStateKey<number>('test');
|
|
|
|
const STRING_KEY = makeStateKey<string>('testString');
|
|
|
|
|
|
|
|
@Component({selector: 'app', template: 'Works!'})
|
|
|
|
class TransferComponent {
|
|
|
|
constructor(private transferStore: TransferState) {}
|
|
|
|
ngOnInit() { this.transferStore.set(TEST_KEY, 10); }
|
|
|
|
}
|
|
|
|
|
|
|
|
@Component({selector: 'esc-app', template: 'Works!'})
|
|
|
|
class EscapedComponent {
|
|
|
|
constructor(private transferStore: TransferState) {}
|
|
|
|
ngOnInit() {
|
|
|
|
this.transferStore.set(STRING_KEY, '</script><script>alert(\'Hello&\' + "World");');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
bootstrap: [TransferComponent],
|
|
|
|
declarations: [TransferComponent],
|
|
|
|
imports: [
|
|
|
|
BrowserModule.withServerTransition({appId: 'transfer'}),
|
|
|
|
ServerModule,
|
|
|
|
ServerTransferStateModule,
|
|
|
|
]
|
|
|
|
})
|
|
|
|
class TransferStoreModule {
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
bootstrap: [EscapedComponent],
|
|
|
|
declarations: [EscapedComponent],
|
|
|
|
imports: [
|
|
|
|
BrowserModule.withServerTransition({appId: 'transfer'}),
|
|
|
|
ServerModule,
|
|
|
|
ServerTransferStateModule,
|
|
|
|
]
|
|
|
|
})
|
|
|
|
class EscapedTransferStoreModule {
|
|
|
|
}
|
|
|
|
|
2017-12-17 18:10:54 -05:00
|
|
|
(function() {
|
2016-06-14 22:49:25 -04:00
|
|
|
if (getDOM().supportsDOMEvents()) return; // NODE only
|
|
|
|
|
2017-02-10 19:48:04 -05:00
|
|
|
describe('platform-server integration', () => {
|
2017-02-12 12:16:23 -05:00
|
|
|
beforeEach(() => {
|
|
|
|
if (getPlatform()) destroyPlatform();
|
|
|
|
});
|
2016-06-14 22:49:25 -04:00
|
|
|
|
|
|
|
it('should bootstrap', async(() => {
|
2017-02-12 12:16:23 -05:00
|
|
|
const platform = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
|
|
|
|
|
|
|
platform.bootstrapModule(ExampleModule).then((moduleRef) => {
|
2017-02-22 19:49:46 -05:00
|
|
|
expect(isPlatformServer(moduleRef.injector.get(PLATFORM_ID))).toBe(true);
|
2017-02-12 12:16:23 -05:00
|
|
|
const doc = moduleRef.injector.get(DOCUMENT);
|
2017-03-14 23:48:01 -04:00
|
|
|
|
|
|
|
expect(doc.head).toBe(getDOM().querySelector(doc, 'head'));
|
|
|
|
expect(doc.body).toBe(getDOM().querySelector(doc, 'body'));
|
|
|
|
|
2017-08-08 05:17:40 -04:00
|
|
|
expect(getDOM().getText(doc.documentElement)).toEqual('Works!');
|
2017-03-14 23:48:01 -04:00
|
|
|
|
2017-02-12 12:16:23 -05:00
|
|
|
platform.destroy();
|
|
|
|
});
|
2016-06-14 22:49:25 -04:00
|
|
|
}));
|
2017-02-10 19:29:30 -05:00
|
|
|
|
2017-02-12 12:16:23 -05:00
|
|
|
it('should allow multiple platform instances', async(() => {
|
|
|
|
const platform = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
2017-02-14 19:14:40 -05:00
|
|
|
|
2017-02-12 12:16:23 -05:00
|
|
|
const platform2 = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
2017-02-14 19:14:40 -05:00
|
|
|
|
|
|
|
|
2017-02-12 12:16:23 -05:00
|
|
|
platform.bootstrapModule(ExampleModule).then((moduleRef) => {
|
|
|
|
const doc = moduleRef.injector.get(DOCUMENT);
|
2017-08-08 05:17:40 -04:00
|
|
|
expect(getDOM().getText(doc.documentElement)).toEqual('Works!');
|
2017-02-12 12:16:23 -05:00
|
|
|
platform.destroy();
|
|
|
|
});
|
2017-02-14 19:14:40 -05:00
|
|
|
|
2017-02-12 12:16:23 -05:00
|
|
|
platform2.bootstrapModule(ExampleModule2).then((moduleRef) => {
|
|
|
|
const doc = moduleRef.injector.get(DOCUMENT);
|
2017-08-08 05:17:40 -04:00
|
|
|
expect(getDOM().getText(doc.documentElement)).toEqual('Works too!');
|
2017-02-12 12:16:23 -05:00
|
|
|
platform2.destroy();
|
|
|
|
});
|
|
|
|
}));
|
2017-02-14 19:14:40 -05:00
|
|
|
|
2017-03-13 16:22:03 -04:00
|
|
|
it('adds title to the document using Title service', async(() => {
|
|
|
|
const platform = platformDynamicServer([{
|
|
|
|
provide: INITIAL_CONFIG,
|
|
|
|
useValue:
|
|
|
|
{document: '<html><head><title></title></head><body><app></app></body></html>'}
|
|
|
|
}]);
|
|
|
|
platform.bootstrapModule(TitleAppModule).then(ref => {
|
|
|
|
const state = ref.injector.get(PlatformState);
|
|
|
|
const doc = ref.injector.get(DOCUMENT);
|
|
|
|
const title = getDOM().querySelector(doc, 'title');
|
|
|
|
expect(getDOM().getText(title)).toBe('Test App Title');
|
|
|
|
expect(state.renderToString()).toContain('<title>Test App Title</title>');
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
|
2017-03-14 18:38:24 -04:00
|
|
|
it('should get base href from document', async(() => {
|
|
|
|
const platform = platformDynamicServer([{
|
|
|
|
provide: INITIAL_CONFIG,
|
|
|
|
useValue:
|
|
|
|
{document: '<html><head><base href="/"></head><body><app></app></body></html>'}
|
|
|
|
}]);
|
|
|
|
platform.bootstrapModule(ExampleModule).then((moduleRef) => {
|
|
|
|
const location = moduleRef.injector.get(PlatformLocation);
|
|
|
|
expect(location.getBaseHrefFromDOM()).toEqual('/');
|
|
|
|
platform.destroy();
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
|
2017-02-22 19:06:21 -05:00
|
|
|
it('adds styles with ng-transition attribute', async(() => {
|
|
|
|
const platform = platformDynamicServer([{
|
|
|
|
provide: INITIAL_CONFIG,
|
|
|
|
useValue: {document: '<html><head></head><body><app></app></body></html>'}
|
|
|
|
}]);
|
2017-02-14 14:34:05 -05:00
|
|
|
platform.bootstrapModule(ExampleStylesModule).then(ref => {
|
2017-02-22 19:06:21 -05:00
|
|
|
const doc = ref.injector.get(DOCUMENT);
|
|
|
|
const head = getDOM().getElementsByTagName(doc, 'head')[0];
|
|
|
|
const styles: any[] = head.children as any;
|
|
|
|
expect(styles.length).toBe(1);
|
|
|
|
expect(getDOM().getText(styles[0])).toContain('color: red');
|
|
|
|
expect(getDOM().getAttribute(styles[0], 'ng-transition')).toBe('example-styles');
|
2017-02-14 14:34:05 -05:00
|
|
|
});
|
|
|
|
}));
|
|
|
|
|
2017-02-13 18:17:40 -05:00
|
|
|
it('copies known properties to attributes', async(() => {
|
|
|
|
const platform = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
|
|
|
platform.bootstrapModule(ImageExampleModule).then(ref => {
|
|
|
|
const appRef: ApplicationRef = ref.injector.get(ApplicationRef);
|
|
|
|
const app = appRef.components[0].location.nativeElement;
|
|
|
|
const img = getDOM().getElementsByTagName(app, 'img')[0] as any;
|
2017-08-08 05:17:40 -04:00
|
|
|
expect(img.attributes['src'].value).toEqual('link');
|
2017-02-13 18:17:40 -05:00
|
|
|
});
|
|
|
|
}));
|
|
|
|
|
2017-02-12 12:16:23 -05:00
|
|
|
describe('PlatformLocation', () => {
|
|
|
|
it('is injectable', async(() => {
|
|
|
|
const platform = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
|
|
|
platform.bootstrapModule(ExampleModule).then(appRef => {
|
|
|
|
const location: PlatformLocation = appRef.injector.get(PlatformLocation);
|
|
|
|
expect(location.pathname).toBe('/');
|
|
|
|
platform.destroy();
|
|
|
|
});
|
|
|
|
}));
|
2017-02-14 22:48:48 -05:00
|
|
|
it('is configurable via INITIAL_CONFIG', () => {
|
|
|
|
platformDynamicServer([{
|
|
|
|
provide: INITIAL_CONFIG,
|
|
|
|
useValue: {document: '<app></app>', url: 'http://test.com/deep/path?query#hash'}
|
|
|
|
}])
|
|
|
|
.bootstrapModule(ExampleModule)
|
|
|
|
.then(appRef => {
|
|
|
|
const location: PlatformLocation = appRef.injector.get(PlatformLocation);
|
|
|
|
expect(location.pathname).toBe('/deep/path');
|
|
|
|
expect(location.search).toBe('?query');
|
|
|
|
expect(location.hash).toBe('#hash');
|
|
|
|
});
|
|
|
|
});
|
2017-02-16 13:18:55 -05:00
|
|
|
it('handles empty search and hash portions of the url', () => {
|
|
|
|
platformDynamicServer([{
|
|
|
|
provide: INITIAL_CONFIG,
|
|
|
|
useValue: {document: '<app></app>', url: 'http://test.com/deep/path'}
|
|
|
|
}])
|
|
|
|
.bootstrapModule(ExampleModule)
|
|
|
|
.then(appRef => {
|
|
|
|
const location: PlatformLocation = appRef.injector.get(PlatformLocation);
|
|
|
|
expect(location.pathname).toBe('/deep/path');
|
|
|
|
expect(location.search).toBe('');
|
|
|
|
expect(location.hash).toBe('');
|
|
|
|
});
|
|
|
|
});
|
2017-02-12 12:16:23 -05:00
|
|
|
it('pushState causes the URL to update', async(() => {
|
|
|
|
const platform = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
|
|
|
platform.bootstrapModule(ExampleModule).then(appRef => {
|
|
|
|
const location: PlatformLocation = appRef.injector.get(PlatformLocation);
|
|
|
|
location.pushState(null, 'Test', '/foo#bar');
|
|
|
|
expect(location.pathname).toBe('/foo');
|
|
|
|
expect(location.hash).toBe('#bar');
|
|
|
|
platform.destroy();
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
it('allows subscription to the hash state', done => {
|
|
|
|
const platform =
|
|
|
|
platformDynamicServer([{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
|
|
|
platform.bootstrapModule(ExampleModule).then(appRef => {
|
|
|
|
const location: PlatformLocation = appRef.injector.get(PlatformLocation);
|
|
|
|
expect(location.pathname).toBe('/');
|
|
|
|
location.onHashChange((e: any) => {
|
|
|
|
expect(e.type).toBe('hashchange');
|
|
|
|
expect(e.oldUrl).toBe('/');
|
|
|
|
expect(e.newUrl).toBe('/foo#bar');
|
|
|
|
platform.destroy();
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
location.pushState(null, 'Test', '/foo#bar');
|
|
|
|
});
|
|
|
|
});
|
2017-02-14 19:14:40 -05:00
|
|
|
});
|
|
|
|
|
2017-02-12 12:16:23 -05:00
|
|
|
describe('render', () => {
|
|
|
|
let doc: string;
|
|
|
|
let called: boolean;
|
|
|
|
let expectedOutput =
|
2017-09-09 18:12:13 -04:00
|
|
|
'<html><head></head><body><app ng-version="0.0.0-PLACEHOLDER">Works!<h1 innertext="fine">fine</h1></app></body></html>';
|
2017-02-14 19:14:40 -05:00
|
|
|
|
2017-02-12 12:16:23 -05:00
|
|
|
beforeEach(() => {
|
|
|
|
// PlatformConfig takes in a parsed document so that it can be cached across requests.
|
|
|
|
doc = '<html><head></head><body><app></app></body></html>';
|
|
|
|
called = false;
|
|
|
|
});
|
|
|
|
afterEach(() => { expect(called).toBe(true); });
|
|
|
|
|
2018-02-27 17:06:06 -05:00
|
|
|
it('using long form should work', async(() => {
|
2017-02-12 12:16:23 -05:00
|
|
|
const platform =
|
|
|
|
platformDynamicServer([{provide: INITIAL_CONFIG, useValue: {document: doc}}]);
|
|
|
|
|
|
|
|
platform.bootstrapModule(AsyncServerModule)
|
|
|
|
.then((moduleRef) => {
|
|
|
|
const applicationRef: ApplicationRef = moduleRef.injector.get(ApplicationRef);
|
2018-02-27 17:06:06 -05:00
|
|
|
return applicationRef.isStable.pipe(first((isStable: boolean) => isStable))
|
|
|
|
.toPromise();
|
2017-02-12 12:16:23 -05:00
|
|
|
})
|
|
|
|
.then((b) => {
|
|
|
|
expect(platform.injector.get(PlatformState).renderToString()).toBe(expectedOutput);
|
|
|
|
platform.destroy();
|
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('using renderModule should work', async(() => {
|
|
|
|
renderModule(AsyncServerModule, {document: doc}).then(output => {
|
|
|
|
expect(output).toBe(expectedOutput);
|
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('using renderModuleFactory should work',
|
|
|
|
async(inject([PlatformRef], (defaultPlatform: PlatformRef) => {
|
|
|
|
const compilerFactory: CompilerFactory =
|
|
|
|
defaultPlatform.injector.get(CompilerFactory, null);
|
|
|
|
const moduleFactory =
|
|
|
|
compilerFactory.createCompiler().compileModuleSync(AsyncServerModule);
|
|
|
|
renderModuleFactory(moduleFactory, {document: doc}).then(output => {
|
|
|
|
expect(output).toBe(expectedOutput);
|
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
})));
|
2017-03-14 18:40:55 -04:00
|
|
|
|
|
|
|
it('works with SVG elements', async(() => {
|
|
|
|
renderModule(SVGServerModule, {document: doc}).then(output => {
|
|
|
|
expect(output).toBe(
|
|
|
|
'<html><head></head><body><app ng-version="0.0.0-PLACEHOLDER">' +
|
|
|
|
'<svg><use xlink:href="#clear"></use></svg></app></body></html>');
|
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
}));
|
2017-03-13 20:31:03 -04:00
|
|
|
|
|
|
|
it('works with animation', async(() => {
|
|
|
|
renderModule(AnimationServerModule, {document: doc}).then(output => {
|
2017-08-08 05:17:40 -04:00
|
|
|
expect(output).toContain('Works!');
|
|
|
|
expect(output).toContain('ng-trigger-myAnimation');
|
2017-03-13 20:31:03 -04:00
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
}));
|
2017-03-14 17:09:24 -04:00
|
|
|
|
|
|
|
it('should handle ViewEncapsulation.Native', async(() => {
|
|
|
|
renderModule(NativeExampleModule, {document: doc}).then(output => {
|
|
|
|
expect(output).not.toBe('');
|
|
|
|
expect(output).toContain('color: red');
|
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
}));
|
2017-03-28 16:33:07 -04:00
|
|
|
|
2018-05-28 03:20:53 -04:00
|
|
|
|
|
|
|
it('sets a prefix for the _nghost and _ngcontent attributes', async(() => {
|
|
|
|
renderModule(ExampleStylesModule, {document: doc}).then(output => {
|
|
|
|
expect(output).toMatch(
|
|
|
|
/<html><head><style ng-transition="example-styles">div\[_ngcontent-sc\d+\] {color: blue; } \[_nghost-sc\d+\] { color: red; }<\/style><\/head><body><app _nghost-sc\d+="" ng-version="0.0.0-PLACEHOLDER"><div _ngcontent-sc\d+="">Works!<\/div><\/app><\/body><\/html>/);
|
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
|
2017-03-28 16:33:07 -04:00
|
|
|
it('should handle false values on attributes', async(() => {
|
2017-07-19 16:58:23 -04:00
|
|
|
renderModule(FalseAttributesModule, {document: doc}).then(output => {
|
2017-03-28 16:33:07 -04:00
|
|
|
expect(output).toBe(
|
|
|
|
'<html><head></head><body><app ng-version="0.0.0-PLACEHOLDER">' +
|
|
|
|
'<my-child ng-reflect-attr="false">Works!</my-child></app></body></html>');
|
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
}));
|
2017-07-19 16:58:23 -04:00
|
|
|
|
|
|
|
it('should handle element property "name"', async(() => {
|
|
|
|
renderModule(NameModule, {document: doc}).then(output => {
|
|
|
|
expect(output).toBe(
|
|
|
|
'<html><head></head><body><app ng-version="0.0.0-PLACEHOLDER">' +
|
|
|
|
'<input name=""></app></body></html>');
|
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
}));
|
2017-09-04 03:38:42 -04:00
|
|
|
|
2018-05-24 19:04:04 -04:00
|
|
|
it('should work with sanitizer to handle "innerHTML"', async(() => {
|
|
|
|
// Clear out any global states. These should be set when platform-server
|
|
|
|
// is initialized.
|
|
|
|
(global as any).Node = undefined;
|
|
|
|
(global as any).Document = undefined;
|
|
|
|
renderModule(HTMLTypesModule, {document: doc}).then(output => {
|
|
|
|
expect(output).toBe(
|
|
|
|
'<html><head></head><body><app ng-version="0.0.0-PLACEHOLDER">' +
|
|
|
|
'<div innerhtml="<b>foo</b> bar"><b>foo</b> bar</div></app></body></html>');
|
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
|
2017-09-04 03:38:42 -04:00
|
|
|
it('should call render hook', async(() => {
|
|
|
|
renderModule(RenderHookModule, {document: doc}).then(output => {
|
|
|
|
// title should be added by the render hook.
|
|
|
|
expect(output).toBe(
|
|
|
|
'<html><head><title>RenderHook</title></head><body>' +
|
|
|
|
'<app ng-version="0.0.0-PLACEHOLDER">Works!</app></body></html>');
|
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
|
2018-03-10 12:14:58 -05:00
|
|
|
it('should call multiple render hooks', async(() => {
|
2017-09-04 03:38:42 -04:00
|
|
|
const consoleSpy = spyOn(console, 'warn');
|
|
|
|
renderModule(MultiRenderHookModule, {document: doc}).then(output => {
|
|
|
|
// title should be added by the render hook.
|
|
|
|
expect(output).toBe(
|
|
|
|
'<html><head><title>RenderHook</title><meta name="description"></head>' +
|
|
|
|
'<body><app ng-version="0.0.0-PLACEHOLDER">Works!</app></body></html>');
|
|
|
|
expect(consoleSpy).toHaveBeenCalled();
|
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
}));
|
2017-02-12 12:16:23 -05:00
|
|
|
});
|
2017-02-10 20:00:27 -05:00
|
|
|
|
|
|
|
describe('http', () => {
|
|
|
|
it('can inject Http', async(() => {
|
|
|
|
const platform = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
|
|
|
platform.bootstrapModule(ExampleModule).then(ref => {
|
|
|
|
expect(ref.injector.get(Http) instanceof Http).toBeTruthy();
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
it('can make Http requests', async(() => {
|
|
|
|
const platform = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
|
|
|
platform.bootstrapModule(ExampleModule).then(ref => {
|
|
|
|
const mock = ref.injector.get(MockBackend);
|
|
|
|
const http = ref.injector.get(Http);
|
2017-11-15 11:43:35 -05:00
|
|
|
ref.injector.get<NgZone>(NgZone).run(() => {
|
2017-02-10 20:00:27 -05:00
|
|
|
NgZone.assertInAngularZone();
|
|
|
|
mock.connections.subscribe((mc: MockConnection) => {
|
|
|
|
NgZone.assertInAngularZone();
|
2017-03-21 13:29:13 -04:00
|
|
|
expect(mc.request.url).toBe('http://localhost/testing');
|
2017-02-10 20:00:27 -05:00
|
|
|
mc.mockRespond(new Response(new ResponseOptions({body: 'success!', status: 200})));
|
|
|
|
});
|
2017-03-21 13:29:13 -04:00
|
|
|
http.get('http://localhost/testing').subscribe(resp => {
|
2017-02-10 20:00:27 -05:00
|
|
|
NgZone.assertInAngularZone();
|
|
|
|
expect(resp.text()).toBe('success!');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
it('requests are macrotasks', async(() => {
|
|
|
|
const platform = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
|
|
|
platform.bootstrapModule(ExampleModule).then(ref => {
|
|
|
|
const mock = ref.injector.get(MockBackend);
|
|
|
|
const http = ref.injector.get(Http);
|
2017-11-15 11:43:35 -05:00
|
|
|
expect(ref.injector.get<NgZone>(NgZone).hasPendingMacrotasks).toBeFalsy();
|
|
|
|
ref.injector.get<NgZone>(NgZone).run(() => {
|
2017-02-10 20:00:27 -05:00
|
|
|
NgZone.assertInAngularZone();
|
|
|
|
mock.connections.subscribe((mc: MockConnection) => {
|
2017-11-15 11:43:35 -05:00
|
|
|
expect(ref.injector.get<NgZone>(NgZone).hasPendingMacrotasks).toBeTruthy();
|
2017-02-10 20:00:27 -05:00
|
|
|
mc.mockRespond(new Response(new ResponseOptions({body: 'success!', status: 200})));
|
|
|
|
});
|
2017-03-21 13:29:13 -04:00
|
|
|
http.get('http://localhost/testing').subscribe(resp => {
|
|
|
|
expect(resp.text()).toBe('success!');
|
|
|
|
});
|
2017-02-10 20:00:27 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
it('works when HttpModule is included before ServerModule', async(() => {
|
|
|
|
const platform = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
|
|
|
platform.bootstrapModule(HttpBeforeExampleModule).then(ref => {
|
|
|
|
const mock = ref.injector.get(MockBackend);
|
|
|
|
const http = ref.injector.get(Http);
|
2017-11-15 11:43:35 -05:00
|
|
|
expect(ref.injector.get<NgZone>(NgZone).hasPendingMacrotasks).toBeFalsy();
|
|
|
|
ref.injector.get<NgZone>(NgZone).run(() => {
|
2017-02-10 20:00:27 -05:00
|
|
|
NgZone.assertInAngularZone();
|
|
|
|
mock.connections.subscribe((mc: MockConnection) => {
|
2017-11-15 11:43:35 -05:00
|
|
|
expect(ref.injector.get<NgZone>(NgZone).hasPendingMacrotasks).toBeTruthy();
|
2017-02-10 20:00:27 -05:00
|
|
|
mc.mockRespond(new Response(new ResponseOptions({body: 'success!', status: 200})));
|
|
|
|
});
|
2017-03-21 13:29:13 -04:00
|
|
|
http.get('http://localhost/testing').subscribe(resp => {
|
|
|
|
expect(resp.text()).toBe('success!');
|
|
|
|
});
|
2017-02-10 20:00:27 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
it('works when HttpModule is included after ServerModule', async(() => {
|
|
|
|
const platform = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
|
|
|
platform.bootstrapModule(HttpAfterExampleModule).then(ref => {
|
|
|
|
const mock = ref.injector.get(MockBackend);
|
|
|
|
const http = ref.injector.get(Http);
|
2017-11-15 11:43:35 -05:00
|
|
|
expect(ref.injector.get<NgZone>(NgZone).hasPendingMacrotasks).toBeFalsy();
|
|
|
|
ref.injector.get<NgZone>(NgZone).run(() => {
|
2017-02-10 20:00:27 -05:00
|
|
|
NgZone.assertInAngularZone();
|
|
|
|
mock.connections.subscribe((mc: MockConnection) => {
|
2017-11-15 11:43:35 -05:00
|
|
|
expect(ref.injector.get<NgZone>(NgZone).hasPendingMacrotasks).toBeTruthy();
|
2017-02-10 20:00:27 -05:00
|
|
|
mc.mockRespond(new Response(new ResponseOptions({body: 'success!', status: 200})));
|
|
|
|
});
|
2017-03-21 13:29:13 -04:00
|
|
|
http.get('http://localhost/testing').subscribe(resp => {
|
|
|
|
expect(resp.text()).toBe('success!');
|
|
|
|
});
|
2017-02-10 20:00:27 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}));
|
2017-03-21 13:29:13 -04:00
|
|
|
it('throws when given a relative URL', async(() => {
|
|
|
|
const platform = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
|
|
|
platform.bootstrapModule(ExampleModule).then(ref => {
|
|
|
|
const http = ref.injector.get(Http);
|
|
|
|
expect(() => http.get('/testing'))
|
|
|
|
.toThrowError(
|
|
|
|
'URLs requested via Http on the server must be absolute. URL: /testing');
|
|
|
|
});
|
|
|
|
}));
|
2017-02-10 20:00:27 -05:00
|
|
|
});
|
2017-07-18 15:45:47 -04:00
|
|
|
describe('HttpClient', () => {
|
|
|
|
it('can inject HttpClient', async(() => {
|
|
|
|
const platform = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
2018-03-10 12:14:58 -05:00
|
|
|
platform.bootstrapModule(HttpClientExampleModule).then(ref => {
|
2017-07-18 15:45:47 -04:00
|
|
|
expect(ref.injector.get(HttpClient) instanceof HttpClient).toBeTruthy();
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
it('can make HttpClient requests', async(() => {
|
|
|
|
const platform = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
2018-03-10 12:14:58 -05:00
|
|
|
platform.bootstrapModule(HttpClientExampleModule).then(ref => {
|
2017-07-18 15:45:47 -04:00
|
|
|
const mock = ref.injector.get(HttpTestingController) as HttpTestingController;
|
|
|
|
const http = ref.injector.get(HttpClient);
|
2017-11-15 11:43:35 -05:00
|
|
|
ref.injector.get<NgZone>(NgZone).run(() => {
|
2017-07-18 15:45:47 -04:00
|
|
|
http.get('http://localhost/testing').subscribe(body => {
|
|
|
|
NgZone.assertInAngularZone();
|
|
|
|
expect(body).toEqual('success!');
|
|
|
|
});
|
|
|
|
mock.expectOne('http://localhost/testing').flush('success!');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
it('requests are macrotasks', async(() => {
|
|
|
|
const platform = platformDynamicServer(
|
|
|
|
[{provide: INITIAL_CONFIG, useValue: {document: '<app></app>'}}]);
|
2018-03-10 12:14:58 -05:00
|
|
|
platform.bootstrapModule(HttpClientExampleModule).then(ref => {
|
2017-07-18 15:45:47 -04:00
|
|
|
const mock = ref.injector.get(HttpTestingController) as HttpTestingController;
|
|
|
|
const http = ref.injector.get(HttpClient);
|
2017-11-15 11:43:35 -05:00
|
|
|
ref.injector.get<NgZone>(NgZone).run(() => {
|
2017-07-18 15:45:47 -04:00
|
|
|
http.get('http://localhost/testing').subscribe(body => {
|
|
|
|
expect(body).toEqual('success!');
|
|
|
|
});
|
2017-11-15 11:43:35 -05:00
|
|
|
expect(ref.injector.get<NgZone>(NgZone).hasPendingMacrotasks).toBeTruthy();
|
2017-07-18 15:45:47 -04:00
|
|
|
mock.expectOne('http://localhost/testing').flush('success!');
|
2017-11-15 11:43:35 -05:00
|
|
|
expect(ref.injector.get<NgZone>(NgZone).hasPendingMacrotasks).toBeFalsy();
|
2017-07-18 15:45:47 -04:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
});
|
2017-09-11 03:18:55 -04:00
|
|
|
|
|
|
|
describe('ServerTransferStoreModule', () => {
|
|
|
|
let called = false;
|
|
|
|
const defaultExpectedOutput =
|
|
|
|
'<html><head></head><body><app ng-version="0.0.0-PLACEHOLDER">Works!</app><script id="transfer-state" type="application/json">{&q;test&q;:10}</script></body></html>';
|
|
|
|
|
|
|
|
beforeEach(() => { called = false; });
|
|
|
|
afterEach(() => { expect(called).toBe(true); });
|
|
|
|
|
|
|
|
it('adds transfer script tag when using renderModule', async(() => {
|
|
|
|
renderModule(TransferStoreModule, {document: '<app></app>'}).then(output => {
|
|
|
|
expect(output).toBe(defaultExpectedOutput);
|
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('adds transfer script tag when using renderModuleFactory',
|
|
|
|
async(inject([PlatformRef], (defaultPlatform: PlatformRef) => {
|
|
|
|
const compilerFactory: CompilerFactory =
|
|
|
|
defaultPlatform.injector.get(CompilerFactory, null);
|
|
|
|
const moduleFactory =
|
|
|
|
compilerFactory.createCompiler().compileModuleSync(TransferStoreModule);
|
|
|
|
renderModuleFactory(moduleFactory, {document: '<app></app>'}).then(output => {
|
|
|
|
expect(output).toBe(defaultExpectedOutput);
|
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
})));
|
|
|
|
|
|
|
|
it('cannot break out of <script> tag in serialized output', async(() => {
|
|
|
|
renderModule(EscapedTransferStoreModule, {
|
|
|
|
document: '<esc-app></esc-app>'
|
|
|
|
}).then(output => {
|
|
|
|
expect(output).toBe(
|
|
|
|
'<html><head></head><body><esc-app ng-version="0.0.0-PLACEHOLDER">Works!</esc-app>' +
|
|
|
|
'<script id="transfer-state" type="application/json">' +
|
|
|
|
'{&q;testString&q;:&q;&l;/script&g;&l;script&g;' +
|
|
|
|
'alert(&s;Hello&a;&s; + \\&q;World\\&q;);&q;}</script></body></html>');
|
|
|
|
called = true;
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
});
|
2017-02-14 19:14:40 -05:00
|
|
|
});
|
2017-12-16 17:42:55 -05:00
|
|
|
})();
|