Skip to main content

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}