From 0d752739a071bd5397ee7c9fbd93f5b8eeed43ec Mon Sep 17 00:00:00 2001 From: Tobias Koppers Date: Tue, 7 Mar 2023 18:10:05 +0100 Subject: [PATCH] improve and enable issue path handling (vercel/turbo#4017) ### Description adds the issue trace again, adds some basic descriptions e. g. to pages ### Testing Instructions snapshot issues [X] Auto label --- crates/next-core/src/app_source.rs | 43 +++-- crates/next-core/src/next_config.rs | 23 ++- crates/next-core/src/page_source.rs | 93 +++++----- crates/next-core/src/router.rs | 27 +++ crates/next-dev-tests/tests/integration.rs | 4 +- ...tar__0__star__) is very dynamic-5efef4.txt | 16 ++ ...__utf-8__quo__) is very dynamic-f84517.txt | 16 ++ ...tar__0__star__) is very dynamic-dd45bc.txt | 16 ++ .../Issue while running loader-6d7f75.txt | 10 ++ .../Issue while running loader-9bd07c.txt | 10 ++ .../Issue while running loader-bb7ea7.txt | 10 ++ .../Issue while running loader-ebe27b.txt | 10 ++ ...rror resolving commonjs request-e56948.txt | 3 + ...ving EcmaScript Modules request-6adb45.txt | 3 + ...ving EcmaScript Modules request-d56c20.txt | 3 + ...ving EcmaScript Modules request-eb66f2.txt | 3 + ...rror resolving commonjs request-d35c6e.txt | 3 + ...rror resolving commonjs request-d795f8.txt | 3 + ...__.__b__quo__]) is very dynamic-1b7b2d.txt | 3 + ...rror resolving commonjs request-0baf48.txt | 3 + ...__.__c__quo__]) is very dynamic-c9f4f1.txt | 3 + crates/next-dev/src/lib.rs | 3 +- crates/node-file-trace/src/lib.rs | 13 +- crates/turbopack-cli-utils/src/issue.rs | 82 +++++---- crates/turbopack-cli-utils/src/lib.rs | 1 + crates/turbopack-core/Cargo.toml | 1 + crates/turbopack-core/src/issue/mod.rs | 165 ++++++++++++----- crates/turbopack-core/src/reference/mod.rs | 11 +- .../src/source/issue_context.rs | 166 ++++++++++++++++++ crates/turbopack-dev-server/src/source/mod.rs | 16 +- .../src/update/protocol.rs | 4 + .../src/render/rendered_source.rs | 8 +- .../turbopack-node/src/transforms/postcss.rs | 9 +- crates/turbopack-tests/tests/snapshot.rs | 8 +- .../unexpected export __star__-b4c42d.txt | 3 + .../unexpected export __star__-30c177.txt | 3 + ...neration for chunk item errored-798a9e.txt | 3 + ...rror resolving commonjs request-69aa17.txt | 3 + ...ving EcmaScript Modules request-cd3585.txt | 3 + 39 files changed, 632 insertions(+), 175 deletions(-) create mode 100644 crates/turbopack-dev-server/src/source/issue_context.rs diff --git a/crates/next-core/src/app_source.rs b/crates/next-core/src/app_source.rs index 2de52ed0c07c1..409292cba2b58 100644 --- a/crates/next-core/src/app_source.rs +++ b/crates/next-core/src/app_source.rs @@ -1,6 +1,7 @@ use std::{ collections::{BTreeMap, HashMap}, io::Write, + iter::once, }; use anyhow::{anyhow, Result}; @@ -394,7 +395,9 @@ async fn create_app_source_for_directory( intermediate_output_path_root: FileSystemPathVc, ) -> Result { let AppStructure { - item, ref children, .. + item, + ref children, + directory, } = *app_structure.await?; let mut sources = Vec::new(); @@ -473,21 +476,31 @@ async fn create_app_source_for_directory( return Ok(NoContentSourceVc::new().into()); } } - for child in children.iter() { - sources.push(create_app_source_for_directory( - *child, - context_ssr, - context, - project_path, - env, - server_root, - runtime_entries, - fallback_page, - intermediate_output_path_root, - )); - } - Ok(CombinedContentSource { sources }.cell().into()) + let source = CombinedContentSource { sources } + .cell() + .as_content_source() + .issue_context(directory, "Next.js App Router"); + + Ok(CombinedContentSource { + sources: once(source) + .chain(children.iter().map(|child| { + create_app_source_for_directory( + *child, + context_ssr, + context, + project_path, + env, + server_root, + runtime_entries, + fallback_page, + intermediate_output_path_root, + ) + })) + .collect(), + } + .cell() + .into()) } /// The renderer for pages in app directory diff --git a/crates/next-core/src/next_config.rs b/crates/next-core/src/next_config.rs index 2b3352bf14376..1df9a67e9275d 100644 --- a/crates/next-core/src/next_config.rs +++ b/crates/next-core/src/next_config.rs @@ -15,6 +15,7 @@ use turbopack_core::{ changed::any_content_changed, context::AssetContext, ident::AssetIdentVc, + issue::IssueContextExt, reference_type::{EntryReferenceSubType, ReferenceType}, resolve::{ find_context_file, @@ -539,6 +540,22 @@ fn next_configs() -> StringsVc { #[turbo_tasks::function] pub async fn load_next_config(execution_context: ExecutionContextVc) -> Result { + let ExecutionContext { project_path, .. } = *execution_context.await?; + let find_config_result = find_context_file(project_path, next_configs()); + let config_file = match &*find_config_result.await? { + FindContextFileResult::Found(config_path, _) => Some(*config_path), + FindContextFileResult::NotFound(_) => None, + }; + load_next_config_internal(execution_context, config_file) + .issue_context(config_file, "Loading Next.js config") + .await +} + +#[turbo_tasks::function] +pub async fn load_next_config_internal( + execution_context: ExecutionContextVc, + config_file: Option, +) -> Result { let ExecutionContext { project_path, intermediate_output_path, @@ -552,11 +569,7 @@ pub async fn load_next_config(execution_context: ExecutionContextVc) -> Result Some(SourceAssetVc::new(*config_path)), - FindContextFileResult::NotFound(_) => None, - }; + let config_asset = config_file.map(SourceAssetVc::new); let config_changed = config_asset.map_or_else(CompletionVc::immutable, |config_asset| { // This invalidates the execution when anything referenced by the config file diff --git a/crates/next-core/src/page_source.rs b/crates/next-core/src/page_source.rs index efe188633562b..4a28dedf1a5da 100644 --- a/crates/next-core/src/page_source.rs +++ b/crates/next-core/src/page_source.rs @@ -277,17 +277,21 @@ pub async fn create_page_source( let fallback_source = AssetGraphContentSourceVc::new_eager(server_root, fallback_page.as_asset()); - Ok(CombinedContentSource { + let source = CombinedContentSource { sources: vec![ // Match _next/404 first to ensure rewrites work properly. - force_not_found_source, + force_not_found_source.issue_context(pages_dir, "Next.js pages directory not found"), page_source, - fallback_source.into(), - fallback_not_found_source, + fallback_source + .as_content_source() + .issue_context(pages_dir, "Next.js pages directory fallback"), + fallback_not_found_source + .issue_context(pages_dir, "Next.js pages directory not found fallback"), ], } .cell() - .into()) + .into(); + Ok(source) } /// Handles a single page file in the pages directory @@ -555,54 +559,53 @@ async fn create_page_source_for_directory( let mut sources = vec![]; for item in items.iter() { - match *item.await? { + let source = match *item.await? { PagesStructureItem::Page { page, specificity, url, - } => { - sources.push(create_page_source_for_file( - project_path, - env, - server_context, - server_data_context, - client_context, - pages_dir, - specificity, - SourceAssetVc::new(page).into(), - runtime_entries, - fallback_page, - server_root, - url, - false, - output_root, - output_root, - )); - } + } => create_page_source_for_file( + project_path, + env, + server_context, + server_data_context, + client_context, + pages_dir, + specificity, + SourceAssetVc::new(page).into(), + runtime_entries, + fallback_page, + server_root, + url, + false, + output_root, + output_root, + ) + .issue_context(page, "Next.js pages directory"), PagesStructureItem::Api { api, specificity, url, - } => { - sources.push(create_page_source_for_file( - project_path, - env, - server_context, - server_data_context, - client_context, - pages_dir, - specificity, - SourceAssetVc::new(api).into(), - runtime_entries, - fallback_page, - server_root, - url, - true, - output_root, - output_root, - )); - } - } + } => create_page_source_for_file( + project_path, + env, + server_context, + server_data_context, + client_context, + pages_dir, + specificity, + SourceAssetVc::new(api).into(), + runtime_entries, + fallback_page, + server_root, + url, + true, + output_root, + output_root, + ) + .issue_context(api, "Next.js pages api directory"), + }; + sources.push(source); } for child in children.iter() { diff --git a/crates/next-core/src/router.rs b/crates/next-core/src/router.rs index 6eacde43e086a..30e7b8592c197 100644 --- a/crates/next-core/src/router.rs +++ b/crates/next-core/src/router.rs @@ -17,6 +17,7 @@ use turbopack_core::{ context::{AssetContext, AssetContextVc}, environment::{EnvironmentIntention::Middleware, ServerAddrVc}, ident::AssetIdentVc, + issue::IssueVc, reference_type::{EcmaScriptModulesReferenceSubType, ReferenceType}, resolve::{find_context_file, FindContextFileResult}, source_asset::SourceAssetVc, @@ -316,6 +317,32 @@ pub async fn route( next_config: NextConfigVc, server_addr: ServerAddrVc, routes_changed: CompletionVc, +) -> Result { + let RouterRequest { + ref method, + ref pathname, + .. + } = *request.await?; + IssueVc::attach_description( + format!("Next.js Routing for {} {}", method, pathname), + route_internal( + execution_context, + request, + next_config, + server_addr, + routes_changed, + ), + ) + .await +} + +#[turbo_tasks::function] +async fn route_internal( + execution_context: ExecutionContextVc, + request: RouterRequestVc, + next_config: NextConfigVc, + server_addr: ServerAddrVc, + routes_changed: CompletionVc, ) -> Result { let ExecutionContext { project_path, diff --git a/crates/next-dev-tests/tests/integration.rs b/crates/next-dev-tests/tests/integration.rs index 9eed0bcd845bd..afb784d7ab5c6 100644 --- a/crates/next-dev-tests/tests/integration.rs +++ b/crates/next-dev-tests/tests/integration.rs @@ -494,8 +494,8 @@ impl IssueReporter for TestIssueReporter { _source: TransientValue, ) -> Result { let issue_tx = self.issue_tx.get_untracked().clone(); - for issue in captured_issues.iter() { - let plain = issue.into_plain(); + for (issue, path) in captured_issues.iter_with_shortest_path() { + let plain = issue.into_plain(path); issue_tx.send((plain.await?, plain.dbg().await?)).await?; } diff --git a/crates/next-dev-tests/tests/integration/next/tailwind/basic/issues/lint TP1004 fs.existsSync(__q____q____q____star__0__star__) is very dynamic-5efef4.txt b/crates/next-dev-tests/tests/integration/next/tailwind/basic/issues/lint TP1004 fs.existsSync(__q____q____q____star__0__star__) is very dynamic-5efef4.txt index 8eea1e7d94059..52708253ed078 100644 --- a/crates/next-dev-tests/tests/integration/next/tailwind/basic/issues/lint TP1004 fs.existsSync(__q____q____q____star__0__star__) is very dynamic-5efef4.txt +++ b/crates/next-dev-tests/tests/integration/next/tailwind/basic/issues/lint TP1004 fs.existsSync(__q____q____q____star__0__star__) is very dynamic-5efef4.txt @@ -22,4 +22,20 @@ PlainIssue { }, ), sub_issues: [], + processing_path: Some( + [ + PlainIssueProcessingPathItem { + context: Some( + "[project]/crates/next-dev-tests/tests/integration/next/tailwind/basic/input/pages/index.jsx", + ), + description: "Next.js pages directory", + }, + PlainIssueProcessingPathItem { + context: Some( + "[project]/crates/next-dev-tests/tests/integration/next/tailwind/basic/input/styles/globals.css", + ), + description: "PostCSS processing", + }, + ], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/next/tailwind/basic/issues/lint TP1004 fs.readFileSync(__q____q____q____star__0__star__, __quo__utf-8__quo__) is very dynamic-f84517.txt b/crates/next-dev-tests/tests/integration/next/tailwind/basic/issues/lint TP1004 fs.readFileSync(__q____q____q____star__0__star__, __quo__utf-8__quo__) is very dynamic-f84517.txt index 6dc6703ee92a3..5648e1f03d9d0 100644 --- a/crates/next-dev-tests/tests/integration/next/tailwind/basic/issues/lint TP1004 fs.readFileSync(__q____q____q____star__0__star__, __quo__utf-8__quo__) is very dynamic-f84517.txt +++ b/crates/next-dev-tests/tests/integration/next/tailwind/basic/issues/lint TP1004 fs.readFileSync(__q____q____q____star__0__star__, __quo__utf-8__quo__) is very dynamic-f84517.txt @@ -22,4 +22,20 @@ PlainIssue { }, ), sub_issues: [], + processing_path: Some( + [ + PlainIssueProcessingPathItem { + context: Some( + "[project]/crates/next-dev-tests/tests/integration/next/tailwind/basic/input/pages/index.jsx", + ), + description: "Next.js pages directory", + }, + PlainIssueProcessingPathItem { + context: Some( + "[project]/crates/next-dev-tests/tests/integration/next/tailwind/basic/input/styles/globals.css", + ), + description: "PostCSS processing", + }, + ], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/next/tailwind/basic/issues/lint TP1006 path.resolve(__q____q____q____star__0__star__) is very dynamic-dd45bc.txt b/crates/next-dev-tests/tests/integration/next/tailwind/basic/issues/lint TP1006 path.resolve(__q____q____q____star__0__star__) is very dynamic-dd45bc.txt index d5b11a8683725..161e29a15fb28 100644 --- a/crates/next-dev-tests/tests/integration/next/tailwind/basic/issues/lint TP1006 path.resolve(__q____q____q____star__0__star__) is very dynamic-dd45bc.txt +++ b/crates/next-dev-tests/tests/integration/next/tailwind/basic/issues/lint TP1006 path.resolve(__q____q____q____star__0__star__) is very dynamic-dd45bc.txt @@ -22,4 +22,20 @@ PlainIssue { }, ), sub_issues: [], + processing_path: Some( + [ + PlainIssueProcessingPathItem { + context: Some( + "[project]/crates/next-dev-tests/tests/integration/next/tailwind/basic/input/pages/index.jsx", + ), + description: "Next.js pages directory", + }, + PlainIssueProcessingPathItem { + context: Some( + "[project]/crates/next-dev-tests/tests/integration/next/tailwind/basic/input/styles/globals.css", + ), + description: "PostCSS processing", + }, + ], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-6d7f75.txt b/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-6d7f75.txt index 59d208adda506..2241ae4b42e35 100644 --- a/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-6d7f75.txt +++ b/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-6d7f75.txt @@ -8,4 +8,14 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [ + PlainIssueProcessingPathItem { + context: Some( + "[project]/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/input/pages/index.js", + ), + description: "Next.js pages directory", + }, + ], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-9bd07c.txt b/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-9bd07c.txt index 07ccf8f70868c..bd8631682017d 100644 --- a/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-9bd07c.txt +++ b/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-9bd07c.txt @@ -8,4 +8,14 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [ + PlainIssueProcessingPathItem { + context: Some( + "[project]/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/input/pages/index.js", + ), + description: "Next.js pages directory", + }, + ], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-bb7ea7.txt b/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-bb7ea7.txt index 165480b58ae16..a3f786e0f9334 100644 --- a/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-bb7ea7.txt +++ b/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-bb7ea7.txt @@ -8,4 +8,14 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [ + PlainIssueProcessingPathItem { + context: Some( + "[project]/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/input/pages/index.js", + ), + description: "Next.js pages directory", + }, + ], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-ebe27b.txt b/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-ebe27b.txt index eb1c8dd1bf2d6..c699897b1f31f 100644 --- a/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-ebe27b.txt +++ b/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/issues/Issue while running loader-ebe27b.txt @@ -8,4 +8,14 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [ + PlainIssueProcessingPathItem { + context: Some( + "[project]/crates/next-dev-tests/tests/integration/next/webpack-loaders/emitted-errors/input/pages/index.js", + ), + description: "Next.js pages directory", + }, + ], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/turbopack/basic/comptime/issues/Error resolving commonjs request-e56948.txt b/crates/next-dev-tests/tests/integration/turbopack/basic/comptime/issues/Error resolving commonjs request-e56948.txt index d16398e3da94b..ce6c02a75e1b7 100644 --- a/crates/next-dev-tests/tests/integration/turbopack/basic/comptime/issues/Error resolving commonjs request-e56948.txt +++ b/crates/next-dev-tests/tests/integration/turbopack/basic/comptime/issues/Error resolving commonjs request-e56948.txt @@ -8,4 +8,7 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/named-chunks/issues/Error resolving EcmaScript Modules request-6adb45.txt b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/named-chunks/issues/Error resolving EcmaScript Modules request-6adb45.txt index 55df2e44d15b1..ffd512e299e47 100644 --- a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/named-chunks/issues/Error resolving EcmaScript Modules request-6adb45.txt +++ b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/named-chunks/issues/Error resolving EcmaScript Modules request-6adb45.txt @@ -8,4 +8,7 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/named-chunks/issues/Error resolving EcmaScript Modules request-d56c20.txt b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/named-chunks/issues/Error resolving EcmaScript Modules request-d56c20.txt index aa2e87a241931..6089213174c9d 100644 --- a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/named-chunks/issues/Error resolving EcmaScript Modules request-d56c20.txt +++ b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/named-chunks/issues/Error resolving EcmaScript Modules request-d56c20.txt @@ -8,4 +8,7 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/named-chunks/issues/Error resolving EcmaScript Modules request-eb66f2.txt b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/named-chunks/issues/Error resolving EcmaScript Modules request-eb66f2.txt index 0a5132b1984d0..6b51430b76c82 100644 --- a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/named-chunks/issues/Error resolving EcmaScript Modules request-eb66f2.txt +++ b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/named-chunks/issues/Error resolving EcmaScript Modules request-eb66f2.txt @@ -8,4 +8,7 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/parsing/issues/Error resolving commonjs request-d35c6e.txt b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/parsing/issues/Error resolving commonjs request-d35c6e.txt index c28c72870bbae..eee4b209f6df5 100644 --- a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/parsing/issues/Error resolving commonjs request-d35c6e.txt +++ b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/parsing/issues/Error resolving commonjs request-d35c6e.txt @@ -8,4 +8,7 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies-context/issues/Error resolving commonjs request-d795f8.txt b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies-context/issues/Error resolving commonjs request-d795f8.txt index 21f2b4cb5d625..6ace76ced5de6 100644 --- a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies-context/issues/Error resolving commonjs request-d795f8.txt +++ b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies-context/issues/Error resolving commonjs request-d795f8.txt @@ -8,4 +8,7 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies-context/issues/lint TP1002 require([__quo__.__b__quo__]) is very dynamic-1b7b2d.txt b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies-context/issues/lint TP1002 require([__quo__.__b__quo__]) is very dynamic-1b7b2d.txt index f67dfec901570..0b8bccd3f65a9 100644 --- a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies-context/issues/lint TP1002 require([__quo__.__b__quo__]) is very dynamic-1b7b2d.txt +++ b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies-context/issues/lint TP1002 require([__quo__.__b__quo__]) is very dynamic-1b7b2d.txt @@ -22,4 +22,7 @@ PlainIssue { }, ), sub_issues: [], + processing_path: Some( + [], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies/issues/Error resolving commonjs request-0baf48.txt b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies/issues/Error resolving commonjs request-0baf48.txt index df46f9ef9d4de..93749c4ed025e 100644 --- a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies/issues/Error resolving commonjs request-0baf48.txt +++ b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies/issues/Error resolving commonjs request-0baf48.txt @@ -8,4 +8,7 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [], + ), } \ No newline at end of file diff --git a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies/issues/lint TP1002 require([__quo__.__c__quo__]) is very dynamic-c9f4f1.txt b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies/issues/lint TP1002 require([__quo__.__c__quo__]) is very dynamic-c9f4f1.txt index ff2f926d2d096..eee4b6c7deb9d 100644 --- a/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies/issues/lint TP1002 require([__quo__.__c__quo__]) is very dynamic-c9f4f1.txt +++ b/crates/next-dev-tests/tests/integration/webpack/chunks/__skipped__/weak-dependencies/issues/lint TP1002 require([__quo__.__c__quo__]) is very dynamic-c9f4f1.txt @@ -22,4 +22,7 @@ PlainIssue { }, ), sub_issues: [], + processing_path: Some( + [], + ), } \ No newline at end of file diff --git a/crates/next-dev/src/lib.rs b/crates/next-dev/src/lib.rs index 0295e4050fba5..5b5a5e18f6349 100644 --- a/crates/next-dev/src/lib.rs +++ b/crates/next-dev/src/lib.rs @@ -10,7 +10,7 @@ use std::{ future::{join, Future}, io::{stdout, Write}, net::{IpAddr, SocketAddr}, - path::MAIN_SEPARATOR, + path::{PathBuf, MAIN_SEPARATOR}, sync::Arc, time::{Duration, Instant}, }; @@ -206,6 +206,7 @@ impl NextDevServerBuilder { let browserslist_query = self.browserslist_query; let log_options = Arc::new(LogOptions { current_dir: current_dir().unwrap(), + project_dir: PathBuf::from(project_dir.clone()), show_all, log_detail, log_level: self.log_level, diff --git a/crates/node-file-trace/src/lib.rs b/crates/node-file-trace/src/lib.rs index 8b26929cc52d2..dae7f31fe50fb 100644 --- a/crates/node-file-trace/src/lib.rs +++ b/crates/node-file-trace/src/lib.rs @@ -44,7 +44,7 @@ use turbopack_core::{ compile_time_info::{CompileTimeDefinesVc, CompileTimeInfo}, context::{AssetContext, AssetContextVc}, environment::{EnvironmentIntention, EnvironmentVc, ExecutionEnvironment, NodeJsEnvironment}, - issue::{IssueReporter, IssueSeverity, IssueVc}, + issue::{IssueContextExt, IssueReporter, IssueSeverity, IssueVc}, reference::all_assets, resolve::options::{ImportMapping, ResolvedMap}, source_asset::SourceAssetVc, @@ -459,6 +459,7 @@ async fn run>( let resolve_options = TransientInstance::new(resolve_options.unwrap_or_default()); let log_options = TransientInstance::new(LogOptions { current_dir: dir.clone(), + project_dir: dir.clone(), show_all, log_detail, log_level: log_level.map_or_else(|| IssueSeverity::Error, |l| l.0), @@ -542,13 +543,9 @@ async fn main_operation( ) .await?; for module in modules.iter() { - let set = all_assets(*module); - IssueVc::attach_context( - module.ident().path(), - "gathering list of assets".to_string(), - set, - ) - .await?; + let set = all_assets(*module) + .issue_context(module.ident().path(), "gathering list of assets") + .await?; for asset in set.await?.iter() { let path = asset.ident().path().await?; result.insert(path.path.to_string()); diff --git a/crates/turbopack-cli-utils/src/issue.rs b/crates/turbopack-cli-utils/src/issue.rs index 9fed3d3a04369..837fe42f8d55b 100644 --- a/crates/turbopack-cli-utils/src/issue.rs +++ b/crates/turbopack-cli-utils/src/issue.rs @@ -1,8 +1,9 @@ use std::{ + borrow::Cow, cmp::min, collections::{hash_map::Entry, HashMap, HashSet}, fmt::Write as _, - path::PathBuf, + path::{Path, PathBuf}, str::FromStr, sync::{Arc, Mutex}, }; @@ -12,16 +13,14 @@ use crossterm::style::{StyledContent, Stylize}; use owo_colors::{OwoColorize as _, Style}; use turbo_tasks::{ primitives::BoolVc, RawVc, ReadRef, TransientInstance, TransientValue, TryJoinIterExt, - ValueToString, }; use turbo_tasks_fs::{ - attach::AttachedFileSystemVc, source_context::{get_source_context, SourceContextLine}, - to_sys_path, FileLinesContent, FileSystemPathVc, + FileLinesContent, }; use turbopack_core::issue::{ - CapturedIssues, Issue, IssueProcessingPathItem, IssueReporter, IssueReporterVc, IssueSeverity, - OptionIssueProcessingPathItemsVc, PlainIssue, PlainIssueSource, + CapturedIssues, IssueReporter, IssueReporterVc, IssueSeverity, PlainIssue, + PlainIssueProcessingPathItem, PlainIssueProcessingPathItemReadRef, PlainIssueSource, }; #[derive(Clone, Copy, PartialEq, Eq, Debug)] @@ -173,32 +172,32 @@ fn format_source_content(source: &PlainIssueSource, formatted_issue: &mut String } } -async fn format_optional_path( - path: &OptionIssueProcessingPathItemsVc, +fn format_optional_path( + path: &Option>, formatted_issue: &mut String, ) -> Result<()> { - if let Some(path) = &*path.await? { + if let Some(path) = path { let mut last_context = None; for item in path.iter().rev() { - let IssueProcessingPathItem { - context, - description, - } = &*item.await?; + let PlainIssueProcessingPathItem { + ref context, + ref description, + } = **item; if let Some(context) = context { - let context = context.resolve().await?; - if last_context == Some(context) { - writeln!(formatted_issue, " at {}", &*description.await?)?; + let option_context = Some(context.clone()); + if last_context == option_context { + writeln!(formatted_issue, " at {}", description)?; } else { writeln!( formatted_issue, " at {} ({})", - context.to_string().await?.bright_blue(), - &*description.await? + context.to_string().bright_blue(), + description )?; - last_context = Some(context); + last_context = option_context; } } else { - writeln!(formatted_issue, " at {}", &*description.await?)?; + writeln!(formatted_issue, " at {}", description)?; last_context = None; } } @@ -301,6 +300,7 @@ const ORDERED_GROUPS: &[IssueSeverity] = &[ #[derive(Debug, Clone)] pub struct LogOptions { pub current_dir: PathBuf, + pub project_dir: PathBuf, pub show_all: bool, pub log_detail: bool, pub log_level: IssueSeverity, @@ -438,6 +438,7 @@ impl IssueReporter for ConsoleUi { let issues = &*issues; let LogOptions { ref current_dir, + ref project_dir, show_all, log_detail, log_level, @@ -448,18 +449,14 @@ impl IssueReporter for ConsoleUi { let issues = issues .iter_with_shortest_path() .map(|(issue, path)| async move { - // (issue.) - let plain_issue = issue.into_plain(); + let plain_issue = issue.into_plain(path); let id = plain_issue.internal_hash().await?; - Ok((plain_issue.await?, path, issue.context(), *id)) + Ok((plain_issue.await?, *id)) }) .try_join() .await?; - let issue_ids = issues - .iter() - .map(|(_, _, _, id)| *id) - .collect::>(); + let issue_ids = issues.iter().map(|(_, id)| *id).collect::>(); let mut new_ids = self .seen .lock() @@ -467,7 +464,7 @@ impl IssueReporter for ConsoleUi { .new_ids(source.into_value(), issue_ids); let mut has_fatal = false; - for (plain_issue, path, context, id) in issues { + for (plain_issue, id) in issues { if !new_ids.remove(&id) { continue; } @@ -477,9 +474,10 @@ impl IssueReporter for ConsoleUi { has_fatal = true; } - let context_path = make_relative_to_cwd(context, current_dir).await?; + let context_path = make_relative_to_cwd(&plain_issue.context, project_dir, current_dir); let category = &plain_issue.category; let title = &plain_issue.title; + let processing_path = &*plain_issue.processing_path; let severity_map = grouped_issues .entry(severity) .or_insert_with(Default::default); @@ -487,7 +485,7 @@ impl IssueReporter for ConsoleUi { .entry(category.clone()) .or_insert_with(Default::default); let issues = category_map - .entry(context_path.clone()) + .entry(context_path.to_string()) .or_insert_with(Default::default); let mut styled_issue = if let Some(source) = &plain_issue.source { @@ -522,7 +520,7 @@ impl IssueReporter for ConsoleUi { if !documentation_link.is_empty() { writeln!(&mut styled_issue, "\ndocumentation: {documentation_link}")?; } - format_optional_path(&path, &mut styled_issue).await?; + format_optional_path(processing_path, &mut styled_issue)?; } issues.push(styled_issue); } @@ -616,21 +614,21 @@ impl IssueReporter for ConsoleUi { } } -async fn make_relative_to_cwd(path: FileSystemPathVc, cwd: &PathBuf) -> Result { - let path = if let Some(fs) = AttachedFileSystemVc::resolve_from(path.fs()).await? { - fs.get_inner_fs_path(path) - } else { - path - }; - if let Some(sys_path) = to_sys_path(path).await? { - let relative = sys_path +fn make_relative_to_cwd<'a>(path: &'a str, project_dir: &Path, cwd: &Path) -> Cow<'a, str> { + if let Some(path_in_project) = path.strip_prefix("[project]/") { + let abs_path = if std::path::MAIN_SEPARATOR != '/' { + project_dir.join(path_in_project.replace('/', std::path::MAIN_SEPARATOR_STR)) + } else { + project_dir.join(path_in_project) + }; + let relative = abs_path .strip_prefix(cwd) - .unwrap_or(&sys_path) + .unwrap_or(&abs_path) .to_string_lossy() .to_string(); - Ok(relative) + relative.into() } else { - Ok(path.to_string().await?.clone_value()) + path.into() } } diff --git a/crates/turbopack-cli-utils/src/lib.rs b/crates/turbopack-cli-utils/src/lib.rs index ddddfe14531db..b185c55cb1a7a 100644 --- a/crates/turbopack-cli-utils/src/lib.rs +++ b/crates/turbopack-cli-utils/src/lib.rs @@ -1,6 +1,7 @@ #![feature(async_closure)] #![feature(min_specialization)] #![feature(round_char_boundary)] +#![feature(main_separator_str)] pub mod issue; diff --git a/crates/turbopack-core/Cargo.toml b/crates/turbopack-core/Cargo.toml index f2b164c25fb26..2d4e024c00b04 100644 --- a/crates/turbopack-core/Cargo.toml +++ b/crates/turbopack-core/Cargo.toml @@ -38,4 +38,5 @@ turbo-tasks-build = { workspace = true } rstest = { workspace = true } [features] +default = ["issue_path"] issue_path = [] diff --git a/crates/turbopack-core/src/issue/mod.rs b/crates/turbopack-core/src/issue/mod.rs index 295e7d95b1da9..24e7e46353495 100644 --- a/crates/turbopack-core/src/issue/mod.rs +++ b/crates/turbopack-core/src/issue/mod.rs @@ -7,13 +7,12 @@ pub mod unsupported_module; use std::{ cmp::Ordering, fmt::{Display, Formatter}, - future::IntoFuture, sync::Arc, }; use anyhow::Result; +use async_trait::async_trait; use auto_hash_map::AutoSet; -use futures::FutureExt; use turbo_tasks::{ emit, primitives::{BoolVc, StringReadRef, StringVc, U64Vc}, @@ -159,9 +158,51 @@ impl ValueToString for IssueProcessingPathItem { } } +#[turbo_tasks::value_impl] +impl IssueProcessingPathItemVc { + #[turbo_tasks::function] + pub async fn into_plain(self) -> Result { + let this = self.await?; + Ok(PlainIssueProcessingPathItem { + context: if let Some(context) = this.context { + Some(context.to_string().await?) + } else { + None + }, + description: this.description.await?, + } + .cell()) + } +} + #[turbo_tasks::value(transparent)] pub struct OptionIssueProcessingPathItems(Option>); +#[turbo_tasks::value_impl] +impl OptionIssueProcessingPathItemsVc { + #[turbo_tasks::function] + pub fn none() -> Self { + OptionIssueProcessingPathItemsVc::cell(None) + } + + #[turbo_tasks::function] + pub async fn into_plain(self) -> Result { + Ok(PlainIssueProcessingPathVc::cell( + if let Some(items) = &*self.await? { + Some( + items + .iter() + .map(|item| item.into_plain()) + .try_join() + .await?, + ) + } else { + None + }, + )) + } +} + #[turbo_tasks::value] struct RootIssueProcessingPath(IssueVc); @@ -198,7 +239,7 @@ impl IssueProcessingPath for ItemIssueProcessingPath { let mut shortest: Option<&Vec<_>> = None; for path in paths.iter().filter_map(|p| p.as_ref()) { if let Some(old) = shortest { - match old.cmp(path) { + match old.len().cmp(&path.len()) { Ordering::Greater => { shortest = Some(path); } @@ -246,9 +287,9 @@ impl IssueVc { impl IssueVc { #[allow(unused_variables, reason = "behind feature flag")] - pub async fn attach_context( - context: FileSystemPathVc, - description: String, + pub async fn attach_context( + context: impl Into> + Send, + description: impl Into + Send, source: T, ) -> Result { #[cfg(feature = "issue_path")] @@ -258,8 +299,8 @@ impl IssueVc { emit( ItemIssueProcessingPathVc::cell(ItemIssueProcessingPath( Some(IssueProcessingPathItemVc::cell(IssueProcessingPathItem { - context: Some(context), - description: StringVc::cell(description), + context: context.into(), + description: StringVc::cell(description.into()), })), children, )) @@ -271,27 +312,11 @@ impl IssueVc { } #[allow(unused_variables, reason = "behind feature flag")] - pub async fn attach_description( - description: String, + pub async fn attach_description( + description: impl Into + Send, source: T, ) -> Result { - #[cfg(feature = "issue_path")] - { - let children = source.take_collectibles().await?; - if !children.is_empty() { - emit( - ItemIssueProcessingPathVc::cell(ItemIssueProcessingPath( - Some(IssueProcessingPathItemVc::cell(IssueProcessingPathItem { - context: None, - description: StringVc::cell(description), - })), - children, - )) - .as_issue_processing_path(), - ); - } - } - Ok(source) + Self::attach_context(None, description, source).await } /// Returns all issues from `source` in a list with their associated @@ -373,7 +398,7 @@ impl CapturedIssues { #[cfg(feature = "issue_path")] let path = self.processing_path.shortest_path(*issue); #[cfg(not(feature = "issue_path"))] - let path = OptionIssueProcessingPathItemsVc::cell(None); + let path = OptionIssueProcessingPathItemsVc::none(); (*issue, path) }) } @@ -382,7 +407,16 @@ impl CapturedIssues { let mut list = self .issues .iter() - .map(|issue| issue.into_plain().into_future()) + .map(|&issue| async move { + #[cfg(feature = "issue_path")] + return issue + .into_plain(self.processing_path.shortest_path(issue)) + .await; + #[cfg(not(feature = "issue_path"))] + return issue + .into_plain(OptionIssueProcessingPathItemsVc::none()) + .await; + }) .try_join() .await?; list.sort_by(|a, b| ReadRef::ptr_cmp(a, b)); @@ -453,6 +487,7 @@ pub struct PlainIssue { pub source: Option, pub sub_issues: Vec, + pub processing_path: PlainIssueProcessingPathReadRef, } impl PlainIssue { @@ -497,7 +532,10 @@ impl PlainIssueVc { #[turbo_tasks::value_impl] impl IssueVc { #[turbo_tasks::function] - pub async fn into_plain(self) -> Result { + pub async fn into_plain( + self, + processing_path: OptionIssueProcessingPathItemsVc, + ) -> Result { Ok(PlainIssue { severity: *self.severity().await?, context: self.context().to_string().await?.clone_value(), @@ -506,24 +544,26 @@ impl IssueVc { description: self.description().await?.clone_value(), detail: self.detail().await?.clone_value(), documentation_link: self.documentation_link().await?.clone_value(), - source: self - .source() - .into_future() - .then(|s| async { - if let Some(s) = *s? { - return Ok(Some(s.into_plain().await?)); - } - - anyhow::Ok(None) - }) - .await?, + source: { + if let Some(s) = *self.source().await? { + Some(s.into_plain().await?) + } else { + None + } + }, sub_issues: self .sub_issues() .await? .iter() - .map(|i| async move { anyhow::Ok(i.into_plain().await?) }) + .map(|i| async move { + anyhow::Ok( + i.into_plain(OptionIssueProcessingPathItemsVc::none()) + .await?, + ) + }) .try_join() .await?, + processing_path: processing_path.into_plain().await?, } .cell()) } @@ -577,6 +617,17 @@ impl PlainAssetVc { } } +#[turbo_tasks::value(transparent, serialization = "none")] +#[derive(Clone, Debug)] +pub struct PlainIssueProcessingPath(Option>); + +#[turbo_tasks::value(serialization = "none")] +#[derive(Clone, Debug)] +pub struct PlainIssueProcessingPathItem { + pub context: Option, + pub description: StringReadRef, +} + #[turbo_tasks::value_trait] pub trait IssueReporter { fn report_issues( @@ -585,3 +636,33 @@ pub trait IssueReporter { source: TransientValue, ) -> BoolVc; } + +#[async_trait] +pub trait IssueContextExt +where + Self: Sized, +{ + async fn issue_context( + self, + context: impl Into> + Send, + description: impl Into + Send, + ) -> Result; + async fn issue_description(self, description: impl Into + Send) -> Result; +} + +#[async_trait] +impl IssueContextExt for T +where + T: CollectiblesSource + Copy + Send, +{ + async fn issue_context( + self, + context: impl Into> + Send, + description: impl Into + Send, + ) -> Result { + IssueVc::attach_context(context, description, self).await + } + async fn issue_description(self, description: impl Into + Send) -> Result { + IssueVc::attach_description(description, self).await + } +} diff --git a/crates/turbopack-core/src/reference/mod.rs b/crates/turbopack-core/src/reference/mod.rs index bc44c45790ecf..1abfa294163bd 100644 --- a/crates/turbopack-core/src/reference/mod.rs +++ b/crates/turbopack-core/src/reference/mod.rs @@ -5,7 +5,7 @@ use turbo_tasks::{primitives::StringVc, ValueToString, ValueToStringVc}; use crate::{ asset::{Asset, AssetVc, AssetsVc}, - issue::IssueVc, + issue::IssueContextExt, resolve::{PrimaryResolveResult, ResolveResult, ResolveResultVc}, }; pub mod source_map; @@ -128,12 +128,9 @@ pub async fn all_assets(asset: AssetVc) -> Result { let mut assets = HashSet::new(); assets.insert(asset); while let Some((parent, references)) = queue.pop_front() { - IssueVc::attach_context( - parent.ident().path(), - "expanding references of asset".to_string(), - references, - ) - .await?; + let references = references + .issue_context(parent.ident().path(), "expanding references of asset") + .await?; for asset in references.await?.iter() { if assets.insert(*asset) { queue.push_back((*asset, all_referenced_assets(*asset))); diff --git a/crates/turbopack-dev-server/src/source/issue_context.rs b/crates/turbopack-dev-server/src/source/issue_context.rs new file mode 100644 index 0000000000000..8e460ca210132 --- /dev/null +++ b/crates/turbopack-dev-server/src/source/issue_context.rs @@ -0,0 +1,166 @@ +use anyhow::Result; +use turbo_tasks::{primitives::StringVc, Value}; +use turbo_tasks_fs::FileSystemPathVc; +use turbopack_core::{ + introspect::{Introspectable, IntrospectableChildrenVc, IntrospectableVc}, + issue::IssueContextExt, +}; + +use super::{ + ContentSource, ContentSourceContentVc, ContentSourceData, ContentSourceDataVaryVc, + ContentSourceResult, ContentSourceResultVc, ContentSourceVc, ContentSourcesVc, + GetContentSourceContent, GetContentSourceContentVc, +}; + +#[turbo_tasks::value] +pub struct IssueContextContentSource { + context: Option, + description: String, + source: ContentSourceVc, +} + +#[turbo_tasks::value_impl] +impl IssueContextContentSourceVc { + #[turbo_tasks::function] + pub fn new_context( + context: FileSystemPathVc, + description: &str, + source: ContentSourceVc, + ) -> Self { + IssueContextContentSource { + context: Some(context), + description: description.to_string(), + source, + } + .cell() + } + + #[turbo_tasks::function] + pub fn new_description(description: &str, source: ContentSourceVc) -> Self { + IssueContextContentSource { + context: None, + description: description.to_string(), + source, + } + .cell() + } +} + +#[turbo_tasks::value_impl] +impl ContentSource for IssueContextContentSource { + #[turbo_tasks::function] + async fn get( + self_vc: IssueContextContentSourceVc, + path: &str, + data: Value, + ) -> Result { + let this = self_vc.await?; + let result = this + .source + .get(path, data) + .issue_context(this.context, &this.description) + .await?; + if let ContentSourceResult::Result { + get_content, + specificity, + } = *result.await? + { + Ok(ContentSourceResult::Result { + get_content: IssueContextGetContentSourceContent { + get_content, + source: self_vc, + } + .cell() + .into(), + specificity, + } + .cell()) + } else { + Ok(result) + } + } + + #[turbo_tasks::function] + fn get_children(&self) -> ContentSourcesVc { + ContentSourcesVc::cell(vec![self.source]) + } +} + +#[turbo_tasks::value] +struct IssueContextGetContentSourceContent { + get_content: GetContentSourceContentVc, + source: IssueContextContentSourceVc, +} + +#[turbo_tasks::value_impl] +impl GetContentSourceContent for IssueContextGetContentSourceContent { + #[turbo_tasks::function] + async fn vary(&self) -> Result { + let source = self.source.await?; + let result = self + .get_content + .vary() + .issue_context(source.context, &source.description) + .await?; + Ok(result) + } + + #[turbo_tasks::function] + async fn get(&self, data: Value) -> Result { + let source = self.source.await?; + let result = self + .get_content + .get(data) + .issue_context(source.context, &source.description) + .await?; + Ok(result) + } +} + +#[turbo_tasks::value_impl] +impl Introspectable for IssueContextContentSource { + #[turbo_tasks::function] + async fn ty(&self) -> Result { + Ok( + if let Some(source) = IntrospectableVc::resolve_from(self.source).await? { + source.ty() + } else { + StringVc::cell("IssueContextContentSource".to_string()) + }, + ) + } + + #[turbo_tasks::function] + async fn title(&self) -> Result { + Ok( + if let Some(source) = IntrospectableVc::resolve_from(self.source).await? { + let title = source.title().await?; + StringVc::cell(format!("{}: {}", self.description, title)) + } else { + StringVc::cell(self.description.clone()) + }, + ) + } + + #[turbo_tasks::function] + async fn details(&self) -> Result { + Ok( + if let Some(source) = IntrospectableVc::resolve_from(self.source).await? { + source.details() + } else { + StringVc::cell(String::new()) + }, + ) + } + + #[turbo_tasks::function] + async fn children(&self) -> Result { + Ok( + if let Some(source) = IntrospectableVc::resolve_from(self.source).await? { + source.children() + } else { + IntrospectableChildrenVc::cell(Default::default()) + }, + ) + } +} diff --git a/crates/turbopack-dev-server/src/source/mod.rs b/crates/turbopack-dev-server/src/source/mod.rs index 51fb977af057d..b08ee62c47d78 100644 --- a/crates/turbopack-dev-server/src/source/mod.rs +++ b/crates/turbopack-dev-server/src/source/mod.rs @@ -2,6 +2,7 @@ pub mod asset_graph; pub mod combined; pub mod conditional; pub mod headers; +pub mod issue_context; pub mod lazy_instantiated; pub mod query; pub mod request; @@ -17,10 +18,13 @@ use std::{collections::BTreeSet, sync::Arc}; use anyhow::Result; use serde::{Deserialize, Serialize, Serializer}; use turbo_tasks::{trace::TraceRawVcs, Value}; -use turbo_tasks_fs::rope::Rope; +use turbo_tasks_fs::{rope::Rope, FileSystemPathVc}; use turbopack_core::version::VersionedContentVc; -use self::{headers::Headers, query::Query, specificity::SpecificityVc}; +use self::{ + headers::Headers, issue_context::IssueContextContentSourceVc, query::Query, + specificity::SpecificityVc, +}; /// The result of proxying a request to another HTTP server. #[turbo_tasks::value(shared)] @@ -460,6 +464,14 @@ pub trait ContentSource { } } +#[turbo_tasks::value_impl] +impl ContentSourceVc { + #[turbo_tasks::function] + pub fn issue_context(self, context: FileSystemPathVc, description: &str) -> ContentSourceVc { + IssueContextContentSourceVc::new_context(context, description, self).into() + } +} + #[turbo_tasks::value(transparent)] pub struct ContentSources(Vec); diff --git a/crates/turbopack-dev-server/src/update/protocol.rs b/crates/turbopack-dev-server/src/update/protocol.rs index a19144262fdd9..3168f46a58f68 100644 --- a/crates/turbopack-dev-server/src/update/protocol.rs +++ b/crates/turbopack-dev-server/src/update/protocol.rs @@ -145,11 +145,15 @@ impl<'a> From<&'a PlainIssue> for Issue<'a> { detail: &plain.detail, source, sub_issues: plain.sub_issues.iter().map(|p| p.deref().into()).collect(), + // TODO(WEB-691) formatting the issue should be handled by the error overlay. + // The browser could handle error formatting in a better way than the text only + // formatting here formatted: format_issue( plain, None, &LogOptions { current_dir: PathBuf::new(), + project_dir: PathBuf::new(), show_all: true, log_detail: true, log_level: IssueSeverity::Info, diff --git a/crates/turbopack-node/src/render/rendered_source.rs b/crates/turbopack-node/src/render/rendered_source.rs index f59e194d1c658..ae4e83d154769 100644 --- a/crates/turbopack-node/src/render/rendered_source.rs +++ b/crates/turbopack-node/src/render/rendered_source.rs @@ -8,6 +8,7 @@ use turbopack_core::{ introspect::{ asset::IntrospectableAssetVc, Introspectable, IntrospectableChildrenVc, IntrospectableVc, }, + issue::IssueContextExt, reference::AssetReference, resolve::PrimaryResolveResult, }; @@ -223,7 +224,12 @@ impl GetContentSourceContent for NodeRenderGetContentResult { path: format!("/{}", source.pathname.await?), } .cell(), - ); + ) + .issue_context( + entry.module.ident().path(), + format!("server-side rendering /{}", source.pathname.await?), + ) + .await?; Ok(match *result.await? { StaticResult::Content { content, diff --git a/crates/turbopack-node/src/transforms/postcss.rs b/crates/turbopack-node/src/transforms/postcss.rs index ee0d3e6cc3e8a..d892095762941 100644 --- a/crates/turbopack-node/src/transforms/postcss.rs +++ b/crates/turbopack-node/src/transforms/postcss.rs @@ -14,6 +14,7 @@ use turbopack_core::{ changed::any_content_changed, context::{AssetContext, AssetContextVc}, ident::AssetIdentVc, + issue::IssueContextExt, reference_type::{EntryReferenceSubType, ReferenceType}, resolve::{find_context_file, FindContextFileResult}, source_asset::SourceAssetVc, @@ -118,7 +119,13 @@ impl Asset for PostCssTransformedAsset { #[turbo_tasks::function] async fn content(self_vc: PostCssTransformedAssetVc) -> Result { - Ok(self_vc.process().await?.content) + let this = self_vc.await?; + Ok(self_vc + .process() + .issue_context(this.source.ident().path(), "PostCSS processing") + .await? + .await? + .content) } } diff --git a/crates/turbopack-tests/tests/snapshot.rs b/crates/turbopack-tests/tests/snapshot.rs index 4f0d87616428b..c84fd5594c3e2 100644 --- a/crates/turbopack-tests/tests/snapshot.rs +++ b/crates/turbopack-tests/tests/snapshot.rs @@ -104,11 +104,11 @@ async fn run(resource: &'static str) -> Result<()> { .await?; let plain_issues = captured_issues - .iter() - .map(|issue_vc| async move { + .iter_with_shortest_path() + .map(|(issue_vc, path)| async move { Ok(( - issue_vc.into_plain().await?, - issue_vc.into_plain().dbg().await?, + issue_vc.into_plain(path).await?, + issue_vc.into_plain(path).dbg().await?, )) }) .try_join() diff --git a/crates/turbopack-tests/tests/snapshot/export-alls/cjs-2/issues/unexpected export __star__-b4c42d.txt b/crates/turbopack-tests/tests/snapshot/export-alls/cjs-2/issues/unexpected export __star__-b4c42d.txt index 4249b8d285135..4d82202bd482f 100644 --- a/crates/turbopack-tests/tests/snapshot/export-alls/cjs-2/issues/unexpected export __star__-b4c42d.txt +++ b/crates/turbopack-tests/tests/snapshot/export-alls/cjs-2/issues/unexpected export __star__-b4c42d.txt @@ -8,4 +8,7 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [], + ), } \ No newline at end of file diff --git a/crates/turbopack-tests/tests/snapshot/export-alls/cjs-script/issues/unexpected export __star__-30c177.txt b/crates/turbopack-tests/tests/snapshot/export-alls/cjs-script/issues/unexpected export __star__-30c177.txt index 499d725706229..2e59a9c516941 100644 --- a/crates/turbopack-tests/tests/snapshot/export-alls/cjs-script/issues/unexpected export __star__-30c177.txt +++ b/crates/turbopack-tests/tests/snapshot/export-alls/cjs-script/issues/unexpected export __star__-30c177.txt @@ -8,4 +8,7 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [], + ), } \ No newline at end of file diff --git a/crates/turbopack-tests/tests/snapshot/imports/json/issues/Code generation for chunk item errored-798a9e.txt b/crates/turbopack-tests/tests/snapshot/imports/json/issues/Code generation for chunk item errored-798a9e.txt index a14c644d4e476..36533d45dd1c1 100644 --- a/crates/turbopack-tests/tests/snapshot/imports/json/issues/Code generation for chunk item errored-798a9e.txt +++ b/crates/turbopack-tests/tests/snapshot/imports/json/issues/Code generation for chunk item errored-798a9e.txt @@ -8,4 +8,7 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [], + ), } \ No newline at end of file diff --git a/crates/turbopack-tests/tests/snapshot/imports/resolve_error_cjs/issues/Error resolving commonjs request-69aa17.txt b/crates/turbopack-tests/tests/snapshot/imports/resolve_error_cjs/issues/Error resolving commonjs request-69aa17.txt index 867bc25118453..5046e04f28000 100644 --- a/crates/turbopack-tests/tests/snapshot/imports/resolve_error_cjs/issues/Error resolving commonjs request-69aa17.txt +++ b/crates/turbopack-tests/tests/snapshot/imports/resolve_error_cjs/issues/Error resolving commonjs request-69aa17.txt @@ -8,4 +8,7 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [], + ), } \ No newline at end of file diff --git a/crates/turbopack-tests/tests/snapshot/imports/resolve_error_esm/issues/Error resolving EcmaScript Modules request-cd3585.txt b/crates/turbopack-tests/tests/snapshot/imports/resolve_error_esm/issues/Error resolving EcmaScript Modules request-cd3585.txt index f797c849313c7..4258654d1eba2 100644 --- a/crates/turbopack-tests/tests/snapshot/imports/resolve_error_esm/issues/Error resolving EcmaScript Modules request-cd3585.txt +++ b/crates/turbopack-tests/tests/snapshot/imports/resolve_error_esm/issues/Error resolving EcmaScript Modules request-cd3585.txt @@ -8,4 +8,7 @@ PlainIssue { documentation_link: "", source: None, sub_issues: [], + processing_path: Some( + [], + ), } \ No newline at end of file