logo
  • Guide
  • Config
  • Plugin
  • API
  • Examples
  • Community
  • Modern.js 2.x Docs
  • English
    • 简体中文
    • English
    • Start
      Introduction
      Quick Start
      Upgrading
      Glossary
      Tech Stack
      Core Concept
      Page Entry
      Build Engine
      Web Server
      Basic Features
      Routes
      Routing
      Config Routes
      Data Solution
      Data Fetching
      Data Writing
      Data Caching
      Rendering
      Server-Side Rendering
      Streaming SSR
      Rendering Cache
      Static Site Generation
      Render Preprocessing
      Styling
      Styling
      Use CSS Modules
      Using CSS-in-JS
      Using Tailwind CSS
      HTML Template
      Import Static Assets
      Import JSON Files
      Import SVG Assets
      Import Wasm Assets
      Debug
      Data Mocking
      Network Proxy
      Using Rsdoctor
      Using Storybook
      Testing
      Playwright
      Vitest
      Jest
      Cypress
      Path Alias
      Environment Variables
      Output Files
      Deploy Application
      Advanced Features
      Using Rspack
      Using BFF
      Basic Usage
      Runtime Framework
      Extend BFF Server
      Extend Request SDK
      File Upload
      Cross-Project Invocation
      Optimize Page Performance
      Code Splitting
      Inline Static Assets
      Bundle Size Optimization
      React Compiler
      Improve Build Performance
      Browser Compatibility
      Low-Level Tools
      Source Code Build Mode
      Server Monitor
      Monitors
      Logs Events
      Metrics Events
      Internationalization
      Basic Concepts
      Quick Start
      Configuration
      Locale Detection
      Resource Loading
      Routing Integration
      API Reference
      Advanced Usage
      Best Practices
      Custom Web Server
      Topic Detail
      Module Federation
      Introduction
      Getting Started
      Application-Level Modules
      Server-Side Rendering
      Deployment
      Integrating Internationalization
      FAQ
      Dependencies FAQ
      CLI FAQ
      Build FAQ
      HMR FAQ
      Deprecated
      📝 Edit this page
      Previous pageLow-Level ToolsNext pageMonitors

      #Source Code Build Mode

      The source code build mode is used in the monorepo development scenario, allowing developers to directly reference the source code of other sub-projects within the monorepo for development.

      #Use Cases

      In a monorepo, there are two main ways for projects to reference each other: artifact referencing and source code referencing. Let's use a simple monorepo as an example to illustrate the use case of source code referencing.

      For example, the monorepo contains an app application and a lib:

      monorepo
      ├── app
      └── lib
          └── src
              └── index.ts

      The app is built using Modern.js and relies on some methods from the lib:

      {
        "name": "app",
        "dependencies": {
          "lib": "workspace:*"
        }
      }

      #Artifact Referencing

      When using artifact referencing, the current project references the artifacts built from other sub-projects.

      In the example above, the lib is written in TypeScript. Typically, we need to build the lib's code in advance to generate JavaScript artifacts so that the app can reference it correctly. When the lib's code is updated, we need to rebuild it (or use tsc --watch) to ensure that the app can use the latest code.

      The advantages of this approach are:

      • The build processes of each sub-project are completely independent and can have different build configurations.
      • Build caching can be applied to individual sub-projects.

      The disadvantages are:

      • The HMR chain becomes longer during local development.
      • The process becomes cumbersome when a project contains multiple lib packages.

      #Source Code Referencing

      When using source code referencing, the current project references the source code of other sub-projects for building.

      In the example mentioned earlier, when you enable the source code build mode and add the relevant configuration in the lib directory, Modern.js will automatically reference the src/index.ts source code of the lib. This means that you don't need to build the lib's code in advance, and when the source code of the lib is updated, it can trigger automatic hot updates for the app.

      The advantages of this approach are:

      • The sub-project does not rely on a build tool and does not require build configurations. The code of the sub-project will be compiled by the build tool of the current project.
      • There is no need to execute the build process for the sub-projects in advance.
      • HMR is more efficient during local development.

      The disadvantages are:

      • The current project needs to support syntax features used by sub-projects and follow the same syntax specifications, such as using a consistent version of decorator syntax. If the current project and sub-projects require different build configurations, building from source code may not be suitable.
      • The current project requires compiling more code, which may result in longer build times.

      #Building from Source Code

      #Enabling Configuration

      You can enable source code build mode by setting experiments.sourceBuild to true.

      export default {
        experiments: {
          sourceBuild: true,
        },
      };

      #Configuring Sub-projects

      When the source code build mode is enabled, the Modern.js will prioritize reading the file specified in the source field of the sub-project during the build process. Therefore, you need to configure the source field in the package.json file of the sub-project and point it to the source code file.

      For example, in the following example, when the lib package is referenced, the ./src/index.ts file will be read for building:

      package.json
      {
        "name": "lib",
        "main": "./dist/index.js",
        "source": "./src/index.ts"
      }

      If the sub-project uses exports field, you also need to add the source field in the exports field.

      package.json
      {
        "name": "lib",
        "exports": {
          ".": {
            "source": "./src/index.ts",
            "default": "./dist/index.js"
          },
          "./features": {
            "source": "./src/features/index.ts",
            "default": "./dist/features/index.js"
          }
        }
      }

      #Configure Project Reference

      In a TypeScript project, you need to use the capability provided by TypeScript called Project Reference. It helps you develop source code more effectively.

      #Introduction

      Project reference provides the following capabilities:

      • It allows TypeScript to correctly recognize the types of other sub-projects without the need to build them.
      • When you navigate the code in VS Code, it automatically takes you to the corresponding source code file of the module.
      • Modern.js reads the project reference configuration and automatically recognizes the tsconfig.compilerOptions.path configuration of the sub-project, so that the use of aliases in the sub-project works correctly.

      #Example

      In the example mentioned earlier, since the app project references the lib sub-project, we need to configure the references options in the app project's tsconfig.json to point to the relative directory of the lib:

      app/tsconfig.json
      {
        "references": [
          {
            "path": "../lib"
          }
        ]
      }

      At the same time, we need to set composite to true in the lib project's tsconfig.json:

      lib/A/tsconfig.json
      {
        "compilerOptions": {
          "composite": true
        },
      }

      After adding these two options, the project reference is already configured. You can restart VS Code to see the effects of the configuration.

      Note that the above example is a simplified one. In real monorepo projects, there may be more complex dependency relationships. You need to add a complete references configuration for the functionality to work correctly.

      Tip

      If you want to learn more about project reference, please refer to the official documentation on TypeScript - Project References.

      #Caveat

      When using source code build mode, there are a few things to keep in mind:

      1. Ensure that the current project can compile the syntax or features used in the sub-project. For example, if the sub-project uses Stylus to write CSS, the current app needs to support Stylus compilation.
      2. Ensure that the current project has the same code syntax and features as the sub-project, such as consistent syntax versions for decorators.
      3. Source code building may have some limitations. When encountering issues, you can remove the source field from the sub-project's package.json and debug using the built artifacts of the sub-project.
      4. When composite: true is enabled, TypeScript will generate *.tsbuildinfo temporary files. You need to add these temporary files to the .gitignore file.
      .gitignore
      *.tsbuildinfo