Skip to content

Commit

Permalink
refactor(turbopack/next-api): Make VcArc use OperationVc
Browse files Browse the repository at this point in the history
  • Loading branch information
bgw committed Jan 13, 2025
1 parent 1d7e10d commit f8efda7
Show file tree
Hide file tree
Showing 11 changed files with 125 additions and 136 deletions.
26 changes: 12 additions & 14 deletions crates/napi/src/next_api/endpoint.rs
Original file line number Diff line number Diff line change
Expand Up @@ -10,9 +10,7 @@ use next_api::{
},
};
use tracing::Instrument;
use turbo_tasks::{
get_effects, Completion, Effects, OperationVc, ReadRef, ResolvedVc, Vc, VcValueType,
};
use turbo_tasks::{get_effects, Completion, Effects, OperationVc, ReadRef, Vc, VcValueType};
use turbopack_core::{
diagnostics::PlainDiagnostic,
error::PrettyPrintError,
Expand Down Expand Up @@ -91,10 +89,10 @@ impl From<Option<WrittenEndpoint>> for NapiWrittenEndpoint {
// some async functions (in this case `endpoint_write_to_disk`) can cause
// higher-ranked lifetime errors. See https://github.com/rust-lang/rust/issues/102211
// 2. the type_complexity clippy lint.
pub struct ExternalEndpoint(pub VcArc<Vc<Box<dyn Endpoint>>>);
pub struct ExternalEndpoint(pub VcArc<Box<dyn Endpoint>>);

impl Deref for ExternalEndpoint {
type Target = VcArc<Vc<Box<dyn Endpoint>>>;
type Target = VcArc<Box<dyn Endpoint>>;

fn deref(&self) -> &Self::Target {
&self.0
Expand Down Expand Up @@ -135,9 +133,9 @@ struct WrittenEndpointWithIssues {

#[turbo_tasks::function]
async fn get_written_endpoint_with_issues(
endpoint: ResolvedVc<Box<dyn Endpoint>>,
endpoint_op: OperationVc<Box<dyn Endpoint>>,
) -> Result<Vc<WrittenEndpointWithIssues>> {
let write_to_disk = endpoint_write_to_disk_operation(endpoint);
let write_to_disk = endpoint_write_to_disk_operation(endpoint_op);
let (written, issues, diagnostics, effects) =
strongly_consistent_catch_collectables(write_to_disk).await?;
Ok(WrittenEndpointWithIssues {
Expand All @@ -155,10 +153,10 @@ pub async fn endpoint_write_to_disk(
#[napi(ts_arg_type = "{ __napiType: \"Endpoint\" }")] endpoint: External<ExternalEndpoint>,
) -> napi::Result<TurbopackResult<NapiWrittenEndpoint>> {
let turbo_tasks = endpoint.turbo_tasks().clone();
let endpoint = ***endpoint;
let endpoint_op = ***endpoint;
let (written, issues, diags) = turbo_tasks
.run_once(async move {
let operation = get_written_endpoint_with_issues(endpoint);
let operation = get_written_endpoint_with_issues(endpoint_op);
let WrittenEndpointWithIssues {
written,
issues,
Expand Down Expand Up @@ -191,8 +189,8 @@ pub fn endpoint_server_changed_subscribe(
func,
move || {
async move {
let operation = subscribe_issues_and_diags(endpoint, issues);
let result = operation.strongly_consistent().await?;
let vc = subscribe_issues_and_diags(endpoint, issues);
let result = vc.strongly_consistent().await?;
result.effects.apply().await?;
Ok(result)
}
Expand Down Expand Up @@ -241,10 +239,10 @@ impl Eq for EndpointIssuesAndDiags {}

#[turbo_tasks::function]
async fn subscribe_issues_and_diags(
endpoint: ResolvedVc<Box<dyn Endpoint>>,
endpoint_op: OperationVc<Box<dyn Endpoint>>,
should_include_issues: bool,
) -> Result<Vc<EndpointIssuesAndDiags>> {
let changed = endpoint_server_changed_operation(endpoint);
let changed = endpoint_server_changed_operation(endpoint_op);

if should_include_issues {
let (changed_value, issues, diagnostics, effects) =
Expand Down Expand Up @@ -280,7 +278,7 @@ pub fn endpoint_client_changed_subscribe(
func,
move || {
async move {
let changed = endpoint.client_changed();
let changed = endpoint.connect().client_changed();
// We don't capture issues and diagnostics here since we don't want to be
// notified when they change
changed.strongly_consistent().await?;
Expand Down
62 changes: 36 additions & 26 deletions crates/napi/src/next_api/project.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,11 +8,14 @@ use napi::{
};
use next_api::{
entrypoints::Entrypoints,
operation::{
EntrypointsOperation, InstrumentationOperation, MiddlewareOperation, RouteOperation,
},
project::{
DefineEnv, DraftModeOptions, Instrumentation, Middleware, PartialProjectOptions, Project,
ProjectContainer, ProjectOptions, WatchOptions,
DefineEnv, DraftModeOptions, PartialProjectOptions, Project, ProjectContainer,
ProjectOptions, WatchOptions,
},
route::{Endpoint, Route},
route::Endpoint,
};
use next_core::tracing_presets::{
TRACING_NEXT_OVERVIEW_TARGETS, TRACING_NEXT_TARGETS, TRACING_NEXT_TURBOPACK_TARGETS,
Expand All @@ -25,7 +28,8 @@ use tracing::Instrument;
use tracing_subscriber::{layer::SubscriberExt, util::SubscriberInitExt, Registry};
use turbo_rcstr::RcStr;
use turbo_tasks::{
get_effects, Completion, Effects, ReadRef, ResolvedVc, TransientInstance, UpdateInfo, Vc,
get_effects, Completion, Effects, OperationVc, ReadRef, ResolvedVc, TransientInstance,
UpdateInfo, Vc,
};
use turbo_tasks_fs::{
get_relative_path_to, util::uri_from_file, DiskFileSystem, FileContent, FileSystem,
Expand Down Expand Up @@ -517,56 +521,56 @@ struct NapiRoute {
}

impl NapiRoute {
fn from_route(pathname: String, value: Route, turbo_tasks: &NextTurboTasks) -> Self {
let convert_endpoint = |endpoint: Vc<Box<dyn Endpoint>>| {
fn from_route(pathname: String, value: RouteOperation, turbo_tasks: &NextTurboTasks) -> Self {
let convert_endpoint = |endpoint: OperationVc<Box<dyn Endpoint>>| {
Some(External::new(ExternalEndpoint(VcArc::new(
turbo_tasks.clone(),
endpoint,
))))
};
match value {
Route::Page {
RouteOperation::Page {
html_endpoint,
data_endpoint,
} => NapiRoute {
pathname,
r#type: "page",
html_endpoint: convert_endpoint(*html_endpoint),
data_endpoint: convert_endpoint(*data_endpoint),
html_endpoint: convert_endpoint(html_endpoint),
data_endpoint: convert_endpoint(data_endpoint),
..Default::default()
},
Route::PageApi { endpoint } => NapiRoute {
RouteOperation::PageApi { endpoint } => NapiRoute {
pathname,
r#type: "page-api",
endpoint: convert_endpoint(*endpoint),
endpoint: convert_endpoint(endpoint),
..Default::default()
},
Route::AppPage(pages) => NapiRoute {
RouteOperation::AppPage(pages) => NapiRoute {
pathname,
r#type: "app-page",
pages: Some(
pages
.into_iter()
.map(|page_route| AppPageNapiRoute {
original_name: Some(page_route.original_name),
original_name: Some(page_route.original_name.into_owned()),
html_endpoint: convert_endpoint(page_route.html_endpoint),
rsc_endpoint: convert_endpoint(page_route.rsc_endpoint),
})
.collect(),
),
..Default::default()
},
Route::AppRoute {
RouteOperation::AppRoute {
original_name,
endpoint,
} => NapiRoute {
pathname,
original_name: Some(original_name),
original_name: Some(original_name.into_owned()),
r#type: "app-route",
endpoint: convert_endpoint(*endpoint),
endpoint: convert_endpoint(endpoint),
..Default::default()
},
Route::Conflict => NapiRoute {
RouteOperation::Conflict => NapiRoute {
pathname,
r#type: "conflict",
..Default::default()
Expand All @@ -581,7 +585,7 @@ struct NapiMiddleware {
}

impl NapiMiddleware {
fn from_middleware(value: &Middleware, turbo_tasks: &NextTurboTasks) -> Result<Self> {
fn from_middleware(value: &MiddlewareOperation, turbo_tasks: &NextTurboTasks) -> Result<Self> {
Ok(NapiMiddleware {
endpoint: External::new(ExternalEndpoint(VcArc::new(
turbo_tasks.clone(),
Expand All @@ -598,7 +602,10 @@ struct NapiInstrumentation {
}

impl NapiInstrumentation {
fn from_instrumentation(value: &Instrumentation, turbo_tasks: &NextTurboTasks) -> Result<Self> {
fn from_instrumentation(
value: &InstrumentationOperation,
turbo_tasks: &NextTurboTasks,
) -> Result<Self> {
Ok(NapiInstrumentation {
node_js: External::new(ExternalEndpoint(VcArc::new(
turbo_tasks.clone(),
Expand All @@ -622,9 +629,9 @@ struct NapiEntrypoints {
pub pages_error_endpoint: External<ExternalEndpoint>,
}

#[turbo_tasks::value(serialization = "none", local)]
#[turbo_tasks::value(serialization = "none")]
struct EntrypointsWithIssues {
entrypoints: ReadRef<Entrypoints>,
entrypoints: ReadRef<EntrypointsOperation>,
issues: Arc<Vec<ReadRef<PlainIssue>>>,
diagnostics: Arc<Vec<ReadRef<PlainDiagnostic>>>,
effects: Arc<Effects>,
Expand All @@ -634,7 +641,8 @@ struct EntrypointsWithIssues {
async fn get_entrypoints_with_issues(
container: ResolvedVc<ProjectContainer>,
) -> Result<Vc<EntrypointsWithIssues>> {
let entrypoints_operation = project_container_entrypoints_operation(container);
let entrypoints_operation =
EntrypointsOperation::new(project_container_entrypoints_operation(container));
let entrypoints = entrypoints_operation
.connect()
.strongly_consistent()
Expand All @@ -653,6 +661,8 @@ async fn get_entrypoints_with_issues(

#[turbo_tasks::function(operation)]
fn project_container_entrypoints_operation(
// the container is a long-lived object with internally mutable state, there's no risk of it
// becoming stale
container: ResolvedVc<ProjectContainer>,
) -> Vc<Entrypoints> {
container.entrypoints()
Expand Down Expand Up @@ -710,15 +720,15 @@ pub fn project_entrypoints_subscribe(
.transpose()?,
pages_document_endpoint: External::new(ExternalEndpoint(VcArc::new(
turbo_tasks.clone(),
*entrypoints.pages_document_endpoint,
entrypoints.pages_document_endpoint,
))),
pages_app_endpoint: External::new(ExternalEndpoint(VcArc::new(
turbo_tasks.clone(),
*entrypoints.pages_app_endpoint,
entrypoints.pages_app_endpoint,
))),
pages_error_endpoint: External::new(ExternalEndpoint(VcArc::new(
turbo_tasks.clone(),
*entrypoints.pages_error_endpoint,
entrypoints.pages_error_endpoint,
))),
},
issues: issues
Expand Down Expand Up @@ -760,7 +770,7 @@ async fn hmr_update(
}

#[turbo_tasks::function(operation)]
async fn project_hmr_update_operation(
fn project_hmr_update_operation(
project: ResolvedVc<Project>,
identifier: RcStr,
state: ResolvedVc<VersionState>,
Expand Down
9 changes: 4 additions & 5 deletions crates/napi/src/next_api/utils.rs
Original file line number Diff line number Diff line change
Expand Up @@ -175,13 +175,12 @@ pub fn create_turbo_tasks(
#[derive(Clone)]
pub struct VcArc<T> {
turbo_tasks: NextTurboTasks,
/// The Vc. Must be resolved, otherwise you are referencing an inactive
/// operation.
vc: T,
/// The Vc. Must be unresolved, otherwise you are referencing an inactive operation.
vc: OperationVc<T>,
}

impl<T> VcArc<T> {
pub fn new(turbo_tasks: NextTurboTasks, vc: T) -> Self {
pub fn new(turbo_tasks: NextTurboTasks, vc: OperationVc<T>) -> Self {
Self { turbo_tasks, vc }
}

Expand All @@ -191,7 +190,7 @@ impl<T> VcArc<T> {
}

impl<T> Deref for VcArc<T> {
type Target = T;
type Target = OperationVc<T>;

fn deref(&self) -> &Self::Target {
&self.vc
Expand Down
14 changes: 7 additions & 7 deletions crates/next-api/src/app.rs
Original file line number Diff line number Diff line change
Expand Up @@ -732,8 +732,8 @@ pub fn app_entry_point_to_route(
pages
.into_iter()
.map(|page| AppPageRoute {
original_name: page.to_string(),
html_endpoint: Vc::upcast(
original_name: RcStr::from(page.to_string()),
html_endpoint: ResolvedVc::upcast(
AppEndpoint {
ty: AppEndpointType::Page {
ty: AppPageEndpointType::Html,
Expand All @@ -742,9 +742,9 @@ pub fn app_entry_point_to_route(
app_project,
page: page.clone(),
}
.cell(),
.resolved_cell(),
),
rsc_endpoint: Vc::upcast(
rsc_endpoint: ResolvedVc::upcast(
AppEndpoint {
ty: AppEndpointType::Page {
ty: AppPageEndpointType::Rsc,
Expand All @@ -753,7 +753,7 @@ pub fn app_entry_point_to_route(
app_project,
page,
}
.cell(),
.resolved_cell(),
),
})
.collect(),
Expand All @@ -763,7 +763,7 @@ pub fn app_entry_point_to_route(
path,
root_layouts,
} => Route::AppRoute {
original_name: page.to_string(),
original_name: page.to_string().into(),
endpoint: ResolvedVc::upcast(
AppEndpoint {
ty: AppEndpointType::Route { path, root_layouts },
Expand All @@ -774,7 +774,7 @@ pub fn app_entry_point_to_route(
),
},
AppEntrypoint::AppMetadata { page, metadata } => Route::AppRoute {
original_name: page.to_string(),
original_name: page.to_string().into(),
endpoint: ResolvedVc::upcast(
AppEndpoint {
ty: AppEndpointType::Metadata { metadata },
Expand Down
2 changes: 1 addition & 1 deletion crates/next-api/src/entrypoints.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ use crate::{
route::{Endpoint, Route},
};

#[turbo_tasks::value(shared, local)]
#[turbo_tasks::value(shared)]
pub struct Entrypoints {
pub routes: FxIndexMap<RcStr, Route>,
pub middleware: Option<Middleware>,
Expand Down
12 changes: 5 additions & 7 deletions crates/next-api/src/global_module_id_strategy.rs
Original file line number Diff line number Diff line change
Expand Up @@ -31,14 +31,12 @@ impl GlobalModuleIdStrategyBuilder {
preprocessed_module_ids.push(preprocess_module_ids(*entrypoints.pages_document_endpoint));

if let Some(middleware) = &entrypoints.middleware {
preprocessed_module_ids.push(preprocess_module_ids(middleware.endpoint));
preprocessed_module_ids.push(preprocess_module_ids(*middleware.endpoint));
}

if let Some(instrumentation) = &entrypoints.instrumentation {
let node_js = instrumentation.node_js;
let edge = instrumentation.edge;
preprocessed_module_ids.push(preprocess_module_ids(node_js));
preprocessed_module_ids.push(preprocess_module_ids(edge));
preprocessed_module_ids.push(preprocess_module_ids(*instrumentation.node_js));
preprocessed_module_ids.push(preprocess_module_ids(*instrumentation.edge));
}

for (_, route) in entrypoints.routes.iter() {
Expand All @@ -56,9 +54,9 @@ impl GlobalModuleIdStrategyBuilder {
Route::AppPage(page_routes) => {
for page_route in page_routes {
preprocessed_module_ids
.push(preprocess_module_ids(page_route.html_endpoint));
.push(preprocess_module_ids(*page_route.html_endpoint));
preprocessed_module_ids
.push(preprocess_module_ids(page_route.rsc_endpoint));
.push(preprocess_module_ids(*page_route.rsc_endpoint));
}
}
Route::AppRoute {
Expand Down
1 change: 1 addition & 0 deletions crates/next-api/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ mod loadable_manifest;
mod middleware;
mod module_graph;
mod nft_json;
pub mod operation;
mod pages;
pub mod paths;
pub mod project;
Expand Down
Loading

0 comments on commit f8efda7

Please sign in to comment.