Skip to content

Commit

Permalink
Add option for inlining typeof window (vercel/turborepo#8211)
Browse files Browse the repository at this point in the history
### Description

In Next.js with webpack we have a feature to inline `typeof window` in
server/client environments. That allows e.g.

```
"use client"
if(typeof window === 'undefined') {
  import('server-only-package')
}
```

Today with Turbopack above code causes `server-only-package` to still be
loaded in the browser compilation. Which causes issues like
#66058 where the `if`/`else`
condition is used to use different modules for server and client.

This change is required to pass the test in
#66058

Pushing up the Next.js changes in that PR.

<!--
  ✍️ Write a short summary of your work.
  If necessary, include relevant screenshots.
-->

### Testing Instructions

<!--
  Give a quick description of steps to test your changes.
-->

---------

Co-authored-by: Donny/강동윤 <[email protected]>
  • Loading branch information
timneutkens and kdy1 authored May 25, 2024
1 parent c0a05bb commit 6c91ba5
Show file tree
Hide file tree
Showing 4 changed files with 36 additions and 1 deletion.
1 change: 1 addition & 0 deletions crates/turbopack-ecmascript/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,7 @@ swc_core = { workspace = true, features = [
"ecma_transforms",
"ecma_transforms_module",
"ecma_transforms_react",
"ecma_transforms_optimization",
"ecma_transforms_typescript",
"ecma_transforms_proposal",
"ecma_quote",
Expand Down
18 changes: 17 additions & 1 deletion crates/turbopack-ecmascript/src/transform/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,13 +3,15 @@ use std::{fmt::Debug, hash::Hash, sync::Arc};
use anyhow::Result;
use async_trait::async_trait;
use swc_core::{
atoms::JsWord,
base::SwcComments,
common::{chain, comments::Comments, util::take::Take, Mark, SourceMap},
common::{chain, collections::AHashMap, comments::Comments, util::take::Take, Mark, SourceMap},
ecma::{
ast::{Module, ModuleItem, Program, Script},
preset_env::{self, Targets},
transforms::{
base::{feature::FeatureFlag, helpers::inject_helpers, Assumptions},
optimization::inline_globals2,
react::react,
},
visit::{FoldWith, VisitMutWith},
Expand Down Expand Up @@ -39,6 +41,9 @@ pub enum EcmascriptInputTransform {
// swc.jsc.transform.react.runtime,
runtime: Vc<Option<String>>,
},
GlobalTypeofs {
window_value: String,
},
// These options are subset of swc_core::ecma::transforms::typescript::Config, but
// it doesn't derive `Copy` so repeating values in here
TypeScript {
Expand Down Expand Up @@ -134,6 +139,17 @@ impl EcmascriptInputTransform {
..
} = ctx;
match self {
EcmascriptInputTransform::GlobalTypeofs { window_value } => {
let mut typeofs: AHashMap<JsWord, JsWord> = Default::default();
typeofs.insert("window".into(), JsWord::from(&**window_value));

program.visit_mut_with(&mut inline_globals2(
Default::default(),
Default::default(),
Default::default(),
Arc::new(typeofs),
));
}
EcmascriptInputTransform::React {
development,
refresh,
Expand Down
10 changes: 10 additions & 0 deletions crates/turbopack/src/module_options/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,7 @@ impl ModuleOptions {
import_externals,
ignore_dynamic_requests,
use_swc_css,
ref enable_typeof_window_inlining,
..
} = *module_options_context.await?;
if !rules.is_empty() {
Expand Down Expand Up @@ -130,6 +131,15 @@ impl ModuleOptions {
transforms.push(EcmascriptInputTransform::PresetEnv(env));
}

if let Some(enable_typeof_window_inlining) = enable_typeof_window_inlining {
transforms.push(EcmascriptInputTransform::GlobalTypeofs {
window_value: match enable_typeof_window_inlining {
TypeofWindow::Object => "object".to_string(),
TypeofWindow::Undefined => "undefined".to_string(),
},
});
}

let ts_transform = if let Some(options) = enable_typescript_transform {
let options = options.await?;
Some(EcmascriptInputTransform::TypeScript {
Expand Down
8 changes: 8 additions & 0 deletions crates/turbopack/src/module_options/module_options_context.rs
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,13 @@ pub enum DecoratorsKind {
Ecma,
}

/// The types when replacing `typeof window` with a constant.
#[derive(Clone, PartialEq, Eq, Debug, TraceRawVcs, Serialize, Deserialize)]
pub enum TypeofWindow {
Object,
Undefined,
}

/// Configuration options for the decorators transform.
/// This is not part of Typescript transform: while there are typescript
/// specific transforms (legay decorators), there is an ecma decorator transform
Expand Down Expand Up @@ -105,6 +112,7 @@ pub struct JsxTransformOptions {
#[derive(Default, Clone)]
#[serde(default)]
pub struct ModuleOptionsContext {
pub enable_typeof_window_inlining: Option<TypeofWindow>,
pub enable_jsx: Option<Vc<JsxTransformOptions>>,
pub enable_postcss_transform: Option<Vc<PostCssTransformOptions>>,
pub enable_webpack_loaders: Option<Vc<WebpackLoadersOptions>>,
Expand Down

0 comments on commit 6c91ba5

Please sign in to comment.