Skip to content

Commit

Permalink
docs: add website text
Browse files Browse the repository at this point in the history
Signed-off-by: Ashley Claymore <[email protected]>
  • Loading branch information
acutmore committed Sep 18, 2024
1 parent 8c610b0 commit a0a5693
Show file tree
Hide file tree
Showing 7 changed files with 213 additions and 26 deletions.
2 changes: 2 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,8 @@ class C extends Array {
}
```

Try it out in the [playground](https://bloomberg.github.io/ts-blank-space/play/).

## Rationale

The benefits of this library are:
Expand Down
13 changes: 12 additions & 1 deletion website/_includes/base.html
Original file line number Diff line number Diff line change
Expand Up @@ -16,13 +16,18 @@
body {
display: grid;
grid-template-columns: 1fr min(45rem, 90%) 1fr;
padding-bottom: 50vh;
}
body > * {
grid-column: 2;
}
header {
padding-bottom: 2em;
}
.gh-link svg {
display: inline-block;
height: 2em;
}
</style>
</head>
<body>
Expand All @@ -31,7 +36,13 @@
<h1 class="cds-t-h1">ts-</h1>
</div>
<div style="background-color: #f90; flex-grow: 1">
<h1 class="cds-t-h1">blank-space</h1>
<h1 class="cds-t-h1" style="float: left">blank-space</h1>
<menu style="float: right; padding: 5px">
<a href="./play" class="cds-t-h4" style="margin-right: 5px">playground</a>
<a href="https://github.com/bloomberg/ts-blank-space" class="gh-link"
>{% include './github-mark.svg' %}</a
>
</menu>
</div>
</header>
<main class="cds-prose">{{ content }}</main>
Expand Down
1 change: 1 addition & 0 deletions website/_includes/github-mark.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added website/assets/benchmark-chart-1.jpg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added website/assets/benchmark-chart-2.jpg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
204 changes: 183 additions & 21 deletions website/index.md
Original file line number Diff line number Diff line change
Expand Up @@ -3,40 +3,202 @@ layout: base.html
title: ts-blank-space
---

TypeScript goes in:
## What is `ts-blank-space`?

[`ts-blank-space`](https://github.com/bloomberg/ts-blank-space) is a fast type-stripping compiler that converts TypeScript to JavaScript. It supports a modern subset of TypeScript by erasing the types and replacing them with whitespace. That's it. It is **not** a type checker and does not perform any other code transformations.

The underlying technique can improve build performance and simplify development. The implementation is pure TypeScript. It is simple enough to read and understand in a few minutes because it is only 700 lines of code and reuses the original TypeScript parser.

## Type Stripping

The core idea is to use spaces as a substitute for type annotations. You can think of `ts-blank-space` as a reference implementation of this type-stripping technique. Much like other compiler techniques seen in the JS ecosystem such as tree-shaking, it is a reusable idea that any TypeScript compiler could implement. [Some already do\!](https://github.com/swc-project/swc/pull/9144)

Let's look at some examples. If you use the classic TypeScript (`tsc`) compiler today without customizing any options, you'll see your source code undergo significant transformations.

<!-- prettier-ignore -->
```typescript
class C<T> extends Array<T> implements I {
// ^^^ ^^^^^^^^^^^^^^^^
private field!: string;
// ^^^^^^^ ^^^^^^^^^

method<T>(this: T, a?: string): void {
// ^^^ ^^^^^^^^ ^^^^^^^^^ ^^^^^^
}
```javascript
class Cat<T> {
public whiskers: number;
public tail: T;

constructor(count: number, tail: T) {
this.whiskers = count;
this.tail = tail;
}
}

throw Error();
```

JavaScript + space comes out:
is compiled to:

<!-- prettier-ignore -->
```javascript
class C extends Array {
// ^^^ ^^^^^^^^^^^^^^^^
field ;
// ^^^^^^^ ^^^^^^^^^
var Cat = /** @class */ (function () {
function Cat(count, tail) {
this.whiskers = count;
this.tail = tail;
}
return Cat;
}());
throw Error();
```

That's because TypeScript is performing "down-leveling" to convert your modern JavaScript into something that can run on older JavaScript engines. This moves the code around which, without also generating sourcemaps, confuses error stacks and breakpoint locations.

method ( a ) {
// ^^^ ^^^^^^^^ ^^^^^^^^^ ^^^^^^
Let's try disabling down-leveling using the `tsconfig.json` option `target: "esnext"`:

<!-- prettier-ignore -->
```javascript
class Cat {
whiskers;
tail;
constructor(count, tail) {
this.whiskers = count;
this.tail = tail;
}
}
throw Error();
```

This looks much closer to the original code with the types erased. But the output is still visually smaller and occupies fewer lines. Meaning we're going to have inaccurate error stacks.

```text
// Original source
Error
at Object.<anonymous> (cat.ts:11:7)
```

```text
// With TypeScript's output
Error
at Object.<anonymous> (cat.js:8:7)
```

One solution to this relocation problem is to ask TypeScript to generate sourcemaps. Debuggers and runtimes know how to use sourcemaps to translate the filenames and coordinates used for breakpoints and error stacks. But could we do better?

Yes\! Here's what the generated code looks like with `ts-blank-space`:

<!-- prettier-ignore -->
```javascript
class Cat {
whiskers ;
tail ;

constructor(count , tail ) {
this.whiskers = count;
this.tail = tail;
}
}

throw Error();
```

This might look a little wacky, but by leaving the code coordinates unchanged, we've eliminated the need to generate sourcemaps. There's no encoding and decoding locations. Error stack coordinates are always accurate\!

Finally, [we can teach](https://github.com/bloomberg/ts-blank-space/blob/8a8eddf77e3a9c81d6a58d9ee08c808a4b338f51/loader/hooks.js#L28) the JS engine and debugger about the original filename by adding one line (`//# sourceURL=cat.ts`) to the end of the JS file. This magic comment is part of the [sourcemap specification](https://tc39.es/source-map/#linking-evald-code-to-named-generated-code) and is [supported](https://github.com/v8/v8/blob/8393fca199737e1a137f5f0d6bc9468748b06542/src/parsing/scanner.cc#L252-L271) [natively](https://github.com/WebKit/WebKit/blob/79271e75c2fec165d6badfa191b6b9a6edeee611/Source/JavaScriptCore/parser/Lexer.cpp#L2555-L2560), naming files correctly in error stacks with no loss of performance.

## Build Performance Benefits

The typical build operations to emit JavaScript code from TypeScript source involves:

1. Parsing the TypeScript into TypeScript's Abstract Syntax Tree (AST)
2. Transforming this into a JavaScript AST
3. Code generation from the JavaScript AST
4. Sourcemap mapping calculation from the generated code to the original locations in the AST

`ts-blank-space` replaces steps 2, 3, 4 with basic string substitution on spans extracted from the AST. The AST is used merely to discover the start and end points of the type annotations in order to shake them off.

Skipping this work is what makes it a swift approach. Further time is saved by decreasing the number of sourcemaps that need to be [merged](https://github.com/ampproject/remapping) during build scenarios that involve a sequence of transforms and corresponding sourcemaps.

### Build Performance Results

Here's a benchmark of compiling a huge TypeScript file ([52 KLoC](https://gist.github.com/acutmore/63cbfa9367d11ff5ef496f03999243e7/7e907f90496e086edba26e0f220f7a696c9aecd2#file-checker-txt)) ten times using a variety of build tools.

![a bar chart comparing the times to run the benchmark. From fastest to slowest the results are: esbuild, swc, ts-blank-space, swc/wasm-typescript, sucrase, swc/wasm, esbuild-wasm, typescript, and babel. esbuild and swc are categorized as having native binaries in contrast to ts-blank-space being distributed as JavaScript](./assets/benchmark-chart-1.jpg)

For the supported subset of TypeScript, `ts-blank-space` is 5.6x faster than `tsc`. Today it appears to be the fastest emitter written in JavaScript and even beats the Wasm-based emitters. Credit for this speed should primarily be attributed to the highly-optimized TypeScript parser which is doing the heavy lifting. It is also worth calling out the impressive speed of `sucrase`, which is also written in JavaScript and is only marginally slower while supporting more syntax.

#### Performance Results with AST Re-use

In Bloomberg, we want to use the TypeScript AST for JavaScript-based analysis in addition to emitting JavaScript. This means we are already paying the cost of generating the TypeScript AST and `ts-blank-space` can operate on this as-is.

Here's a similar benchmark in which we measure the combined time to (a) emit JavaScript, and (b) generate a JavaScript-accessible AST. `babel`, `@swc/core`, and `@swc/wasm` have APIs to produce the AST. `sucrase` and `esbuild` [do](https://github.com/alangpierce/sucrase/blob/main/README.md#what-sucrase-is-not) [not](https://esbuild.github.io/plugins/#plugin-api-limitations) provide access to an AST, for these we fallback to using `typescript` to provide the AST.

![a bar chart with the same results as the previous chart, however with AST access overhead added to the times. Now the results from fastest to slowest are ts-blank-space, esbuild, swc, sucrase, swc/wasm-typescript, swc/wasm, esbuild-wasm, typescript, and babel. ts-blank-space, typescript, and babel have essentially no AST access overhead in contrast to the other libraries that have more significant overheads](./assets/benchmark-chart-2.jpg)

With these requirements we've found `ts-blank-space` provides the fastest overall result.

On large files such as that used in the benchmark, we've seen Garbage Collection (GC) account for up to 17% of build time. With internal benchmarks of real-life code, the file size is typically smaller and we've seen `ts-blank-space` go even faster with GC dropping to nearer 3%.

The full code and results of the benchmark can be found in [the `perf` folder of the repository](https://github.com/bloomberg/ts-blank-space/tree/main/perf).

## Development Time Benefits

### Simpler Startup

Debuggers require sourcemaps to be loaded and processed before app startup in order to reliably set breakpoints in terms of generated coordinates ahead of first execution. Eliminating sourcemaps means fewer moving parts and potentially faster startup.

### Faster Errors

Avoiding sourcemaps can also increase runtime performance with respect to Error stacks. Node has the ability to toggle whether runtime processing of Error stacks uses sourcemaps. Node includes [a microbenchmark](https://github.com/nodejs/node/blob/main/benchmark/es/error-stack.js) that demonstrates the cost of reading a mapped error stack is approximately twice that of an unmapped error stack.

```text
> node benchmark/es/error-stack.js
es/error-stack.js n=100000 method="without-sourcemap": 107,059.95251565645
es/error-stack.js n=100000 method="sourcemap": 55,918.729592805634
```

## Playground
### Resilience

Bloomberg is a heavy user of sourcemaps and we have folk helping to [make them even better.](https://ecma-international.org/news/ecma-tc39-ecmascript-initiates-a-new-task-group-to-standardize-source-maps/) For sourcemaps to work, many different tools all need to align end-to-end, which can lead to them [sometimes failing](https://x.com/robpalmer2/status/1583581325289656320). The good news is that with `ts-blank-space` your debugging experience will be based on accurate locations even if your environment is misconfigured or when sourcemaps would have been inaccessible.

## Maintenance Benefits

The default parser used by `ts-blank-space` is the official TypeScript parser. This maximizes alignment with the upstream TypeScript project. Specifically it means that as new syntax is added to the TypeScript language, it should be trivial for `ts-blank-space` to support that syntax.

This reduces the effort and compatibility risk associated with maintaining another parser. It also means the project benefits from any performance or correctness improvements that are made to the central TypeScript parser.

Beyond the parser, the remaining code is only 700 lines of TypeScript. This small size makes it easy to review and understand by TypeScript users.

## Limitations

`ts-blank-space` [does not support](https://github.com/bloomberg/ts-blank-space/blob/main/docs/unsupported_syntax.md) all TypeScript syntax and so may not be suitable for your needs.

JSX is tolerated but not compiled, meaning it preserves the tags in the output: `*.tsx` becomes `*.jsx`.

We refer to the supported subset as **_Modern TypeScript_** because it represents nearly all TypeScript syntax, except for TypeScript-specific features that may be considered [legacy](https://www.typescriptlang.org/docs/handbook/2/modules.html#typescript-namespaces) or [discouraged](https://www.totaltypescript.com/why-i-dont-like-typescript-enums) in some way \- despite not being formally deprecated.

These unsupported TypeScript features already have alternatives we prefer:

- Prefix-style type assertions (`<type>value`)
- ...should be replaced with `as`\-style type assertions.
- CommonJS-specific syntax `(import =` and `export =)`
- ...should be replaced with standard ES modules \- CommonJS is not supported.
- Instantiated Namespaces (`namespace Foo` and `module Foo`)
- ...should be [replaced](https://www.typescriptlang.org/docs/handbook/2/modules.html#typescript-namespaces) with standard ES modules.

The following unsupported TypeScript features must be avoided when using `ts-blank-space`. Unfortunately the alternatives are not always as concise. We would be excited if more powerful and coherent alternatives to these were to succeed as JavaScript language proposals.

- Enums (`enum`)
- [Alternative patterns](https://www.typescriptlang.org/docs/handbook/enums.html#objects-vs-enums) include const objects and string unions.
- Constructor Parameter Properties (`constructor(public prop) {}`)
- Explicit field declarations and assignments are needed instead.

Note that [ambient Namespaces](https://www.typescriptlang.org/docs/handbook/namespaces.html#ambient-namespaces) and [ambient Enums](https://www.typescriptlang.org/docs/handbook/enums.html#ambient-enums) **are** supported, i.e. `declare namespace`, `declare enum`.

We've successfully used this TypeScript subset in Bloomberg for many years, following the clear mental model that [TypeScript is JavaScript \+ Types](https://www.bloomberg.com/company/stories/10-insights-adopting-typescript-at-scale/). During that time, the primary unsupported feature that users said they missed was Enums. Most of the time that need was appeased via the [alternative patterns](https://www.typescriptlang.org/docs/handbook/enums.html#objects-vs-enums).

## Getting Started

`ts-blank-space` is just a library that transforms strings. It is a single function. It is not a CLI or toolchain.

- If you want to see how the code is transformed, please use the [playground](./play).
- The quickest way to use this library for real is to use the example [Node loader](https://github.com/bloomberg/ts-blank-space#nodejs-loader).
- For anything more, you will need to integrate the library into a CLI or toolchain.

Checkout the [playground](./play).
## Should I use this library?

## Links
Only if you want to. While `ts-blank-space` is battle-tested and used in production at Bloomberg, the idea is more important than the implementation.

- [github.com/bloomberg/ts-blank-space](https://github.com/bloomberg/ts-blank-space)
`ts-blank-space` has a singular purpose and a limited scope. Many users will continue to be better served by existing tools with a more comprehensive scope. And who knows \- if those tools find ways to leverage the ideas presented here, everyone wins 😉
19 changes: 15 additions & 4 deletions website/play/index.html
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,11 @@
menu > * {
padding: 1ch;
}

.gh-link svg {
display: inline-block;
height: 2em;
}
</style>
</head>
<body>
Expand All @@ -54,8 +59,8 @@
<h1 class="cds-t-h1">ts-</h1>
</div>
<div style="background-color: #f90; grid-area: header-js">
<h1 class="cds-t-h1">blank-space</h1>
<menu>
<h1 class="cds-t-h1" style="float: left">blank-space</h1>
<menu style="float: right">
<label title="Control if output should be shown as a diff"
>diff: <input type="checkbox" id="diff-check" />
</label>
Expand All @@ -65,7 +70,7 @@ <h1 class="cds-t-h1">blank-space</h1>
</label>
<label title="Select which parser mode to use for the input text"
>parser:
<select id="lang-select" style="width: auto; display: inline-block">
<select id="lang-select" style="width: auto; display: inline-block; padding: 5px">
<option value="ts">.ts</option>
<option value="tsx">.tsx</option>
</select>
Expand All @@ -77,14 +82,20 @@ <h1 class="cds-t-h1">blank-space</h1>
>
💾
</button>
<a
href="https://github.com/bloomberg/ts-blank-space"
class="gh-link"
style="float: right; padding: 5px"
>{% include 'github-mark.svg' %}</a
>
</menu>
</div>
<div id="container-ts" class="editor" style="grid-area: editor-ts"></div>
<div id="container-js" class="editor" style="grid-area: editor-js"></div>
</div>
<div id="container-diff" class="editor"></div>
<footer style="text-align: center; padding-top: 10px">
<a href="https://github.com/bloomberg/ts-blank-space">github.com/bloomberg/ts-blank-space</a>
<a href="..">bloomberg/ts-blank-space</a>
</footer>
<script src="./play.js"></script>
</body>
Expand Down

0 comments on commit a0a5693

Please sign in to comment.