ANGULAR UNIVERSAL
in
data:image/s3,"s3://crabby-images/f45bf/f45bfe6bda0b11e74eaf144527f99aeb08b43827" alt=""
data:image/s3,"s3://crabby-images/7a9e1/7a9e1046b6d0a12e78bbb89e8564efcd48c87ff5" alt=""
2019
Hi!
I'm
Craig
2019
This is a deep dive!
data:image/s3,"s3://crabby-images/f5e1c/f5e1cd8842f58d0762dc9240700e8656f1f4e3e4" alt=""
@phenomnominal
Assumptions:
You know some Angular
You've heard of Angular Universal
What we're covering:
How does Angular Universal work?
What problems does it solve?
How do we make our applications work with Universal?
Why would you not want to use Universal?
@phenomnominal
2019
how does Angular Universal work?
Angular
universal.app/data/:id
<html>
<link
rel="stylesheet"
href="/static/styles.css">
<script
src="/static/script.js">
<body>
<universal-app>
<div
class="loading >
</div>
</universal-app>
</body>
</html>
rel="stylesheet"
href="/static/styles.css"
src="/static/script.js"
class="loading"
"stylesheet"
"/static/styles.css"
"/static/script.js"
"loading"
Generic application loading state
Static resources
data:image/s3,"s3://crabby-images/c0c06/c0c062876543333e4ef82e21ef11569907dc5af2" alt=""
data:image/s3,"s3://crabby-images/16ad9/16ad9284448d1daca9a251f3e2bb873af0b022c8" alt=""
@phenomnominal
2019
data:image/s3,"s3://crabby-images/c0c06/c0c062876543333e4ef82e21ef11569907dc5af2" alt=""
data:image/s3,"s3://crabby-images/16ad9/16ad9284448d1daca9a251f3e2bb873af0b022c8" alt=""
universal.app/static/script.js
universal.app/static/styles.css
() {
...
}
{
...
}
...
...
function () {
...
}
html {
...
}
Angular
@phenomnominal
2019
data:image/s3,"s3://crabby-images/c0c06/c0c062876543333e4ef82e21ef11569907dc5af2" alt=""
data:image/s3,"s3://crabby-images/16ad9/16ad9284448d1daca9a251f3e2bb873af0b022c8" alt=""
universal.app/api/search.json?query=hi
universal.app/api/data/:id.json
Angular
{
"id": "1234567890", ...
}
{
"results": [{
"id": 1234567890,
"content": "Hi", ...
}]
}
@phenomnominal
2019
What can we do to remove some of those round trips?
data:image/s3,"s3://crabby-images/5f32d/5f32deb7d314ade04b5dce0b03d6fd55030e7078" alt=""
@phenomnominal
2019
data:image/s3,"s3://crabby-images/c0c06/c0c062876543333e4ef82e21ef11569907dc5af2" alt=""
data:image/s3,"s3://crabby-images/16ad9/16ad9284448d1daca9a251f3e2bb873af0b022c8" alt=""
Angular
Universal
universal.app/data/:id
<html>
<link
rel="stylesheet"
href="/static/styles.css">
<script
src="/static/script.js">
<body>
<universal-app>
<ua-homepage>
<ua-search>
<ua-content>
</ua-homepage>
</universal-app>
</body>
<script
id="server-app-state">
</script>
</html>
"stylesheet"
"/static/styles.css"
"/static/script.js"
"server-app-state"
rel=" "
href=" "
src=" "
id=
Real application content!
Inlined server request data!
@phenomnominal
2019
@phenomnominal
What problems does it solve?
Performance
SEO/No JavaScript environments
Social media links
Angular
Universal
2019
Performance
Slower "Time to First Byte" due to server processing time
Faster "First Paint" over client-only application, particularly for mobile devices
This can be mitigated with caching - both on the server & on the client with a service worker
Slower "Time to Interactive" due to client-side rehydration
Faster discovery of resources required for your application
Measure, experiment, and find what works for your app
@phenomnominal
2019
SEO/No JS
Some crawlers have JavaScript capabilities, but who knows how that really works (not me!)
Support for browsers with JavaScript disabled, or old engines.
Check out Igor Minar's talk from ng-conf 2018 for how angular.io handles SEO without Angular Univeral
If you're competing against other websites, server rendering can give you an advantage over client-only SEO
@phenomnominal
2019
data:image/s3,"s3://crabby-images/635a2/635a2a19d2eb258c5641ca57a088fa7dea05e37b" alt=""
data:image/s3,"s3://crabby-images/d31af/d31afa742a61ebb775e01978adafa226c2dbc8c9" alt=""
data:image/s3,"s3://crabby-images/eb115/eb115f280263a45826efd73a7a7c8ff4c7d50467" alt=""
Social links
data:image/s3,"s3://crabby-images/0021c/0021c7237678f4dc2b7130c419321fcfe39b8bef" alt=""
Pre-rendered content for Facebook, Twitter, etc.
@phenomnominal
2019
data:image/s3,"s3://crabby-images/676f0/676f0d9a6079ad50b77d3ad3f563461aa70904aa" alt=""
data:image/s3,"s3://crabby-images/676f0/676f0d9a6079ad50b77d3ad3f563461aa70904aa" alt=""
@phenomnominal
2019
data:image/s3,"s3://crabby-images/0b486/0b486ca87485fd81b573448e3d8cee84018fd0c8" alt=""
data:image/s3,"s3://crabby-images/0b486/0b486ca87485fd81b573448e3d8cee84018fd0c8" alt=""
data:image/s3,"s3://crabby-images/0b486/0b486ca87485fd81b573448e3d8cee84018fd0c8" alt=""
data:image/s3,"s3://crabby-images/b0b50/b0b503cf375db83652950e7c9413cf6e923c1370" alt=""
data:image/s3,"s3://crabby-images/147d3/147d3fe23567578a7ca7fbce9eb09f952ea1dd47" alt=""
data:image/s3,"s3://crabby-images/bfb55/bfb55d1b595b94359af32ad74102c9712b506143" alt=""
data:image/s3,"s3://crabby-images/22f52/22f524ba226e7586c6b2ad137312f4bdb5769571" alt=""
data:image/s3,"s3://crabby-images/22f52/22f524ba226e7586c6b2ad137312f4bdb5769571" alt=""
data:image/s3,"s3://crabby-images/22f52/22f524ba226e7586c6b2ad137312f4bdb5769571" alt=""
@phenomnominal
2019
How do we make our
applications work with Angular Universal?
REAL WORLD!
@phenomnominal
2019
data:image/s3,"s3://crabby-images/1a76a/1a76a2b809809cf141ecf41346797fabc48201d4" alt=""
data:image/s3,"s3://crabby-images/a5d77/a5d771b57589f7fefcd96c7b0715c093c6b56ff5" alt=""
@phenomnominal
2019
data:image/s3,"s3://crabby-images/9b10e/9b10e03e9de172e181716fa8171e72b923a0c6bf" alt=""
We can rebuild it!
@phenomnominal
2019
@phenomnominal
2019
Angular CLI
ng new the-real-world && cd the-real-world
ng add @nguniversal/express-engine --clientProject the-real-world
npm run build:ssr
npm run serve:ssr
@phenomnominal
2019
Demo!
@phenomnominal
2019
A new app Angular CLI app
Build steps for two different versions of the app
All the necessary Angular Universal dependencies
A new server.ts file which contains the server code
A modified app.module.ts that uses withServerTransition
A new app.server.module.ts file for the server application
A main.ts that waits for DOMContentLoaded before bootstrap
A new main.server.ts file for the server bootstrap
@phenomnominal
2019
Build steps for server
data:image/s3,"s3://crabby-images/0eb27/0eb2759c8471811ffe47d37b296493052ecdf9f9" alt=""
JavaScript runtime (no DOM)
Driven by Chrome's V8 engine
data:image/s3,"s3://crabby-images/fab73/fab73a3bd688e7615bce0f5381fd14149d6be649" alt=""
Use everyone else's code via
data:image/s3,"s3://crabby-images/5c5c6/5c5c6140b6bd32693ce558a4e0743a86c151f831" alt=""
data:image/s3,"s3://crabby-images/c7e17/c7e17696b55d0a62daec1a81eadae13776f22a97" alt=""
@phenomnominal
2019
Require the built Universal app
Set up the engine for running the application
Pass all requests to the engine
Create the Express server
Start the server
@phenomnominal
2019
Node.js
const app = express();
const {AppServerModuleNgFactory, LAZY_MODULE_MAP} = require('./dist/server/main');
app.engine('html', ngExpressEngine({
bootstrap: AppServerModuleNgFactory,
providers: [
provideModuleMap(LAZY_MODULE_MAP)
]
}));
app.set('view engine', 'html');
app.set('views', DIST_FOLDER);
app.get('*.*', express.static(DIST_FOLDER, {
maxAge: '1y'
}));
app.get('*', (req, res) => {
res.render('index', { req });
});
app.listen(PORT, () => {
console.log(`Node Express server listening on http://localhost:${PORT}`);
});
@phenomnominal
2019
TIME TRAVEL
@phenomnominal
2019
Demo!
@phenomnominal
2019
⭐️ @angular/cli application
⭐️ Beautiful 90s inspired table layouts
⭐️ Cast member data requested from API
⭐️ D3 rendered SVG world map
⭐️ @ngrx/store + router-state + entity
😔 Client render only
@phenomnominal
2019
document is not defined
@phenomnominal
2019
data:image/s3,"s3://crabby-images/70620/706201118f6aa7d1d98b3fd3f8833087c03dbfbc" alt=""
How do we run this on the server?
data:image/s3,"s3://crabby-images/cf00a/cf00a991e78c6493a6a2c5ab2d88c115de85e8f1" alt=""
@phenomnominal
2019
Run your Angular application anywhere...
https://www.youtube.com/watch?v=_trUBHaUAR0
@phenomnominal
2019
Angular
Universal
anywhere!
@phenomnominal
2019
@angular/platform-browser
@angular/platform-browser-dynamic
@angular/platform-server
@angular/platform-webworker // Gone!!
@angular/platform-webworker-dynamic // Gone!!
@angular/platform-terminal // !!!
How do we run this anywhere?
@phenomnominal
2019
One implementation for all environments
One implementation for a specific environment
Different implementations for each environment
Different functionality for each environment
Four patterns
data:image/s3,"s3://crabby-images/b24c5/b24c52e5b502ec4b335189a783688e13ac9432d9" alt=""
@phenomnominal
2019
data:image/s3,"s3://crabby-images/870e7/870e744c26d51ddf646af5820d1f34d1f282f2bf" alt=""
data:image/s3,"s3://crabby-images/9dcb5/9dcb5ff631307c3cdbc6b591021aa02ccd32672d" alt=""
data:image/s3,"s3://crabby-images/b12ce/b12ce9b2e6d2ff08419776c22eb1844af6e3eaf6" alt=""
Different implementations for each environment
@phenomnominal
2019
data:image/s3,"s3://crabby-images/b24c5/b24c52e5b502ec4b335189a783688e13ac9432d9" alt=""
Platform-specific modules
Remember app.module.ts & app.server.module.ts?
@phenomnominal
2019
// app.module.ts
import { NgModule } from '@angular/core';
import { BrowserModule } from '@angular/platform-browser';
import { RouterModule } from '@angular/router';
import { AppComponent } from './app.component';
@NgModule({
bootstrap: [AppComponent],
declarations: [AppComponent],
imports: [
BrowserModule.withServerTransition({ appId: 'serverApp' }),
RouterModule.forRoot([{
// routes
}], {
initialNavigation: 'enabled'
})
]
})
export class AppModule { }
// app.server.module.ts
import { NgModule } from '@angular/core';
import { ServerModule } from '@angular/platform-server';
import { ModuleMapLoaderModule } from '@nguniversal/module-map-ngfactory-loader';
import { AppModule } from './app.module';
import { AppComponent } from './app.component';
@NgModule({
bootstrap: [AppComponent],
imports: [
AppModule,
ModuleMapLoaderModule,
ServerModule
]
})
export class AppServerModule { }
Services
Provide different implementations for different environments:
Server implementation for Renderer
Server implementation for HttpClient
@phenomnominal
2019
// @angular/platform-server
@NgModule({
exports: [BrowserModule],
imports: [HttpModule, HttpClientModule, NoopAnimationsModule],
providers: [
SERVER_RENDER_PROVIDERS,
SERVER_HTTP_PROVIDERS,
// ...
]
})
export class ServerModule { }
One implementation for all environments
@phenomnominal
2019
data:image/s3,"s3://crabby-images/870e7/870e744c26d51ddf646af5820d1f34d1f282f2bf" alt=""
Avoiding the DOM
data:image/s3,"s3://crabby-images/c056b/c056b97859de86429fa552740cf9945497b4248f" alt=""
Do not use the DOM directly! There's almost always a way to do the same thing with Angular's abstractions.
@phenomnominal
2019
Components
@Input, @Output, @HostBinding, @ViewChild, @ContentChild, ngStyle, ngClass.
You may occasionally need to reach for the Renderer:
Normal dependency injection
@phenomnominal
2019
import { Component, Renderer2 } from '@angular/core'
@Component({
// ...
})
export class MyComponent {
constructor (
private _renderer: Renderer2
) { }
}
But what if you have to use the DOM?
data:image/s3,"s3://crabby-images/47d2a/47d2a766ab48211075f5ab83aa91a79f5be3a7d4" alt=""
@phenomnominal
2019
Be explicit about it!
Be defensive!
Explicit null type is good
Only run code if window is available.
@phenomnominal
2019
@Injectable({
providedIn: 'root'
})
export class WindowRef<T extends Window = Window> {
private readonly _window: T | null;
constructor () {
this._window = this._getWindow();
}
public useWindow <U> (handler: (window: T) => U): U | null {
return this._window ? handler(this._window): null;
}
public _getWindow (): T | null {
return typeof window !== 'undefined' ? window as T : null;
}
}
Provide an alternative
Use custom services to abstract around DOM APIs.
Fall back to the actual DOM
Optional injection token!
Use the optional value
@phenomnominal
2019
export const URL_LOCATION_TOKEN = new InjectionToken('URL_LOCATION');
@Injectable({
// ...
})
export class UrlLocationService {
constructor (
private _window: WindowRef,
@Optional() @Inject(URL_LOCATION_TOKEN) private _location: Location
) {
this._location = this._location || this._window.useWindow(w => w.location);
}
public getHostname (): string {
return this._location.hostname;
}
}
Hack it?
Some third-party code won't want to play nicely...
Set up globals
Require troublesome module
Clean up after
Directly check for window
@phenomnominal
2019
declare var global: any
if (typeof window === 'undefined') {
(global).window = {};
(global).requestAnimationFrame = () => void 0;
require('../src/cursor.js');
delete (global).window;
delete (global).requestAnimationFrame;
}
Check the platform...
data:image/s3,"s3://crabby-images/edab5/edab5580e1b088e0ac088cfeddb1a8e690c31d75" alt=""
As a last resort (please!)
@phenomnominal
2019
If you really must...
Injectable PLATFORM_ID token
Use the method for the specific platform
Check the platform...
@phenomnominal
2019
import { isPlatformBrowser, isPlatformServer } from '@angular/common';
import { Inject, PLATFORM_ID } from '@angular/core';
@Component({
// ...
})
export class MyComponent {
constructor (
@Inject(PLATFORM_ID) private _platformId: Object
) { }
public myExpensiveDomHeavyOperation (): void {
if (!isPlatformBrowser(this._platformId) {
return;
}
this._useTheDom();
}
}
One implementation for a specific environment
@phenomnominal
2019
data:image/s3,"s3://crabby-images/9dcb5/9dcb5ff631307c3cdbc6b591021aa02ccd32672d" alt=""
Add @Optional injection annotation
Trusty if statement
Check for the presence of an injected service
Just turn it off
@phenomnominal
2019
@Component({
// ...
})
export class MapComponent implements AfterViewInit {
@ViewChild('map', { static: true }) public map: ElementRef;
constructor (
@Optional() private _mapService: MapService
) { }
public ngAfterViewInit (): void {
if (!this._mapService) {
return;
}
this._init();
}
}
Turn things off at a provider level
Provide null for specific platform
Corresponding injection token
@phenomnominal
2019
Just turn it off
@NgModule({
imports: [
// ...
],
bootstrap: [AppComponent],
providers: [
{ provide: VideoService, useValue: null }
]
})
export class AppServerModule { }
@phenomnominal
2019
Demo!
@phenomnominal
2019
@phenomnominal
2019
😕 data requested from API two times
⭐️ No errors!
😕 flickery re-render at after bootstrap
⭐️ Fast!
@phenomnominal
2019
Different functionality for each environment
data:image/s3,"s3://crabby-images/b12ce/b12ce9b2e6d2ff08419776c22eb1844af6e3eaf6" alt=""
The whole application runs twice.
This means duplicate API calls!
This call happens twice
@phenomnominal
2019
State Transfer
@Component({
// ...
})
export class AppComponent implements OnInit {
public castMember: CastMember;
constructor (
private _castMemberDataService: CastMemberDataService
) { }
public ngOnInit (): void {
this._castMemberDataService.getCastMember()
.subscribe(data => this.castMember = data);
}
}
Inject TransferState
Create a state key
Set the state on the first run
Re-use it
@phenomnominal
2019
State Transfer
import { TransferState, makeStateKey } from '@angular/platform-browser';
const CAST_MEMBER = makeStateKey('castMember');
@Component({
// ...
})
export class AppComponent implements OnInit {
public castMember: CastMember
constructor (
private _castMemberDataService: CastMemberDataService,
private _state: TransferState
) { }
public ngOnInit (): void {
this.castMember = this.state.get(CAST_MEMBER, null);
if (!this.castMember) {
this._castMemberDataService.getCastMember()
.subscribe(data => {
this.castMember = data;
this.state.set(CAST_MEMBER, data);
});
}
}
}
Or just transfer your whole store state!
One state key for your whole store
Read the whole store and serialise
@phenomnominal
2019
State Transfer
@NgModule({
imports: [BrowserTransferStateModule]
})
export class ServerStateModule {
constructor (
private _transferState: TransferState,
private _store: Store<State>
) {
this._transferState.onSerialize(STATE_KEY, () => {
let stateToTransfer = null;
this._store.pipe(
select(state => state),
take(1)
).subscribe(state => stateToTransfer = state);
return stateToTransfer;
});
}
}
Use the same key on the client
Dispatch a new action with the whole serialised store
@phenomnominal
2019
State Transfer
Or just transfer your whole store state!
@NgModule({
imports: [BrowserTransferStateModule]
})
export class ClientStateModule {
constructor (
private _transferState: TransferState,
private _store: Store<State>
) {
if (this._transferState.hasKey(STATE_KEY)) {
const state = this._transferState.get<State>(STATE_KEY, {});
this._transferState.remove(STATE_KEY);
this.store.dispatch(new TransferStateAction(state));
}
}
}
Use a simple cache layer to prevent duplicated calls
Split the GET efffect into two parts
Mark each API response with a cache time
Only GET if the cache has expired
@phenomnominal
2019
State Transfer
@Effect()
public getCastMemberDataEffect$ = this._actions.pipe(
ofType<GetCastMemberDataAction>(GetCastMemberDataAction.TYPE),
switchMap(action =>
this._store.pipe(
select(CastMemberDataSelectors.currentCastMemberData),
take(1),
filter(storeItem => this._cacheService.shouldFetch(storeItem, { expiryTime: 120000 })),
map(() => new GetCastMemberDataFromApi(action.options)
)
)
@Effect()
public getCastMemberDataFromApiEffect$ = this._actions.pipe(
ofType<GetCastMemberDataFromApi>(GetCastMemberDataFromApi.TYPE),
switchMap(action =>
this._castMemberDataService.getAsteroidData(action.options).pipe(
map(() => new GetCastMemberDataFromApiSucccess(action.options, response, new Data().toString())),
catchError(err => of(new GetCastMembersDataFromApiFail(actions, options, err)))
)
)
The state transfer data is inlined into the HTML response as a blob of JSON. This means...
Don't send too much data!
The whole store must be JSON serialisable!
@phenomnominal
2019
State Transfer
@phenomnominal
2019
Demo!
@phenomnominal
2019
One implementation for all environments
One implementation for a specific environment
Different implementations for each environment
data:image/s3,"s3://crabby-images/b24c5/b24c52e5b502ec4b335189a783688e13ac9432d9" alt=""
data:image/s3,"s3://crabby-images/870e7/870e744c26d51ddf646af5820d1f34d1f282f2bf" alt=""
data:image/s3,"s3://crabby-images/9dcb5/9dcb5ff631307c3cdbc6b591021aa02ccd32672d" alt=""
Different functionality for each environment
data:image/s3,"s3://crabby-images/b12ce/b12ce9b2e6d2ff08419776c22eb1844af6e3eaf6" alt=""
COOL.
@phenomnominal
2019
Why wouldn't you use Angular Universal?
@phenomnominal
2019
Performance
Universal is not a magical silver rocket.
If performance is your only goal, just ship less JavaScript
It can definitely make things worse if you're not careful!
@phenomnominal
2019
Complexity
You now need a "real" server, not just static files.
Harder to get working, harder mental model, harder to get people shipping value
There will be lots more code - logging, monitoring
@phenomnominal
2019
It's very slow...
Bazel will fix it?
@phenomnominal
2019
WorkFlow/DX
data:image/s3,"s3://crabby-images/99fcd/99fcded009c98edd4ae70f3d76a8b3832756e8af" alt=""
So it's just a dumpster fire?
Lint rules for using DOM APIs
Lint rules for using isBrowser
Meta-reducer for unserialisable data
Test for too much serialised data
Universal E2E tests
Fix it before you break it
@phenomnominal
2019
data:image/s3,"s3://crabby-images/20e87/20e878539fdeb81e1576801d3ee4a39d847fe1f5" alt=""
data:image/s3,"s3://crabby-images/995c1/995c1981301542e9d59b97fb14d66b7d5e6e3794" alt=""
data:image/s3,"s3://crabby-images/ea743/ea7432845553cc0a0db500b9be3a47bf343d4150" alt=""
data:image/s3,"s3://crabby-images/3ead3/3ead38e4dbeadeac59a0b8be2f7a24e98876182e" alt=""
data:image/s3,"s3://crabby-images/b489f/b489f601984bf26db4265f06f4c941f5df55afa6" alt=""
@phenomnominal
2019
data:image/s3,"s3://crabby-images/dd3a9/dd3a9f49acf4a98219a468b51fff26f541d0fb3a" alt=""
Should i Use Angular UniversaL?
If you want/need your Angular app to be crawled by Search Engines
@phenomnominal
2019
If you want/need improved No JS/old browser support
@phenomnominal
2019
data:image/s3,"s3://crabby-images/19ff7/19ff703e97c7ca80e8a7b8db67ec8445cbf487bc" alt=""
data:image/s3,"s3://crabby-images/6b673/6b673a25b2998fbfe46d3f1ede46d36fc92a5a4e" alt=""
data:image/s3,"s3://crabby-images/902fc/902fc07e92bf33be5c5d05d69a4af202293f5985" alt=""
If you want/need content previews for Social Media
@phenomnominal
2019
data:image/s3,"s3://crabby-images/2eeea/2eeeaa6db0283bdcad16d68164443bdfeb4940a6" alt=""
data:image/s3,"s3://crabby-images/e7839/e7839363c913c8c71cc3822ca25e791fd972b982" alt=""
data:image/s3,"s3://crabby-images/d4458/d4458f8fe44f510696b436dbcf666c25e599ea96" alt=""
data:image/s3,"s3://crabby-images/d53c6/d53c611258760de809edf3dc3648ace6f9d1478f" alt=""
data:image/s3,"s3://crabby-images/eba20/eba20caac783085dd0bb1715e5883d5813835af6" alt=""
data:image/s3,"s3://crabby-images/21ab2/21ab2f74f9933870e4b2b0fab547df6be391131b" alt=""
AND if you're willing to fight for better perceived performance
@phenomnominal
2019
data:image/s3,"s3://crabby-images/921e5/921e52a592a2758ff389ad83cc1770cb990579d6" alt=""
Angular
Universal
data:image/s3,"s3://crabby-images/a0680/a06809e50b490a04e051216c0f5f99893cad52c6" alt=""
data:image/s3,"s3://crabby-images/a0680/a06809e50b490a04e051216c0f5f99893cad52c6" alt=""
data:image/s3,"s3://crabby-images/a0680/a06809e50b490a04e051216c0f5f99893cad52c6" alt=""
data:image/s3,"s3://crabby-images/a0680/a06809e50b490a04e051216c0f5f99893cad52c6" alt=""
@phenomnominal
2019
data:image/s3,"s3://crabby-images/fb17e/fb17ee23e6702fc70febeaa1de71ab0734b4d130" alt=""
data:image/s3,"s3://crabby-images/fb17e/fb17ee23e6702fc70febeaa1de71ab0734b4d130" alt=""
data:image/s3,"s3://crabby-images/fb17e/fb17ee23e6702fc70febeaa1de71ab0734b4d130" alt=""
data:image/s3,"s3://crabby-images/fb17e/fb17ee23e6702fc70febeaa1de71ab0734b4d130" alt=""
data:image/s3,"s3://crabby-images/5cff2/5cff29f6b01978997b7cb98380dbf2dbd60afaee" alt=""
data:image/s3,"s3://crabby-images/5cff2/5cff29f6b01978997b7cb98380dbf2dbd60afaee" alt=""
data:image/s3,"s3://crabby-images/fb17e/fb17ee23e6702fc70febeaa1de71ab0734b4d130" alt=""
data:image/s3,"s3://crabby-images/5cff2/5cff29f6b01978997b7cb98380dbf2dbd60afaee" alt=""
@phenomnominal
2019
data:image/s3,"s3://crabby-images/db02c/db02cc55c6798b238b5991584e05a473eb03c810" alt=""
myspace/phenomnominal
geocities/phenomnomnominal
Angular Universal in the Real World
By Craig Spence
Angular Universal in the Real World
Craig Spence - Angular Denver 2019 - Angular Universal in the Real World
- 3,138