| # TypeScript rules for Bazel |
| |
| Circle CI | Bazel CI |
| :---: | :---: |
| [](https://circleci.com/gh/bazelbuild/rules_typescript) | [](https://buildkite.com/bazel/typescript-rules-typescript-postsubmit) |
| |
| **WARNING: this is beta-quality software. Breaking changes are likely. Not recommended for production use without expert support.** |
| |
| The TypeScript rules integrate the TypeScript compiler with Bazel. |
| |
| ## API Docs |
| |
| Generated documentation for using each rule is at: |
| http://tsetse.info/api/ |
| |
| ## Installation |
| |
| First, install a current Bazel distribution. |
| |
| Add the `@bazel/typescript` npm package to your `package.json` `devDependencies`. |
| Optionally add the `@bazel/karma` npm package if you would like to use the |
| `ts_web_test`, `ts_web_test_suite`, `karma_web_test` or `karma_web_test_suite` rules. |
| |
| ``` |
| { |
| ... |
| "devDependencies": { |
| "@bazel/typescript": "0.23.0", |
| "@bazel/karma": "0.23.0", |
| ... |
| }, |
| ... |
| } |
| ``` |
| |
| Create a `BUILD.bazel` file in your project root: |
| |
| ```python |
| package(default_visibility = ["//visibility:public"]) |
| exports_files(["tsconfig.json"]) |
| ``` |
| |
| Next create a `WORKSPACE` file in your project root (or edit the existing one) |
| containing: |
| |
| ```python |
| load("@bazel_tools//tools/build_defs/repo:http.bzl", "http_archive") |
| |
| # Fetch rules_nodejs |
| http_archive( |
| name = "build_bazel_rules_nodejs", |
| urls = ["https://github.com/bazelbuild/rules_nodejs/archive/0.16.8.zip"], |
| strip_prefix = "rules_nodejs-0.16.8", |
| ) |
| |
| # Setup the NodeJS toolchain |
| load("@build_bazel_rules_nodejs//:defs.bzl", "node_repositories", "yarn_install") |
| node_repositories() |
| |
| # Setup Bazel managed npm dependencies with the `yarn_install` rule. |
| # The name of this rule should be set to `npm` so that `ts_library` and `ts_web_test_suite` |
| # can find your npm dependencies by default in the `@npm` workspace. You may |
| # also use the `npm_install` rule with a `package-lock.json` file if you prefer. |
| # See https://github.com/bazelbuild/rules_nodejs#dependencies for more info. |
| yarn_install( |
| name = "npm", |
| package_json = "//:package.json", |
| yarn_lock = "//:yarn.lock", |
| ) |
| |
| # Install all Bazel dependencies needed for npm packages that supply Bazel rules |
| load("@npm//:install_bazel_dependencies.bzl", "install_bazel_dependencies") |
| install_bazel_dependencies() |
| |
| # Fetch transitive Bazel dependencies of build_bazel_rules_typescript |
| load("@build_bazel_rules_typescript//:package.bzl", "rules_typescript_dependencies") |
| rules_typescript_dependencies() |
| |
| # Fetch transitive Bazel dependencies of build_bazel_rules_karma |
| # ONLY REQUIRED if you are using the @bazel/karma npm package |
| load("@build_bazel_rules_karma//:package.bzl", "rules_karma_dependencies") |
| rules_karma_dependencies() |
| |
| # Setup TypeScript toolchain |
| load("@build_bazel_rules_typescript//:defs.bzl", "ts_setup_workspace") |
| ts_setup_workspace() |
| |
| # Setup web testing, choose browsers we can test on |
| # ONLY REQUIRED if you are using the @bazel/karma npm package |
| load("@io_bazel_rules_webtesting//web:repositories.bzl", "browser_repositories", "web_test_repositories") |
| |
| web_test_repositories() |
| browser_repositories( |
| chromium = True, |
| ) |
| ``` |
| |
| # Self-managed npm dependencies |
| |
| We recommend you use Bazel managed dependencies but if you would like |
| Bazel to also install a `node_modules` in your workspace you can also |
| point the `node_repositories` repository rule in your WORKSPACE file to |
| your `package.json`. |
| |
| ```python |
| node_repositories(package_json = ["//:package.json"]) |
| ``` |
| |
| You can then run `yarn` in your workspace with: |
| |
| ```sh |
| $ bazel run @nodejs//:yarn |
| ``` |
| |
| To use your workspace `node_modules` folder as a dependency in `ts_library` and |
| other rules, add the following to your root `BUILD.bazel` file: |
| |
| ```python |
| filegroup( |
| name = "node_modules", |
| srcs = glob( |
| include = [ |
| "node_modules/**/*.js", |
| "node_modules/**/*.d.ts", |
| "node_modules/**/*.json", |
| "node_modules/.bin/*", |
| ], |
| exclude = [ |
| # Files under test & docs may contain file names that |
| # are not legal Bazel labels (e.g., |
| # node_modules/ecstatic/test/public/中文/檔案.html) |
| "node_modules/**/test/**", |
| "node_modules/**/docs/**", |
| # Files with spaces in the name are not legal Bazel labels |
| "node_modules/**/* */**", |
| "node_modules/**/* *", |
| ], |
| ), |
| ) |
| |
| # Create a tsc_wrapped compiler rule to use in the ts_library |
| # compiler attribute when using self-managed dependencies |
| nodejs_binary( |
| name = "@bazel/typescript/tsc_wrapped", |
| entry_point = "@bazel/typescript/internal/tsc_wrapped/tsc_wrapped.js", |
| # The --expose-gc node option is required for tsc_wrapped |
| templated_args = ["--node_options=--expose-gc"], |
| # Point bazel to your node_modules to find the entry point |
| node_modules = ["//:node_modules"], |
| ) |
| |
| # Create a karma rule to use in ts_web_test_suite karma |
| # attribute when using self-managed dependencies |
| nodejs_binary( |
| name = "karma/karma", |
| entry_point = "karma/bin/karma", |
| # Point bazel to your node_modules to find the entry point |
| node_modules = ["//:node_modules"], |
| ) |
| ``` |
| |
| See https://github.com/bazelbuild/rules_nodejs#dependencies for more information on |
| managing npm dependencies with Bazel. |
| |
| ## Usage |
| |
| ### Compiling TypeScript: `ts_library` |
| |
| The `ts_library` rule invokes the TypeScript compiler on one compilation unit, |
| or "library" (generally one directory of source files). |
| |
| Create a `BUILD` file next to your sources: |
| |
| ```python |
| package(default_visibility=["//visibility:public"]) |
| load("@build_bazel_rules_typescript//:defs.bzl", "ts_library") |
| |
| ts_library( |
| name = "my_code", |
| srcs = glob(["*.ts"]), |
| deps = ["//path/to/other:library"], |
| ) |
| ``` |
| |
| If your ts_library target has npm dependencies you can specify these |
| with fine grained npm dependency targets created by the `yarn_install` or |
| `npm_install` rules: |
| |
| ```python |
| ts_library( |
| name = "my_code", |
| srcs = glob(["*.ts"]), |
| deps = [ |
| "@npm//@types/node", |
| "@npm//@types/foo", |
| "@npm//foo", |
| "//path/to/other:library", |
| ], |
| ) |
| ``` |
| |
| You can also you the `@npm//@types` target which will include all |
| packages in the `@types` scope as dependencies. |
| |
| If you are using self-managed npm dependencies, you can use the |
| `node_modules` attribute in `ts_library` and point it to the |
| `//:node_modules` filegroup defined in your root `BUILD.bazel` file. |
| You'll also need to override the `compiler` attribute if you do this |
| as the Bazel-managed deps and self-managed cannot be used together |
| in the same rule. |
| |
| ```python |
| ts_library( |
| name = "my_code", |
| srcs = glob(["*.ts"]), |
| deps = ["//path/to/other:library"], |
| node_modules = "//:node_modules", |
| compiler = "//:@bazel/typescript/tsc_wrapped", |
| ) |
| ``` |
| |
| To build a `ts_library` target run: |
| |
| `bazel build //path/to/package:target` |
| |
| The resulting `.d.ts` file paths will be printed. Additionally, the `.js` |
| outputs from TypeScript will be written to disk, next to the `.d.ts` files <sup>1</sup>. |
| |
| Note that the `tsconfig.json` file used for compilation should be the same one |
| your editor references, to keep consistent settings for the TypeScript compiler. |
| By default, `ts_library` uses the `tsconfig.json` file in the workspace root |
| directory. See the notes about the `tsconfig` attribute in the [ts_library API docs]. |
| |
| > <sup>1</sup> The |
| > [declarationDir](https://www.typescriptlang.org/docs/handbook/compiler-options.html) |
| > compiler option will be silently overwritten if present. |
| |
| [ts_library API docs]: http://tsetse.info/api/build_defs.html#ts_library |
| |
| ### Serving TypeScript for development |
| |
| There are two choices for development mode: |
| |
| 1. Use the `ts_devserver` rule to bring up our simple, fast development server. |
| This is intentionally very simple, to help you get started quickly. However, |
| since there are many development servers available, we do not want to mirror |
| their features in yet another server we maintain. |
| 1. Teach your real frontend server to serve files from Bazel's output directory. |
| This is not yet documented. Choose this option if you have an existing server |
| used in development mode, or if your requirements exceed what the |
| `ts_devserver` supports. Be careful that your development round-trip stays |
| fast (should be under two seconds). |
| |
| To use `ts_devserver`, you simply `load` the rule, and call it with `deps` that |
| point to your `ts_library` target(s): |
| |
| ```python |
| load("@build_bazel_rules_typescript//:defs.bzl", "ts_devserver", "ts_library") |
| |
| ts_library( |
| name = "app", |
| srcs = ["app.ts"], |
| ) |
| |
| ts_devserver( |
| name = "devserver", |
| # We'll collect all the devmode JS sources from these TypeScript libraries |
| deps = [":app"], |
| # This is the path we'll request from the browser, see index.html |
| serving_path = "/bundle.js", |
| # The devserver can serve our static files too |
| static_files = ["index.html"], |
| ) |
| ``` |
| |
| The `index.html` should be the same one you use for production, and it should |
| load the JavaScript bundle from the path indicated in `serving_path`. |
| |
| If you don't have an index.html file, a simple one will be generated by the |
| `ts_devserver`. |
| |
| See `examples/app` in this repository for a working example. To run the |
| devserver, we recommend you use [ibazel]: |
| |
| ```sh |
| $ ibazel run examples/app:devserver |
| ``` |
| |
| `ibazel` will keep the devserver program running, and provides a LiveReload |
| server so the browser refreshes the application automatically when each build |
| finishes. |
| |
| [ibazel]: https://github.com/bazelbuild/bazel-watcher |
| |
| ## Writing TypeScript code for Bazel |
| |
| Bazel's TypeScript compiler has your workspace path mapped, so you can import |
| from an absolute path starting from your workspace. |
| |
| `/WORKSPACE`: |
| ```python |
| workspace(name = "myworkspace") |
| ``` |
| |
| `/some/long/path/to/deeply/nested/subdirectory.ts`: |
| ```javascript |
| import {thing} from 'myworkspace/place'; |
| ``` |
| |
| will import from `/place.ts`. |
| |
| |
| Since this is an extension to the vanillia TypeScript compiler, editors which use the TypeScript language services to provide code completion and inline type checking will not be able to resolve the modules. In the above example, adding |
| ```json |
| "paths": { |
| "myworkspace/*": ["*"] |
| } |
| ``` |
| to `tsconfig.json` will fix the imports for the common case of using absolute paths. |
| See https://www.typescriptlang.org/docs/handbook/module-resolution.html#path-mapping for more details on the paths syntax. |
| |
| Similarly, you can use path mapping to teach the editor how to resolve imports |
| from `ts_library` rules which set the `module_name` attribute. |
| |
| ## Notes |
| |
| If you'd like a "watch mode", try https://github.com/bazelbuild/bazel-watcher |
| (note, it's also quite new). |
| |
| At some point, we plan to release a tool similar to [gazelle] to generate the |
| BUILD files from your source code. |
| |
| In the meantime, we suggest associating the `.bazel` extension with Python in |
| your editor, so that you get useful syntax highlighting. |
| |
| [gazelle]: https://github.com/bazelbuild/rules_go/tree/master/go/tools/gazelle |