6
6
* found in the LICENSE file at https://angular.io/license
7
7
*/
8
8
9
- import { join , normalize , strings } from '@angular-devkit/core' ;
9
+ import { isJsonObject , join , normalize , strings } from '@angular-devkit/core' ;
10
10
import {
11
11
Rule ,
12
+ SchematicContext ,
12
13
SchematicsException ,
13
14
Tree ,
14
15
apply ,
@@ -19,6 +20,7 @@ import {
19
20
schematic ,
20
21
url ,
21
22
} from '@angular-devkit/schematics' ;
23
+ import { posix } from 'node:path' ;
22
24
import { Schema as ServerOptions } from '../server/schema' ;
23
25
import { DependencyType , addDependency , readWorkspace , updateWorkspace } from '../utility' ;
24
26
import { JSONFile } from '../utility/json-file' ;
@@ -33,21 +35,24 @@ import { Schema as SSROptions } from './schema';
33
35
34
36
const SERVE_SSR_TARGET_NAME = 'serve-ssr' ;
35
37
const PRERENDER_TARGET_NAME = 'prerender' ;
38
+ const DEFAULT_BROWSER_DIR = 'browser' ;
39
+ const DEFAULT_MEDIA_DIR = 'media' ;
40
+ const DEFAULT_SERVER_DIR = 'server' ;
36
41
37
- async function getOutputPath (
42
+ async function getLegacyOutputPaths (
38
43
host : Tree ,
39
44
projectName : string ,
40
45
target : 'server' | 'build' ,
41
46
) : Promise < string > {
42
47
// Generate new output paths
43
48
const workspace = await readWorkspace ( host ) ;
44
49
const project = workspace . projects . get ( projectName ) ;
45
- const serverTarget = project ?. targets . get ( target ) ;
46
- if ( ! serverTarget || ! serverTarget . options ) {
50
+ const architectTarget = project ?. targets . get ( target ) ;
51
+ if ( ! architectTarget ? .options ) {
47
52
throw new SchematicsException ( `Cannot find 'options' for ${ projectName } ${ target } target.` ) ;
48
53
}
49
54
50
- const { outputPath } = serverTarget . options ;
55
+ const { outputPath } = architectTarget . options ;
51
56
if ( typeof outputPath !== 'string' ) {
52
57
throw new SchematicsException (
53
58
`outputPath for ${ projectName } ${ target } target is not a string.` ,
@@ -57,6 +62,52 @@ async function getOutputPath(
57
62
return outputPath ;
58
63
}
59
64
65
+ async function getApplicationBuilderOutputPaths (
66
+ host : Tree ,
67
+ projectName : string ,
68
+ ) : Promise < { browser : string ; server : string ; base : string } > {
69
+ // Generate new output paths
70
+ const target = 'build' ;
71
+ const workspace = await readWorkspace ( host ) ;
72
+ const project = workspace . projects . get ( projectName ) ;
73
+ const architectTarget = project ?. targets . get ( target ) ;
74
+
75
+ if ( ! architectTarget ?. options ) {
76
+ throw new SchematicsException ( `Cannot find 'options' for ${ projectName } ${ target } target.` ) ;
77
+ }
78
+
79
+ const { outputPath } = architectTarget . options ;
80
+ if ( outputPath === null || outputPath === undefined ) {
81
+ throw new SchematicsException (
82
+ `outputPath for ${ projectName } ${ target } target is undeined or null.` ,
83
+ ) ;
84
+ }
85
+
86
+ const defaultDirs = {
87
+ server : DEFAULT_SERVER_DIR ,
88
+ browser : DEFAULT_BROWSER_DIR ,
89
+ } ;
90
+
91
+ if ( outputPath && isJsonObject ( outputPath ) ) {
92
+ return {
93
+ ...defaultDirs ,
94
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
95
+ ...( outputPath as any ) ,
96
+ } ;
97
+ }
98
+
99
+ if ( typeof outputPath !== 'string' ) {
100
+ throw new SchematicsException (
101
+ `outputPath for ${ projectName } ${ target } target is not a string.` ,
102
+ ) ;
103
+ }
104
+
105
+ return {
106
+ base : outputPath ,
107
+ ...defaultDirs ,
108
+ } ;
109
+ }
110
+
60
111
function addScriptsRule ( { project } : SSROptions , isUsingApplicationBuilder : boolean ) : Rule {
61
112
return async ( host ) => {
62
113
const pkgPath = '/package.json' ;
@@ -66,11 +117,11 @@ function addScriptsRule({ project }: SSROptions, isUsingApplicationBuilder: bool
66
117
}
67
118
68
119
if ( isUsingApplicationBuilder ) {
69
- const distPath = await getOutputPath ( host , project , 'build' ) ;
120
+ const { base , server } = await getApplicationBuilderOutputPaths ( host , project ) ;
70
121
pkg . scripts ??= { } ;
71
- pkg . scripts [ `serve:ssr:${ project } ` ] = `node ${ distPath } / server/server.mjs` ;
122
+ pkg . scripts [ `serve:ssr:${ project } ` ] = `node ${ posix . join ( base , server ) } /server.mjs` ;
72
123
} else {
73
- const serverDist = await getOutputPath ( host , project , 'server' ) ;
124
+ const serverDist = await getLegacyOutputPaths ( host , project , 'server' ) ;
74
125
pkg . scripts = {
75
126
...pkg . scripts ,
76
127
'dev:ssr' : `ng run ${ project } :${ SERVE_SSR_TARGET_NAME } ` ,
@@ -111,15 +162,40 @@ function updateApplicationBuilderTsConfigRule(options: SSROptions): Rule {
111
162
function updateApplicationBuilderWorkspaceConfigRule (
112
163
projectRoot : string ,
113
164
options : SSROptions ,
165
+ { logger } : SchematicContext ,
114
166
) : Rule {
115
167
return updateWorkspace ( ( workspace ) => {
116
168
const buildTarget = workspace . projects . get ( options . project ) ?. targets . get ( 'build' ) ;
117
169
if ( ! buildTarget ) {
118
170
return ;
119
171
}
120
172
173
+ let outputPath = buildTarget . options ?. outputPath ;
174
+ if ( outputPath && isJsonObject ( outputPath ) ) {
175
+ if ( outputPath . browser === '' ) {
176
+ const base = outputPath . base as string ;
177
+ logger . warn (
178
+ `The output location of the browser build has been updated from "${ base } " to "${ posix . join (
179
+ base ,
180
+ DEFAULT_BROWSER_DIR ,
181
+ ) } ".
182
+ You might need to adjust your deployment pipeline.` ,
183
+ ) ;
184
+
185
+ if (
186
+ ( outputPath . media && outputPath . media !== DEFAULT_MEDIA_DIR ) ||
187
+ ( outputPath . server && outputPath . server !== DEFAULT_SERVER_DIR )
188
+ ) {
189
+ delete outputPath . browser ;
190
+ } else {
191
+ outputPath = outputPath . base ;
192
+ }
193
+ }
194
+ }
195
+
121
196
buildTarget . options = {
122
197
...buildTarget . options ,
198
+ outputPath,
123
199
prerender : true ,
124
200
ssr : {
125
201
entry : join ( normalize ( projectRoot ) , 'server.ts' ) ,
@@ -238,23 +314,22 @@ function addDependencies(isUsingApplicationBuilder: boolean): Rule {
238
314
239
315
function addServerFile ( options : ServerOptions , isStandalone : boolean ) : Rule {
240
316
return async ( host ) => {
317
+ const projectName = options . project ;
241
318
const workspace = await readWorkspace ( host ) ;
242
- const project = workspace . projects . get ( options . project ) ;
319
+ const project = workspace . projects . get ( projectName ) ;
243
320
if ( ! project ) {
244
- throw new SchematicsException ( `Invalid project name (${ options . project } )` ) ;
321
+ throw new SchematicsException ( `Invalid project name (${ projectName } )` ) ;
245
322
}
323
+ const isUsingApplicationBuilder =
324
+ project ?. targets ?. get ( 'build' ) ?. builder === Builders . Application ;
246
325
247
- const browserDistDirectory = await getOutputPath ( host , options . project , 'build' ) ;
326
+ const browserDistDirectory = isUsingApplicationBuilder
327
+ ? ( await getApplicationBuilderOutputPaths ( host , projectName ) ) . browser
328
+ : await getLegacyOutputPaths ( host , projectName , 'build' ) ;
248
329
249
330
return mergeWith (
250
331
apply (
251
- url (
252
- `./files/${
253
- project ?. targets ?. get ( 'build' ) ?. builder === Builders . Application
254
- ? 'application-builder'
255
- : 'server-builder'
256
- } `,
257
- ) ,
332
+ url ( `./files/${ isUsingApplicationBuilder ? 'application-builder' : 'server-builder' } ` ) ,
258
333
[
259
334
applyTemplates ( {
260
335
...strings ,
@@ -270,7 +345,7 @@ function addServerFile(options: ServerOptions, isStandalone: boolean): Rule {
270
345
}
271
346
272
347
export default function ( options : SSROptions ) : Rule {
273
- return async ( host ) => {
348
+ return async ( host , context ) => {
274
349
const browserEntryPoint = await getMainFilePath ( host , options . project ) ;
275
350
const isStandalone = isStandaloneApp ( host , browserEntryPoint ) ;
276
351
@@ -289,7 +364,7 @@ export default function (options: SSROptions): Rule {
289
364
} ) ,
290
365
...( isUsingApplicationBuilder
291
366
? [
292
- updateApplicationBuilderWorkspaceConfigRule ( clientProject . root , options ) ,
367
+ updateApplicationBuilderWorkspaceConfigRule ( clientProject . root , options , context ) ,
293
368
updateApplicationBuilderTsConfigRule ( options ) ,
294
369
]
295
370
: [
0 commit comments