-
Notifications
You must be signed in to change notification settings - Fork 522
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(typescript): add ts_project rule
This is a very thin layer on top of vanilla tsc
- Loading branch information
Showing
22 changed files
with
340 additions
and
15 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
{ | ||
"compilerOptions": { | ||
"jsx": "react", | ||
"lib": ["ES2015", "DOM"], | ||
"rootDirs": [ | ||
".", | ||
"bazel-out/darwin-fastbuild/bin", | ||
], | ||
"listFiles": true, | ||
"traceResolution": true | ||
}, | ||
"files": ["index.tsx", "types.d.ts"] | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,192 @@ | ||
"ts_project rule" | ||
|
||
load("@build_bazel_rules_nodejs//:providers.bzl", "DeclarationInfo", "NpmPackageInfo", "run_node") | ||
|
||
_ATTRS = { | ||
# TODO: limit to the extensions we expect? | ||
# ts, tsx, js, jsx, json | ||
"srcs": attr.label_list(allow_files = True), | ||
"extends": attr.label_list(allow_files = [".json"]), | ||
"tsc": attr.label(default = Label("@npm//typescript/bin:tsc"), executable = True, cfg = "host"), | ||
"tsconfig": attr.label(mandatory = True, allow_single_file = [".json"]), | ||
"deps": attr.label_list(providers = [DeclarationInfo]), | ||
} | ||
|
||
_OUTPUTS = { | ||
"js_outs": attr.output_list(), | ||
"map_outs": attr.output_list(), | ||
"typing_maps_outs": attr.output_list(), | ||
"typings_outs": attr.output_list(), | ||
} | ||
|
||
_TsConfigInfo = provider( | ||
doc = """Passes tsconfig.json files to downstream compilations so that TypeScript can read them. | ||
This is needed to support Project References""", | ||
fields = { | ||
"tsconfigs": "depset of tsconfig.json files", | ||
}, | ||
) | ||
|
||
def _ts_project_impl(ctx): | ||
arguments = ctx.actions.args() | ||
arguments.add_all([ | ||
"-p", | ||
ctx.file.tsconfig.short_path, | ||
"--outDir", | ||
"/".join([ctx.bin_dir.path, ctx.label.package]), | ||
]) | ||
if len(ctx.outputs.typings_outs) > 0: | ||
arguments.add_all([ | ||
"--declarationDir", | ||
"/".join([ctx.bin_dir.path, ctx.label.package]), | ||
]) | ||
|
||
deps_depsets = [] | ||
for dep in ctx.attr.deps: | ||
if _TsConfigInfo in dep: | ||
deps_depsets.append(dep[_TsConfigInfo].tsconfigs) | ||
if NpmPackageInfo in dep: | ||
# TODO: we could maybe filter these to be tsconfig.json or *.d.ts only | ||
# we don't expect tsc wants to read any other files from npm packages. | ||
deps_depsets.append(dep[NpmPackageInfo].sources) | ||
elif DeclarationInfo in dep: | ||
deps_depsets.append(dep[DeclarationInfo].transitive_declarations) | ||
|
||
inputs = ctx.files.srcs + depset(transitive = deps_depsets).to_list() + [ctx.file.tsconfig] | ||
if ctx.attr.extends: | ||
inputs.extend(ctx.files.extends) | ||
outputs = ctx.outputs.js_outs + ctx.outputs.map_outs + ctx.outputs.typings_outs + ctx.outputs.typing_maps_outs | ||
|
||
if len(outputs) == 0: | ||
return [] | ||
|
||
run_node( | ||
ctx, | ||
inputs = inputs, | ||
arguments = [arguments], | ||
outputs = outputs, | ||
executable = "tsc", | ||
progress_message = "Compiling TypeScript project %s" % ctx.file.tsconfig.short_path, | ||
) | ||
|
||
runtime_files = depset(ctx.outputs.js_outs + ctx.outputs.map_outs) | ||
typings_files = ctx.outputs.typings_outs + [s for s in ctx.files.srcs if s.path.endswith(".d.ts")] | ||
|
||
return [ | ||
DeclarationInfo( | ||
declarations = depset(typings_files), | ||
transitive_declarations = depset(typings_files, transitive = [ | ||
dep[DeclarationInfo].transitive_declarations | ||
for dep in ctx.attr.deps | ||
]), | ||
), | ||
DefaultInfo( | ||
files = runtime_files, | ||
runfiles = ctx.runfiles( | ||
transitive_files = runtime_files, | ||
collect_default = True, | ||
), | ||
), | ||
_TsConfigInfo(tsconfigs = depset([ctx.file.tsconfig], transitive = [ | ||
dep[_TsConfigInfo].tsconfigs | ||
for dep in ctx.attr.deps | ||
if _TsConfigInfo in dep | ||
])), | ||
] | ||
|
||
ts_project = rule( | ||
implementation = _ts_project_impl, | ||
attrs = dict(_ATTRS, **_OUTPUTS), | ||
) | ||
|
||
def _out_paths(srcs, ext): | ||
return [f[:f.rindex(".")] + ext for f in srcs if not f.endswith(".d.ts")] | ||
|
||
def ts_project_macro( | ||
name = "tsconfig", | ||
srcs = None, | ||
deps = [], | ||
extends = None, | ||
declaration = True, | ||
source_map = False, | ||
declaration_map = False, | ||
emit_declaration_only = False, | ||
**kwargs): | ||
"""Compiles one TypeScript project using `tsc -p` | ||
Unlike `ts_library`, this rule is the thinnest possible layer of Bazel awareness on top | ||
of the TypeScript compiler. Any code that works with `tsc` should work with `ts_project` | ||
with a few caveats: | ||
- Bazel requires that the `outDir` (and `declarationDir`) be set to | ||
`bazel-out/[arch]/bin/path/to/package` | ||
so we override whatever settings appear in your tsconfig. | ||
- Bazel expects that each output is produced by a single rule. | ||
Thus if you have two `ts_project` rules with overlapping sources (the same .ts file | ||
appears in more than one) then you get an error if you try to build both together. | ||
Worse, if you build them separately then the output directory will contain whichever | ||
one you happened to build most recently. This is highly discouraged. | ||
> Note, in order for TypeScript to find referenced projects in the bazel-out folder, | ||
> we recommend that the base tsconfig contain a rootDirs section that includes all | ||
> possible locations they may appear. | ||
> | ||
> We hope this will not be needed in some future release of TypeScript. | ||
> Follow https://github.com/microsoft/TypeScript/issues/37257 for more info. | ||
> | ||
> For example, if the base tsconfig file relative to the workspace root is | ||
> `path/to/tsconfig.json` then you should configure like: | ||
``` | ||
"compilerOptions": { | ||
"rootDirs": [ | ||
".", | ||
"../../bazel-out/darwin-fastbuild/bin/path/to", | ||
"../../bazel-out/k8-fastbuild/bin/path/to", | ||
"../../bazel-out/x64_windows-fastbuild/bin/path/to", | ||
"../../bazel-out/darwin-dbg/bin/path/to", | ||
"../../bazel-out/k8-dbg/bin/path/to", | ||
"../../bazel-out/x64_windows-dbg/bin/path/to", | ||
] | ||
} | ||
``` | ||
If your dependencies are not installed into a workspace called `npm`, or if you want to | ||
use a custom TypeScript compiler binary, you can pass `tsc = "@some_wksp//path/to:tsc_bin"` | ||
to this rule to override the compiler. | ||
Args: | ||
name: The basename (no `.json` extension) of the tsconfig file that should be compiled. | ||
srcs: List of labels of TypeScript source files to be provided to the compiler. | ||
If absent, defaults to `**/*.ts` (all TypeScript files in the package). | ||
deps: List of labels of other rules that produce TypeScript typings (.d.ts files) | ||
extends: List of labels of tsconfig file(s) referenced in `extends` section of tsconfig. | ||
Must include any tsconfig files "chained" by extends clauses. | ||
declaration: if the `declaration` or `composite` bit are set in the tsconfig. | ||
Instructs Bazel to expect a `.d.ts` output for each `.ts` source. | ||
source_map: if the `sourceMap` bit is set in the tsconfig. | ||
Instructs Bazel to expect a `.js.map` output for each `.ts` source. | ||
declaration_map: if the `declarationMap` bit is set in the tsconfig. | ||
Instructs Bazel to expect a `.d.ts.map` output for each `.ts` source. | ||
emit_declaration_only: if the `emitDeclarationOnly` bit is set in the tsconfig. | ||
Instructs Bazel *not* to expect `.js` outputs for `.ts` sources. | ||
""" | ||
|
||
if srcs == None: | ||
srcs = native.glob(["**/*.ts"]) | ||
tsconfig = name + ".json" | ||
|
||
ts_project( | ||
name = name, | ||
srcs = srcs, | ||
deps = deps, | ||
tsconfig = tsconfig, | ||
extends = extends, | ||
js_outs = _out_paths(srcs, ".js") if not emit_declaration_only else [], | ||
map_outs = _out_paths(srcs, ".js.map") if source_map else [], | ||
typings_outs = _out_paths(srcs, ".d.ts") if declaration else [], | ||
typing_maps_outs = _out_paths(srcs, ".d.ts.map") if declaration_map else [], | ||
**kwargs | ||
) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
exports_files(["tsconfig-base.json"]) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,9 @@ | ||
load("@npm_bazel_typescript//:index.bzl", "ts_project") | ||
|
||
ts_project( | ||
name = "tsconfig", # This will use ./tsconfig.json | ||
srcs = [":a.ts"], | ||
extends = ["//packages/typescript/test/ts_project:tsconfig-base.json"], | ||
source_map = True, | ||
visibility = ["//packages/typescript/test:__subpackages__"], | ||
) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
export const a: string = 'hello'; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
{ | ||
"extends": "../tsconfig-base.json", | ||
"compilerOptions": { | ||
"sourceMap": true | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,31 @@ | ||
load("@npm_bazel_jasmine//:index.from_src.bzl", "jasmine_node_test") | ||
load("@npm_bazel_typescript//:index.bzl", "ts_project") | ||
|
||
package(default_visibility = ["//packages/typescript/test:__subpackages__"]) | ||
|
||
ts_project( | ||
name = "tsconfig", # This will use ./tsconfig.json | ||
srcs = [":b.ts"], | ||
extends = ["//packages/typescript/test/ts_project:tsconfig-base.json"], | ||
deps = ["//packages/typescript/test/ts_project/a:tsconfig"], | ||
) | ||
|
||
ts_project( | ||
name = "tsconfig-test", # This will use ./tsconfig-test.json | ||
testonly = True, | ||
srcs = [":b.spec.ts"], | ||
extends = ["//packages/typescript/test/ts_project:tsconfig-base.json"], | ||
deps = [ | ||
":tsconfig", | ||
"@npm//@types/jasmine", | ||
], | ||
) | ||
|
||
jasmine_node_test( | ||
name = "test", | ||
srcs = ["b.spec.js"], | ||
data = [ | ||
":tsconfig", | ||
#"//packages/typescript/test/ts_project/a:tsconfig" | ||
], | ||
) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
import {sayHello} from './b'; | ||
|
||
describe('b', () => { | ||
it('should say hello', () => { | ||
let captured: string = ''; | ||
console.log = (s: string) => captured = s; | ||
sayHello(' world'); | ||
expect(captured).toBe('hello world'); | ||
}); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
import {a} from '../a/a'; | ||
|
||
export function sayHello(f: string) { | ||
console.log(a + f); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
{ | ||
"extends": "../tsconfig-base.json", | ||
"references": [ | ||
{"path": "./"} | ||
], | ||
"include": ["*.spec.ts"] | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
{ | ||
"extends": "../tsconfig-base.json", | ||
"references": [ | ||
{"path": "../a"} | ||
], | ||
"exclude": ["*.spec.ts"] | ||
} |
Oops, something went wrong.