diff --git a/README.md b/README.md
index 18d44e1..5aaaa6d 100644
--- a/README.md
+++ b/README.md
@@ -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:
diff --git a/website/_includes/base.html b/website/_includes/base.html
index d490b96..d1fb102 100644
--- a/website/_includes/base.html
+++ b/website/_includes/base.html
@@ -16,6 +16,7 @@
body {
display: grid;
grid-template-columns: 1fr min(45rem, 90%) 1fr;
+ padding-bottom: 50vh;
}
body > * {
grid-column: 2;
@@ -23,6 +24,10 @@
header {
padding-bottom: 2em;
}
+ .gh-link svg {
+ display: inline-block;
+ height: 2em;
+ }
@@ -31,7 +36,13 @@
ts-
-
blank-space
+
blank-space
+
{{ content }}
diff --git a/website/_includes/github-mark.svg b/website/_includes/github-mark.svg
new file mode 100644
index 0000000..21b14ea
--- /dev/null
+++ b/website/_includes/github-mark.svg
@@ -0,0 +1 @@
+
diff --git a/website/assets/benchmark-chart-1.jpg b/website/assets/benchmark-chart-1.jpg
new file mode 100644
index 0000000..b17947e
Binary files /dev/null and b/website/assets/benchmark-chart-1.jpg differ
diff --git a/website/assets/benchmark-chart-2.jpg b/website/assets/benchmark-chart-2.jpg
new file mode 100644
index 0000000..fd2b616
Binary files /dev/null and b/website/assets/benchmark-chart-2.jpg differ
diff --git a/website/index.md b/website/index.md
index f7b0f07..be647bf 100644
--- a/website/index.md
+++ b/website/index.md
@@ -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.
-```typescript
-class C extends Array implements I {
-// ^^^ ^^^^^^^^^^^^^^^^
- private field!: string;
-// ^^^^^^^ ^^^^^^^^^
-
- method(this: T, a?: string): void {
-// ^^^ ^^^^^^^^ ^^^^^^^^^ ^^^^^^
- }
+```javascript
+class Cat {
+ 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:
```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"`:
+
+
+```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. (cat.ts:11:7)
+```
+
+```text
+// With TypeScript's output
+Error
+ at Object. (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`:
+
+
+```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 (`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 😉
diff --git a/website/play/index.html b/website/play/index.html
index e4e0ff2..0555ade 100644
--- a/website/play/index.html
+++ b/website/play/index.html
@@ -46,6 +46,11 @@
menu > * {
padding: 1ch;
}
+
+ .gh-link svg {
+ display: inline-block;
+ height: 2em;
+ }
@@ -54,8 +59,8 @@