next_core/next_app/app_client_references_chunks.rs
1use anyhow::Result;
2use tracing::Instrument;
3use turbo_rcstr::rcstr;
4use turbo_tasks::{
5 FxIndexMap, FxIndexSet, ResolvedVc, TryFlatJoinIterExt, TryJoinIterExt, ValueToStringRef, Vc,
6};
7use turbopack_core::{
8 chunk::{ChunkGroupResult, ChunkingContext, availability_info::AvailabilityInfo},
9 module::Module,
10 module_graph::{ModuleGraph, chunk_group_info::ChunkGroup},
11 output::{OutputAsset, OutputAssets, OutputAssetsWithReferenced},
12};
13
14use crate::{
15 next_client_reference::{
16 ClientReferenceType,
17 ecmascript_client_reference::ecmascript_client_reference_module::{
18 ecmascript_client_reference_merge_tag, ecmascript_client_reference_merge_tag_ssr,
19 },
20 visit_client_reference::ClientReferenceGraphResult,
21 },
22 next_server_component::server_component_module::NextServerComponentModule,
23};
24
25#[turbo_tasks::value]
26pub struct ClientReferencesChunks {
27 #[bincode(with = "turbo_bincode::indexmap")]
28 pub client_component_client_chunks:
29 FxIndexMap<ClientReferenceType, ResolvedVc<ChunkGroupResult>>,
30 #[bincode(with = "turbo_bincode::indexmap")]
31 pub client_component_ssr_chunks:
32 FxIndexMap<ClientReferenceType, ResolvedVc<OutputAssetsWithReferenced>>,
33 #[bincode(with = "turbo_bincode::indexmap")]
34 pub layout_segment_client_chunks:
35 FxIndexMap<ResolvedVc<NextServerComponentModule>, ResolvedVc<OutputAssetsWithReferenced>>,
36}
37
38/// Computes all client references chunks.
39///
40/// This returns a map from client reference type to the chunks that the reference
41/// type needs to load.
42#[turbo_tasks::function]
43pub async fn get_app_client_references_chunks(
44 app_client_references: Vc<ClientReferenceGraphResult>,
45 module_graph: Vc<ModuleGraph>,
46 client_chunking_context: Vc<Box<dyn ChunkingContext>>,
47 client_availability_info: AvailabilityInfo,
48 ssr_chunking_context: Option<Vc<Box<dyn ChunkingContext>>>,
49) -> Result<Vc<ClientReferencesChunks>> {
50 async move {
51 // TODO Reconsider this. Maybe it need to be true in production.
52 let separate_chunk_group_per_client_reference = false;
53 let app_client_references = app_client_references.await?;
54 if separate_chunk_group_per_client_reference {
55 todo!();
56 // let app_client_references_chunks: Vec<(_, (_, Option<_>))> = app_client_references
57 // .client_references
58 // .iter()
59 // .map(|client_reference| async move {
60 // Ok((
61 // client_reference.ty,
62 // match client_reference.ty {
63 // ClientReferenceType::EcmascriptClientReference(
64 // ecmascript_client_reference,
65 // ) => {
66 // let ecmascript_client_reference_ref =
67 // ecmascript_client_reference.await?;
68
69 // let client_chunk_group = client_chunking_context
70 // .root_chunk_group(
71 // module_graph,
72 // *ResolvedVc::upcast(
73 // ecmascript_client_reference_ref.client_module,
74 // ),
75 // )
76 // .await?;
77
78 // (
79 // (
80 // client_chunk_group.assets,
81 // client_chunk_group.availability_info,
82 // ),
83 // if let Some(ssr_chunking_context) = ssr_chunking_context {
84 // let ssr_chunk_group = ssr_chunking_context
85 // .root_chunk_group(
86 // *ResolvedVc::upcast(
87 // ecmascript_client_reference_ref.ssr_module,
88 // ),
89 // module_graph,
90 // )
91 // .await?;
92
93 // Some((
94 // ssr_chunk_group.assets,
95 // ssr_chunk_group.availability_info,
96 // ))
97 // } else {
98 // None
99 // },
100 // )
101 // }
102 // ClientReferenceType::CssClientReference(css_client_reference) => {
103 // let client_chunk_group = client_chunking_context
104 // .root_chunk_group(
105 // *ResolvedVc::upcast(css_client_reference),
106 // module_graph,
107 // )
108 // .await?;
109
110 // (
111 // (
112 // client_chunk_group.assets,
113 // client_chunk_group.availability_info,
114 // ),
115 // None,
116 // )
117 // }
118 // },
119 // ))
120 // })
121 // .try_join()
122 // .await?;
123
124 // Ok(ClientReferencesChunks {
125 // client_component_client_chunks: app_client_references_chunks
126 // .iter()
127 // .map(|&(client_reference_ty, (client_chunks, _))| {
128 // (client_reference_ty, client_chunks)
129 // })
130 // .collect(),
131 // client_component_ssr_chunks: app_client_references_chunks
132 // .iter()
133 // .flat_map(|&(client_reference_ty, (_, ssr_chunks))| {
134 // ssr_chunks.map(|ssr_chunks| (client_reference_ty, ssr_chunks))
135 // })
136 // .collect(),
137 // layout_segment_client_chunks: FxIndexMap::default(),
138 // }
139 // .cell())
140 } else {
141 let mut client_references_by_server_component: FxIndexMap<_, Vec<_>> =
142 FxIndexMap::default();
143 let mut framework_reference_types = Vec::new();
144 for &server_component in app_client_references.server_component_entries.iter() {
145 client_references_by_server_component
146 .entry(server_component)
147 .or_default();
148 }
149 for client_reference in app_client_references.client_references.iter() {
150 if let Some(server_component) = client_reference.server_component {
151 client_references_by_server_component
152 .entry(server_component)
153 .or_default()
154 .push(client_reference.ty);
155 } else {
156 framework_reference_types.push(client_reference.ty);
157 }
158 }
159 // Framework components need to go into first layout segment
160 if let Some((_, list)) = client_references_by_server_component.first_mut() {
161 list.extend(framework_reference_types);
162 }
163
164 let chunk_group_info = module_graph.chunk_group_info();
165
166 let mut current_client_chunk_group = ChunkGroupResult {
167 assets: ResolvedVc::cell(vec![]),
168 referenced_assets: ResolvedVc::cell(vec![]),
169 references: ResolvedVc::cell(vec![]),
170 availability_info: client_availability_info,
171 }
172 .resolved_cell();
173 let mut current_ssr_chunk_group = ChunkGroupResult::empty_resolved();
174
175 let mut layout_segment_client_chunks = FxIndexMap::default();
176 let mut client_component_ssr_chunks = FxIndexMap::default();
177 let mut client_component_client_chunks = FxIndexMap::default();
178
179 for (server_component, client_reference_types) in
180 client_references_by_server_component.into_iter()
181 {
182 let parent_chunk_group = *chunk_group_info
183 .get_index_of(ChunkGroup::Shared(ResolvedVc::upcast(server_component)))
184 .await?;
185
186 let base_ident = server_component.ident().owned().await?;
187
188 let server_path = server_component.server_path().owned().await?;
189 let is_layout = server_path.file_stem() == Some("layout");
190 let server_component_path = server_path.to_string_ref().await?;
191
192 let ssr_modules = client_reference_types
193 .iter()
194 .map(|client_reference_ty| async move {
195 Ok(match client_reference_ty {
196 ClientReferenceType::EcmascriptClientReference(
197 ecmascript_client_reference,
198 ) => {
199 let ecmascript_client_reference_ref =
200 ecmascript_client_reference.await?;
201
202 Some(ResolvedVc::upcast(
203 ecmascript_client_reference_ref.ssr_module,
204 ))
205 }
206 _ => None,
207 })
208 })
209 .try_flat_join()
210 .await?;
211
212 let ssr_chunk_group = if !ssr_modules.is_empty()
213 && let Some(ssr_chunking_context) = ssr_chunking_context
214 {
215 let availability_info = current_ssr_chunk_group.await?.availability_info;
216 let _span = tracing::info_span!(
217 "server side rendering",
218 layout_segment = display(&server_component_path),
219 )
220 .entered();
221
222 Some(
223 ssr_chunking_context.chunk_group(
224 base_ident
225 .clone()
226 .with_modifier(rcstr!("ssr modules"))
227 .into_vc(),
228 ChunkGroup::IsolatedMerged {
229 parent: parent_chunk_group,
230 merge_tag: ecmascript_client_reference_merge_tag_ssr(),
231 entries: ssr_modules,
232 },
233 module_graph,
234 availability_info,
235 ),
236 )
237 } else {
238 None
239 };
240
241 let client_modules = client_reference_types
242 .iter()
243 .map(|client_reference_ty| async move {
244 Ok(match client_reference_ty {
245 ClientReferenceType::EcmascriptClientReference(
246 ecmascript_client_reference,
247 ) => {
248 ResolvedVc::upcast(ecmascript_client_reference.await?.client_module)
249 }
250 ClientReferenceType::CssClientReference(css_client_reference) => {
251 ResolvedVc::upcast(*css_client_reference)
252 }
253 })
254 })
255 .try_join()
256 .await?;
257 let client_chunk_group = if !client_modules.is_empty() {
258 let availability_info = current_client_chunk_group.await?.availability_info;
259 let _span = tracing::info_span!(
260 "client side rendering",
261 layout_segment = display(&server_component_path),
262 )
263 .entered();
264
265 Some(client_chunking_context.chunk_group(
266 base_ident.with_modifier(rcstr!("client modules")).into_vc(),
267 ChunkGroup::IsolatedMerged {
268 parent: parent_chunk_group,
269 merge_tag: ecmascript_client_reference_merge_tag(),
270 entries: client_modules,
271 },
272 module_graph,
273 availability_info,
274 ))
275 } else {
276 None
277 };
278
279 if let Some(client_chunk_group) = client_chunk_group {
280 let client_chunk_group = current_client_chunk_group
281 .concatenate(client_chunk_group)
282 .to_resolved()
283 .await?;
284
285 if is_layout {
286 current_client_chunk_group = client_chunk_group;
287 }
288
289 let assets = client_chunk_group
290 .output_assets_with_referenced()
291 .to_resolved()
292 .await?;
293 layout_segment_client_chunks.insert(server_component, assets);
294
295 for &client_reference_ty in client_reference_types.iter() {
296 if let ClientReferenceType::EcmascriptClientReference(_) =
297 client_reference_ty
298 {
299 client_component_client_chunks
300 .insert(client_reference_ty, client_chunk_group);
301 }
302 }
303 }
304
305 if let Some(ssr_chunk_group) = ssr_chunk_group {
306 let ssr_chunk_group = current_ssr_chunk_group
307 .concatenate(ssr_chunk_group)
308 .to_resolved()
309 .await?;
310
311 if is_layout {
312 current_ssr_chunk_group = ssr_chunk_group;
313 }
314
315 let assets = ssr_chunk_group
316 .output_assets_with_referenced()
317 .to_resolved()
318 .await?;
319 for &client_reference_ty in client_reference_types.iter() {
320 if let ClientReferenceType::EcmascriptClientReference(_) =
321 client_reference_ty
322 {
323 client_component_ssr_chunks.insert(client_reference_ty, assets);
324 }
325 }
326 }
327 }
328
329 Ok(ClientReferencesChunks {
330 client_component_client_chunks,
331 client_component_ssr_chunks,
332 layout_segment_client_chunks,
333 }
334 .cell())
335 }
336 }
337 .instrument(tracing::info_span!("process client references"))
338 .await
339}
340
341/// Flattens all client-side output assets from `client_references_chunks` so the
342/// page's HMR chunk list can subscribe to updates for chunks built outside the
343/// entry's own module graph (each `chunk_group(IsolatedMerged)` call for a
344/// client component group generates chunks separately).
345#[turbo_tasks::function]
346pub async fn get_client_references_chunks_for_hmr(
347 client_references_chunks: Vc<ClientReferencesChunks>,
348) -> Result<Vc<OutputAssets>> {
349 let client_references_chunks_ref = client_references_chunks.await?;
350 let mut extras: FxIndexSet<ResolvedVc<Box<dyn OutputAsset>>> = client_references_chunks_ref
351 .layout_segment_client_chunks
352 .values()
353 .map(|&assets| async move {
354 let primary = assets.primary_assets().await?;
355 Ok(primary.iter().copied().collect::<Vec<_>>())
356 })
357 .try_flat_join()
358 .await?
359 .into_iter()
360 .collect();
361 for &chunk_group in client_references_chunks_ref
362 .client_component_client_chunks
363 .values()
364 {
365 // Use all_assets() (not primary_assets()) to also follow async loader references
366 // transitively. This ensures that dynamic imports within 'use client' pages are
367 // covered by the page's HMR subscription, not just the page module itself.
368 extras.extend(chunk_group.all_assets().await?.iter().copied());
369 }
370 // client_component_ssr_chunks are intentionally excluded: they run on the server
371 // (Node.js/Edge), not in the browser, so they don't belong in the client HMR chunk list.
372 Ok(Vc::cell(extras.into_iter().collect()))
373}