From 3e81f2cf1500a3589d8236bd04e6ad52534e7f60 Mon Sep 17 00:00:00 2001
From: Volker Schukai <volker.schukai@schukai.com>
Date: Wed, 20 Nov 2024 14:31:32 +0100
Subject: [PATCH] feat: new javascript generation #9

---
 .gitignore                                    |     3 +
 README.md                                     |    40 +
 examples/example5/package-lock.json           |    61 +
 examples/example5/package.json                |    14 +
 examples/example5/scripts/.gitkeep            |     0
 examples/example5/source/lib.mjs              |     4 +
 examples/example5/source/main.css             |     5 +
 examples/example5/source/main.mjs             |     7 +
 examples/example5/source/page.mjs             |     6 +
 examples/example5/styles/.gitkeep             |     0
 examples/example5/test.html                   |    26 +
 flake.lock                                    |     6 +-
 source/.idea/.gitignore                       |     8 +
 source/.idea/codeStyles/codeStyleConfig.xml   |     5 +
 source/.idea/misc.xml                         |    20 +
 source/.idea/modules.xml                      |     8 +
 source/.idea/source.iml                       |    10 +
 source/.idea/vcs.xml                          |     6 +
 source/command.go                             |    48 +-
 source/go.mod                                 |    21 +-
 source/go.sum                                 |    30 +-
 source/html/cut.go                            |     2 +-
 source/html/generate.go                       |     2 +-
 source/html/sync.go                           |     2 +-
 source/javascript/generate.go                 |   251 +
 source/main.go                                |     3 +-
 .../github.com/evanw/esbuild/LICENSE.md       |    21 +
 .../esbuild/internal/api_helpers/use_timer.go |     7 +
 .../evanw/esbuild/internal/ast/ast.go         |   812 +
 .../evanw/esbuild/internal/bundler/bundler.go |  3331 +++
 .../evanw/esbuild/internal/cache/cache.go     |   115 +
 .../evanw/esbuild/internal/cache/cache_ast.go |   190 +
 .../evanw/esbuild/internal/cache/cache_fs.go  |    52 +
 .../evanw/esbuild/internal/compat/compat.go   |    92 +
 .../esbuild/internal/compat/css_table.go      |   361 +
 .../evanw/esbuild/internal/compat/js_table.go |   910 +
 .../evanw/esbuild/internal/config/config.go   |   842 +
 .../evanw/esbuild/internal/config/globals.go  |  1014 +
 .../evanw/esbuild/internal/css_ast/css_ast.go |  1205 ++
 .../internal/css_ast/css_decl_table.go        |   698 +
 .../esbuild/internal/css_lexer/css_lexer.go   |  1081 +
 .../internal/css_parser/css_color_spaces.go   |   620 +
 .../esbuild/internal/css_parser/css_decls.go  |   538 +
 .../css_parser/css_decls_animation.go         |   119 +
 .../css_parser/css_decls_border_radius.go     |   217 +
 .../internal/css_parser/css_decls_box.go      |   206 +
 .../css_parser/css_decls_box_shadow.go        |   106 +
 .../internal/css_parser/css_decls_color.go    |   938 +
 .../internal/css_parser/css_decls_composes.go |   103 +
 .../css_parser/css_decls_container.go         |    53 +
 .../internal/css_parser/css_decls_font.go     |   138 +
 .../css_parser/css_decls_font_family.go       |   162 +
 .../css_parser/css_decls_font_weight.go       |    25 +
 .../internal/css_parser/css_decls_gradient.go |  1057 +
 .../css_parser/css_decls_list_style.go        |   179 +
 .../css_parser/css_decls_transform.go         |   347 +
 .../internal/css_parser/css_nesting.go        |   490 +
 .../esbuild/internal/css_parser/css_parser.go |  2374 ++
 .../css_parser/css_parser_selector.go         |   979 +
 .../internal/css_parser/css_reduce_calc.go    |   605 +
 .../internal/css_printer/css_printer.go       |  1141 +
 .../evanw/esbuild/internal/fs/error_other.go  |     9 +
 .../esbuild/internal/fs/error_wasm+windows.go |    17 +
 .../evanw/esbuild/internal/fs/filepath.go     |   649 +
 .../evanw/esbuild/internal/fs/fs.go           |   287 +
 .../evanw/esbuild/internal/fs/fs_mock.go      |   294 +
 .../evanw/esbuild/internal/fs/fs_real.go      |   543 +
 .../evanw/esbuild/internal/fs/fs_zip.go       |   405 +
 .../evanw/esbuild/internal/fs/iswin_other.go  |     9 +
 .../evanw/esbuild/internal/fs/iswin_wasm.go   |    25 +
 .../esbuild/internal/fs/iswin_windows.go      |     8 +
 .../evanw/esbuild/internal/fs/modkey_other.go |    35 +
 .../evanw/esbuild/internal/fs/modkey_unix.go  |    41 +
 .../evanw/esbuild/internal/graph/graph.go     |   431 +
 .../evanw/esbuild/internal/graph/input.go     |   127 +
 .../evanw/esbuild/internal/graph/meta.go      |   205 +
 .../evanw/esbuild/internal/helpers/bitset.go  |    27 +
 .../evanw/esbuild/internal/helpers/comment.go |    29 +
 .../evanw/esbuild/internal/helpers/dataurl.go |    72 +
 .../evanw/esbuild/internal/helpers/float.go   |   158 +
 .../evanw/esbuild/internal/helpers/glob.go    |    58 +
 .../evanw/esbuild/internal/helpers/hash.go    |    14 +
 .../evanw/esbuild/internal/helpers/joiner.go  |    86 +
 .../evanw/esbuild/internal/helpers/mime.go    |    49 +
 .../evanw/esbuild/internal/helpers/path.go    |    22 +
 .../evanw/esbuild/internal/helpers/quote.go   |   142 +
 .../esbuild/internal/helpers/serializer.go    |    26 +
 .../evanw/esbuild/internal/helpers/stack.go   |    50 +
 .../evanw/esbuild/internal/helpers/strings.go |    41 +
 .../evanw/esbuild/internal/helpers/timer.go   |    94 +
 .../evanw/esbuild/internal/helpers/typos.go   |    38 +
 .../evanw/esbuild/internal/helpers/utf.go     |   230 +
 .../esbuild/internal/helpers/waitgroup.go     |    37 +
 .../evanw/esbuild/internal/js_ast/js_ast.go   |  1841 ++
 .../esbuild/internal/js_ast/js_ast_helpers.go |  2973 +++
 .../evanw/esbuild/internal/js_ast/js_ident.go |   247 +
 .../evanw/esbuild/internal/js_ast/unicode.go  |  2065 ++
 .../esbuild/internal/js_lexer/js_lexer.go     |  2665 +++
 .../evanw/esbuild/internal/js_lexer/tables.go |   382 +
 .../internal/js_parser/global_name_parser.go  |    49 +
 .../esbuild/internal/js_parser/js_parser.go   | 18021 ++++++++++++++++
 .../internal/js_parser/js_parser_lower.go     |  2131 ++
 .../js_parser/js_parser_lower_class.go        |  2573 +++
 .../esbuild/internal/js_parser/json_parser.go |   238 +
 .../internal/js_parser/sourcemap_parser.go    |   277 +
 .../esbuild/internal/js_parser/ts_parser.go   |  1999 ++
 .../esbuild/internal/js_printer/js_printer.go |  4924 +++++
 .../evanw/esbuild/internal/linker/debug.go    |   148 +
 .../evanw/esbuild/internal/linker/linker.go   |  7154 ++++++
 .../evanw/esbuild/internal/logger/logger.go   |  2045 ++
 .../esbuild/internal/logger/logger_darwin.go  |    34 +
 .../esbuild/internal/logger/logger_linux.go   |    34 +
 .../esbuild/internal/logger/logger_other.go   |    16 +
 .../esbuild/internal/logger/logger_windows.go |   136 +
 .../evanw/esbuild/internal/logger/msg_ids.go  |   371 +
 .../evanw/esbuild/internal/renamer/renamer.go |   662 +
 .../esbuild/internal/resolver/dataurl.go      |    76 +
 .../esbuild/internal/resolver/package_json.go |  1462 ++
 .../esbuild/internal/resolver/resolver.go     |  2923 +++
 .../internal/resolver/testExpectations.json   |   311 +
 .../internal/resolver/tsconfig_json.go        |   481 +
 .../esbuild/internal/resolver/yarnpnp.go      |   665 +
 .../evanw/esbuild/internal/runtime/runtime.go |   604 +
 .../esbuild/internal/sourcemap/sourcemap.go   |   834 +
 .../evanw/esbuild/internal/xxhash/LICENSE.txt |    22 +
 .../evanw/esbuild/internal/xxhash/README.md   |     1 +
 .../evanw/esbuild/internal/xxhash/xxhash.go   |   235 +
 .../esbuild/internal/xxhash/xxhash_other.go   |    74 +
 .../github.com/evanw/esbuild/pkg/api/api.go   |   718 +
 .../evanw/esbuild/pkg/api/api_impl.go         |  2530 +++
 .../evanw/esbuild/pkg/api/api_js_table.go     |    50 +
 .../evanw/esbuild/pkg/api/favicon.go          |    31 +
 .../evanw/esbuild/pkg/api/serve_other.go      |   990 +
 .../evanw/esbuild/pkg/api/serve_wasm.go       |    21 +
 .../evanw/esbuild/pkg/api/watcher.go          |   187 +
 .../application/{xflags => xflags.git}/.envrc |     0
 .../{xflags => xflags.git}/.gitignore         |     0
 .../{xflags => xflags.git}/.gitlab-ci.yml     |     0
 .../{xflags => xflags.git}/CHANGELOG.md       |     0
 .../{xflags => xflags.git}/CONTRIBUTING.md    |     0
 .../{xflags => xflags.git}/LICENSE            |     0
 .../{xflags => xflags.git}/README.md          |     0
 .../application/{xflags => xflags.git}/api.go |     0
 .../{xflags => xflags.git}/command.go         |     0
 .../{xflags => xflags.git}/devenv.lock        |     0
 .../{xflags => xflags.git}/devenv.nix         |     0
 .../{xflags => xflags.git}/devenv.yaml        |     0
 .../application/{xflags => xflags.git}/doc.go |     0
 .../{xflags => xflags.git}/error.go           |     0
 .../{xflags => xflags.git}/execute.go         |     0
 .../{xflags => xflags.git}/flake.lock         |     0
 .../{xflags => xflags.git}/flake.nix          |     0
 .../{xflags => xflags.git}/help-util.go       |     0
 .../{xflags => xflags.git}/help.go            |     0
 .../{xflags => xflags.git}/hint.go            |     0
 .../{xflags => xflags.git}/mapping.go         |     2 +-
 .../{xflags => xflags.git}/parse.go           |     0
 .../{xflags => xflags.git}/release.json       |     0
 .../{xflags => xflags.git}/setting.go         |     0
 .../{xflags => xflags.git}/tags.go            |     0
 .../{xflags => xflags.git}/type.go            |     0
 .../go/markup/{html => html.git}/LICENSE      |     0
 .../{html => html.git}/engine/engine.go       |     0
 .../markup/{html => html.git}/engine/error.go |     0
 .../libraries/go/utilities/pathfinder/.envrc  |     2 -
 .../go/utilities/pathfinder/.gitignore        |   155 -
 .../go/utilities/pathfinder/.gitlab-ci.yml    |    43 -
 .../go/utilities/pathfinder/CONTRIBUTING.md   |    58 -
 .../libraries/go/utilities/pathfinder/LICENSE |    14 -
 .../go/utilities/pathfinder/README.md         |    69 -
 .../go/utilities/pathfinder/error.go          |    39 -
 .../libraries/go/utilities/pathfinder/find.go |    51 -
 .../go/utilities/pathfinder/flake.lock        |   181 -
 .../go/utilities/pathfinder/flake.nix         |   158 -
 .../libraries/go/utilities/pathfinder/get.go  |    74 -
 .../go/utilities/pathfinder/pathfinder.iml    |    12 -
 .../go/utilities/pathfinder/release.json      |     1 -
 .../libraries/go/utilities/pathfinder/set.go  |   293 -
 source/vendor/golang.org/x/net/html/doc.go    |     7 +-
 source/vendor/golang.org/x/net/html/iter.go   |    56 +
 source/vendor/golang.org/x/net/html/node.go   |     4 +
 .../golang.org/x/sys/unix/ioctl_linux.go      |    96 +
 .../vendor/golang.org/x/sys/unix/mkerrors.sh  |    12 +
 .../golang.org/x/sys/unix/syscall_linux.go    |     1 +
 .../x/sys/unix/syscall_zos_s390x.go           |   104 +-
 .../golang.org/x/sys/unix/zerrors_linux.go    |    22 +
 .../x/sys/unix/zerrors_linux_386.go           |    14 +
 .../x/sys/unix/zerrors_linux_amd64.go         |    14 +
 .../x/sys/unix/zerrors_linux_arm.go           |    14 +
 .../x/sys/unix/zerrors_linux_arm64.go         |    14 +
 .../x/sys/unix/zerrors_linux_loong64.go       |    14 +
 .../x/sys/unix/zerrors_linux_mips.go          |    14 +
 .../x/sys/unix/zerrors_linux_mips64.go        |    14 +
 .../x/sys/unix/zerrors_linux_mips64le.go      |    14 +
 .../x/sys/unix/zerrors_linux_mipsle.go        |    14 +
 .../x/sys/unix/zerrors_linux_ppc.go           |    14 +
 .../x/sys/unix/zerrors_linux_ppc64.go         |    14 +
 .../x/sys/unix/zerrors_linux_ppc64le.go       |    14 +
 .../x/sys/unix/zerrors_linux_riscv64.go       |    14 +
 .../x/sys/unix/zerrors_linux_s390x.go         |    14 +
 .../x/sys/unix/zerrors_linux_sparc64.go       |    14 +
 .../golang.org/x/sys/unix/zsyscall_linux.go   |    10 +
 .../golang.org/x/sys/unix/ztypes_linux.go     |   120 +-
 .../golang.org/x/sys/unix/ztypes_zos_s390x.go |     6 +
 .../x/sys/windows/syscall_windows.go          |    34 +-
 .../golang.org/x/sys/windows/types_windows.go |   126 +
 .../x/sys/windows/zsyscall_windows.go         |    53 +
 source/vendor/modules.txt                     |    50 +-
 208 files changed, 90024 insertions(+), 1213 deletions(-)
 create mode 100644 examples/example5/package-lock.json
 create mode 100644 examples/example5/package.json
 create mode 100644 examples/example5/scripts/.gitkeep
 create mode 100644 examples/example5/source/lib.mjs
 create mode 100644 examples/example5/source/main.css
 create mode 100644 examples/example5/source/main.mjs
 create mode 100644 examples/example5/source/page.mjs
 create mode 100644 examples/example5/styles/.gitkeep
 create mode 100644 examples/example5/test.html
 create mode 100644 source/.idea/.gitignore
 create mode 100644 source/.idea/codeStyles/codeStyleConfig.xml
 create mode 100644 source/.idea/misc.xml
 create mode 100644 source/.idea/modules.xml
 create mode 100644 source/.idea/source.iml
 create mode 100644 source/.idea/vcs.xml
 create mode 100644 source/javascript/generate.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/LICENSE.md
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/api_helpers/use_timer.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/ast/ast.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/bundler/bundler.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/cache/cache.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/cache/cache_ast.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/cache/cache_fs.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/compat/compat.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/compat/css_table.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/compat/js_table.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/config/config.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/config/globals.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_ast/css_ast.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_ast/css_decl_table.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_lexer/css_lexer.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_color_spaces.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_animation.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_border_radius.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_box.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_box_shadow.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_color.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_composes.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_container.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_font.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_font_family.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_font_weight.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_gradient.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_list_style.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_transform.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_nesting.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_parser.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_parser_selector.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_parser/css_reduce_calc.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/css_printer/css_printer.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/fs/error_other.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/fs/error_wasm+windows.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/fs/filepath.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/fs/fs.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/fs/fs_mock.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/fs/fs_real.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/fs/fs_zip.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/fs/iswin_other.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/fs/iswin_wasm.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/fs/iswin_windows.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/fs/modkey_other.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/fs/modkey_unix.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/graph/graph.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/graph/input.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/graph/meta.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/bitset.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/comment.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/dataurl.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/float.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/glob.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/hash.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/joiner.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/mime.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/path.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/quote.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/serializer.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/stack.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/strings.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/timer.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/typos.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/utf.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/helpers/waitgroup.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/js_ast/js_ast.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/js_ast/js_ast_helpers.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/js_ast/js_ident.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/js_ast/unicode.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/js_lexer/js_lexer.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/js_lexer/tables.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/js_parser/global_name_parser.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/js_parser/js_parser.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/js_parser/js_parser_lower.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/js_parser/js_parser_lower_class.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/js_parser/json_parser.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/js_parser/sourcemap_parser.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/js_parser/ts_parser.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/js_printer/js_printer.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/linker/debug.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/linker/linker.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/logger/logger.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/logger/logger_darwin.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/logger/logger_linux.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/logger/logger_other.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/logger/logger_windows.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/logger/msg_ids.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/renamer/renamer.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/resolver/dataurl.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/resolver/package_json.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/resolver/resolver.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/resolver/testExpectations.json
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/resolver/tsconfig_json.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/resolver/yarnpnp.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/runtime/runtime.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/sourcemap/sourcemap.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/xxhash/LICENSE.txt
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/xxhash/README.md
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/xxhash/xxhash.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/internal/xxhash/xxhash_other.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/pkg/api/api.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/pkg/api/api_impl.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/pkg/api/api_js_table.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/pkg/api/favicon.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/pkg/api/serve_other.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/pkg/api/serve_wasm.go
 create mode 100644 source/vendor/github.com/evanw/esbuild/pkg/api/watcher.go
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/.envrc (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/.gitignore (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/.gitlab-ci.yml (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/CHANGELOG.md (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/CONTRIBUTING.md (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/LICENSE (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/README.md (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/api.go (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/command.go (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/devenv.lock (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/devenv.nix (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/devenv.yaml (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/doc.go (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/error.go (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/execute.go (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/flake.lock (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/flake.nix (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/help-util.go (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/help.go (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/hint.go (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/mapping.go (97%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/parse.go (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/release.json (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/setting.go (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/tags.go (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/application/{xflags => xflags.git}/type.go (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/markup/{html => html.git}/LICENSE (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/markup/{html => html.git}/engine/engine.go (100%)
 rename source/vendor/gitlab.schukai.com/oss/libraries/go/markup/{html => html.git}/engine/error.go (100%)
 delete mode 100644 source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/.envrc
 delete mode 100644 source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/.gitignore
 delete mode 100644 source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/.gitlab-ci.yml
 delete mode 100644 source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/CONTRIBUTING.md
 delete mode 100644 source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/LICENSE
 delete mode 100644 source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/README.md
 delete mode 100644 source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/error.go
 delete mode 100644 source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/find.go
 delete mode 100644 source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/flake.lock
 delete mode 100644 source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/flake.nix
 delete mode 100644 source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/get.go
 delete mode 100644 source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/pathfinder.iml
 delete mode 100644 source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/release.json
 delete mode 100644 source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/set.go
 create mode 100644 source/vendor/golang.org/x/net/html/iter.go

diff --git a/.gitignore b/.gitignore
index 38e0e60..342599d 100644
--- a/.gitignore
+++ b/.gitignore
@@ -313,3 +313,6 @@ Taskfile.yaml
 
 /environment/do-secrets.json
 documentation/manual/de/book/
+/examples/example5/scripts/main.mjs
+/examples/example5/scripts/page.mjs
+/examples/example5/styles/main.css
diff --git a/README.md b/README.md
index b52d1c3..45e3a8a 100644
--- a/README.md
+++ b/README.md
@@ -259,7 +259,47 @@ the task `task update-code` must be called.
 The hash is currently always null, as a vendor directory is used 
 in the project. This is created with `go mod vendor`.
 
+#### JavaScript
 
+##### generate
+
+Bob can use **ESBuild** to transform JavaScript code directly from an HTML file. 
+For this, the relevant `<script>` tags must include specific attributes. 
+
+ESBuild is licensed under the (https://github.com/evanw/esbuild?tab=MIT-1-ov-file#readme)[MIT license].
+
+Here’s an example:
+
+```html
+<script 
+    data-bob-source="source/main.mjs" 
+    data-bob-script-dist="scripts/main.mjs" 
+    data-bob-style-dist="styles/main.css" 
+    src="/scripts/main.mjs" 
+    type="module">
+</script>
+```
+
+**Attribute Explanation:**
+
+- **`data-bob-source`**  
+  Specifies the file to be used as the source for the build process (e.g., `source/page.mjs`).
+
+- **`data-bob-target`** *(optional)*  
+  Defines the target JavaScript format. The default is `esnext`. A common alternative is `es6`.
+
+- **`data-bob-script-dist`**  
+  Specifies the path to the output JavaScript file relative to the template, e.g., `scripts/page.mjs`.
+
+- **`data-bob-style-dist`**  
+  Defines the path to the output styles file, e.g., `styles/page.css`.
+
+- **`src`**  
+  Indicates the URL where the script is served in the browser. This value is **not** used by ESBuild for the build process.
+
+### Notes:
+- These attributes help separate development and delivery paths clearly.
+- `src` is used solely for delivery and has no impact on the ESBuild process.
 
 
 ## Questions
diff --git a/examples/example5/package-lock.json b/examples/example5/package-lock.json
new file mode 100644
index 0000000..5b8ea29
--- /dev/null
+++ b/examples/example5/package-lock.json
@@ -0,0 +1,61 @@
+{
+  "name": "example5",
+  "version": "1.0.0",
+  "lockfileVersion": 3,
+  "requires": true,
+  "packages": {
+    "": {
+      "name": "example5",
+      "version": "1.0.0",
+      "license": "ISC",
+      "dependencies": {
+        "@schukai/monster": "^3.88.0"
+      }
+    },
+    "node_modules/@floating-ui/core": {
+      "version": "1.6.8",
+      "resolved": "https://registry.npmjs.org/@floating-ui/core/-/core-1.6.8.tgz",
+      "integrity": "sha512-7XJ9cPU+yI2QeLS+FCSlqNFZJq8arvswefkZrYI1yQBbftw6FyrZOxYSh+9S7z7TpeWlRt9zJ5IhM1WIL334jA==",
+      "license": "MIT",
+      "dependencies": {
+        "@floating-ui/utils": "^0.2.8"
+      }
+    },
+    "node_modules/@floating-ui/dom": {
+      "version": "1.6.12",
+      "resolved": "https://registry.npmjs.org/@floating-ui/dom/-/dom-1.6.12.tgz",
+      "integrity": "sha512-NP83c0HjokcGVEMeoStg317VD9W7eDlGK7457dMBANbKA6GJZdc7rjujdgqzTaz93jkGgc5P/jeWbaCHnMNc+w==",
+      "license": "MIT",
+      "dependencies": {
+        "@floating-ui/core": "^1.6.0",
+        "@floating-ui/utils": "^0.2.8"
+      }
+    },
+    "node_modules/@floating-ui/utils": {
+      "version": "0.2.8",
+      "resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.2.8.tgz",
+      "integrity": "sha512-kym7SodPp8/wloecOpcmSnWJsK7M0E5Wg8UcFA+uO4B9s5d0ywXOEro/8HM9x0rW+TljRzul/14UYz3TleT3ig==",
+      "license": "MIT"
+    },
+    "node_modules/@popperjs/core": {
+      "version": "2.11.8",
+      "resolved": "https://registry.npmjs.org/@popperjs/core/-/core-2.11.8.tgz",
+      "integrity": "sha512-P1st0aksCrn9sGZhp8GMYwBnQsbvAWsZAX44oXNNvLHGqAOcoVxmjZiohstwQ7SqKnbR47akdNi+uleWD8+g6A==",
+      "license": "MIT",
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/popperjs"
+      }
+    },
+    "node_modules/@schukai/monster": {
+      "version": "3.88.0",
+      "resolved": "https://registry.npmjs.org/@schukai/monster/-/monster-3.88.0.tgz",
+      "integrity": "sha512-hwV5hlpubSjxkOS3LZazS0+Up0Cp5OJ+oxpV+2iNGtXUwhiRuvtVmfNlsuN9MxXGu4pOoXzbFUk/jFmPuGOLmA==",
+      "license": "AGPL 3.0",
+      "dependencies": {
+        "@floating-ui/dom": "^1.6.12",
+        "@popperjs/core": "^2.11.8"
+      }
+    }
+  }
+}
diff --git a/examples/example5/package.json b/examples/example5/package.json
new file mode 100644
index 0000000..047a733
--- /dev/null
+++ b/examples/example5/package.json
@@ -0,0 +1,14 @@
+{
+  "name": "example5",
+  "version": "1.0.0",
+  "main": "index.js",
+  "scripts": {
+    "test": "echo \"Error: no test specified\" && exit 1"
+  },
+  "author": "",
+  "license": "ISC",
+  "description": "",
+  "dependencies": {
+    "@schukai/monster": "^3.88.0"
+  }
+}
diff --git a/examples/example5/scripts/.gitkeep b/examples/example5/scripts/.gitkeep
new file mode 100644
index 0000000..e69de29
diff --git a/examples/example5/source/lib.mjs b/examples/example5/source/lib.mjs
new file mode 100644
index 0000000..a5da3be
--- /dev/null
+++ b/examples/example5/source/lib.mjs
@@ -0,0 +1,4 @@
+
+
+
+export const A=1;
\ No newline at end of file
diff --git a/examples/example5/source/main.css b/examples/example5/source/main.css
new file mode 100644
index 0000000..5dbb4c5
--- /dev/null
+++ b/examples/example5/source/main.css
@@ -0,0 +1,5 @@
+
+
+div {
+    color: red;
+}
\ No newline at end of file
diff --git a/examples/example5/source/main.mjs b/examples/example5/source/main.mjs
new file mode 100644
index 0000000..176cb62
--- /dev/null
+++ b/examples/example5/source/main.mjs
@@ -0,0 +1,7 @@
+import  "@schukai/monster/source/components/form/button.mjs";
+import {A} from "./lib.mjs";
+import "./main.css";
+
+console.log(A);
+
+
diff --git a/examples/example5/source/page.mjs b/examples/example5/source/page.mjs
new file mode 100644
index 0000000..d865ece
--- /dev/null
+++ b/examples/example5/source/page.mjs
@@ -0,0 +1,6 @@
+import {A} from "./lib.mjs";
+
+
+console.log(A);
+
+
diff --git a/examples/example5/styles/.gitkeep b/examples/example5/styles/.gitkeep
new file mode 100644
index 0000000..e69de29
diff --git a/examples/example5/test.html b/examples/example5/test.html
new file mode 100644
index 0000000..0f5063a
--- /dev/null
+++ b/examples/example5/test.html
@@ -0,0 +1,26 @@
+<!DOCTYPE html><html lang="en" data-attributes="lang path:lang"><head>
+
+    <meta charset="utf-8"/>
+
+    <link rel="apple-touch-icon" href="/apple-touch-icon.png"/>
+
+    <script data-bob-source="source/main.mjs" data-bob-script-dist="scripts/main.mjs" data-bob-style-dist="styles/main.css" src="/scripts/main.mjs" type="module"></script>
+    <script data-bob-source="source/page.mjs" data-bob-target=es6 data-bob-script-dist="scripts/page.mjs" data-bob-style-dist="styles/page.css" src="/scripts/page.mjs" type="module"></script>
+
+    <script type="application/json" data-monster-role="translations" data-bob-reference="the-translation" data-replace="path:translations.the-translation.content">
+      {
+        "key5": "translation4"
+      }
+    </script>
+  </head>
+
+  <body>
+    <header>
+      <div class="gradient"></div>
+    </header>
+
+
+  <monster-button>test</monster-button>
+  
+
+</body></html>
\ No newline at end of file
diff --git a/flake.lock b/flake.lock
index ec97ba0..c89e91a 100644
--- a/flake.lock
+++ b/flake.lock
@@ -2,11 +2,11 @@
   "nodes": {
     "nixpkgs": {
       "locked": {
-        "lastModified": 1730883749,
-        "narHash": "sha256-mwrFF0vElHJP8X3pFCByJR365Q2463ATp2qGIrDUdlE=",
+        "lastModified": 1731797254,
+        "narHash": "sha256-df3dJApLPhd11AlueuoN0Q4fHo/hagP75LlM5K1sz9g=",
         "owner": "NixOS",
         "repo": "nixpkgs",
-        "rev": "dba414932936fde69f0606b4f1d87c5bc0003ede",
+        "rev": "e8c38b73aeb218e27163376a2d617e61a2ad9b59",
         "type": "github"
       },
       "original": {
diff --git a/source/.idea/.gitignore b/source/.idea/.gitignore
new file mode 100644
index 0000000..13566b8
--- /dev/null
+++ b/source/.idea/.gitignore
@@ -0,0 +1,8 @@
+# Default ignored files
+/shelf/
+/workspace.xml
+# Editor-based HTTP Client requests
+/httpRequests/
+# Datasource local storage ignored files
+/dataSources/
+/dataSources.local.xml
diff --git a/source/.idea/codeStyles/codeStyleConfig.xml b/source/.idea/codeStyles/codeStyleConfig.xml
new file mode 100644
index 0000000..a55e7a1
--- /dev/null
+++ b/source/.idea/codeStyles/codeStyleConfig.xml
@@ -0,0 +1,5 @@
+<component name="ProjectCodeStyleConfiguration">
+  <state>
+    <option name="PREFERRED_PROJECT_CODE_STYLE" value="Default" />
+  </state>
+</component>
\ No newline at end of file
diff --git a/source/.idea/misc.xml b/source/.idea/misc.xml
new file mode 100644
index 0000000..aa935ab
--- /dev/null
+++ b/source/.idea/misc.xml
@@ -0,0 +1,20 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<project version="4">
+  <component name="ProjectRootManager" version="2" languageLevel="JDK_21" default="true" project-jdk-name="21 (4)" project-jdk-type="JavaSDK">
+    <output url="file://$PROJECT_DIR$/out" />
+  </component>
+  <component name="accountSettings">
+    <option name="activeProfile" value="profile:default" />
+    <option name="activeRegion" value="eu-west-1" />
+    <option name="recentlyUsedProfiles">
+      <list>
+        <option value="profile:default" />
+      </list>
+    </option>
+    <option name="recentlyUsedRegions">
+      <list>
+        <option value="eu-west-1" />
+      </list>
+    </option>
+  </component>
+</project>
\ No newline at end of file
diff --git a/source/.idea/modules.xml b/source/.idea/modules.xml
new file mode 100644
index 0000000..66f3350
--- /dev/null
+++ b/source/.idea/modules.xml
@@ -0,0 +1,8 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<project version="4">
+  <component name="ProjectModuleManager">
+    <modules>
+      <module fileurl="file://$PROJECT_DIR$/.idea/source.iml" filepath="$PROJECT_DIR$/.idea/source.iml" />
+    </modules>
+  </component>
+</project>
\ No newline at end of file
diff --git a/source/.idea/source.iml b/source/.idea/source.iml
new file mode 100644
index 0000000..25ed3f6
--- /dev/null
+++ b/source/.idea/source.iml
@@ -0,0 +1,10 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<module type="JAVA_MODULE" version="4">
+  <component name="Go" enabled="true" />
+  <component name="NewModuleRootManager" inherit-compiler-output="true">
+    <exclude-output />
+    <content url="file://$MODULE_DIR$" />
+    <orderEntry type="inheritedJdk" />
+    <orderEntry type="sourceFolder" forTests="false" />
+  </component>
+</module>
\ No newline at end of file
diff --git a/source/.idea/vcs.xml b/source/.idea/vcs.xml
new file mode 100644
index 0000000..6c0b863
--- /dev/null
+++ b/source/.idea/vcs.xml
@@ -0,0 +1,6 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<project version="4">
+  <component name="VcsDirectoryMappings">
+    <mapping directory="$PROJECT_DIR$/.." vcs="Git" />
+  </component>
+</project>
\ No newline at end of file
diff --git a/source/command.go b/source/command.go
index 901faab..20bd3ea 100644
--- a/source/command.go
+++ b/source/command.go
@@ -4,15 +4,17 @@ import (
 	"fmt"
 	"github.com/charmbracelet/log"
 	html2 "gitlab.schukai.com/oss/bob/html"
+	"gitlab.schukai.com/oss/bob/javascript"
 	"gitlab.schukai.com/oss/bob/release"
 	"gitlab.schukai.com/oss/bob/style"
 	template2 "gitlab.schukai.com/oss/bob/template"
 	"gitlab.schukai.com/oss/bob/types"
-	"gitlab.schukai.com/oss/libraries/go/application/xflags"
+	xflags "gitlab.schukai.com/oss/libraries/go/application/xflags.git"
 	"gopkg.in/yaml.v3"
 	"os"
 	"path"
 	"path/filepath"
+	"strings"
 )
 
 type Definition struct {
@@ -25,6 +27,12 @@ type Definition struct {
 			DataFile string `short:"d" long:"data-file" description:"Name of the main data file" default:"data.yaml"`
 		} `command:"prepare" description:"Prepare content from a file" call:"PrepareTemplate"`
 	} `command:"template" description:"Template commands"`
+	Javascript struct {
+		Generate struct {
+			Input       string `short:"i" long:"input" description:"Directory with prepared html files" required:"true"`
+			Development bool   `short:"d" long:"development" description:"Development mode" default:"false"`
+		} `command:"generate" description:"Generate javascript files from a file" call:"GenerateJavascript"`
+	} `command:"javascript" description:"Javascript related commands"`
 	HTML struct {
 		Generate struct {
 			Input     string `short:"i" long:"input" description:"Directory with prepared html files" required:"true"`
@@ -67,6 +75,44 @@ func (d *Definition) CutHTML(s *xflags.Settings[Definition]) {
 	}
 }
 
+func (d *Definition) GenerateJavascript(s *xflags.Settings[Definition]) {
+
+	skipDirs := map[string]bool{
+		"node_modules": true,
+	}
+
+	err := filepath.Walk(d.Javascript.Generate.Input, func(p string, info os.FileInfo, err error) error {
+
+		if err != nil {
+			return err
+		}
+
+		for _, part := range strings.Split(p, string(filepath.Separator)) {
+			if skipDirs[part] {
+				return filepath.SkipDir
+			}
+		}
+
+		if info.IsDir() {
+			return nil
+		}
+
+		ext := filepath.Ext(p)
+		if ext != ".html" {
+			return nil
+		}
+
+		log.Info("Generate " + p)
+
+		return javascript.ParseHTMLFile(p, d.Javascript.Generate.Development)
+	})
+
+	if err != nil {
+		s.AddError(err)
+	}
+
+}
+
 func (d *Definition) SyncHTML(s *xflags.Settings[Definition]) {
 
 	err := html2.SyncHtml(d.HTML.Sync.Specification)
diff --git a/source/go.mod b/source/go.mod
index 1492ff2..dabdb4b 100644
--- a/source/go.mod
+++ b/source/go.mod
@@ -7,12 +7,22 @@ toolchain go1.22.4
 require (
 	github.com/andybalholm/cascadia v1.3.2
 	github.com/charmbracelet/log v0.4.0
+	github.com/evanw/esbuild v0.24.0
 	github.com/tdewolff/parse/v2 v2.7.19
+	gitlab.schukai.com/oss/libraries/go/application/configuration.git v1.22.9
+	gitlab.schukai.com/oss/libraries/go/application/xflags.git v1.16.5
+	gitlab.schukai.com/oss/libraries/go/markup/html.git v0.4.7
+	gitlab.schukai.com/oss/libraries/go/services/job-queues.git v1.20.2
+	gitlab.schukai.com/oss/libraries/go/utilities/pathfinder.git v0.9.5
+	gitlab.schukai.com/oss/libraries/go/utilities/watch.git v0.4.2
+	golang.org/x/crypto v0.29.0
+	golang.org/x/net v0.31.0
+	gopkg.in/yaml.v3 v3.0.1
+)
+
+require (
 	gitlab.schukai.com/oss/libraries/go/application/xflags v1.16.3
 	gitlab.schukai.com/oss/libraries/go/markup/html v0.4.6
-	golang.org/x/crypto v0.28.0
-	golang.org/x/net v0.30.0
-	gopkg.in/yaml.v3 v3.0.1
 )
 
 require (
@@ -29,8 +39,7 @@ require (
 	github.com/volker-schukai/tokenizer v1.0.0 // indirect
 	gitlab.schukai.com/oss/libraries/go/utilities/data.git v0.2.2 // indirect
 	gitlab.schukai.com/oss/libraries/go/utilities/pathfinder v0.9.4 // indirect
-	gitlab.schukai.com/oss/libraries/go/utilities/pathfinder.git v0.9.5 // indirect
-	golang.org/x/exp v0.0.0-20241009180824-f66d83c29e7c // indirect
-	golang.org/x/sys v0.26.0 // indirect
+	golang.org/x/exp v0.0.0-20241108190413-2d47ceb2692f // indirect
+	golang.org/x/sys v0.27.0 // indirect
 	gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c // indirect
 )
diff --git a/source/go.sum b/source/go.sum
index c1fc4fb..f936d7d 100644
--- a/source/go.sum
+++ b/source/go.sum
@@ -12,6 +12,8 @@ github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ3
 github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
 github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
 github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
+github.com/evanw/esbuild v0.24.0 h1:GZ78naTLp7FKr+K7eNuM/SLs5maeiHYRPsTg6kmdsSE=
+github.com/evanw/esbuild v0.24.0/go.mod h1:D2vIQZqV/vIf/VRHtViaUtViZmG7o+kKmlBfVQuRi48=
 github.com/go-logfmt/logfmt v0.6.0 h1:wGYYu3uicYdqXVgoYbvnkrPVXkuLM1p1ifugDMEdRi4=
 github.com/go-logfmt/logfmt v0.6.0/go.mod h1:WYhtIu8zTZfxdn5+rREduYbwxfcBr/Vr6KEVveWlfTs=
 github.com/google/gofuzz v1.2.0 h1:xRy4A+RhZaiKjJ1bPfwQ8sedCA+YS2YcCHW6ec7JMi0=
@@ -50,22 +52,33 @@ github.com/tdewolff/test v1.0.11-0.20231101010635-f1265d231d52/go.mod h1:6DAvZli
 github.com/volker-schukai/tokenizer v1.0.0 h1:wF4haFoCodq7lgAk8c+th/DZmpFpL2WVD8wDzAGU1mA=
 github.com/volker-schukai/tokenizer v1.0.0/go.mod h1:LPw7lLIxUnZgeg96818N7IvwLE1x8ya31J/Aa0aCq9M=
 github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY=
+gitlab.schukai.com/oss/libraries/go/application/configuration.git v1.22.9/go.mod h1:pZ+8vTAodhn3BfWMXMe18q3Ys0Dc9v3MPHY4jY4vClk=
 gitlab.schukai.com/oss/libraries/go/application/xflags v1.16.3 h1:IWRCQOsZZPkoh/vIzjsF8BnqT4VVbOlCtfeuaYV5qEQ=
 gitlab.schukai.com/oss/libraries/go/application/xflags v1.16.3/go.mod h1:e+uFr/73kXoSozlAewBBqKsAUCIichlcvNDyj/0fj9Q=
+gitlab.schukai.com/oss/libraries/go/application/xflags v1.16.5 h1:lghTHrRwlF7YSXkG/KRlJvnZuK3Z8HeM7aEiEbFXIEQ=
+gitlab.schukai.com/oss/libraries/go/application/xflags v1.16.5/go.mod h1:6477TWP2W0vWWJ6Ctu/Mdh5SsjKUVVPFDgFfM4YdtcU=
+gitlab.schukai.com/oss/libraries/go/application/xflags.git v1.16.5 h1:YD1skbZRu3iUHafIhGIEcZqPHJXA65Zj0AcSgsTQJAk=
+gitlab.schukai.com/oss/libraries/go/application/xflags.git v1.16.5/go.mod h1:6477TWP2W0vWWJ6Ctu/Mdh5SsjKUVVPFDgFfM4YdtcU=
 gitlab.schukai.com/oss/libraries/go/markup/html v0.4.6 h1:eMold9Nl6ZkygVF1K1lTA3ROGz/mlEIcPt9aUUJC33c=
 gitlab.schukai.com/oss/libraries/go/markup/html v0.4.6/go.mod h1:FAzz3QWPCqQG54ou0zLnF6j3/ZQgGSTGsTHLShc3UFU=
+gitlab.schukai.com/oss/libraries/go/markup/html.git v0.4.7 h1:LxNtSNRXV9Ay3rKDH/KaVLORd0IpUClbfHo9tXG3Y4Q=
+gitlab.schukai.com/oss/libraries/go/markup/html.git v0.4.7/go.mod h1:vlqd5glmCGm0rdM/QXWq/ifGdIM/wq7ynSzHFnl4LzU=
+gitlab.schukai.com/oss/libraries/go/services/job-queues.git v1.20.2/go.mod h1:SZmOS4cms25c9UIOq9S2qTKHrPLmKYk0GqqtVCt4dxk=
+gitlab.schukai.com/oss/libraries/go/utilities/data.git v0.2.0 h1:JVxMHiA8zFVjJDhNl65XeYrhdMkzB+5dyrBUEZ982WU=
+gitlab.schukai.com/oss/libraries/go/utilities/data.git v0.2.0/go.mod h1:BsR4Y9jsvISplkW6UoLFRGxQX69/AUmP1SXRwWhx31o=
 gitlab.schukai.com/oss/libraries/go/utilities/data.git v0.2.2 h1:jRlVTikl73AL1y9OfYxdZ4OYG8Hkbl/8ezbwd9r5l44=
 gitlab.schukai.com/oss/libraries/go/utilities/data.git v0.2.2/go.mod h1:Vl5kzzMjpy1LGe+RUi2pTnZvZFP53Th4JChP9dbkOVk=
 gitlab.schukai.com/oss/libraries/go/utilities/pathfinder v0.9.4 h1:/+fgcPeXqz5tRrT+EZXA2vGV+OWV9R+5hEBpRJUpp80=
 gitlab.schukai.com/oss/libraries/go/utilities/pathfinder v0.9.4/go.mod h1:36psT3WHelpcXWXVp8D33IXvUIpaAXEtrQYYOODUbjE=
 gitlab.schukai.com/oss/libraries/go/utilities/pathfinder.git v0.9.5 h1:PiDmw3O3UDubKILC+t7fKs+m9670a+b8SkrHq5Rkk9M=
 gitlab.schukai.com/oss/libraries/go/utilities/pathfinder.git v0.9.5/go.mod h1:HwXjaFUAxLv+qTN63xrPBO5DEVGuZNF859t29bhTsFQ=
+gitlab.schukai.com/oss/libraries/go/utilities/watch.git v0.4.2/go.mod h1:7Rv8r5eVw5FDRp/K3KlDmjNMrdj7JA7eC3o6s7JBcHU=
 golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
 golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
-golang.org/x/crypto v0.28.0 h1:GBDwsMXVQi34v5CCYUm2jkJvu4cbtru2U4TN2PSyQnw=
-golang.org/x/crypto v0.28.0/go.mod h1:rmgy+3RHxRZMyY0jjAJShp2zgEdOqj2AO7U0pYmeQ7U=
-golang.org/x/exp v0.0.0-20241009180824-f66d83c29e7c h1:7dEasQXItcW1xKJ2+gg5VOiBnqWrJc+rq0DPKyvvdbY=
-golang.org/x/exp v0.0.0-20241009180824-f66d83c29e7c/go.mod h1:NQtJDoLvd6faHhE7m4T/1IY708gDefGGjR/iUW8yQQ8=
+golang.org/x/crypto v0.29.0 h1:L5SG1JTTXupVV3n6sUqMTeWbjAyfPwoda2DLX8J8FrQ=
+golang.org/x/crypto v0.29.0/go.mod h1:+F4F4N5hv6v38hfeYwTdx20oUvLLc+QfrE9Ax9HtgRg=
+golang.org/x/exp v0.0.0-20241108190413-2d47ceb2692f h1:XdNn9LlyWAhLVp6P/i8QYBW+hlyhrhei9uErw2B5GJo=
+golang.org/x/exp v0.0.0-20241108190413-2d47ceb2692f/go.mod h1:D5SMRVC3C2/4+F/DB1wZsLRnSNimn2Sp/NPsCrsv8ak=
 golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
 golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
 golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
@@ -73,8 +86,8 @@ golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v
 golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c=
 golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
 golang.org/x/net v0.9.0/go.mod h1:d48xBJpPfHeWQsugry2m+kC02ZBRGRgulfHnEXEuWns=
-golang.org/x/net v0.30.0 h1:AcW1SDZMkb8IpzCdQUaIq2sP4sZ4zw+55h6ynffypl4=
-golang.org/x/net v0.30.0/go.mod h1:2wGyMJ5iFasEhkwi13ChkO/t1ECNC4X4eBKkVFyYFlU=
+golang.org/x/net v0.31.0 h1:68CPQngjLL0r2AlUKiSxtQFKvzRVbnzLwMUn5SzcLHo=
+golang.org/x/net v0.31.0/go.mod h1:P4fl1q7dY2hnZFxEk4pPSkDHF+QqjitcnDjUQyMM+pM=
 golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
 golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
 golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
@@ -82,12 +95,13 @@ golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5h
 golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
+golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 golang.org/x/sys v0.7.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
-golang.org/x/sys v0.26.0 h1:KHjCJyddX0LoSTb3J+vWpupP9p0oznkqVk/IfjymZbo=
-golang.org/x/sys v0.26.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
+golang.org/x/sys v0.27.0 h1:wBqf8DvsY9Y/2P8gAfPDEYNuS30J4lPHJxXSb/nJZ+s=
+golang.org/x/sys v0.27.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
 golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
 golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
 golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
diff --git a/source/html/cut.go b/source/html/cut.go
index 00f9b8d..35d0344 100644
--- a/source/html/cut.go
+++ b/source/html/cut.go
@@ -3,7 +3,7 @@ package html
 import (
 	"github.com/andybalholm/cascadia"
 	"gitlab.schukai.com/oss/bob/types"
-	"gitlab.schukai.com/oss/libraries/go/markup/html/engine"
+	engine "gitlab.schukai.com/oss/libraries/go/markup/html.git/engine"
 	"golang.org/x/net/html"
 	"gopkg.in/yaml.v3"
 	"os"
diff --git a/source/html/generate.go b/source/html/generate.go
index 394c4cd..566f1da 100644
--- a/source/html/generate.go
+++ b/source/html/generate.go
@@ -5,7 +5,7 @@ import (
 	"github.com/andybalholm/cascadia"
 	"github.com/charmbracelet/log"
 	"gitlab.schukai.com/oss/bob/types"
-	"gitlab.schukai.com/oss/libraries/go/markup/html/engine"
+	"gitlab.schukai.com/oss/libraries/go/markup/html.git/engine"
 	"golang.org/x/net/html"
 	"gopkg.in/yaml.v3"
 	"os"
diff --git a/source/html/sync.go b/source/html/sync.go
index b2f619b..745a59a 100644
--- a/source/html/sync.go
+++ b/source/html/sync.go
@@ -4,7 +4,7 @@ import (
 	"fmt"
 	"github.com/andybalholm/cascadia"
 	"gitlab.schukai.com/oss/bob/types"
-	"gitlab.schukai.com/oss/libraries/go/markup/html/engine"
+	engine "gitlab.schukai.com/oss/libraries/go/markup/html.git/engine"
 	"golang.org/x/net/html"
 	"gopkg.in/yaml.v3"
 	"os"
diff --git a/source/javascript/generate.go b/source/javascript/generate.go
new file mode 100644
index 0000000..4a9e720
--- /dev/null
+++ b/source/javascript/generate.go
@@ -0,0 +1,251 @@
+package javascript
+
+import (
+	"github.com/charmbracelet/log"
+	"golang.org/x/net/html"
+	"os"
+	"path"
+	"strings"
+
+	"github.com/evanw/esbuild/pkg/api"
+)
+
+func ParseHTMLFile(p string, development bool) error {
+	data, err := os.ReadFile(p)
+	if err != nil {
+		return err
+	}
+
+	doc, err := html.Parse(strings.NewReader(string(data)))
+	if err != nil {
+		return err
+	}
+
+	var f func(*html.Node)
+	f = func(n *html.Node) {
+
+		var src, source, scriptDist, styleDist, target string
+
+		if n.Type == html.ElementNode && n.Data == "script" {
+			for _, attr := range n.Attr {
+				if attr.Key == "data-bob-source" {
+					source = attr.Val
+				} else if attr.Key == "data-bob-script-dist" {
+					scriptDist = attr.Val
+				} else if attr.Key == "data-bob-style-dist" {
+					styleDist = attr.Val
+				} else if attr.Key == "data-bob-target" {
+					target = attr.Val
+				} else if attr.Key == "src" {
+					src = attr.Val
+				}
+			}
+
+			if src != "" {
+
+				if !path.IsAbs(source) {
+					source = path.Dir(p) + "/" + source
+
+					if _, err := os.Stat(source); os.IsNotExist(err) {
+						log.Error("File does not exist: " + source)
+						return
+					}
+
+				}
+
+				if !path.IsAbs(scriptDist) {
+					scriptDist = path.Dir(p) + "/" + scriptDist
+				}
+
+				if !path.IsAbs(styleDist) {
+					styleDist = path.Dir(p) + "/" + styleDist
+				}
+
+				log.Info("Script: " + src + " " + source + " " + scriptDist + " " + styleDist)
+
+				runESBuild(source, path.Dir(p), scriptDist, styleDist, development, target)
+
+			}
+
+		}
+		for c := n.FirstChild; c != nil; c = c.NextSibling {
+			f(c)
+		}
+	}
+
+	f(doc)
+
+	return nil
+}
+
+func runESBuild(source, dist, scriptDist, styleDist string, development bool, target string) {
+
+	// get temporary directory
+	//tempDir, err := os.MkdirTemp("", "build-bob")
+	//if err != nil {
+	//	fmt.Println("Fehler beim Erstellen des temporären Verzeichnisses:", err)
+	//	return
+	//}
+	//
+	//defer func() {
+	//	err := os.RemoveAll(tempDir)
+	//	if err != nil {
+	//		fmt.Println("Fehler beim Löschen des temporären Verzeichnisses:", err)
+	//	}
+	//
+	//}()
+
+	var treeShaking = api.TreeShakingTrue
+	if development {
+		treeShaking = api.TreeShakingFalse
+	}
+
+	keepNames := development
+	esbuildTarget := api.ESNext
+	if target != "" {
+		switch {
+		case target == "es5" || target == "es2015":
+			esbuildTarget = api.ES5
+			keepNames = false
+		case target == "es6" || target == "es2016":
+			esbuildTarget = api.ES2016
+		case target == "es7" || target == "es2017":
+			esbuildTarget = api.ES2017
+		case target == "es8" || target == "es2018":
+			esbuildTarget = api.ES2018
+		case target == "es9" || target == "es2019":
+			esbuildTarget = api.ES2019
+		case target == "es10" || target == "es2020":
+			esbuildTarget = api.ES2020
+		case target == "es11" || target == "es2021":
+			esbuildTarget = api.ES2021
+		case target == "es12" || target == "es2022":
+			esbuildTarget = api.ES2022
+		case target == "es13" || target == "es2023":
+			esbuildTarget = api.ES2023
+		case target == "es14" || target == "es2024":
+			esbuildTarget = api.ES2024
+		default:
+			log.Error("Unknown target: " + target + ". Using ESNext")
+
+		}
+	}
+
+	footer := map[string]string{
+		"js": `
+/*
+ * Copyright protects this code. Use, reproduction, or
+ * modification of this code without prior written permission from the copyright holder
+ * is strictly prohibited. For inquiries regarding licenses or usage rights,
+ * please contact schukai GmbH.
+ */`,
+		"css": `
+/*
+ * Copyright protects this code. Use, reproduction, or
+ * modification of this code without prior written permission from the copyright holder
+ * is strictly prohibited. For inquiries regarding licenses or usage rights,
+ * please contact schukai GmbH.
+ */
+`,
+	}
+
+	var sourceMap = api.SourceMapNone
+	if development {
+		sourceMap = api.SourceMapInline
+	}
+
+	result := api.Build(api.BuildOptions{
+		EntryPoints: []string{source},
+		Outfile:     path.Join(path.Base(scriptDist)),
+		Bundle:      true,
+		Write:       false,
+		LogLevel:    api.LogLevelInfo,
+		Target:      esbuildTarget,
+
+		MinifySyntax:      !development,
+		MinifyWhitespace:  !development,
+		MinifyIdentifiers: !development,
+		TreeShaking:       treeShaking,
+		KeepNames:         keepNames,
+
+		Sourcemap:     sourceMap,
+		LegalComments: api.LegalCommentsExternal,
+
+		Footer: footer,
+	})
+
+	if len(result.Errors) > 0 {
+		for _, err := range result.Errors {
+			log.Error(err.Text)
+		}
+	}
+
+	for _, warning := range result.Warnings {
+		log.Warn(warning.Text)
+	}
+
+	for _, file := range result.OutputFiles {
+		switch path.Ext(file.Path) {
+		case ".mjs":
+			err := os.WriteFile(scriptDist, file.Contents, os.ModePerm)
+			if err != nil {
+				log.Error(err.Error())
+			} else {
+				log.Info("Saved " + scriptDist)
+			}
+
+		case ".js":
+			err := os.WriteFile(scriptDist, file.Contents, os.ModePerm)
+			if err != nil {
+				log.Error(err.Error())
+			} else {
+				log.Info("Saved " + scriptDist)
+			}
+
+		case ".css":
+			err := os.WriteFile(styleDist, file.Contents, os.ModePerm)
+			if err != nil {
+				log.Error(err.Error())
+			} else {
+				log.Info("Saved " + styleDist)
+			}
+
+		case ".txt":
+
+			content := file.Contents
+			if strings.TrimSpace(string(content)) == "" {
+				continue
+			}
+
+			if strings.Contains(file.Path, "LEGAL") {
+
+				if development {
+					log.Info("Legal information not saved in development mode")
+					continue
+				}
+
+				if strings.Contains(file.Path, "js") {
+					out := path.Join(path.Dir(scriptDist), path.Base(file.Path))
+					err := os.WriteFile(out, file.Contents, os.ModePerm)
+					if err != nil {
+						log.Error(err.Error())
+					} else {
+						log.Info("Saved " + out)
+					}
+				} else {
+					out := path.Join(path.Dir(styleDist), path.Base(file.Path))
+					err := os.WriteFile(out, file.Contents, os.ModePerm)
+					if err != nil {
+						log.Error(err.Error())
+					} else {
+						log.Info("Saved " + out)
+					}
+				}
+
+			}
+
+		}
+
+	}
+
+}
diff --git a/source/main.go b/source/main.go
index 06be5eb..c9e634b 100644
--- a/source/main.go
+++ b/source/main.go
@@ -2,7 +2,8 @@ package main
 
 import (
 	"fmt"
-	"gitlab.schukai.com/oss/libraries/go/application/xflags"
+	xflags "gitlab.schukai.com/oss/libraries/go/application/xflags.git"
+
 	"os"
 )
 
diff --git a/source/vendor/github.com/evanw/esbuild/LICENSE.md b/source/vendor/github.com/evanw/esbuild/LICENSE.md
new file mode 100644
index 0000000..2027e8d
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/LICENSE.md
@@ -0,0 +1,21 @@
+MIT License
+
+Copyright (c) 2020 Evan Wallace
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
diff --git a/source/vendor/github.com/evanw/esbuild/internal/api_helpers/use_timer.go b/source/vendor/github.com/evanw/esbuild/internal/api_helpers/use_timer.go
new file mode 100644
index 0000000..3b36fe2
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/api_helpers/use_timer.go
@@ -0,0 +1,7 @@
+package api_helpers
+
+// This flag is set by the CLI to activate the timer. It's put here instead of
+// by the timer to discourage code from checking this flag. Only the code that
+// creates the root timer should check this flag. Other code should check that
+// the timer is not null to detect if the timer is being used or not.
+var UseTimer bool
diff --git a/source/vendor/github.com/evanw/esbuild/internal/ast/ast.go b/source/vendor/github.com/evanw/esbuild/internal/ast/ast.go
new file mode 100644
index 0000000..67d2e5b
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/ast/ast.go
@@ -0,0 +1,812 @@
+package ast
+
+// This file contains data structures that are used with the AST packages for
+// both JavaScript and CSS. This helps the bundler treat both AST formats in
+// a somewhat format-agnostic manner.
+
+import (
+	"sort"
+
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type ImportKind uint8
+
+const (
+	// An entry point provided by the user
+	ImportEntryPoint ImportKind = iota
+
+	// An ES6 import or re-export statement
+	ImportStmt
+
+	// A call to "require()"
+	ImportRequire
+
+	// An "import()" expression with a string argument
+	ImportDynamic
+
+	// A call to "require.resolve()"
+	ImportRequireResolve
+
+	// A CSS "@import" rule
+	ImportAt
+
+	// A CSS "composes" declaration
+	ImportComposesFrom
+
+	// A CSS "url(...)" token
+	ImportURL
+)
+
+func (kind ImportKind) StringForMetafile() string {
+	switch kind {
+	case ImportStmt:
+		return "import-statement"
+	case ImportRequire:
+		return "require-call"
+	case ImportDynamic:
+		return "dynamic-import"
+	case ImportRequireResolve:
+		return "require-resolve"
+	case ImportAt:
+		return "import-rule"
+	case ImportComposesFrom:
+		return "composes-from"
+	case ImportURL:
+		return "url-token"
+	case ImportEntryPoint:
+		return "entry-point"
+	default:
+		panic("Internal error")
+	}
+}
+
+func (kind ImportKind) IsFromCSS() bool {
+	switch kind {
+	case ImportAt, ImportComposesFrom, ImportURL:
+		return true
+	}
+	return false
+}
+
+func (kind ImportKind) MustResolveToCSS() bool {
+	switch kind {
+	case ImportAt, ImportComposesFrom:
+		return true
+	}
+	return false
+}
+
+type ImportRecordFlags uint16
+
+const (
+	// Sometimes the parser creates an import record and decides it isn't needed.
+	// For example, TypeScript code may have import statements that later turn
+	// out to be type-only imports after analyzing the whole file.
+	IsUnused ImportRecordFlags = 1 << iota
+
+	// If this is true, the import contains syntax like "* as ns". This is used
+	// to determine whether modules that have no exports need to be wrapped in a
+	// CommonJS wrapper or not.
+	ContainsImportStar
+
+	// If this is true, the import contains an import for the alias "default",
+	// either via the "import x from" or "import {default as x} from" syntax.
+	ContainsDefaultAlias
+
+	// If this is true, the import contains an import for the alias "__esModule",
+	// via the "import {__esModule} from" syntax.
+	ContainsESModuleAlias
+
+	// If true, this "export * from 'path'" statement is evaluated at run-time by
+	// calling the "__reExport()" helper function
+	CallsRunTimeReExportFn
+
+	// Tell the printer to wrap this call to "require()" in "__toESM(...)"
+	WrapWithToESM
+
+	// Tell the printer to wrap this ESM exports object in "__toCJS(...)"
+	WrapWithToCJS
+
+	// Tell the printer to use the runtime "__require()" instead of "require()"
+	CallRuntimeRequire
+
+	// True for the following cases:
+	//
+	//   try { require('x') } catch { handle }
+	//   try { await import('x') } catch { handle }
+	//   try { require.resolve('x') } catch { handle }
+	//   import('x').catch(handle)
+	//   import('x').then(_, handle)
+	//
+	// In these cases we shouldn't generate an error if the path could not be
+	// resolved.
+	HandlesImportErrors
+
+	// If true, this was originally written as a bare "import 'file'" statement
+	WasOriginallyBareImport
+
+	// If true, this import can be removed if it's unused
+	IsExternalWithoutSideEffects
+
+	// If true, "assert { type: 'json' }" was present
+	AssertTypeJSON
+
+	// If true, do not generate "external": true in the metafile
+	ShouldNotBeExternalInMetafile
+
+	// CSS "@import" of an empty file should be removed
+	WasLoadedWithEmptyLoader
+
+	// Unique keys are randomly-generated strings that are used to replace paths
+	// in the source code after it's printed. These must not ever be split apart.
+	ContainsUniqueKey
+)
+
+func (flags ImportRecordFlags) Has(flag ImportRecordFlags) bool {
+	return (flags & flag) != 0
+}
+
+type ImportRecord struct {
+	AssertOrWith *ImportAssertOrWith
+	GlobPattern  *GlobPattern
+	Path         logger.Path
+	Range        logger.Range
+
+	// If the "HandlesImportErrors" flag is present, then this is the location
+	// of the error handler. This is used for error reporting.
+	ErrorHandlerLoc logger.Loc
+
+	// The resolved source index for an internal import (within the bundle) or
+	// invalid for an external import (not included in the bundle)
+	SourceIndex Index32
+
+	// Files imported via the "copy" loader use this instead of "SourceIndex"
+	// because they are sort of like external imports, and are not bundled.
+	CopySourceIndex Index32
+
+	Flags ImportRecordFlags
+	Kind  ImportKind
+}
+
+type AssertOrWithKeyword uint8
+
+const (
+	AssertKeyword AssertOrWithKeyword = iota
+	WithKeyword
+)
+
+func (kw AssertOrWithKeyword) String() string {
+	if kw == AssertKeyword {
+		return "assert"
+	}
+	return "with"
+}
+
+type ImportAssertOrWith struct {
+	Entries            []AssertOrWithEntry
+	KeywordLoc         logger.Loc
+	InnerOpenBraceLoc  logger.Loc
+	InnerCloseBraceLoc logger.Loc
+	OuterOpenBraceLoc  logger.Loc
+	OuterCloseBraceLoc logger.Loc
+	Keyword            AssertOrWithKeyword
+}
+
+type AssertOrWithEntry struct {
+	Key             []uint16 // An identifier or a string
+	Value           []uint16 // Always a string
+	KeyLoc          logger.Loc
+	ValueLoc        logger.Loc
+	PreferQuotedKey bool
+}
+
+func FindAssertOrWithEntry(assertions []AssertOrWithEntry, name string) *AssertOrWithEntry {
+	for _, assertion := range assertions {
+		if helpers.UTF16EqualsString(assertion.Key, name) {
+			return &assertion
+		}
+	}
+	return nil
+}
+
+type GlobPattern struct {
+	Parts       []helpers.GlobPart
+	ExportAlias string
+	Kind        ImportKind
+}
+
+// This stores a 32-bit index where the zero value is an invalid index. This is
+// a better alternative to storing the index as a pointer since that has the
+// same properties but takes up more space and costs an extra pointer traversal.
+type Index32 struct {
+	flippedBits uint32
+}
+
+func MakeIndex32(index uint32) Index32 {
+	return Index32{flippedBits: ^index}
+}
+
+func (i Index32) IsValid() bool {
+	return i.flippedBits != 0
+}
+
+func (i Index32) GetIndex() uint32 {
+	return ^i.flippedBits
+}
+
+type SymbolKind uint8
+
+const (
+	// An unbound symbol is one that isn't declared in the file it's referenced
+	// in. For example, using "window" without declaring it will be unbound.
+	SymbolUnbound SymbolKind = iota
+
+	// This has special merging behavior. You're allowed to re-declare these
+	// symbols more than once in the same scope. These symbols are also hoisted
+	// out of the scope they are declared in to the closest containing function
+	// or module scope. These are the symbols with this kind:
+	//
+	// - Function arguments
+	// - Function statements
+	// - Variables declared using "var"
+	//
+	SymbolHoisted
+	SymbolHoistedFunction
+
+	// There's a weird special case where catch variables declared using a simple
+	// identifier (i.e. not a binding pattern) block hoisted variables instead of
+	// becoming an error:
+	//
+	//   var e = 0;
+	//   try { throw 1 } catch (e) {
+	//     print(e) // 1
+	//     var e = 2
+	//     print(e) // 2
+	//   }
+	//   print(e) // 0 (since the hoisting stops at the catch block boundary)
+	//
+	// However, other forms are still a syntax error:
+	//
+	//   try {} catch (e) { let e }
+	//   try {} catch ({e}) { var e }
+	//
+	// This symbol is for handling this weird special case.
+	SymbolCatchIdentifier
+
+	// Generator and async functions are not hoisted, but still have special
+	// properties such as being able to overwrite previous functions with the
+	// same name
+	SymbolGeneratorOrAsyncFunction
+
+	// This is the special "arguments" variable inside functions
+	SymbolArguments
+
+	// Classes can merge with TypeScript namespaces.
+	SymbolClass
+
+	// Class names are not allowed to be referenced by computed property keys
+	SymbolClassInComputedPropertyKey
+
+	// A class-private identifier (i.e. "#foo").
+	SymbolPrivateField
+	SymbolPrivateMethod
+	SymbolPrivateGet
+	SymbolPrivateSet
+	SymbolPrivateGetSetPair
+	SymbolPrivateStaticField
+	SymbolPrivateStaticMethod
+	SymbolPrivateStaticGet
+	SymbolPrivateStaticSet
+	SymbolPrivateStaticGetSetPair
+
+	// Labels are in their own namespace
+	SymbolLabel
+
+	// TypeScript enums can merge with TypeScript namespaces and other TypeScript
+	// enums.
+	SymbolTSEnum
+
+	// TypeScript namespaces can merge with classes, functions, TypeScript enums,
+	// and other TypeScript namespaces.
+	SymbolTSNamespace
+
+	// In TypeScript, imports are allowed to silently collide with symbols within
+	// the module. Presumably this is because the imports may be type-only.
+	SymbolImport
+
+	// Assigning to a "const" symbol will throw a TypeError at runtime
+	SymbolConst
+
+	// Injected symbols can be overridden by provided defines
+	SymbolInjected
+
+	// Properties can optionally be renamed to shorter names
+	SymbolMangledProp
+
+	// CSS identifiers that are never renamed
+	SymbolGlobalCSS
+
+	// CSS identifiers that are renamed to be unique to the file they are in
+	SymbolLocalCSS
+
+	// This annotates all other symbols that don't have special behavior
+	SymbolOther
+)
+
+func (kind SymbolKind) IsPrivate() bool {
+	return kind >= SymbolPrivateField && kind <= SymbolPrivateStaticGetSetPair
+}
+
+func (kind SymbolKind) IsHoisted() bool {
+	return kind == SymbolHoisted || kind == SymbolHoistedFunction
+}
+
+func (kind SymbolKind) IsHoistedOrFunction() bool {
+	return kind.IsHoisted() || kind == SymbolGeneratorOrAsyncFunction
+}
+
+func (kind SymbolKind) IsFunction() bool {
+	return kind == SymbolHoistedFunction || kind == SymbolGeneratorOrAsyncFunction
+}
+
+func (kind SymbolKind) IsUnboundOrInjected() bool {
+	return kind == SymbolUnbound || kind == SymbolInjected
+}
+
+var InvalidRef Ref = Ref{^uint32(0), ^uint32(0)}
+
+// Files are parsed in parallel for speed. We want to allow each parser to
+// generate symbol IDs that won't conflict with each other. We also want to be
+// able to quickly merge symbol tables from all files into one giant symbol
+// table.
+//
+// We can accomplish both goals by giving each symbol ID two parts: a source
+// index that is unique to the parser goroutine, and an inner index that
+// increments as the parser generates new symbol IDs. Then a symbol map can
+// be an array of arrays indexed first by source index, then by inner index.
+// The maps can be merged quickly by creating a single outer array containing
+// all inner arrays from all parsed files.
+type Ref struct {
+	SourceIndex uint32
+	InnerIndex  uint32
+}
+
+type LocRef struct {
+	Loc logger.Loc
+	Ref Ref
+}
+
+type ImportItemStatus uint8
+
+const (
+	ImportItemNone ImportItemStatus = iota
+
+	// The linker doesn't report import/export mismatch errors
+	ImportItemGenerated
+
+	// The printer will replace this import with "undefined"
+	ImportItemMissing
+)
+
+type SymbolFlags uint16
+
+const (
+	// Certain symbols must not be renamed or minified. For example, the
+	// "arguments" variable is declared by the runtime for every function.
+	// Renaming can also break any identifier used inside a "with" statement.
+	MustNotBeRenamed SymbolFlags = 1 << iota
+
+	// In React's version of JSX, lower-case names are strings while upper-case
+	// names are identifiers. If we are preserving JSX syntax (i.e. not
+	// transforming it), then we need to be careful to name the identifiers
+	// something with a capital letter so further JSX processing doesn't treat
+	// them as strings instead.
+	MustStartWithCapitalLetterForJSX
+
+	// If true, this symbol is the target of a "__name" helper function call.
+	// This call is special because it deliberately doesn't count as a use
+	// of the symbol (otherwise keeping names would disable tree shaking)
+	// so "UseCountEstimate" is not incremented. This flag helps us know to
+	// avoid optimizing this symbol when "UseCountEstimate" is 1 in this case.
+	DidKeepName
+
+	// Sometimes we lower private symbols even if they are supported. For example,
+	// consider the following TypeScript code:
+	//
+	//   class Foo {
+	//     #foo = 123
+	//     bar = this.#foo
+	//   }
+	//
+	// If "useDefineForClassFields: false" is set in "tsconfig.json", then "bar"
+	// must use assignment semantics instead of define semantics. We can compile
+	// that to this code:
+	//
+	//   class Foo {
+	//     constructor() {
+	//       this.#foo = 123;
+	//       this.bar = this.#foo;
+	//     }
+	//     #foo;
+	//   }
+	//
+	// However, we can't do the same for static fields:
+	//
+	//   class Foo {
+	//     static #foo = 123
+	//     static bar = this.#foo
+	//   }
+	//
+	// Compiling these static fields to something like this would be invalid:
+	//
+	//   class Foo {
+	//     static #foo;
+	//   }
+	//   Foo.#foo = 123;
+	//   Foo.bar = Foo.#foo;
+	//
+	// Thus "#foo" must be lowered even though it's supported. Another case is
+	// when we're converting top-level class declarations to class expressions
+	// to avoid the TDZ and the class shadowing symbol is referenced within the
+	// class body:
+	//
+	//   class Foo {
+	//     static #foo = Foo
+	//   }
+	//
+	// This cannot be converted into something like this:
+	//
+	//   var Foo = class {
+	//     static #foo;
+	//   };
+	//   Foo.#foo = Foo;
+	//
+	PrivateSymbolMustBeLowered
+
+	// This is used to remove the all but the last function re-declaration if a
+	// function is re-declared multiple times like this:
+	//
+	//   function foo() { console.log(1) }
+	//   function foo() { console.log(2) }
+	//
+	RemoveOverwrittenFunctionDeclaration
+
+	// This flag is to avoid warning about this symbol more than once. It only
+	// applies to the "module" and "exports" unbound symbols.
+	DidWarnAboutCommonJSInESM
+
+	// If this is present, the symbol could potentially be overwritten. This means
+	// it's not safe to make assumptions about this symbol from the initializer.
+	CouldPotentiallyBeMutated
+
+	// This flags all symbols that were exported from the module using the ES6
+	// "export" keyword, either directly on the declaration or using "export {}".
+	WasExported
+
+	// This means the symbol is a normal function that has no body statements.
+	IsEmptyFunction
+
+	// This means the symbol is a normal function that takes a single argument
+	// and returns that argument.
+	IsIdentityFunction
+
+	// If true, calls to this symbol can be unwrapped (i.e. removed except for
+	// argument side effects) if the result is unused.
+	CallCanBeUnwrappedIfUnused
+)
+
+func (flags SymbolFlags) Has(flag SymbolFlags) bool {
+	return (flags & flag) != 0
+}
+
+// Note: the order of values in this struct matters to reduce struct size.
+type Symbol struct {
+	// This is used for symbols that represent items in the import clause of an
+	// ES6 import statement. These should always be referenced by EImportIdentifier
+	// instead of an EIdentifier. When this is present, the expression should
+	// be printed as a property access off the namespace instead of as a bare
+	// identifier.
+	//
+	// For correctness, this must be stored on the symbol instead of indirectly
+	// associated with the Ref for the symbol somehow. In ES6 "flat bundling"
+	// mode, re-exported symbols are collapsed using MergeSymbols() and renamed
+	// symbols from other files that end up at this symbol must be able to tell
+	// if it has a namespace alias.
+	NamespaceAlias *NamespaceAlias
+
+	// This is the name that came from the parser. Printed names may be renamed
+	// during minification or to avoid name collisions. Do not use the original
+	// name during printing.
+	OriginalName string
+
+	// Used by the parser for single pass parsing. Symbols that have been merged
+	// form a linked-list where the last link is the symbol to use. This link is
+	// an invalid ref if it's the last link. If this isn't invalid, you need to
+	// FollowSymbols to get the real one.
+	Link Ref
+
+	// An estimate of the number of uses of this symbol. This is used to detect
+	// whether a symbol is used or not. For example, TypeScript imports that are
+	// unused must be removed because they are probably type-only imports. This
+	// is an estimate and may not be completely accurate due to oversights in the
+	// code. But it should always be non-zero when the symbol is used.
+	UseCountEstimate uint32
+
+	// This is for generating cross-chunk imports and exports for code splitting.
+	ChunkIndex Index32
+
+	// This is used for minification. Symbols that are declared in sibling scopes
+	// can share a name. A good heuristic (from Google Closure Compiler) is to
+	// assign names to symbols from sibling scopes in declaration order. That way
+	// local variable names are reused in each global function like this, which
+	// improves gzip compression:
+	//
+	//   function x(a, b) { ... }
+	//   function y(a, b, c) { ... }
+	//
+	// The parser fills this in for symbols inside nested scopes. There are three
+	// slot namespaces: regular symbols, label symbols, and private symbols.
+	NestedScopeSlot Index32
+
+	// Boolean values should all be flags instead to save space
+	Flags SymbolFlags
+
+	Kind SymbolKind
+
+	// We automatically generate import items for property accesses off of
+	// namespace imports. This lets us remove the expensive namespace imports
+	// while bundling in many cases, replacing them with a cheap import item
+	// instead:
+	//
+	//   import * as ns from 'path'
+	//   ns.foo()
+	//
+	// That can often be replaced by this, which avoids needing the namespace:
+	//
+	//   import {foo} from 'path'
+	//   foo()
+	//
+	// However, if the import is actually missing then we don't want to report a
+	// compile-time error like we do for real import items. This status lets us
+	// avoid this. We also need to be able to replace such import items with
+	// undefined, which this status is also used for.
+	ImportItemStatus ImportItemStatus
+}
+
+// You should call "MergeSymbols" instead of calling this directly
+func (newSymbol *Symbol) MergeContentsWith(oldSymbol *Symbol) {
+	newSymbol.UseCountEstimate += oldSymbol.UseCountEstimate
+	if oldSymbol.Flags.Has(MustNotBeRenamed) && !newSymbol.Flags.Has(MustNotBeRenamed) {
+		newSymbol.OriginalName = oldSymbol.OriginalName
+		newSymbol.Flags |= MustNotBeRenamed
+	}
+	if oldSymbol.Flags.Has(MustStartWithCapitalLetterForJSX) {
+		newSymbol.Flags |= MustStartWithCapitalLetterForJSX
+	}
+}
+
+type SlotNamespace uint8
+
+const (
+	SlotDefault SlotNamespace = iota
+	SlotLabel
+	SlotPrivateName
+	SlotMangledProp
+	SlotMustNotBeRenamed
+)
+
+func (s *Symbol) SlotNamespace() SlotNamespace {
+	if s.Kind == SymbolUnbound || s.Flags.Has(MustNotBeRenamed) {
+		return SlotMustNotBeRenamed
+	}
+	if s.Kind.IsPrivate() {
+		return SlotPrivateName
+	}
+	if s.Kind == SymbolLabel {
+		return SlotLabel
+	}
+	if s.Kind == SymbolMangledProp {
+		return SlotMangledProp
+	}
+	return SlotDefault
+}
+
+type SlotCounts [4]uint32
+
+func (a *SlotCounts) UnionMax(b SlotCounts) {
+	for i := range *a {
+		ai := &(*a)[i]
+		bi := b[i]
+		if *ai < bi {
+			*ai = bi
+		}
+	}
+}
+
+type NamespaceAlias struct {
+	Alias        string
+	NamespaceRef Ref
+}
+
+type SymbolMap struct {
+	// This could be represented as a "map[Ref]Symbol" but a two-level array was
+	// more efficient in profiles. This appears to be because it doesn't involve
+	// a hash. This representation also makes it trivial to quickly merge symbol
+	// maps from multiple files together. Each file only generates symbols in a
+	// single inner array, so you can join the maps together by just make a
+	// single outer array containing all of the inner arrays. See the comment on
+	// "Ref" for more detail.
+	SymbolsForSource [][]Symbol
+}
+
+func NewSymbolMap(sourceCount int) SymbolMap {
+	return SymbolMap{make([][]Symbol, sourceCount)}
+}
+
+func (sm SymbolMap) Get(ref Ref) *Symbol {
+	return &sm.SymbolsForSource[ref.SourceIndex][ref.InnerIndex]
+}
+
+// Returns the canonical ref that represents the ref for the provided symbol.
+// This may not be the provided ref if the symbol has been merged with another
+// symbol.
+func FollowSymbols(symbols SymbolMap, ref Ref) Ref {
+	symbol := symbols.Get(ref)
+	if symbol.Link == InvalidRef {
+		return ref
+	}
+
+	link := FollowSymbols(symbols, symbol.Link)
+
+	// Only write if needed to avoid concurrent map update hazards
+	if symbol.Link != link {
+		symbol.Link = link
+	}
+
+	return link
+}
+
+// Use this before calling "FollowSymbols" from separate threads to avoid
+// concurrent map update hazards. In Go, mutating a map is not threadsafe
+// but reading from a map is. Calling "FollowAllSymbols" first ensures that
+// all mutation is done up front.
+func FollowAllSymbols(symbols SymbolMap) {
+	for sourceIndex, inner := range symbols.SymbolsForSource {
+		for symbolIndex := range inner {
+			FollowSymbols(symbols, Ref{uint32(sourceIndex), uint32(symbolIndex)})
+		}
+	}
+}
+
+// Makes "old" point to "new" by joining the linked lists for the two symbols
+// together. That way "FollowSymbols" on both "old" and "new" will result in
+// the same ref.
+func MergeSymbols(symbols SymbolMap, old Ref, new Ref) Ref {
+	if old == new {
+		return new
+	}
+
+	oldSymbol := symbols.Get(old)
+	if oldSymbol.Link != InvalidRef {
+		oldSymbol.Link = MergeSymbols(symbols, oldSymbol.Link, new)
+		return oldSymbol.Link
+	}
+
+	newSymbol := symbols.Get(new)
+	if newSymbol.Link != InvalidRef {
+		newSymbol.Link = MergeSymbols(symbols, old, newSymbol.Link)
+		return newSymbol.Link
+	}
+
+	oldSymbol.Link = new
+	newSymbol.MergeContentsWith(oldSymbol)
+	return new
+}
+
+// This is a histogram of character frequencies for minification
+type CharFreq [64]int32
+
+func (freq *CharFreq) Scan(text string, delta int32) {
+	if delta == 0 {
+		return
+	}
+
+	// This matches the order in "DefaultNameMinifier"
+	for i, n := 0, len(text); i < n; i++ {
+		c := text[i]
+		switch {
+		case c >= 'a' && c <= 'z':
+			(*freq)[c-'a'] += delta
+		case c >= 'A' && c <= 'Z':
+			(*freq)[c-('A'-26)] += delta
+		case c >= '0' && c <= '9':
+			(*freq)[c+(52-'0')] += delta
+		case c == '_':
+			(*freq)[62] += delta
+		case c == '$':
+			(*freq)[63] += delta
+		}
+	}
+}
+
+func (freq *CharFreq) Include(other *CharFreq) {
+	for i := 0; i < 64; i++ {
+		(*freq)[i] += (*other)[i]
+	}
+}
+
+type NameMinifier struct {
+	head string
+	tail string
+}
+
+var DefaultNameMinifierJS = NameMinifier{
+	head: "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ_$",
+	tail: "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_$",
+}
+
+var DefaultNameMinifierCSS = NameMinifier{
+	head: "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ_",
+	tail: "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_",
+}
+
+type charAndCount struct {
+	char  string
+	count int32
+	index byte
+}
+
+// This type is just so we can use Go's native sort function
+type charAndCountArray []charAndCount
+
+func (a charAndCountArray) Len() int          { return len(a) }
+func (a charAndCountArray) Swap(i int, j int) { a[i], a[j] = a[j], a[i] }
+
+func (a charAndCountArray) Less(i int, j int) bool {
+	ai := a[i]
+	aj := a[j]
+	return ai.count > aj.count || (ai.count == aj.count && ai.index < aj.index)
+}
+
+func (source NameMinifier) ShuffleByCharFreq(freq CharFreq) NameMinifier {
+	// Sort the histogram in descending order by count
+	array := make(charAndCountArray, 64)
+	for i := 0; i < len(source.tail); i++ {
+		array[i] = charAndCount{
+			char:  source.tail[i : i+1],
+			index: byte(i),
+			count: freq[i],
+		}
+	}
+	sort.Sort(array)
+
+	// Compute the identifier start and identifier continue sequences
+	minifier := NameMinifier{}
+	for _, item := range array {
+		if item.char < "0" || item.char > "9" {
+			minifier.head += item.char
+		}
+		minifier.tail += item.char
+	}
+	return minifier
+}
+
+func (minifier NameMinifier) NumberToMinifiedName(i int) string {
+	n_head := len(minifier.head)
+	n_tail := len(minifier.tail)
+
+	j := i % n_head
+	name := minifier.head[j : j+1]
+	i = i / n_head
+
+	for i > 0 {
+		i--
+		j := i % n_tail
+		name += minifier.tail[j : j+1]
+		i = i / n_tail
+	}
+
+	return name
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/bundler/bundler.go b/source/vendor/github.com/evanw/esbuild/internal/bundler/bundler.go
new file mode 100644
index 0000000..67d9e41
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/bundler/bundler.go
@@ -0,0 +1,3331 @@
+package bundler
+
+// The bundler is the core of the "build" and "transform" API calls. Each
+// operation has two phases. The first phase scans the module graph, and is
+// represented by the "ScanBundle" function. The second phase generates the
+// output files from the module graph, and is implemented by the "Compile"
+// function.
+
+import (
+	"bytes"
+	"encoding/base32"
+	"encoding/base64"
+	"fmt"
+	"math/rand"
+	"net/http"
+	"sort"
+	"strings"
+	"sync"
+	"syscall"
+	"time"
+	"unicode"
+	"unicode/utf8"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/cache"
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/config"
+	"github.com/evanw/esbuild/internal/css_parser"
+	"github.com/evanw/esbuild/internal/fs"
+	"github.com/evanw/esbuild/internal/graph"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/js_lexer"
+	"github.com/evanw/esbuild/internal/js_parser"
+	"github.com/evanw/esbuild/internal/logger"
+	"github.com/evanw/esbuild/internal/resolver"
+	"github.com/evanw/esbuild/internal/runtime"
+	"github.com/evanw/esbuild/internal/sourcemap"
+	"github.com/evanw/esbuild/internal/xxhash"
+)
+
+type scannerFile struct {
+	// If "AbsMetadataFile" is present, this will be filled out with information
+	// about this file in JSON format. This is a partial JSON file that will be
+	// fully assembled later.
+	jsonMetadataChunk string
+
+	pluginData interface{}
+	inputFile  graph.InputFile
+}
+
+// This is data related to source maps. It's computed in parallel with linking
+// and must be ready by the time printing happens. This is beneficial because
+// it is somewhat expensive to produce.
+type DataForSourceMap struct {
+	// This data is for the printer. It maps from byte offsets in the file (which
+	// are stored at every AST node) to UTF-16 column offsets (required by source
+	// maps).
+	LineOffsetTables []sourcemap.LineOffsetTable
+
+	// This contains the quoted contents of the original source file. It's what
+	// needs to be embedded in the "sourcesContent" array in the final source
+	// map. Quoting is precomputed because it's somewhat expensive.
+	QuotedContents [][]byte
+}
+
+type Bundle struct {
+	// The unique key prefix is a random string that is unique to every bundling
+	// operation. It is used as a prefix for the unique keys assigned to every
+	// chunk during linking. These unique keys are used to identify each chunk
+	// before the final output paths have been computed.
+	uniqueKeyPrefix string
+
+	fs          fs.FS
+	res         *resolver.Resolver
+	files       []scannerFile
+	entryPoints []graph.EntryPoint
+	options     config.Options
+}
+
+type parseArgs struct {
+	fs              fs.FS
+	log             logger.Log
+	res             *resolver.Resolver
+	caches          *cache.CacheSet
+	prettyPath      string
+	importSource    *logger.Source
+	importWith      *ast.ImportAssertOrWith
+	sideEffects     graph.SideEffects
+	pluginData      interface{}
+	results         chan parseResult
+	inject          chan config.InjectedFile
+	uniqueKeyPrefix string
+	keyPath         logger.Path
+	options         config.Options
+	importPathRange logger.Range
+	sourceIndex     uint32
+	skipResolve     bool
+}
+
+type parseResult struct {
+	resolveResults     []*resolver.ResolveResult
+	globResolveResults map[uint32]globResolveResult
+	file               scannerFile
+	tlaCheck           tlaCheck
+	ok                 bool
+}
+
+type globResolveResult struct {
+	resolveResults map[string]resolver.ResolveResult
+	absPath        string
+	prettyPath     string
+	exportAlias    string
+}
+
+type tlaCheck struct {
+	parent            ast.Index32
+	depth             uint32
+	importRecordIndex uint32
+}
+
+func parseFile(args parseArgs) {
+	source := logger.Source{
+		Index:          args.sourceIndex,
+		KeyPath:        args.keyPath,
+		PrettyPath:     args.prettyPath,
+		IdentifierName: js_ast.GenerateNonUniqueNameFromPath(args.keyPath.Text),
+	}
+
+	var loader config.Loader
+	var absResolveDir string
+	var pluginName string
+	var pluginData interface{}
+
+	if stdin := args.options.Stdin; stdin != nil {
+		// Special-case stdin
+		source.Contents = stdin.Contents
+		loader = stdin.Loader
+		if loader == config.LoaderNone {
+			loader = config.LoaderJS
+		}
+		absResolveDir = args.options.Stdin.AbsResolveDir
+	} else {
+		result, ok := runOnLoadPlugins(
+			args.options.Plugins,
+			args.fs,
+			&args.caches.FSCache,
+			args.log,
+			&source,
+			args.importSource,
+			args.importPathRange,
+			args.pluginData,
+			args.options.WatchMode,
+		)
+		if !ok {
+			if args.inject != nil {
+				args.inject <- config.InjectedFile{
+					Source: source,
+				}
+			}
+			args.results <- parseResult{}
+			return
+		}
+		loader = result.loader
+		absResolveDir = result.absResolveDir
+		pluginName = result.pluginName
+		pluginData = result.pluginData
+	}
+
+	_, base, ext := logger.PlatformIndependentPathDirBaseExt(source.KeyPath.Text)
+
+	// The special "default" loader determines the loader from the file path
+	if loader == config.LoaderDefault {
+		loader = loaderFromFileExtension(args.options.ExtensionToLoader, base+ext)
+	}
+
+	// Reject unsupported import attributes when the loader isn't "copy" (since
+	// "copy" is kind of like "external"). But only do this if this file was not
+	// loaded by a plugin. Plugins are allowed to assign whatever semantics they
+	// want to import attributes.
+	if loader != config.LoaderCopy && pluginName == "" {
+		for _, attr := range source.KeyPath.ImportAttributes.DecodeIntoArray() {
+			var errorText string
+			var errorRange js_lexer.KeyOrValue
+
+			// We only currently handle "type: json"
+			if attr.Key != "type" {
+				errorText = fmt.Sprintf("Importing with the %q attribute is not supported", attr.Key)
+				errorRange = js_lexer.KeyRange
+			} else if attr.Value == "json" {
+				loader = config.LoaderWithTypeJSON
+				continue
+			} else {
+				errorText = fmt.Sprintf("Importing with a type attribute of %q is not supported", attr.Value)
+				errorRange = js_lexer.ValueRange
+			}
+
+			// Everything else is an error
+			r := args.importPathRange
+			if args.importWith != nil {
+				r = js_lexer.RangeOfImportAssertOrWith(*args.importSource, *ast.FindAssertOrWithEntry(args.importWith.Entries, attr.Key), errorRange)
+			}
+			tracker := logger.MakeLineColumnTracker(args.importSource)
+			args.log.AddError(&tracker, r, errorText)
+			if args.inject != nil {
+				args.inject <- config.InjectedFile{
+					Source: source,
+				}
+			}
+			args.results <- parseResult{}
+			return
+		}
+	}
+
+	if loader == config.LoaderEmpty {
+		source.Contents = ""
+	}
+
+	result := parseResult{
+		file: scannerFile{
+			inputFile: graph.InputFile{
+				Source:      source,
+				Loader:      loader,
+				SideEffects: args.sideEffects,
+			},
+			pluginData: pluginData,
+		},
+	}
+
+	defer func() {
+		r := recover()
+		if r != nil {
+			args.log.AddErrorWithNotes(nil, logger.Range{},
+				fmt.Sprintf("panic: %v (while parsing %q)", r, source.PrettyPath),
+				[]logger.MsgData{{Text: helpers.PrettyPrintedStack()}})
+			args.results <- result
+		}
+	}()
+
+	switch loader {
+	case config.LoaderJS, config.LoaderEmpty:
+		ast, ok := args.caches.JSCache.Parse(args.log, source, js_parser.OptionsFromConfig(&args.options))
+		if len(ast.Parts) <= 1 { // Ignore the implicitly-generated namespace export part
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_EmptyAST
+		}
+		result.file.inputFile.Repr = &graph.JSRepr{AST: ast}
+		result.ok = ok
+
+	case config.LoaderJSX:
+		args.options.JSX.Parse = true
+		ast, ok := args.caches.JSCache.Parse(args.log, source, js_parser.OptionsFromConfig(&args.options))
+		if len(ast.Parts) <= 1 { // Ignore the implicitly-generated namespace export part
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_EmptyAST
+		}
+		result.file.inputFile.Repr = &graph.JSRepr{AST: ast}
+		result.ok = ok
+
+	case config.LoaderTS, config.LoaderTSNoAmbiguousLessThan:
+		args.options.TS.Parse = true
+		args.options.TS.NoAmbiguousLessThan = loader == config.LoaderTSNoAmbiguousLessThan
+		ast, ok := args.caches.JSCache.Parse(args.log, source, js_parser.OptionsFromConfig(&args.options))
+		if len(ast.Parts) <= 1 { // Ignore the implicitly-generated namespace export part
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_EmptyAST
+		}
+		result.file.inputFile.Repr = &graph.JSRepr{AST: ast}
+		result.ok = ok
+
+	case config.LoaderTSX:
+		args.options.TS.Parse = true
+		args.options.JSX.Parse = true
+		ast, ok := args.caches.JSCache.Parse(args.log, source, js_parser.OptionsFromConfig(&args.options))
+		if len(ast.Parts) <= 1 { // Ignore the implicitly-generated namespace export part
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_EmptyAST
+		}
+		result.file.inputFile.Repr = &graph.JSRepr{AST: ast}
+		result.ok = ok
+
+	case config.LoaderCSS, config.LoaderGlobalCSS, config.LoaderLocalCSS:
+		ast := args.caches.CSSCache.Parse(args.log, source, css_parser.OptionsFromConfig(loader, &args.options))
+		result.file.inputFile.Repr = &graph.CSSRepr{AST: ast}
+		result.ok = true
+
+	case config.LoaderJSON, config.LoaderWithTypeJSON:
+		expr, ok := args.caches.JSONCache.Parse(args.log, source, js_parser.JSONOptions{
+			UnsupportedJSFeatures: args.options.UnsupportedJSFeatures,
+		})
+		ast := js_parser.LazyExportAST(args.log, source, js_parser.OptionsFromConfig(&args.options), expr, "")
+		if loader == config.LoaderWithTypeJSON {
+			// The exports kind defaults to "none", in which case the linker picks
+			// either ESM or CommonJS depending on the situation. Dynamic imports
+			// causes the linker to pick CommonJS which uses "require()" and then
+			// converts the return value to ESM, which adds extra properties that
+			// aren't supposed to be there when "{ with: { type: 'json' } }" is
+			// present. So if there's an import attribute, we force the type to
+			// be ESM to avoid this.
+			ast.ExportsKind = js_ast.ExportsESM
+		}
+		if pluginName != "" {
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_PureData_FromPlugin
+		} else {
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_PureData
+		}
+		result.file.inputFile.Repr = &graph.JSRepr{AST: ast}
+		result.ok = ok
+
+	case config.LoaderText:
+		encoded := base64.StdEncoding.EncodeToString([]byte(source.Contents))
+		expr := js_ast.Expr{Data: &js_ast.EString{Value: helpers.StringToUTF16(source.Contents)}}
+		ast := js_parser.LazyExportAST(args.log, source, js_parser.OptionsFromConfig(&args.options), expr, "")
+		ast.URLForCSS = "data:text/plain;base64," + encoded
+		if pluginName != "" {
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_PureData_FromPlugin
+		} else {
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_PureData
+		}
+		result.file.inputFile.Repr = &graph.JSRepr{AST: ast}
+		result.ok = true
+
+	case config.LoaderBase64:
+		mimeType := guessMimeType(ext, source.Contents)
+		encoded := base64.StdEncoding.EncodeToString([]byte(source.Contents))
+		expr := js_ast.Expr{Data: &js_ast.EString{Value: helpers.StringToUTF16(encoded)}}
+		ast := js_parser.LazyExportAST(args.log, source, js_parser.OptionsFromConfig(&args.options), expr, "")
+		ast.URLForCSS = "data:" + mimeType + ";base64," + encoded
+		if pluginName != "" {
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_PureData_FromPlugin
+		} else {
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_PureData
+		}
+		result.file.inputFile.Repr = &graph.JSRepr{AST: ast}
+		result.ok = true
+
+	case config.LoaderBinary:
+		encoded := base64.StdEncoding.EncodeToString([]byte(source.Contents))
+		expr := js_ast.Expr{Data: &js_ast.EString{Value: helpers.StringToUTF16(encoded)}}
+		helper := "__toBinary"
+		if args.options.Platform == config.PlatformNode {
+			helper = "__toBinaryNode"
+		}
+		ast := js_parser.LazyExportAST(args.log, source, js_parser.OptionsFromConfig(&args.options), expr, helper)
+		ast.URLForCSS = "data:application/octet-stream;base64," + encoded
+		if pluginName != "" {
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_PureData_FromPlugin
+		} else {
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_PureData
+		}
+		result.file.inputFile.Repr = &graph.JSRepr{AST: ast}
+		result.ok = true
+
+	case config.LoaderDataURL:
+		mimeType := guessMimeType(ext, source.Contents)
+		url := helpers.EncodeStringAsShortestDataURL(mimeType, source.Contents)
+		expr := js_ast.Expr{Data: &js_ast.EString{Value: helpers.StringToUTF16(url)}}
+		ast := js_parser.LazyExportAST(args.log, source, js_parser.OptionsFromConfig(&args.options), expr, "")
+		ast.URLForCSS = url
+		if pluginName != "" {
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_PureData_FromPlugin
+		} else {
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_PureData
+		}
+		result.file.inputFile.Repr = &graph.JSRepr{AST: ast}
+		result.ok = true
+
+	case config.LoaderFile:
+		uniqueKey := fmt.Sprintf("%sA%08d", args.uniqueKeyPrefix, args.sourceIndex)
+		uniqueKeyPath := uniqueKey + source.KeyPath.IgnoredSuffix
+		expr := js_ast.Expr{Data: &js_ast.EString{
+			Value:             helpers.StringToUTF16(uniqueKeyPath),
+			ContainsUniqueKey: true,
+		}}
+		ast := js_parser.LazyExportAST(args.log, source, js_parser.OptionsFromConfig(&args.options), expr, "")
+		ast.URLForCSS = uniqueKeyPath
+		if pluginName != "" {
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_PureData_FromPlugin
+		} else {
+			result.file.inputFile.SideEffects.Kind = graph.NoSideEffects_PureData
+		}
+		result.file.inputFile.Repr = &graph.JSRepr{AST: ast}
+		result.ok = true
+
+		// Mark that this file is from the "file" loader
+		result.file.inputFile.UniqueKeyForAdditionalFile = uniqueKey
+
+	case config.LoaderCopy:
+		uniqueKey := fmt.Sprintf("%sA%08d", args.uniqueKeyPrefix, args.sourceIndex)
+		uniqueKeyPath := uniqueKey + source.KeyPath.IgnoredSuffix
+		result.file.inputFile.Repr = &graph.CopyRepr{
+			URLForCode: uniqueKeyPath,
+		}
+		result.ok = true
+
+		// Mark that this file is from the "copy" loader
+		result.file.inputFile.UniqueKeyForAdditionalFile = uniqueKey
+
+	default:
+		var message string
+		if source.KeyPath.Namespace == "file" && ext != "" {
+			message = fmt.Sprintf("No loader is configured for %q files: %s", ext, source.PrettyPath)
+		} else {
+			message = fmt.Sprintf("Do not know how to load path: %s", source.PrettyPath)
+		}
+		tracker := logger.MakeLineColumnTracker(args.importSource)
+		args.log.AddError(&tracker, args.importPathRange, message)
+	}
+
+	// Only continue now if parsing was successful
+	if result.ok {
+		// Run the resolver on the parse thread so it's not run on the main thread.
+		// That way the main thread isn't blocked if the resolver takes a while.
+		if recordsPtr := result.file.inputFile.Repr.ImportRecords(); args.options.Mode == config.ModeBundle && !args.skipResolve && recordsPtr != nil {
+			// Clone the import records because they will be mutated later
+			records := append([]ast.ImportRecord{}, *recordsPtr...)
+			*recordsPtr = records
+			result.resolveResults = make([]*resolver.ResolveResult, len(records))
+
+			if len(records) > 0 {
+				type cacheEntry struct {
+					resolveResult *resolver.ResolveResult
+					debug         resolver.DebugMeta
+					didLogError   bool
+				}
+
+				type cacheKey struct {
+					kind  ast.ImportKind
+					path  string
+					attrs logger.ImportAttributes
+				}
+				resolverCache := make(map[cacheKey]cacheEntry)
+				tracker := logger.MakeLineColumnTracker(&source)
+
+				for importRecordIndex := range records {
+					// Don't try to resolve imports that are already resolved
+					record := &records[importRecordIndex]
+					if record.SourceIndex.IsValid() {
+						continue
+					}
+
+					// Encode the import attributes
+					var attrs logger.ImportAttributes
+					if record.AssertOrWith != nil && record.AssertOrWith.Keyword == ast.WithKeyword {
+						data := make(map[string]string, len(record.AssertOrWith.Entries))
+						for _, entry := range record.AssertOrWith.Entries {
+							data[helpers.UTF16ToString(entry.Key)] = helpers.UTF16ToString(entry.Value)
+						}
+						attrs = logger.EncodeImportAttributes(data)
+					}
+
+					// Special-case glob pattern imports
+					if record.GlobPattern != nil {
+						prettyPath := helpers.GlobPatternToString(record.GlobPattern.Parts)
+						switch record.GlobPattern.Kind {
+						case ast.ImportRequire:
+							prettyPath = fmt.Sprintf("require(%q)", prettyPath)
+						case ast.ImportDynamic:
+							prettyPath = fmt.Sprintf("import(%q)", prettyPath)
+						}
+						if results, msg := args.res.ResolveGlob(absResolveDir, record.GlobPattern.Parts, record.GlobPattern.Kind, prettyPath); results != nil {
+							if msg != nil {
+								args.log.AddID(msg.ID, msg.Kind, &tracker, record.Range, msg.Data.Text)
+							}
+							if result.globResolveResults == nil {
+								result.globResolveResults = make(map[uint32]globResolveResult)
+							}
+							for key, result := range results {
+								result.PathPair.Primary.ImportAttributes = attrs
+								if result.PathPair.HasSecondary() {
+									result.PathPair.Secondary.ImportAttributes = attrs
+								}
+								results[key] = result
+							}
+							result.globResolveResults[uint32(importRecordIndex)] = globResolveResult{
+								resolveResults: results,
+								absPath:        args.fs.Join(absResolveDir, "(glob)"),
+								prettyPath:     fmt.Sprintf("%s in %s", prettyPath, result.file.inputFile.Source.PrettyPath),
+								exportAlias:    record.GlobPattern.ExportAlias,
+							}
+						} else {
+							args.log.AddError(&tracker, record.Range, fmt.Sprintf("Could not resolve %s", prettyPath))
+						}
+						continue
+					}
+
+					// Ignore records that the parser has discarded. This is used to remove
+					// type-only imports in TypeScript files.
+					if record.Flags.Has(ast.IsUnused) {
+						continue
+					}
+
+					// Cache the path in case it's imported multiple times in this file
+					cacheKey := cacheKey{
+						kind:  record.Kind,
+						path:  record.Path.Text,
+						attrs: attrs,
+					}
+					entry, ok := resolverCache[cacheKey]
+					if ok {
+						result.resolveResults[importRecordIndex] = entry.resolveResult
+					} else {
+						// Run the resolver and log an error if the path couldn't be resolved
+						resolveResult, didLogError, debug := RunOnResolvePlugins(
+							args.options.Plugins,
+							args.res,
+							args.log,
+							args.fs,
+							&args.caches.FSCache,
+							&source,
+							record.Range,
+							source.KeyPath,
+							record.Path.Text,
+							attrs,
+							record.Kind,
+							absResolveDir,
+							pluginData,
+						)
+						if resolveResult != nil {
+							resolveResult.PathPair.Primary.ImportAttributes = attrs
+							if resolveResult.PathPair.HasSecondary() {
+								resolveResult.PathPair.Secondary.ImportAttributes = attrs
+							}
+						}
+						entry = cacheEntry{
+							resolveResult: resolveResult,
+							debug:         debug,
+							didLogError:   didLogError,
+						}
+						resolverCache[cacheKey] = entry
+
+						// All "require.resolve()" imports should be external because we don't
+						// want to waste effort traversing into them
+						if record.Kind == ast.ImportRequireResolve {
+							if resolveResult != nil && resolveResult.PathPair.IsExternal {
+								// Allow path substitution as long as the result is external
+								result.resolveResults[importRecordIndex] = resolveResult
+							} else if !record.Flags.Has(ast.HandlesImportErrors) {
+								args.log.AddID(logger.MsgID_Bundler_RequireResolveNotExternal, logger.Warning, &tracker, record.Range,
+									fmt.Sprintf("%q should be marked as external for use with \"require.resolve\"", record.Path.Text))
+							}
+							continue
+						}
+					}
+
+					// Check whether we should log an error every time the result is nil,
+					// even if it's from the cache. Do this because the error may not
+					// have been logged for nil entries if the previous instances had
+					// the "HandlesImportErrors" flag.
+					if entry.resolveResult == nil {
+						// Failed imports inside a try/catch are silently turned into
+						// external imports instead of causing errors. This matches a common
+						// code pattern for conditionally importing a module with a graceful
+						// fallback.
+						if !entry.didLogError && !record.Flags.Has(ast.HandlesImportErrors) {
+							// Report an error
+							text, suggestion, notes := ResolveFailureErrorTextSuggestionNotes(args.res, record.Path.Text, record.Kind,
+								pluginName, args.fs, absResolveDir, args.options.Platform, source.PrettyPath, entry.debug.ModifiedImportPath)
+							entry.debug.LogErrorMsg(args.log, &source, record.Range, text, suggestion, notes)
+
+							// Only report this error once per unique import path in the file
+							entry.didLogError = true
+							resolverCache[cacheKey] = entry
+						} else if !entry.didLogError && record.Flags.Has(ast.HandlesImportErrors) {
+							// Report a debug message about why there was no error
+							args.log.AddIDWithNotes(logger.MsgID_Bundler_IgnoredDynamicImport, logger.Debug, &tracker, record.Range,
+								fmt.Sprintf("Importing %q was allowed even though it could not be resolved because dynamic import failures appear to be handled here:",
+									record.Path.Text), []logger.MsgData{tracker.MsgData(js_lexer.RangeOfIdentifier(source, record.ErrorHandlerLoc),
+									"The handler for dynamic import failures is here:")})
+						}
+						continue
+					}
+
+					result.resolveResults[importRecordIndex] = entry.resolveResult
+				}
+			}
+		}
+
+		// Attempt to parse the source map if present
+		if loader.CanHaveSourceMap() && args.options.SourceMap != config.SourceMapNone {
+			var sourceMapComment logger.Span
+			switch repr := result.file.inputFile.Repr.(type) {
+			case *graph.JSRepr:
+				sourceMapComment = repr.AST.SourceMapComment
+			case *graph.CSSRepr:
+				sourceMapComment = repr.AST.SourceMapComment
+			}
+
+			if sourceMapComment.Text != "" {
+				tracker := logger.MakeLineColumnTracker(&source)
+
+				if path, contents := extractSourceMapFromComment(args.log, args.fs, &args.caches.FSCache,
+					&source, &tracker, sourceMapComment, absResolveDir); contents != nil {
+					prettyPath := resolver.PrettyPath(args.fs, path)
+					log := logger.NewDeferLog(logger.DeferLogNoVerboseOrDebug, args.log.Overrides)
+
+					sourceMap := js_parser.ParseSourceMap(log, logger.Source{
+						KeyPath:    path,
+						PrettyPath: prettyPath,
+						Contents:   *contents,
+					})
+
+					if msgs := log.Done(); len(msgs) > 0 {
+						var text string
+						if path.Namespace == "file" {
+							text = fmt.Sprintf("The source map %q was referenced by the file %q here:", prettyPath, args.prettyPath)
+						} else {
+							text = fmt.Sprintf("This source map came from the file %q here:", args.prettyPath)
+						}
+						note := tracker.MsgData(sourceMapComment.Range, text)
+						for _, msg := range msgs {
+							msg.Notes = append(msg.Notes, note)
+							args.log.AddMsg(msg)
+						}
+					}
+
+					// If "sourcesContent" entries aren't present, try filling them in
+					// using the file system. This includes both generating the entire
+					// "sourcesContent" array if it's absent as well as filling in
+					// individual null entries in the array if the array is present.
+					if sourceMap != nil && !args.options.ExcludeSourcesContent {
+						// Make sure "sourcesContent" is big enough
+						if len(sourceMap.SourcesContent) < len(sourceMap.Sources) {
+							slice := make([]sourcemap.SourceContent, len(sourceMap.Sources))
+							copy(slice, sourceMap.SourcesContent)
+							sourceMap.SourcesContent = slice
+						}
+
+						// Attempt to fill in null entries using the file system
+						for i, source := range sourceMap.Sources {
+							if sourceMap.SourcesContent[i].Value == nil {
+								var absPath string
+								if args.fs.IsAbs(source) {
+									absPath = source
+								} else if path.Namespace == "file" {
+									absPath = args.fs.Join(args.fs.Dir(path.Text), source)
+								} else {
+									continue
+								}
+								if contents, err, _ := args.caches.FSCache.ReadFile(args.fs, absPath); err == nil {
+									sourceMap.SourcesContent[i].Value = helpers.StringToUTF16(contents)
+								}
+							}
+						}
+					}
+
+					result.file.inputFile.InputSourceMap = sourceMap
+				}
+			}
+		}
+	}
+
+	// Note: We must always send on the "inject" channel before we send on the
+	// "results" channel to avoid deadlock
+	if args.inject != nil {
+		var exports []config.InjectableExport
+
+		if repr, ok := result.file.inputFile.Repr.(*graph.JSRepr); ok {
+			aliases := make([]string, 0, len(repr.AST.NamedExports))
+			for alias := range repr.AST.NamedExports {
+				aliases = append(aliases, alias)
+			}
+			sort.Strings(aliases) // Sort for determinism
+			exports = make([]config.InjectableExport, len(aliases))
+			for i, alias := range aliases {
+				exports[i] = config.InjectableExport{
+					Alias: alias,
+					Loc:   repr.AST.NamedExports[alias].AliasLoc,
+				}
+			}
+		}
+
+		// Once we send on the "inject" channel, the main thread may mutate the
+		// "options" object to populate the "InjectedFiles" field. So we must
+		// only send on the "inject" channel after we're done using the "options"
+		// object so we don't introduce a data race.
+		isCopyLoader := loader == config.LoaderCopy
+		if isCopyLoader && args.skipResolve {
+			// This is not allowed because the import path would have to be rewritten,
+			// but import paths are not rewritten when bundling isn't enabled.
+			args.log.AddError(nil, logger.Range{},
+				fmt.Sprintf("Cannot inject %q with the \"copy\" loader without bundling enabled", source.PrettyPath))
+		}
+		args.inject <- config.InjectedFile{
+			Source:       source,
+			Exports:      exports,
+			IsCopyLoader: isCopyLoader,
+		}
+	}
+
+	args.results <- result
+}
+
+func ResolveFailureErrorTextSuggestionNotes(
+	res *resolver.Resolver,
+	path string,
+	kind ast.ImportKind,
+	pluginName string,
+	fs fs.FS,
+	absResolveDir string,
+	platform config.Platform,
+	originatingFilePath string,
+	modifiedImportPath string,
+) (text string, suggestion string, notes []logger.MsgData) {
+	if modifiedImportPath != "" {
+		text = fmt.Sprintf("Could not resolve %q (originally %q)", modifiedImportPath, path)
+		notes = append(notes, logger.MsgData{Text: fmt.Sprintf(
+			"The path %q was remapped to %q using the alias feature, which then couldn't be resolved. "+
+				"Keep in mind that import path aliases are resolved in the current working directory.",
+			path, modifiedImportPath)})
+		path = modifiedImportPath
+	} else {
+		text = fmt.Sprintf("Could not resolve %q", path)
+	}
+	hint := ""
+
+	if resolver.IsPackagePath(path) && !fs.IsAbs(path) {
+		hint = fmt.Sprintf("You can mark the path %q as external to exclude it from the bundle, which will remove this error and leave the unresolved path in the bundle.", path)
+		if kind == ast.ImportRequire {
+			hint += " You can also surround this \"require\" call with a try/catch block to handle this failure at run-time instead of bundle-time."
+		} else if kind == ast.ImportDynamic {
+			hint += " You can also add \".catch()\" here to handle this failure at run-time instead of bundle-time."
+		}
+		if pluginName == "" && !fs.IsAbs(path) {
+			if query, _ := res.ProbeResolvePackageAsRelative(absResolveDir, path, kind); query != nil {
+				hint = fmt.Sprintf("Use the relative path %q to reference the file %q. "+
+					"Without the leading \"./\", the path %q is being interpreted as a package path instead.",
+					"./"+path, resolver.PrettyPath(fs, query.PathPair.Primary), path)
+				suggestion = string(helpers.QuoteForJSON("./"+path, false))
+			}
+		}
+	}
+
+	if platform != config.PlatformNode {
+		pkg := strings.TrimPrefix(path, "node:")
+		if resolver.BuiltInNodeModules[pkg] {
+			var how string
+			switch logger.API {
+			case logger.CLIAPI:
+				how = "--platform=node"
+			case logger.JSAPI:
+				how = "platform: 'node'"
+			case logger.GoAPI:
+				how = "Platform: api.PlatformNode"
+			}
+			hint = fmt.Sprintf("The package %q wasn't found on the file system but is built into node. "+
+				"Are you trying to bundle for node? You can use %q to do that, which will remove this error.", path, how)
+		}
+	}
+
+	if absResolveDir == "" && pluginName != "" {
+		where := ""
+		if originatingFilePath != "" {
+			where = fmt.Sprintf(" for the file %q", originatingFilePath)
+		}
+		hint = fmt.Sprintf("The plugin %q didn't set a resolve directory%s, "+
+			"so esbuild did not search for %q on the file system.", pluginName, where, path)
+	}
+
+	if hint != "" {
+		if modifiedImportPath != "" {
+			// Add a newline if there's already a paragraph of text
+			notes = append(notes, logger.MsgData{})
+
+			// Don't add a suggestion if the path was rewritten using an alias
+			suggestion = ""
+		}
+		notes = append(notes, logger.MsgData{Text: hint})
+	}
+	return
+}
+
+func isASCIIOnly(text string) bool {
+	for _, c := range text {
+		if c < 0x20 || c > 0x7E {
+			return false
+		}
+	}
+	return true
+}
+
+func guessMimeType(extension string, contents string) string {
+	mimeType := helpers.MimeTypeByExtension(extension)
+	if mimeType == "" {
+		mimeType = http.DetectContentType([]byte(contents))
+	}
+
+	// Turn "text/plain; charset=utf-8" into "text/plain;charset=utf-8"
+	return strings.ReplaceAll(mimeType, "; ", ";")
+}
+
+func extractSourceMapFromComment(
+	log logger.Log,
+	fs fs.FS,
+	fsCache *cache.FSCache,
+	source *logger.Source,
+	tracker *logger.LineColumnTracker,
+	comment logger.Span,
+	absResolveDir string,
+) (logger.Path, *string) {
+	// Support data URLs
+	if parsed, ok := resolver.ParseDataURL(comment.Text); ok {
+		if contents, err := parsed.DecodeData(); err == nil {
+			return logger.Path{Text: source.PrettyPath, IgnoredSuffix: "#sourceMappingURL"}, &contents
+		} else {
+			log.AddID(logger.MsgID_SourceMap_UnsupportedSourceMapComment, logger.Warning, tracker, comment.Range,
+				fmt.Sprintf("Unsupported source map comment: %s", err.Error()))
+			return logger.Path{}, nil
+		}
+	}
+
+	// Relative path in a file with an absolute path
+	if absResolveDir != "" {
+		absPath := fs.Join(absResolveDir, comment.Text)
+		path := logger.Path{Text: absPath, Namespace: "file"}
+		contents, err, originalError := fsCache.ReadFile(fs, absPath)
+		if log.Level <= logger.LevelDebug && originalError != nil {
+			log.AddID(logger.MsgID_None, logger.Debug, tracker, comment.Range, fmt.Sprintf("Failed to read file %q: %s", resolver.PrettyPath(fs, path), originalError.Error()))
+		}
+		if err != nil {
+			kind := logger.Warning
+			if err == syscall.ENOENT {
+				// Don't report a warning because this is likely unactionable
+				kind = logger.Debug
+			}
+			log.AddID(logger.MsgID_SourceMap_MissingSourceMap, kind, tracker, comment.Range,
+				fmt.Sprintf("Cannot read file %q: %s", resolver.PrettyPath(fs, path), err.Error()))
+			return logger.Path{}, nil
+		}
+		return path, &contents
+	}
+
+	// Anything else is unsupported
+	return logger.Path{}, nil
+}
+
+func sanitizeLocation(fs fs.FS, loc *logger.MsgLocation) {
+	if loc != nil {
+		if loc.Namespace == "" {
+			loc.Namespace = "file"
+		}
+		if loc.File != "" {
+			loc.File = resolver.PrettyPath(fs, logger.Path{Text: loc.File, Namespace: loc.Namespace})
+		}
+	}
+}
+
+func logPluginMessages(
+	fs fs.FS,
+	log logger.Log,
+	name string,
+	msgs []logger.Msg,
+	thrown error,
+	importSource *logger.Source,
+	importPathRange logger.Range,
+) bool {
+	didLogError := false
+	tracker := logger.MakeLineColumnTracker(importSource)
+
+	// Report errors and warnings generated by the plugin
+	for _, msg := range msgs {
+		if msg.PluginName == "" {
+			msg.PluginName = name
+		}
+		if msg.Kind == logger.Error {
+			didLogError = true
+		}
+
+		// Sanitize the locations
+		for _, note := range msg.Notes {
+			sanitizeLocation(fs, note.Location)
+		}
+		if msg.Data.Location == nil {
+			msg.Data.Location = tracker.MsgLocationOrNil(importPathRange)
+		} else {
+			sanitizeLocation(fs, msg.Data.Location)
+			if importSource != nil {
+				if msg.Data.Location.File == "" {
+					msg.Data.Location.File = importSource.PrettyPath
+				}
+				msg.Notes = append(msg.Notes, tracker.MsgData(importPathRange,
+					fmt.Sprintf("The plugin %q was triggered by this import", name)))
+			}
+		}
+
+		log.AddMsg(msg)
+	}
+
+	// Report errors thrown by the plugin itself
+	if thrown != nil {
+		didLogError = true
+		text := thrown.Error()
+		log.AddMsg(logger.Msg{
+			PluginName: name,
+			Kind:       logger.Error,
+			Data: logger.MsgData{
+				Text:       text,
+				Location:   tracker.MsgLocationOrNil(importPathRange),
+				UserDetail: thrown,
+			},
+		})
+	}
+
+	return didLogError
+}
+
+func RunOnResolvePlugins(
+	plugins []config.Plugin,
+	res *resolver.Resolver,
+	log logger.Log,
+	fs fs.FS,
+	fsCache *cache.FSCache,
+	importSource *logger.Source,
+	importPathRange logger.Range,
+	importer logger.Path,
+	path string,
+	importAttributes logger.ImportAttributes,
+	kind ast.ImportKind,
+	absResolveDir string,
+	pluginData interface{},
+) (*resolver.ResolveResult, bool, resolver.DebugMeta) {
+	resolverArgs := config.OnResolveArgs{
+		Path:       path,
+		ResolveDir: absResolveDir,
+		Kind:       kind,
+		PluginData: pluginData,
+		Importer:   importer,
+		With:       importAttributes,
+	}
+	applyPath := logger.Path{
+		Text:      path,
+		Namespace: importer.Namespace,
+	}
+	tracker := logger.MakeLineColumnTracker(importSource)
+
+	// Apply resolver plugins in order until one succeeds
+	for _, plugin := range plugins {
+		for _, onResolve := range plugin.OnResolve {
+			if !config.PluginAppliesToPath(applyPath, onResolve.Filter, onResolve.Namespace) {
+				continue
+			}
+
+			result := onResolve.Callback(resolverArgs)
+			pluginName := result.PluginName
+			if pluginName == "" {
+				pluginName = plugin.Name
+			}
+			didLogError := logPluginMessages(fs, log, pluginName, result.Msgs, result.ThrownError, importSource, importPathRange)
+
+			// Plugins can also provide additional file system paths to watch
+			for _, file := range result.AbsWatchFiles {
+				fsCache.ReadFile(fs, file)
+			}
+			for _, dir := range result.AbsWatchDirs {
+				if entries, err, _ := fs.ReadDirectory(dir); err == nil {
+					entries.SortedKeys()
+				}
+			}
+
+			// Stop now if there was an error
+			if didLogError {
+				return nil, true, resolver.DebugMeta{}
+			}
+
+			// The "file" namespace is the default for non-external paths, but not
+			// for external paths. External paths must explicitly specify the "file"
+			// namespace.
+			nsFromPlugin := result.Path.Namespace
+			if result.Path.Namespace == "" && !result.External {
+				result.Path.Namespace = "file"
+			}
+
+			// Otherwise, continue on to the next resolver if this loader didn't succeed
+			if result.Path.Text == "" {
+				if result.External {
+					result.Path = logger.Path{Text: path}
+				} else {
+					continue
+				}
+			}
+
+			// Paths in the file namespace must be absolute paths
+			if result.Path.Namespace == "file" && !fs.IsAbs(result.Path.Text) {
+				if nsFromPlugin == "file" {
+					log.AddError(&tracker, importPathRange,
+						fmt.Sprintf("Plugin %q returned a path in the \"file\" namespace that is not an absolute path: %s", pluginName, result.Path.Text))
+				} else {
+					log.AddError(&tracker, importPathRange,
+						fmt.Sprintf("Plugin %q returned a non-absolute path: %s (set a namespace if this is not a file path)", pluginName, result.Path.Text))
+				}
+				return nil, true, resolver.DebugMeta{}
+			}
+
+			var sideEffectsData *resolver.SideEffectsData
+			if result.IsSideEffectFree {
+				sideEffectsData = &resolver.SideEffectsData{
+					PluginName: pluginName,
+				}
+			}
+
+			return &resolver.ResolveResult{
+				PathPair:               resolver.PathPair{Primary: result.Path, IsExternal: result.External},
+				PluginData:             result.PluginData,
+				PrimarySideEffectsData: sideEffectsData,
+			}, false, resolver.DebugMeta{}
+		}
+	}
+
+	// Resolve relative to the resolve directory by default. All paths in the
+	// "file" namespace automatically have a resolve directory. Loader plugins
+	// can also configure a custom resolve directory for files in other namespaces.
+	result, debug := res.Resolve(absResolveDir, path, kind)
+
+	// Warn when the case used for importing differs from the actual file name
+	if result != nil && result.DifferentCase != nil && !helpers.IsInsideNodeModules(absResolveDir) {
+		diffCase := *result.DifferentCase
+		log.AddID(logger.MsgID_Bundler_DifferentPathCase, logger.Warning, &tracker, importPathRange, fmt.Sprintf(
+			"Use %q instead of %q to avoid issues with case-sensitive file systems",
+			resolver.PrettyPath(fs, logger.Path{Text: fs.Join(diffCase.Dir, diffCase.Actual), Namespace: "file"}),
+			resolver.PrettyPath(fs, logger.Path{Text: fs.Join(diffCase.Dir, diffCase.Query), Namespace: "file"}),
+		))
+	}
+
+	return result, false, debug
+}
+
+type loaderPluginResult struct {
+	pluginData    interface{}
+	absResolveDir string
+	pluginName    string
+	loader        config.Loader
+}
+
+func runOnLoadPlugins(
+	plugins []config.Plugin,
+	fs fs.FS,
+	fsCache *cache.FSCache,
+	log logger.Log,
+	source *logger.Source,
+	importSource *logger.Source,
+	importPathRange logger.Range,
+	pluginData interface{},
+	isWatchMode bool,
+) (loaderPluginResult, bool) {
+	loaderArgs := config.OnLoadArgs{
+		Path:       source.KeyPath,
+		PluginData: pluginData,
+	}
+	tracker := logger.MakeLineColumnTracker(importSource)
+
+	// Apply loader plugins in order until one succeeds
+	for _, plugin := range plugins {
+		for _, onLoad := range plugin.OnLoad {
+			if !config.PluginAppliesToPath(source.KeyPath, onLoad.Filter, onLoad.Namespace) {
+				continue
+			}
+
+			result := onLoad.Callback(loaderArgs)
+			pluginName := result.PluginName
+			if pluginName == "" {
+				pluginName = plugin.Name
+			}
+			didLogError := logPluginMessages(fs, log, pluginName, result.Msgs, result.ThrownError, importSource, importPathRange)
+
+			// Plugins can also provide additional file system paths to watch
+			for _, file := range result.AbsWatchFiles {
+				fsCache.ReadFile(fs, file)
+			}
+			for _, dir := range result.AbsWatchDirs {
+				if entries, err, _ := fs.ReadDirectory(dir); err == nil {
+					entries.SortedKeys()
+				}
+			}
+
+			// Stop now if there was an error
+			if didLogError {
+				if isWatchMode && source.KeyPath.Namespace == "file" {
+					fsCache.ReadFile(fs, source.KeyPath.Text) // Read the file for watch mode tracking
+				}
+				return loaderPluginResult{}, false
+			}
+
+			// Otherwise, continue on to the next loader if this loader didn't succeed
+			if result.Contents == nil {
+				continue
+			}
+
+			source.Contents = *result.Contents
+			loader := result.Loader
+			if loader == config.LoaderNone {
+				loader = config.LoaderJS
+			}
+			if result.AbsResolveDir == "" && source.KeyPath.Namespace == "file" {
+				result.AbsResolveDir = fs.Dir(source.KeyPath.Text)
+			}
+			if isWatchMode && source.KeyPath.Namespace == "file" {
+				fsCache.ReadFile(fs, source.KeyPath.Text) // Read the file for watch mode tracking
+			}
+			return loaderPluginResult{
+				loader:        loader,
+				absResolveDir: result.AbsResolveDir,
+				pluginName:    pluginName,
+				pluginData:    result.PluginData,
+			}, true
+		}
+	}
+
+	// Force disabled modules to be empty
+	if source.KeyPath.IsDisabled() {
+		return loaderPluginResult{loader: config.LoaderEmpty}, true
+	}
+
+	// Read normal modules from disk
+	if source.KeyPath.Namespace == "file" {
+		if contents, err, originalError := fsCache.ReadFile(fs, source.KeyPath.Text); err == nil {
+			source.Contents = contents
+			return loaderPluginResult{
+				loader:        config.LoaderDefault,
+				absResolveDir: fs.Dir(source.KeyPath.Text),
+			}, true
+		} else {
+			if log.Level <= logger.LevelDebug && originalError != nil {
+				log.AddID(logger.MsgID_None, logger.Debug, nil, logger.Range{}, fmt.Sprintf("Failed to read file %q: %s", source.KeyPath.Text, originalError.Error()))
+			}
+			if err == syscall.ENOENT {
+				log.AddError(&tracker, importPathRange,
+					fmt.Sprintf("Could not read from file: %s", source.KeyPath.Text))
+				return loaderPluginResult{}, false
+			} else {
+				log.AddError(&tracker, importPathRange,
+					fmt.Sprintf("Cannot read file %q: %s", resolver.PrettyPath(fs, source.KeyPath), err.Error()))
+				return loaderPluginResult{}, false
+			}
+		}
+	}
+
+	// Native support for data URLs. This is supported natively by node:
+	// https://nodejs.org/docs/latest/api/esm.html#esm_data_imports
+	if source.KeyPath.Namespace == "dataurl" {
+		if parsed, ok := resolver.ParseDataURL(source.KeyPath.Text); ok {
+			if contents, err := parsed.DecodeData(); err != nil {
+				log.AddError(&tracker, importPathRange,
+					fmt.Sprintf("Could not load data URL: %s", err.Error()))
+				return loaderPluginResult{loader: config.LoaderNone}, true
+			} else {
+				source.Contents = contents
+				if mimeType := parsed.DecodeMIMEType(); mimeType != resolver.MIMETypeUnsupported {
+					switch mimeType {
+					case resolver.MIMETypeTextCSS:
+						return loaderPluginResult{loader: config.LoaderCSS}, true
+					case resolver.MIMETypeTextJavaScript:
+						return loaderPluginResult{loader: config.LoaderJS}, true
+					case resolver.MIMETypeApplicationJSON:
+						return loaderPluginResult{loader: config.LoaderJSON}, true
+					}
+				}
+			}
+		}
+	}
+
+	// Otherwise, fail to load the path
+	return loaderPluginResult{loader: config.LoaderNone}, true
+}
+
+func loaderFromFileExtension(extensionToLoader map[string]config.Loader, base string) config.Loader {
+	// Pick the loader with the longest matching extension. So if there's an
+	// extension for ".css" and for ".module.css", we want to match the one for
+	// ".module.css" before the one for ".css".
+	if i := strings.IndexByte(base, '.'); i != -1 {
+		for {
+			if loader, ok := extensionToLoader[base[i:]]; ok {
+				return loader
+			}
+			base = base[i+1:]
+			i = strings.IndexByte(base, '.')
+			if i == -1 {
+				break
+			}
+		}
+	} else {
+		// If there's no extension, explicitly check for an extensionless loader
+		if loader, ok := extensionToLoader[""]; ok {
+			return loader
+		}
+	}
+	return config.LoaderNone
+}
+
+// Identify the path by its lowercase absolute path name with Windows-specific
+// slashes substituted for standard slashes. This should hopefully avoid path
+// issues on Windows where multiple different paths can refer to the same
+// underlying file.
+func canonicalFileSystemPathForWindows(absPath string) string {
+	return strings.ReplaceAll(strings.ToLower(absPath), "\\", "/")
+}
+
+func HashForFileName(hashBytes []byte) string {
+	return base32.StdEncoding.EncodeToString(hashBytes)[:8]
+}
+
+type scanner struct {
+	log             logger.Log
+	fs              fs.FS
+	res             *resolver.Resolver
+	caches          *cache.CacheSet
+	timer           *helpers.Timer
+	uniqueKeyPrefix string
+
+	// These are not guarded by a mutex because it's only ever modified by a single
+	// thread. Note that not all results in the "results" array are necessarily
+	// valid. Make sure to check the "ok" flag before using them.
+	results       []parseResult
+	visited       map[logger.Path]visitedFile
+	resultChannel chan parseResult
+
+	options config.Options
+
+	// Also not guarded by a mutex for the same reason
+	remaining int
+}
+
+type visitedFile struct {
+	sourceIndex uint32
+}
+
+type EntryPoint struct {
+	InputPath                string
+	OutputPath               string
+	InputPathInFileNamespace bool
+}
+
+func generateUniqueKeyPrefix() (string, error) {
+	var data [12]byte
+	rand.Seed(time.Now().UnixNano())
+	if _, err := rand.Read(data[:]); err != nil {
+		return "", err
+	}
+
+	// This is 16 bytes and shouldn't generate escape characters when put into strings
+	return base64.URLEncoding.EncodeToString(data[:]), nil
+}
+
+// This creates a bundle by scanning over the whole module graph starting from
+// the entry points until all modules are reached. Each module has some number
+// of import paths which are resolved to module identifiers (i.e. "onResolve"
+// in the plugin API). Each unique module identifier is loaded once (i.e.
+// "onLoad" in the plugin API).
+func ScanBundle(
+	call config.APICall,
+	log logger.Log,
+	fs fs.FS,
+	caches *cache.CacheSet,
+	entryPoints []EntryPoint,
+	options config.Options,
+	timer *helpers.Timer,
+) Bundle {
+	timer.Begin("Scan phase")
+	defer timer.End("Scan phase")
+
+	applyOptionDefaults(&options)
+
+	// Run "onStart" plugins in parallel. IMPORTANT: We always need to run all
+	// "onStart" callbacks even when the build is cancelled, because plugins may
+	// rely on invariants that are started in "onStart" and ended in "onEnd".
+	// This works because "onEnd" callbacks are always run as well.
+	timer.Begin("On-start callbacks")
+	onStartWaitGroup := sync.WaitGroup{}
+	for _, plugin := range options.Plugins {
+		for _, onStart := range plugin.OnStart {
+			onStartWaitGroup.Add(1)
+			go func(plugin config.Plugin, onStart config.OnStart) {
+				result := onStart.Callback()
+				logPluginMessages(fs, log, plugin.Name, result.Msgs, result.ThrownError, nil, logger.Range{})
+				onStartWaitGroup.Done()
+			}(plugin, onStart)
+		}
+	}
+
+	// Each bundling operation gets a separate unique key
+	uniqueKeyPrefix, err := generateUniqueKeyPrefix()
+	if err != nil {
+		log.AddError(nil, logger.Range{}, fmt.Sprintf("Failed to read from randomness source: %s", err.Error()))
+	}
+
+	// This may mutate "options" by the "tsconfig.json" override settings
+	res := resolver.NewResolver(call, fs, log, caches, &options)
+
+	s := scanner{
+		log:             log,
+		fs:              fs,
+		res:             res,
+		caches:          caches,
+		options:         options,
+		timer:           timer,
+		results:         make([]parseResult, 0, caches.SourceIndexCache.LenHint()),
+		visited:         make(map[logger.Path]visitedFile),
+		resultChannel:   make(chan parseResult),
+		uniqueKeyPrefix: uniqueKeyPrefix,
+	}
+
+	// Always start by parsing the runtime file
+	s.results = append(s.results, parseResult{})
+	s.remaining++
+	go func() {
+		source, ast, ok := globalRuntimeCache.parseRuntime(&options)
+		s.resultChannel <- parseResult{
+			file: scannerFile{
+				inputFile: graph.InputFile{
+					Source: source,
+					Repr: &graph.JSRepr{
+						AST: ast,
+					},
+					OmitFromSourceMapsAndMetafile: true,
+				},
+			},
+			ok: ok,
+		}
+	}()
+
+	// Wait for all "onStart" plugins here before continuing. People sometimes run
+	// setup code in "onStart" that "onLoad" expects to be able to use without
+	// "onLoad" needing to block on the completion of their "onStart" callback.
+	//
+	// We want to enable this:
+	//
+	//   let plugin = {
+	//     name: 'example',
+	//     setup(build) {
+	//       let started = false
+	//       build.onStart(() => started = true)
+	//       build.onLoad({ filter: /.*/ }, () => {
+	//         assert(started === true)
+	//       })
+	//     },
+	//   }
+	//
+	// without people having to write something like this:
+	//
+	//   let plugin = {
+	//     name: 'example',
+	//     setup(build) {
+	//       let started = {}
+	//       started.promise = new Promise(resolve => {
+	//         started.resolve = resolve
+	//       })
+	//       build.onStart(() => {
+	//         started.resolve(true)
+	//       })
+	//       build.onLoad({ filter: /.*/ }, async () => {
+	//         assert(await started.promise === true)
+	//       })
+	//     },
+	//   }
+	//
+	onStartWaitGroup.Wait()
+	timer.End("On-start callbacks")
+
+	// We can check the cancel flag now that all "onStart" callbacks are done
+	if options.CancelFlag.DidCancel() {
+		return Bundle{options: options}
+	}
+
+	s.preprocessInjectedFiles()
+
+	if options.CancelFlag.DidCancel() {
+		return Bundle{options: options}
+	}
+
+	entryPointMeta := s.addEntryPoints(entryPoints)
+
+	if options.CancelFlag.DidCancel() {
+		return Bundle{options: options}
+	}
+
+	s.scanAllDependencies()
+
+	if options.CancelFlag.DidCancel() {
+		return Bundle{options: options}
+	}
+
+	files := s.processScannedFiles(entryPointMeta)
+
+	if options.CancelFlag.DidCancel() {
+		return Bundle{options: options}
+	}
+
+	return Bundle{
+		fs:              fs,
+		res:             s.res,
+		files:           files,
+		entryPoints:     entryPointMeta,
+		uniqueKeyPrefix: uniqueKeyPrefix,
+		options:         s.options,
+	}
+}
+
+type inputKind uint8
+
+const (
+	inputKindNormal inputKind = iota
+	inputKindEntryPoint
+	inputKindStdin
+)
+
+// This returns the source index of the resulting file
+func (s *scanner) maybeParseFile(
+	resolveResult resolver.ResolveResult,
+	prettyPath string,
+	importSource *logger.Source,
+	importPathRange logger.Range,
+	importWith *ast.ImportAssertOrWith,
+	kind inputKind,
+	inject chan config.InjectedFile,
+) uint32 {
+	path := resolveResult.PathPair.Primary
+	visitedKey := path
+	if visitedKey.Namespace == "file" {
+		visitedKey.Text = canonicalFileSystemPathForWindows(visitedKey.Text)
+	}
+
+	// Only parse a given file path once
+	visited, ok := s.visited[visitedKey]
+	if ok {
+		if inject != nil {
+			inject <- config.InjectedFile{}
+		}
+		return visited.sourceIndex
+	}
+
+	visited = visitedFile{
+		sourceIndex: s.allocateSourceIndex(visitedKey, cache.SourceIndexNormal),
+	}
+	s.visited[visitedKey] = visited
+	s.remaining++
+	optionsClone := s.options
+	if kind != inputKindStdin {
+		optionsClone.Stdin = nil
+	}
+
+	// Allow certain properties to be overridden by "tsconfig.json"
+	resolveResult.TSConfigJSX.ApplyTo(&optionsClone.JSX)
+	if resolveResult.TSConfig != nil {
+		optionsClone.TS.Config = *resolveResult.TSConfig
+	}
+	if resolveResult.TSAlwaysStrict != nil {
+		optionsClone.TSAlwaysStrict = resolveResult.TSAlwaysStrict
+	}
+
+	// Set the module type preference using node's module type rules
+	if strings.HasSuffix(path.Text, ".mjs") {
+		optionsClone.ModuleTypeData.Type = js_ast.ModuleESM_MJS
+	} else if strings.HasSuffix(path.Text, ".mts") {
+		optionsClone.ModuleTypeData.Type = js_ast.ModuleESM_MTS
+	} else if strings.HasSuffix(path.Text, ".cjs") {
+		optionsClone.ModuleTypeData.Type = js_ast.ModuleCommonJS_CJS
+	} else if strings.HasSuffix(path.Text, ".cts") {
+		optionsClone.ModuleTypeData.Type = js_ast.ModuleCommonJS_CTS
+	} else if strings.HasSuffix(path.Text, ".js") || strings.HasSuffix(path.Text, ".jsx") ||
+		strings.HasSuffix(path.Text, ".ts") || strings.HasSuffix(path.Text, ".tsx") {
+		optionsClone.ModuleTypeData = resolveResult.ModuleTypeData
+	} else {
+		// The "type" setting in "package.json" only applies to ".js" files
+		optionsClone.ModuleTypeData.Type = js_ast.ModuleUnknown
+	}
+
+	// Enable bundling for injected files so we always do tree shaking. We
+	// never want to include unnecessary code from injected files since they
+	// are essentially bundled. However, if we do this we should skip the
+	// resolving step when we're not bundling. It'd be strange to get
+	// resolution errors when the top-level bundling controls are disabled.
+	skipResolve := false
+	if inject != nil && optionsClone.Mode != config.ModeBundle {
+		optionsClone.Mode = config.ModeBundle
+		skipResolve = true
+	}
+
+	// Special-case pretty-printed paths for data URLs
+	if path.Namespace == "dataurl" {
+		if _, ok := resolver.ParseDataURL(path.Text); ok {
+			prettyPath = path.Text
+			if len(prettyPath) > 65 {
+				prettyPath = prettyPath[:65]
+			}
+			prettyPath = strings.ReplaceAll(prettyPath, "\n", "\\n")
+			if len(prettyPath) > 64 {
+				prettyPath = prettyPath[:64] + "..."
+			}
+			prettyPath = fmt.Sprintf("<%s>", prettyPath)
+		}
+	}
+
+	var sideEffects graph.SideEffects
+	if resolveResult.PrimarySideEffectsData != nil {
+		sideEffects.Kind = graph.NoSideEffects_PackageJSON
+		sideEffects.Data = resolveResult.PrimarySideEffectsData
+	}
+
+	go parseFile(parseArgs{
+		fs:              s.fs,
+		log:             s.log,
+		res:             s.res,
+		caches:          s.caches,
+		keyPath:         path,
+		prettyPath:      prettyPath,
+		sourceIndex:     visited.sourceIndex,
+		importSource:    importSource,
+		sideEffects:     sideEffects,
+		importPathRange: importPathRange,
+		importWith:      importWith,
+		pluginData:      resolveResult.PluginData,
+		options:         optionsClone,
+		results:         s.resultChannel,
+		inject:          inject,
+		skipResolve:     skipResolve,
+		uniqueKeyPrefix: s.uniqueKeyPrefix,
+	})
+
+	return visited.sourceIndex
+}
+
+func (s *scanner) allocateSourceIndex(path logger.Path, kind cache.SourceIndexKind) uint32 {
+	// Allocate a source index using the shared source index cache so that
+	// subsequent builds reuse the same source index and therefore use the
+	// cached parse results for increased speed.
+	sourceIndex := s.caches.SourceIndexCache.Get(path, kind)
+
+	// Grow the results array to fit this source index
+	if newLen := int(sourceIndex) + 1; len(s.results) < newLen {
+		// Reallocate to a bigger array
+		if cap(s.results) < newLen {
+			s.results = append(make([]parseResult, 0, 2*newLen), s.results...)
+		}
+
+		// Grow in place
+		s.results = s.results[:newLen]
+	}
+
+	return sourceIndex
+}
+
+func (s *scanner) allocateGlobSourceIndex(parentSourceIndex uint32, globIndex uint32) uint32 {
+	// Allocate a source index using the shared source index cache so that
+	// subsequent builds reuse the same source index and therefore use the
+	// cached parse results for increased speed.
+	sourceIndex := s.caches.SourceIndexCache.GetGlob(parentSourceIndex, globIndex)
+
+	// Grow the results array to fit this source index
+	if newLen := int(sourceIndex) + 1; len(s.results) < newLen {
+		// Reallocate to a bigger array
+		if cap(s.results) < newLen {
+			s.results = append(make([]parseResult, 0, 2*newLen), s.results...)
+		}
+
+		// Grow in place
+		s.results = s.results[:newLen]
+	}
+
+	return sourceIndex
+}
+
+func (s *scanner) preprocessInjectedFiles() {
+	s.timer.Begin("Preprocess injected files")
+	defer s.timer.End("Preprocess injected files")
+
+	injectedFiles := make([]config.InjectedFile, 0, len(s.options.InjectedDefines)+len(s.options.InjectPaths))
+
+	// These are virtual paths that are generated for compound "--define" values.
+	// They are special-cased and are not available for plugins to intercept.
+	for _, define := range s.options.InjectedDefines {
+		// These should be unique by construction so no need to check for collisions
+		visitedKey := logger.Path{Text: fmt.Sprintf("<define:%s>", define.Name)}
+		sourceIndex := s.allocateSourceIndex(visitedKey, cache.SourceIndexNormal)
+		s.visited[visitedKey] = visitedFile{sourceIndex: sourceIndex}
+		source := logger.Source{
+			Index:          sourceIndex,
+			KeyPath:        visitedKey,
+			PrettyPath:     resolver.PrettyPath(s.fs, visitedKey),
+			IdentifierName: js_ast.EnsureValidIdentifier(visitedKey.Text),
+		}
+
+		// The first "len(InjectedDefine)" injected files intentionally line up
+		// with the injected defines by index. The index will be used to import
+		// references to them in the parser.
+		injectedFiles = append(injectedFiles, config.InjectedFile{
+			Source:     source,
+			DefineName: define.Name,
+		})
+
+		// Generate the file inline here since it has already been parsed
+		expr := js_ast.Expr{Data: define.Data}
+		ast := js_parser.LazyExportAST(s.log, source, js_parser.OptionsFromConfig(&s.options), expr, "")
+		result := parseResult{
+			ok: true,
+			file: scannerFile{
+				inputFile: graph.InputFile{
+					Source: source,
+					Repr:   &graph.JSRepr{AST: ast},
+					Loader: config.LoaderJSON,
+					SideEffects: graph.SideEffects{
+						Kind: graph.NoSideEffects_PureData,
+					},
+				},
+			},
+		}
+
+		// Append to the channel on a goroutine in case it blocks due to capacity
+		s.remaining++
+		go func() { s.resultChannel <- result }()
+	}
+
+	// Add user-specified injected files. Run resolver plugins on these files
+	// so plugins can alter where they resolve to. These are run in parallel in
+	// case any of these plugins block.
+	injectResolveResults := make([]*resolver.ResolveResult, len(s.options.InjectPaths))
+	injectAbsResolveDir := s.fs.Cwd()
+	injectResolveWaitGroup := sync.WaitGroup{}
+	injectResolveWaitGroup.Add(len(s.options.InjectPaths))
+	for i, importPath := range s.options.InjectPaths {
+		go func(i int, importPath string) {
+			var importer logger.Path
+
+			// Add a leading "./" if it's missing, similar to entry points
+			absPath := importPath
+			if !s.fs.IsAbs(absPath) {
+				absPath = s.fs.Join(injectAbsResolveDir, absPath)
+			}
+			dir := s.fs.Dir(absPath)
+			base := s.fs.Base(absPath)
+			if entries, err, originalError := s.fs.ReadDirectory(dir); err == nil {
+				if entry, _ := entries.Get(base); entry != nil && entry.Kind(s.fs) == fs.FileEntry {
+					importer.Namespace = "file"
+					if !s.fs.IsAbs(importPath) && resolver.IsPackagePath(importPath) {
+						importPath = "./" + importPath
+					}
+				}
+			} else if s.log.Level <= logger.LevelDebug && originalError != nil {
+				s.log.AddID(logger.MsgID_None, logger.Debug, nil, logger.Range{}, fmt.Sprintf("Failed to read directory %q: %s", absPath, originalError.Error()))
+			}
+
+			// Run the resolver and log an error if the path couldn't be resolved
+			resolveResult, didLogError, debug := RunOnResolvePlugins(
+				s.options.Plugins,
+				s.res,
+				s.log,
+				s.fs,
+				&s.caches.FSCache,
+				nil,
+				logger.Range{},
+				importer,
+				importPath,
+				logger.ImportAttributes{},
+				ast.ImportEntryPoint,
+				injectAbsResolveDir,
+				nil,
+			)
+			if resolveResult != nil {
+				if resolveResult.PathPair.IsExternal {
+					s.log.AddError(nil, logger.Range{}, fmt.Sprintf("The injected path %q cannot be marked as external", importPath))
+				} else {
+					injectResolveResults[i] = resolveResult
+				}
+			} else if !didLogError {
+				debug.LogErrorMsg(s.log, nil, logger.Range{}, fmt.Sprintf("Could not resolve %q", importPath), "", nil)
+			}
+			injectResolveWaitGroup.Done()
+		}(i, importPath)
+	}
+	injectResolveWaitGroup.Wait()
+
+	if s.options.CancelFlag.DidCancel() {
+		return
+	}
+
+	// Parse all entry points that were resolved successfully
+	results := make([]config.InjectedFile, len(s.options.InjectPaths))
+	j := 0
+	var injectWaitGroup sync.WaitGroup
+	for _, resolveResult := range injectResolveResults {
+		if resolveResult != nil {
+			channel := make(chan config.InjectedFile, 1)
+			s.maybeParseFile(*resolveResult, resolver.PrettyPath(s.fs, resolveResult.PathPair.Primary), nil, logger.Range{}, nil, inputKindNormal, channel)
+			injectWaitGroup.Add(1)
+
+			// Wait for the results in parallel. The results slice is large enough so
+			// it is not reallocated during the computations.
+			go func(i int) {
+				results[i] = <-channel
+				injectWaitGroup.Done()
+			}(j)
+			j++
+		}
+	}
+	injectWaitGroup.Wait()
+	injectedFiles = append(injectedFiles, results[:j]...)
+
+	// It's safe to mutate the options object to add the injected files here
+	// because there aren't any concurrent "parseFile" goroutines at this point.
+	// The only ones that were created by this point are the ones we created
+	// above, and we've already waited for all of them to finish using the
+	// "options" object.
+	s.options.InjectedFiles = injectedFiles
+}
+
+func (s *scanner) addEntryPoints(entryPoints []EntryPoint) []graph.EntryPoint {
+	s.timer.Begin("Add entry points")
+	defer s.timer.End("Add entry points")
+
+	// Reserve a slot for each entry point
+	entryMetas := make([]graph.EntryPoint, 0, len(entryPoints)+1)
+
+	// Treat stdin as an extra entry point
+	if stdin := s.options.Stdin; stdin != nil {
+		stdinPath := logger.Path{Text: "<stdin>"}
+		if stdin.SourceFile != "" {
+			if stdin.AbsResolveDir == "" {
+				stdinPath = logger.Path{Text: stdin.SourceFile}
+			} else if s.fs.IsAbs(stdin.SourceFile) {
+				stdinPath = logger.Path{Text: stdin.SourceFile, Namespace: "file"}
+			} else {
+				stdinPath = logger.Path{Text: s.fs.Join(stdin.AbsResolveDir, stdin.SourceFile), Namespace: "file"}
+			}
+		}
+		resolveResult := resolver.ResolveResult{PathPair: resolver.PathPair{Primary: stdinPath}}
+		sourceIndex := s.maybeParseFile(resolveResult, resolver.PrettyPath(s.fs, stdinPath), nil, logger.Range{}, nil, inputKindStdin, nil)
+		entryMetas = append(entryMetas, graph.EntryPoint{
+			OutputPath:  "stdin",
+			SourceIndex: sourceIndex,
+		})
+	}
+
+	if s.options.CancelFlag.DidCancel() {
+		return nil
+	}
+
+	// Check each entry point ahead of time to see if it's a real file
+	entryPointAbsResolveDir := s.fs.Cwd()
+	for i := range entryPoints {
+		entryPoint := &entryPoints[i]
+		absPath := entryPoint.InputPath
+		if strings.ContainsRune(absPath, '*') {
+			continue // Ignore glob patterns
+		}
+		if !s.fs.IsAbs(absPath) {
+			absPath = s.fs.Join(entryPointAbsResolveDir, absPath)
+		}
+		dir := s.fs.Dir(absPath)
+		base := s.fs.Base(absPath)
+		if entries, err, originalError := s.fs.ReadDirectory(dir); err == nil {
+			if entry, _ := entries.Get(base); entry != nil && entry.Kind(s.fs) == fs.FileEntry {
+				entryPoint.InputPathInFileNamespace = true
+
+				// Entry point paths without a leading "./" are interpreted as package
+				// paths. This happens because they go through general path resolution
+				// like all other import paths so that plugins can run on them. Requiring
+				// a leading "./" for a relative path simplifies writing plugins because
+				// entry points aren't a special case.
+				//
+				// However, requiring a leading "./" also breaks backward compatibility
+				// and makes working with the CLI more difficult. So attempt to insert
+				// "./" automatically when needed. We don't want to unconditionally insert
+				// a leading "./" because the path may not be a file system path. For
+				// example, it may be a URL. So only insert a leading "./" when the path
+				// is an exact match for an existing file.
+				if !s.fs.IsAbs(entryPoint.InputPath) && resolver.IsPackagePath(entryPoint.InputPath) {
+					entryPoint.InputPath = "./" + entryPoint.InputPath
+				}
+			}
+		} else if s.log.Level <= logger.LevelDebug && originalError != nil {
+			s.log.AddID(logger.MsgID_None, logger.Debug, nil, logger.Range{}, fmt.Sprintf("Failed to read directory %q: %s", absPath, originalError.Error()))
+		}
+	}
+
+	if s.options.CancelFlag.DidCancel() {
+		return nil
+	}
+
+	// Add any remaining entry points. Run resolver plugins on these entry points
+	// so plugins can alter where they resolve to. These are run in parallel in
+	// case any of these plugins block.
+	type entryPointInfo struct {
+		results []resolver.ResolveResult
+		isGlob  bool
+	}
+	entryPointInfos := make([]entryPointInfo, len(entryPoints))
+	entryPointWaitGroup := sync.WaitGroup{}
+	entryPointWaitGroup.Add(len(entryPoints))
+	for i, entryPoint := range entryPoints {
+		go func(i int, entryPoint EntryPoint) {
+			var importer logger.Path
+			if entryPoint.InputPathInFileNamespace {
+				importer.Namespace = "file"
+			}
+
+			// Special-case glob patterns here
+			if strings.ContainsRune(entryPoint.InputPath, '*') {
+				if pattern := helpers.ParseGlobPattern(entryPoint.InputPath); len(pattern) > 1 {
+					prettyPattern := fmt.Sprintf("%q", entryPoint.InputPath)
+					if results, msg := s.res.ResolveGlob(entryPointAbsResolveDir, pattern, ast.ImportEntryPoint, prettyPattern); results != nil {
+						keys := make([]string, 0, len(results))
+						for key := range results {
+							keys = append(keys, key)
+						}
+						sort.Strings(keys)
+						info := entryPointInfo{isGlob: true}
+						for _, key := range keys {
+							info.results = append(info.results, results[key])
+						}
+						entryPointInfos[i] = info
+						if msg != nil {
+							s.log.AddID(msg.ID, msg.Kind, nil, logger.Range{}, msg.Data.Text)
+						}
+					} else {
+						s.log.AddError(nil, logger.Range{}, fmt.Sprintf("Could not resolve %q", entryPoint.InputPath))
+					}
+					entryPointWaitGroup.Done()
+					return
+				}
+			}
+
+			// Run the resolver and log an error if the path couldn't be resolved
+			resolveResult, didLogError, debug := RunOnResolvePlugins(
+				s.options.Plugins,
+				s.res,
+				s.log,
+				s.fs,
+				&s.caches.FSCache,
+				nil,
+				logger.Range{},
+				importer,
+				entryPoint.InputPath,
+				logger.ImportAttributes{},
+				ast.ImportEntryPoint,
+				entryPointAbsResolveDir,
+				nil,
+			)
+			if resolveResult != nil {
+				if resolveResult.PathPair.IsExternal {
+					s.log.AddError(nil, logger.Range{}, fmt.Sprintf("The entry point %q cannot be marked as external", entryPoint.InputPath))
+				} else {
+					entryPointInfos[i] = entryPointInfo{results: []resolver.ResolveResult{*resolveResult}}
+				}
+			} else if !didLogError {
+				var notes []logger.MsgData
+				if !s.fs.IsAbs(entryPoint.InputPath) {
+					if query, _ := s.res.ProbeResolvePackageAsRelative(entryPointAbsResolveDir, entryPoint.InputPath, ast.ImportEntryPoint); query != nil {
+						notes = append(notes, logger.MsgData{
+							Text: fmt.Sprintf("Use the relative path %q to reference the file %q. "+
+								"Without the leading \"./\", the path %q is being interpreted as a package path instead.",
+								"./"+entryPoint.InputPath, resolver.PrettyPath(s.fs, query.PathPair.Primary), entryPoint.InputPath),
+						})
+					}
+				}
+				debug.LogErrorMsg(s.log, nil, logger.Range{}, fmt.Sprintf("Could not resolve %q", entryPoint.InputPath), "", notes)
+			}
+			entryPointWaitGroup.Done()
+		}(i, entryPoint)
+	}
+	entryPointWaitGroup.Wait()
+
+	if s.options.CancelFlag.DidCancel() {
+		return nil
+	}
+
+	// Parse all entry points that were resolved successfully
+	for i, info := range entryPointInfos {
+		if info.results == nil {
+			continue
+		}
+
+		for _, resolveResult := range info.results {
+			prettyPath := resolver.PrettyPath(s.fs, resolveResult.PathPair.Primary)
+			sourceIndex := s.maybeParseFile(resolveResult, prettyPath, nil, logger.Range{}, nil, inputKindEntryPoint, nil)
+			outputPath := entryPoints[i].OutputPath
+			outputPathWasAutoGenerated := false
+
+			// If the output path is missing, automatically generate one from the input path
+			if outputPath == "" {
+				if info.isGlob {
+					outputPath = prettyPath
+				} else {
+					outputPath = entryPoints[i].InputPath
+				}
+				windowsVolumeLabel := ""
+
+				// The ":" character is invalid in file paths on Windows except when
+				// it's used as a volume separator. Special-case that here so volume
+				// labels don't break on Windows.
+				if s.fs.IsAbs(outputPath) && len(outputPath) >= 3 && outputPath[1] == ':' {
+					if c := outputPath[0]; (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') {
+						if c := outputPath[2]; c == '/' || c == '\\' {
+							windowsVolumeLabel = outputPath[:3]
+							outputPath = outputPath[3:]
+						}
+					}
+				}
+
+				// For cross-platform robustness, do not allow characters in the output
+				// path that are invalid on Windows. This is especially relevant when
+				// the input path is something other than a file path, such as a URL.
+				outputPath = sanitizeFilePathForVirtualModulePath(outputPath)
+				if windowsVolumeLabel != "" {
+					outputPath = windowsVolumeLabel + outputPath
+				}
+				outputPathWasAutoGenerated = true
+			}
+
+			entryMetas = append(entryMetas, graph.EntryPoint{
+				OutputPath:                 outputPath,
+				SourceIndex:                sourceIndex,
+				OutputPathWasAutoGenerated: outputPathWasAutoGenerated,
+			})
+		}
+	}
+
+	// Turn all automatically-generated output paths into absolute paths
+	for i := range entryMetas {
+		entryPoint := &entryMetas[i]
+		if entryPoint.OutputPathWasAutoGenerated && !s.fs.IsAbs(entryPoint.OutputPath) {
+			entryPoint.OutputPath = s.fs.Join(entryPointAbsResolveDir, entryPoint.OutputPath)
+		}
+	}
+
+	// Automatically compute "outbase" if it wasn't provided
+	if s.options.AbsOutputBase == "" {
+		s.options.AbsOutputBase = lowestCommonAncestorDirectory(s.fs, entryMetas)
+		if s.options.AbsOutputBase == "" {
+			s.options.AbsOutputBase = entryPointAbsResolveDir
+		}
+	}
+
+	// Turn all output paths back into relative paths, but this time relative to
+	// the "outbase" value we computed above
+	for i := range entryMetas {
+		entryPoint := &entryMetas[i]
+		if s.fs.IsAbs(entryPoint.OutputPath) {
+			if !entryPoint.OutputPathWasAutoGenerated {
+				// If an explicit absolute output path was specified, use the path
+				// relative to the "outdir" directory
+				if relPath, ok := s.fs.Rel(s.options.AbsOutputDir, entryPoint.OutputPath); ok {
+					entryPoint.OutputPath = relPath
+				}
+			} else {
+				// Otherwise if the absolute output path was derived from the input
+				// path, use the path relative to the "outbase" directory
+				if relPath, ok := s.fs.Rel(s.options.AbsOutputBase, entryPoint.OutputPath); ok {
+					entryPoint.OutputPath = relPath
+				}
+
+				// Strip the file extension from the output path if there is one so the
+				// "out extension" setting is used instead
+				if last := strings.LastIndexAny(entryPoint.OutputPath, "/.\\"); last != -1 && entryPoint.OutputPath[last] == '.' {
+					entryPoint.OutputPath = entryPoint.OutputPath[:last]
+				}
+			}
+		}
+	}
+
+	return entryMetas
+}
+
+func lowestCommonAncestorDirectory(fs fs.FS, entryPoints []graph.EntryPoint) string {
+	// Ignore any explicitly-specified output paths
+	absPaths := make([]string, 0, len(entryPoints))
+	for _, entryPoint := range entryPoints {
+		if entryPoint.OutputPathWasAutoGenerated {
+			absPaths = append(absPaths, entryPoint.OutputPath)
+		}
+	}
+
+	if len(absPaths) == 0 {
+		return ""
+	}
+
+	lowestAbsDir := fs.Dir(absPaths[0])
+
+	for _, absPath := range absPaths[1:] {
+		absDir := fs.Dir(absPath)
+		lastSlash := 0
+		a := 0
+		b := 0
+
+		for {
+			runeA, widthA := utf8.DecodeRuneInString(absDir[a:])
+			runeB, widthB := utf8.DecodeRuneInString(lowestAbsDir[b:])
+			boundaryA := widthA == 0 || runeA == '/' || runeA == '\\'
+			boundaryB := widthB == 0 || runeB == '/' || runeB == '\\'
+
+			if boundaryA && boundaryB {
+				if widthA == 0 || widthB == 0 {
+					// Truncate to the smaller path if one path is a prefix of the other
+					lowestAbsDir = absDir[:a]
+					break
+				} else {
+					// Track the longest common directory so far
+					lastSlash = a
+				}
+			} else if boundaryA != boundaryB || unicode.ToLower(runeA) != unicode.ToLower(runeB) {
+				// If we're at the top-level directory, then keep the slash
+				if lastSlash < len(absDir) && !strings.ContainsAny(absDir[:lastSlash], "\\/") {
+					lastSlash++
+				}
+
+				// If both paths are different at this point, stop and set the lowest so
+				// far to the common parent directory. Compare using a case-insensitive
+				// comparison to handle paths on Windows.
+				lowestAbsDir = absDir[:lastSlash]
+				break
+			}
+
+			a += widthA
+			b += widthB
+		}
+	}
+
+	return lowestAbsDir
+}
+
+func (s *scanner) scanAllDependencies() {
+	s.timer.Begin("Scan all dependencies")
+	defer s.timer.End("Scan all dependencies")
+
+	// Continue scanning until all dependencies have been discovered
+	for s.remaining > 0 {
+		if s.options.CancelFlag.DidCancel() {
+			return
+		}
+
+		result := <-s.resultChannel
+		s.remaining--
+		if !result.ok {
+			continue
+		}
+
+		// Don't try to resolve paths if we're not bundling
+		if recordsPtr := result.file.inputFile.Repr.ImportRecords(); s.options.Mode == config.ModeBundle && recordsPtr != nil {
+			records := *recordsPtr
+			for importRecordIndex := range records {
+				record := &records[importRecordIndex]
+
+				// This is used for error messages
+				var with *ast.ImportAssertOrWith
+				if record.AssertOrWith != nil && record.AssertOrWith.Keyword == ast.WithKeyword {
+					with = record.AssertOrWith
+				}
+
+				// Skip this import record if the previous resolver call failed
+				resolveResult := result.resolveResults[importRecordIndex]
+				if resolveResult == nil {
+					if globResults := result.globResolveResults[uint32(importRecordIndex)]; globResults.resolveResults != nil {
+						sourceIndex := s.allocateGlobSourceIndex(result.file.inputFile.Source.Index, uint32(importRecordIndex))
+						record.SourceIndex = ast.MakeIndex32(sourceIndex)
+						s.results[sourceIndex] = s.generateResultForGlobResolve(sourceIndex, globResults.absPath,
+							&result.file.inputFile.Source, record.Range, with, record.GlobPattern.Kind, globResults, record.AssertOrWith)
+					}
+					continue
+				}
+
+				path := resolveResult.PathPair.Primary
+				if !resolveResult.PathPair.IsExternal {
+					// Handle a path within the bundle
+					sourceIndex := s.maybeParseFile(*resolveResult, resolver.PrettyPath(s.fs, path),
+						&result.file.inputFile.Source, record.Range, with, inputKindNormal, nil)
+					record.SourceIndex = ast.MakeIndex32(sourceIndex)
+				} else {
+					// Allow this import statement to be removed if something marked it as "sideEffects: false"
+					if resolveResult.PrimarySideEffectsData != nil {
+						record.Flags |= ast.IsExternalWithoutSideEffects
+					}
+
+					// If the path to the external module is relative to the source
+					// file, rewrite the path to be relative to the working directory
+					if path.Namespace == "file" {
+						if relPath, ok := s.fs.Rel(s.options.AbsOutputDir, path.Text); ok {
+							// Prevent issues with path separators being different on Windows
+							relPath = strings.ReplaceAll(relPath, "\\", "/")
+							if resolver.IsPackagePath(relPath) {
+								relPath = "./" + relPath
+							}
+							record.Path.Text = relPath
+						} else {
+							record.Path = path
+						}
+					} else {
+						record.Path = path
+					}
+				}
+			}
+		}
+
+		s.results[result.file.inputFile.Source.Index] = result
+	}
+}
+
+func (s *scanner) generateResultForGlobResolve(
+	sourceIndex uint32,
+	fakeSourcePath string,
+	importSource *logger.Source,
+	importRange logger.Range,
+	importWith *ast.ImportAssertOrWith,
+	kind ast.ImportKind,
+	result globResolveResult,
+	assertions *ast.ImportAssertOrWith,
+) parseResult {
+	keys := make([]string, 0, len(result.resolveResults))
+	for key := range result.resolveResults {
+		keys = append(keys, key)
+	}
+	sort.Strings(keys)
+
+	object := js_ast.EObject{Properties: make([]js_ast.Property, 0, len(result.resolveResults))}
+	importRecords := make([]ast.ImportRecord, 0, len(result.resolveResults))
+	resolveResults := make([]*resolver.ResolveResult, 0, len(result.resolveResults))
+
+	for _, key := range keys {
+		resolveResult := result.resolveResults[key]
+		var value js_ast.Expr
+
+		importRecordIndex := uint32(len(importRecords))
+		var sourceIndex ast.Index32
+
+		if !resolveResult.PathPair.IsExternal {
+			sourceIndex = ast.MakeIndex32(s.maybeParseFile(
+				resolveResult,
+				resolver.PrettyPath(s.fs, resolveResult.PathPair.Primary),
+				importSource,
+				importRange,
+				importWith,
+				inputKindNormal,
+				nil,
+			))
+		}
+
+		path := resolveResult.PathPair.Primary
+
+		// If the path to the external module is relative to the source
+		// file, rewrite the path to be relative to the working directory
+		if path.Namespace == "file" {
+			if relPath, ok := s.fs.Rel(s.options.AbsOutputDir, path.Text); ok {
+				// Prevent issues with path separators being different on Windows
+				relPath = strings.ReplaceAll(relPath, "\\", "/")
+				if resolver.IsPackagePath(relPath) {
+					relPath = "./" + relPath
+				}
+				path.Text = relPath
+			}
+		}
+
+		resolveResults = append(resolveResults, &resolveResult)
+		importRecords = append(importRecords, ast.ImportRecord{
+			Path:         path,
+			SourceIndex:  sourceIndex,
+			AssertOrWith: assertions,
+			Kind:         kind,
+		})
+
+		switch kind {
+		case ast.ImportDynamic:
+			value.Data = &js_ast.EImportString{ImportRecordIndex: importRecordIndex}
+		case ast.ImportRequire:
+			value.Data = &js_ast.ERequireString{ImportRecordIndex: importRecordIndex}
+		default:
+			panic("Internal error")
+		}
+
+		object.Properties = append(object.Properties, js_ast.Property{
+			Key: js_ast.Expr{Data: &js_ast.EString{Value: helpers.StringToUTF16(key)}},
+			ValueOrNil: js_ast.Expr{Data: &js_ast.EArrow{
+				Body:       js_ast.FnBody{Block: js_ast.SBlock{Stmts: []js_ast.Stmt{{Data: &js_ast.SReturn{ValueOrNil: value}}}}},
+				PreferExpr: true,
+			}},
+		})
+	}
+
+	source := logger.Source{
+		KeyPath:    logger.Path{Text: fakeSourcePath, Namespace: "file"},
+		PrettyPath: result.prettyPath,
+		Index:      sourceIndex,
+	}
+	ast := js_parser.GlobResolveAST(s.log, source, importRecords, &object, result.exportAlias)
+
+	// Fill out "nil" for any additional imports (i.e. from the runtime)
+	for len(resolveResults) < len(ast.ImportRecords) {
+		resolveResults = append(resolveResults, nil)
+	}
+
+	return parseResult{
+		resolveResults: resolveResults,
+		file: scannerFile{
+			inputFile: graph.InputFile{
+				Source: source,
+				Repr: &graph.JSRepr{
+					AST: ast,
+				},
+				OmitFromSourceMapsAndMetafile: true,
+			},
+		},
+		ok: true,
+	}
+}
+
+func (s *scanner) processScannedFiles(entryPointMeta []graph.EntryPoint) []scannerFile {
+	s.timer.Begin("Process scanned files")
+	defer s.timer.End("Process scanned files")
+
+	// Build a set of entry point source indices for quick lookup
+	entryPointSourceIndexToMetaIndex := make(map[uint32]uint32, len(entryPointMeta))
+	for i, meta := range entryPointMeta {
+		entryPointSourceIndexToMetaIndex[meta.SourceIndex] = uint32(i)
+	}
+
+	// Check for pretty-printed path collisions
+	importAttributeNameCollisions := make(map[string][]uint32)
+	for sourceIndex := range s.results {
+		if result := &s.results[sourceIndex]; result.ok {
+			prettyPath := result.file.inputFile.Source.PrettyPath
+			importAttributeNameCollisions[prettyPath] = append(importAttributeNameCollisions[prettyPath], uint32(sourceIndex))
+		}
+	}
+
+	// Import attributes can result in the same file being imported multiple
+	// times in different ways. If that happens, append the import attributes
+	// to the pretty-printed file names to disambiguate them. This renaming
+	// must happen before we construct the metafile JSON chunks below.
+	for _, sourceIndices := range importAttributeNameCollisions {
+		if len(sourceIndices) == 1 {
+			continue
+		}
+
+		for _, sourceIndex := range sourceIndices {
+			source := &s.results[sourceIndex].file.inputFile.Source
+			attrs := source.KeyPath.ImportAttributes.DecodeIntoArray()
+			if len(attrs) == 0 {
+				continue
+			}
+
+			var sb strings.Builder
+			sb.WriteString(" with {")
+			for i, attr := range attrs {
+				if i > 0 {
+					sb.WriteByte(',')
+				}
+				sb.WriteByte(' ')
+				if js_ast.IsIdentifier(attr.Key) {
+					sb.WriteString(attr.Key)
+				} else {
+					sb.Write(helpers.QuoteSingle(attr.Key, false))
+				}
+				sb.WriteString(": ")
+				sb.Write(helpers.QuoteSingle(attr.Value, false))
+			}
+			sb.WriteString(" }")
+			source.PrettyPath += sb.String()
+		}
+	}
+
+	// Now that all files have been scanned, process the final file import records
+	for sourceIndex, result := range s.results {
+		if !result.ok {
+			continue
+		}
+
+		sb := strings.Builder{}
+		isFirstImport := true
+
+		// Begin the metadata chunk
+		if s.options.NeedsMetafile {
+			sb.Write(helpers.QuoteForJSON(result.file.inputFile.Source.PrettyPath, s.options.ASCIIOnly))
+			sb.WriteString(fmt.Sprintf(": {\n      \"bytes\": %d,\n      \"imports\": [", len(result.file.inputFile.Source.Contents)))
+		}
+
+		// Don't try to resolve paths if we're not bundling
+		if recordsPtr := result.file.inputFile.Repr.ImportRecords(); s.options.Mode == config.ModeBundle && recordsPtr != nil {
+			records := *recordsPtr
+			tracker := logger.MakeLineColumnTracker(&result.file.inputFile.Source)
+
+			for importRecordIndex := range records {
+				record := &records[importRecordIndex]
+
+				// Save the import attributes to the metafile
+				var metafileWith string
+				if s.options.NeedsMetafile {
+					if with := record.AssertOrWith; with != nil && with.Keyword == ast.WithKeyword && len(with.Entries) > 0 {
+						data := strings.Builder{}
+						data.WriteString(",\n          \"with\": {")
+						for i, entry := range with.Entries {
+							if i > 0 {
+								data.WriteByte(',')
+							}
+							data.WriteString("\n            ")
+							data.Write(helpers.QuoteForJSON(helpers.UTF16ToString(entry.Key), s.options.ASCIIOnly))
+							data.WriteString(": ")
+							data.Write(helpers.QuoteForJSON(helpers.UTF16ToString(entry.Value), s.options.ASCIIOnly))
+						}
+						data.WriteString("\n          }")
+						metafileWith = data.String()
+					}
+				}
+
+				// Skip this import record if the previous resolver call failed
+				resolveResult := result.resolveResults[importRecordIndex]
+				if resolveResult == nil || !record.SourceIndex.IsValid() {
+					if s.options.NeedsMetafile {
+						if isFirstImport {
+							isFirstImport = false
+							sb.WriteString("\n        ")
+						} else {
+							sb.WriteString(",\n        ")
+						}
+						sb.WriteString(fmt.Sprintf("{\n          \"path\": %s,\n          \"kind\": %s,\n          \"external\": true%s\n        }",
+							helpers.QuoteForJSON(record.Path.Text, s.options.ASCIIOnly),
+							helpers.QuoteForJSON(record.Kind.StringForMetafile(), s.options.ASCIIOnly),
+							metafileWith))
+					}
+					continue
+				}
+
+				// Now that all files have been scanned, look for packages that are imported
+				// both with "import" and "require". Rewrite any imports that reference the
+				// "module" package.json field to the "main" package.json field instead.
+				//
+				// This attempts to automatically avoid the "dual package hazard" where a
+				// package has both a CommonJS module version and an ECMAScript module
+				// version and exports a non-object in CommonJS (often a function). If we
+				// pick the "module" field and the package is imported with "require" then
+				// code expecting a function will crash.
+				if resolveResult.PathPair.HasSecondary() {
+					secondaryKey := resolveResult.PathPair.Secondary
+					if secondaryKey.Namespace == "file" {
+						secondaryKey.Text = canonicalFileSystemPathForWindows(secondaryKey.Text)
+					}
+					if secondaryVisited, ok := s.visited[secondaryKey]; ok {
+						record.SourceIndex = ast.MakeIndex32(secondaryVisited.sourceIndex)
+					}
+				}
+
+				// Generate metadata about each import
+				otherResult := &s.results[record.SourceIndex.GetIndex()]
+				otherFile := &otherResult.file
+				if s.options.NeedsMetafile {
+					if isFirstImport {
+						isFirstImport = false
+						sb.WriteString("\n        ")
+					} else {
+						sb.WriteString(",\n        ")
+					}
+					sb.WriteString(fmt.Sprintf("{\n          \"path\": %s,\n          \"kind\": %s,\n          \"original\": %s%s\n        }",
+						helpers.QuoteForJSON(otherFile.inputFile.Source.PrettyPath, s.options.ASCIIOnly),
+						helpers.QuoteForJSON(record.Kind.StringForMetafile(), s.options.ASCIIOnly),
+						helpers.QuoteForJSON(record.Path.Text, s.options.ASCIIOnly),
+						metafileWith))
+				}
+
+				// Validate that imports with "assert { type: 'json' }" were imported
+				// with the JSON loader. This is done to match the behavior of these
+				// import assertions in a real JavaScript runtime. In addition, we also
+				// allow the copy loader since this is sort of like marking the path
+				// as external (the import assertions are kept and the real JavaScript
+				// runtime evaluates them, not us).
+				if record.Flags.Has(ast.AssertTypeJSON) && otherResult.ok && otherFile.inputFile.Loader != config.LoaderJSON && otherFile.inputFile.Loader != config.LoaderCopy {
+					s.log.AddErrorWithNotes(&tracker, record.Range,
+						fmt.Sprintf("The file %q was loaded with the %q loader", otherFile.inputFile.Source.PrettyPath, config.LoaderToString[otherFile.inputFile.Loader]),
+						[]logger.MsgData{
+							tracker.MsgData(js_lexer.RangeOfImportAssertOrWith(result.file.inputFile.Source,
+								*ast.FindAssertOrWithEntry(record.AssertOrWith.Entries, "type"), js_lexer.KeyAndValueRange),
+								"This import assertion requires the loader to be \"json\" instead:"),
+							{Text: "You need to either reconfigure esbuild to ensure that the loader for this file is \"json\" or you need to remove this import assertion."}})
+				}
+
+				switch record.Kind {
+				case ast.ImportComposesFrom:
+					// Using a JavaScript file with CSS "composes" is not allowed
+					if _, ok := otherFile.inputFile.Repr.(*graph.JSRepr); ok && otherFile.inputFile.Loader != config.LoaderEmpty {
+						s.log.AddErrorWithNotes(&tracker, record.Range,
+							fmt.Sprintf("Cannot use \"composes\" with %q", otherFile.inputFile.Source.PrettyPath),
+							[]logger.MsgData{{Text: fmt.Sprintf(
+								"You can only use \"composes\" with CSS files and %q is not a CSS file (it was loaded with the %q loader).",
+								otherFile.inputFile.Source.PrettyPath, config.LoaderToString[otherFile.inputFile.Loader])}})
+					}
+
+				case ast.ImportAt:
+					// Using a JavaScript file with CSS "@import" is not allowed
+					if _, ok := otherFile.inputFile.Repr.(*graph.JSRepr); ok && otherFile.inputFile.Loader != config.LoaderEmpty {
+						s.log.AddErrorWithNotes(&tracker, record.Range,
+							fmt.Sprintf("Cannot import %q into a CSS file", otherFile.inputFile.Source.PrettyPath),
+							[]logger.MsgData{{Text: fmt.Sprintf(
+								"An \"@import\" rule can only be used to import another CSS file and %q is not a CSS file (it was loaded with the %q loader).",
+								otherFile.inputFile.Source.PrettyPath, config.LoaderToString[otherFile.inputFile.Loader])}})
+					}
+
+				case ast.ImportURL:
+					// Using a JavaScript or CSS file with CSS "url()" is not allowed
+					switch otherRepr := otherFile.inputFile.Repr.(type) {
+					case *graph.CSSRepr:
+						s.log.AddErrorWithNotes(&tracker, record.Range,
+							fmt.Sprintf("Cannot use %q as a URL", otherFile.inputFile.Source.PrettyPath),
+							[]logger.MsgData{{Text: fmt.Sprintf(
+								"You can't use a \"url()\" token to reference a CSS file, and %q is a CSS file (it was loaded with the %q loader).",
+								otherFile.inputFile.Source.PrettyPath, config.LoaderToString[otherFile.inputFile.Loader])}})
+
+					case *graph.JSRepr:
+						if otherRepr.AST.URLForCSS == "" && otherFile.inputFile.Loader != config.LoaderEmpty {
+							s.log.AddErrorWithNotes(&tracker, record.Range,
+								fmt.Sprintf("Cannot use %q as a URL", otherFile.inputFile.Source.PrettyPath),
+								[]logger.MsgData{{Text: fmt.Sprintf(
+									"You can't use a \"url()\" token to reference the file %q because it was loaded with the %q loader, which doesn't provide a URL to embed in the resulting CSS.",
+									otherFile.inputFile.Source.PrettyPath, config.LoaderToString[otherFile.inputFile.Loader])}})
+						}
+					}
+				}
+
+				// If the imported file uses the "copy" loader, then move it from
+				// "SourceIndex" to "CopySourceIndex" so we don't end up bundling it.
+				if _, ok := otherFile.inputFile.Repr.(*graph.CopyRepr); ok {
+					record.CopySourceIndex = record.SourceIndex
+					record.SourceIndex = ast.Index32{}
+					continue
+				}
+
+				// If an import from a JavaScript file targets a CSS file, generate a
+				// JavaScript stub to ensure that JavaScript files only ever import
+				// other JavaScript files.
+				if _, ok := result.file.inputFile.Repr.(*graph.JSRepr); ok {
+					if css, ok := otherFile.inputFile.Repr.(*graph.CSSRepr); ok {
+						if s.options.WriteToStdout {
+							s.log.AddError(&tracker, record.Range,
+								fmt.Sprintf("Cannot import %q into a JavaScript file without an output path configured", otherFile.inputFile.Source.PrettyPath))
+						} else if !css.JSSourceIndex.IsValid() {
+							stubKey := otherFile.inputFile.Source.KeyPath
+							if stubKey.Namespace == "file" {
+								stubKey.Text = canonicalFileSystemPathForWindows(stubKey.Text)
+							}
+							sourceIndex := s.allocateSourceIndex(stubKey, cache.SourceIndexJSStubForCSS)
+							source := otherFile.inputFile.Source
+							source.Index = sourceIndex
+							s.results[sourceIndex] = parseResult{
+								file: scannerFile{
+									inputFile: graph.InputFile{
+										Source: source,
+										Loader: otherFile.inputFile.Loader,
+										Repr: &graph.JSRepr{
+											// Note: The actual export object will be filled in by the linker
+											AST: js_parser.LazyExportAST(s.log, source,
+												js_parser.OptionsFromConfig(&s.options), js_ast.Expr{Data: js_ast.ENullShared}, ""),
+											CSSSourceIndex: ast.MakeIndex32(record.SourceIndex.GetIndex()),
+										},
+									},
+								},
+								ok: true,
+							}
+							css.JSSourceIndex = ast.MakeIndex32(sourceIndex)
+						}
+						record.SourceIndex = css.JSSourceIndex
+						if !css.JSSourceIndex.IsValid() {
+							continue
+						}
+					}
+				}
+
+				// Warn about this import if it's a bare import statement without any
+				// imported names (i.e. a side-effect-only import) and the module has
+				// been marked as having no side effects.
+				//
+				// Except don't do this if this file is inside "node_modules" since
+				// it's a bug in the package and the user won't be able to do anything
+				// about it. Note that this can result in esbuild silently generating
+				// broken code. If this actually happens for people, it's probably worth
+				// re-enabling the warning about code inside "node_modules".
+				if record.Flags.Has(ast.WasOriginallyBareImport) && !s.options.IgnoreDCEAnnotations &&
+					!helpers.IsInsideNodeModules(result.file.inputFile.Source.KeyPath.Text) {
+					if otherModule := &s.results[record.SourceIndex.GetIndex()].file.inputFile; otherModule.SideEffects.Kind != graph.HasSideEffects &&
+						// Do not warn if this is from a plugin, since removing the import
+						// would cause the plugin to not run, and running a plugin is a side
+						// effect.
+						otherModule.SideEffects.Kind != graph.NoSideEffects_PureData_FromPlugin &&
+
+						// Do not warn if this has no side effects because the parsed AST
+						// is empty. This is the case for ".d.ts" files, for example.
+						otherModule.SideEffects.Kind != graph.NoSideEffects_EmptyAST {
+
+						var notes []logger.MsgData
+						var by string
+						if data := otherModule.SideEffects.Data; data != nil {
+							if data.PluginName != "" {
+								by = fmt.Sprintf(" by plugin %q", data.PluginName)
+							} else {
+								var text string
+								if data.IsSideEffectsArrayInJSON {
+									text = "It was excluded from the \"sideEffects\" array in the enclosing \"package.json\" file:"
+								} else {
+									text = "\"sideEffects\" is false in the enclosing \"package.json\" file:"
+								}
+								tracker := logger.MakeLineColumnTracker(data.Source)
+								notes = append(notes, tracker.MsgData(data.Range, text))
+							}
+						}
+						s.log.AddIDWithNotes(logger.MsgID_Bundler_IgnoredBareImport, logger.Warning, &tracker, record.Range,
+							fmt.Sprintf("Ignoring this import because %q was marked as having no side effects%s",
+								otherModule.Source.PrettyPath, by), notes)
+					}
+				}
+			}
+		}
+
+		// End the metadata chunk
+		if s.options.NeedsMetafile {
+			if !isFirstImport {
+				sb.WriteString("\n      ")
+			}
+			if repr, ok := result.file.inputFile.Repr.(*graph.JSRepr); ok &&
+				(repr.AST.ExportsKind == js_ast.ExportsCommonJS || repr.AST.ExportsKind == js_ast.ExportsESM) {
+				format := "cjs"
+				if repr.AST.ExportsKind == js_ast.ExportsESM {
+					format = "esm"
+				}
+				sb.WriteString(fmt.Sprintf("],\n      \"format\": %q", format))
+			} else {
+				sb.WriteString("]")
+			}
+			if attrs := result.file.inputFile.Source.KeyPath.ImportAttributes.DecodeIntoArray(); len(attrs) > 0 {
+				sb.WriteString(",\n      \"with\": {")
+				for i, attr := range attrs {
+					if i > 0 {
+						sb.WriteByte(',')
+					}
+					sb.WriteString(fmt.Sprintf("\n        %s: %s",
+						helpers.QuoteForJSON(attr.Key, s.options.ASCIIOnly),
+						helpers.QuoteForJSON(attr.Value, s.options.ASCIIOnly),
+					))
+				}
+				sb.WriteString("\n      }")
+			}
+			sb.WriteString("\n    }")
+		}
+
+		result.file.jsonMetadataChunk = sb.String()
+
+		// If this file is from the "file" or "copy" loaders, generate an additional file
+		if result.file.inputFile.UniqueKeyForAdditionalFile != "" {
+			bytes := []byte(result.file.inputFile.Source.Contents)
+			template := s.options.AssetPathTemplate
+
+			// Use the entry path template instead of the asset path template if this
+			// file is an entry point and uses the "copy" loader. With the "file" loader
+			// the JS stub is the entry point, but with the "copy" loader the file is
+			// the entry point itself.
+			customFilePath := ""
+			useOutputFile := false
+			if result.file.inputFile.Loader == config.LoaderCopy {
+				if metaIndex, ok := entryPointSourceIndexToMetaIndex[uint32(sourceIndex)]; ok {
+					template = s.options.EntryPathTemplate
+					customFilePath = entryPointMeta[metaIndex].OutputPath
+					useOutputFile = s.options.AbsOutputFile != ""
+				}
+			}
+
+			// Add a hash to the file name to prevent multiple files with the same name
+			// but different contents from colliding
+			var hash string
+			if config.HasPlaceholder(template, config.HashPlaceholder) {
+				h := xxhash.New()
+				h.Write(bytes)
+				hash = HashForFileName(h.Sum(nil))
+			}
+
+			// This should use similar logic to how the linker computes output paths
+			var dir, base, ext string
+			if useOutputFile {
+				// If the output path was configured explicitly, use it verbatim
+				dir = "/"
+				base = s.fs.Base(s.options.AbsOutputFile)
+				ext = s.fs.Ext(base)
+				base = base[:len(base)-len(ext)]
+			} else {
+				// Otherwise, derive the output path from the input path
+				// Generate the input for the template
+				_, _, originalExt := logger.PlatformIndependentPathDirBaseExt(result.file.inputFile.Source.KeyPath.Text)
+				dir, base = PathRelativeToOutbase(
+					&result.file.inputFile,
+					&s.options,
+					s.fs,
+					/* avoidIndex */ false,
+					customFilePath,
+				)
+				ext = originalExt
+			}
+
+			// Apply the path template
+			templateExt := strings.TrimPrefix(ext, ".")
+			relPath := config.TemplateToString(config.SubstituteTemplate(template, config.PathPlaceholders{
+				Dir:  &dir,
+				Name: &base,
+				Hash: &hash,
+				Ext:  &templateExt,
+			})) + ext
+
+			// Optionally add metadata about the file
+			var jsonMetadataChunk string
+			if s.options.NeedsMetafile {
+				inputs := fmt.Sprintf("{\n        %s: {\n          \"bytesInOutput\": %d\n        }\n      }",
+					helpers.QuoteForJSON(result.file.inputFile.Source.PrettyPath, s.options.ASCIIOnly),
+					len(bytes),
+				)
+				jsonMetadataChunk = fmt.Sprintf(
+					"{\n      \"imports\": [],\n      \"exports\": [],\n      \"inputs\": %s,\n      \"bytes\": %d\n    }",
+					inputs,
+					len(bytes),
+				)
+			}
+
+			// Generate the additional file to copy into the output directory
+			result.file.inputFile.AdditionalFiles = []graph.OutputFile{{
+				AbsPath:           s.fs.Join(s.options.AbsOutputDir, relPath),
+				Contents:          bytes,
+				JSONMetadataChunk: jsonMetadataChunk,
+			}}
+		}
+
+		s.results[sourceIndex] = result
+	}
+
+	// The linker operates on an array of files, so construct that now. This
+	// can't be constructed earlier because we generate new parse results for
+	// JavaScript stub files for CSS imports above.
+	files := make([]scannerFile, len(s.results))
+	for sourceIndex := range s.results {
+		if result := &s.results[sourceIndex]; result.ok {
+			s.validateTLA(uint32(sourceIndex))
+			files[sourceIndex] = result.file
+		}
+	}
+
+	return files
+}
+
+func (s *scanner) validateTLA(sourceIndex uint32) tlaCheck {
+	result := &s.results[sourceIndex]
+
+	if result.ok && result.tlaCheck.depth == 0 {
+		if repr, ok := result.file.inputFile.Repr.(*graph.JSRepr); ok {
+			result.tlaCheck.depth = 1
+			if repr.AST.LiveTopLevelAwaitKeyword.Len > 0 {
+				result.tlaCheck.parent = ast.MakeIndex32(sourceIndex)
+			}
+
+			for importRecordIndex, record := range repr.AST.ImportRecords {
+				if record.SourceIndex.IsValid() && (record.Kind == ast.ImportRequire || record.Kind == ast.ImportStmt) {
+					parent := s.validateTLA(record.SourceIndex.GetIndex())
+					if !parent.parent.IsValid() {
+						continue
+					}
+
+					// Follow any import chains
+					if record.Kind == ast.ImportStmt && (!result.tlaCheck.parent.IsValid() || parent.depth < result.tlaCheck.depth) {
+						result.tlaCheck.depth = parent.depth + 1
+						result.tlaCheck.parent = record.SourceIndex
+						result.tlaCheck.importRecordIndex = uint32(importRecordIndex)
+						continue
+					}
+
+					// Require of a top-level await chain is forbidden
+					if record.Kind == ast.ImportRequire {
+						var notes []logger.MsgData
+						var tlaPrettyPath string
+						otherSourceIndex := record.SourceIndex.GetIndex()
+
+						// Build up a chain of relevant notes for all of the imports
+						for {
+							parentResult := &s.results[otherSourceIndex]
+							parentRepr := parentResult.file.inputFile.Repr.(*graph.JSRepr)
+
+							if parentRepr.AST.LiveTopLevelAwaitKeyword.Len > 0 {
+								tlaPrettyPath = parentResult.file.inputFile.Source.PrettyPath
+								tracker := logger.MakeLineColumnTracker(&parentResult.file.inputFile.Source)
+								notes = append(notes, tracker.MsgData(parentRepr.AST.LiveTopLevelAwaitKeyword,
+									fmt.Sprintf("The top-level await in %q is here:", tlaPrettyPath)))
+								break
+							}
+
+							if !parentResult.tlaCheck.parent.IsValid() {
+								notes = append(notes, logger.MsgData{Text: "unexpected invalid index"})
+								break
+							}
+
+							otherSourceIndex = parentResult.tlaCheck.parent.GetIndex()
+
+							tracker := logger.MakeLineColumnTracker(&parentResult.file.inputFile.Source)
+							notes = append(notes, tracker.MsgData(
+								parentRepr.AST.ImportRecords[parentResult.tlaCheck.importRecordIndex].Range,
+								fmt.Sprintf("The file %q imports the file %q here:",
+									parentResult.file.inputFile.Source.PrettyPath, s.results[otherSourceIndex].file.inputFile.Source.PrettyPath)))
+						}
+
+						var text string
+						importedPrettyPath := s.results[record.SourceIndex.GetIndex()].file.inputFile.Source.PrettyPath
+
+						if importedPrettyPath == tlaPrettyPath {
+							text = fmt.Sprintf("This require call is not allowed because the imported file %q contains a top-level await",
+								importedPrettyPath)
+						} else {
+							text = fmt.Sprintf("This require call is not allowed because the transitive dependency %q contains a top-level await",
+								tlaPrettyPath)
+						}
+
+						tracker := logger.MakeLineColumnTracker(&result.file.inputFile.Source)
+						s.log.AddErrorWithNotes(&tracker, record.Range, text, notes)
+					}
+				}
+			}
+
+			// Make sure that if we wrap this module in a closure, the closure is also
+			// async. This happens when you call "import()" on this module and code
+			// splitting is off.
+			if result.tlaCheck.parent.IsValid() {
+				repr.Meta.IsAsyncOrHasAsyncDependency = true
+			}
+		}
+	}
+
+	return result.tlaCheck
+}
+
+func DefaultExtensionToLoaderMap() map[string]config.Loader {
+	return map[string]config.Loader{
+		"":            config.LoaderJS, // This represents files without an extension
+		".js":         config.LoaderJS,
+		".mjs":        config.LoaderJS,
+		".cjs":        config.LoaderJS,
+		".jsx":        config.LoaderJSX,
+		".ts":         config.LoaderTS,
+		".cts":        config.LoaderTSNoAmbiguousLessThan,
+		".mts":        config.LoaderTSNoAmbiguousLessThan,
+		".tsx":        config.LoaderTSX,
+		".css":        config.LoaderCSS,
+		".module.css": config.LoaderLocalCSS,
+		".json":       config.LoaderJSON,
+		".txt":        config.LoaderText,
+	}
+}
+
+func applyOptionDefaults(options *config.Options) {
+	if options.ExtensionToLoader == nil {
+		options.ExtensionToLoader = DefaultExtensionToLoaderMap()
+	}
+	if options.OutputExtensionJS == "" {
+		options.OutputExtensionJS = ".js"
+	}
+	if options.OutputExtensionCSS == "" {
+		options.OutputExtensionCSS = ".css"
+	}
+
+	// Configure default path templates
+	if len(options.EntryPathTemplate) == 0 {
+		options.EntryPathTemplate = []config.PathTemplate{
+			{Data: "./", Placeholder: config.DirPlaceholder},
+			{Data: "/", Placeholder: config.NamePlaceholder},
+		}
+	}
+	if len(options.ChunkPathTemplate) == 0 {
+		options.ChunkPathTemplate = []config.PathTemplate{
+			{Data: "./", Placeholder: config.NamePlaceholder},
+			{Data: "-", Placeholder: config.HashPlaceholder},
+		}
+	}
+	if len(options.AssetPathTemplate) == 0 {
+		options.AssetPathTemplate = []config.PathTemplate{
+			{Data: "./", Placeholder: config.NamePlaceholder},
+			{Data: "-", Placeholder: config.HashPlaceholder},
+		}
+	}
+
+	options.ProfilerNames = !options.MinifyIdentifiers
+
+	// Automatically fix invalid configurations of unsupported features
+	fixInvalidUnsupportedJSFeatureOverrides(options, compat.AsyncAwait, compat.AsyncGenerator|compat.ForAwait|compat.TopLevelAwait)
+	fixInvalidUnsupportedJSFeatureOverrides(options, compat.Generator, compat.AsyncGenerator)
+	fixInvalidUnsupportedJSFeatureOverrides(options, compat.ObjectAccessors, compat.ClassPrivateAccessor|compat.ClassPrivateStaticAccessor)
+	fixInvalidUnsupportedJSFeatureOverrides(options, compat.ClassField, compat.ClassPrivateField)
+	fixInvalidUnsupportedJSFeatureOverrides(options, compat.ClassStaticField, compat.ClassPrivateStaticField)
+	fixInvalidUnsupportedJSFeatureOverrides(options, compat.Class,
+		compat.ClassField|compat.ClassPrivateAccessor|compat.ClassPrivateBrandCheck|compat.ClassPrivateField|
+			compat.ClassPrivateMethod|compat.ClassPrivateStaticAccessor|compat.ClassPrivateStaticField|
+			compat.ClassPrivateStaticMethod|compat.ClassStaticBlocks|compat.ClassStaticField)
+
+	// If we're not building for the browser, automatically disable support for
+	// inline </script> and </style> tags if there aren't currently any overrides
+	if options.Platform != config.PlatformBrowser {
+		if !options.UnsupportedJSFeatureOverridesMask.Has(compat.InlineScript) {
+			options.UnsupportedJSFeatures |= compat.InlineScript
+		}
+		if !options.UnsupportedCSSFeatureOverridesMask.Has(compat.InlineStyle) {
+			options.UnsupportedCSSFeatures |= compat.InlineStyle
+		}
+	}
+}
+
+func fixInvalidUnsupportedJSFeatureOverrides(options *config.Options, implies compat.JSFeature, implied compat.JSFeature) {
+	// If this feature is unsupported, that implies that the other features must also be unsupported
+	if options.UnsupportedJSFeatureOverrides.Has(implies) {
+		options.UnsupportedJSFeatures |= implied
+		options.UnsupportedJSFeatureOverrides |= implied
+		options.UnsupportedJSFeatureOverridesMask |= implied
+	}
+}
+
+type Linker func(
+	options *config.Options,
+	timer *helpers.Timer,
+	log logger.Log,
+	fs fs.FS,
+	res *resolver.Resolver,
+	inputFiles []graph.InputFile,
+	entryPoints []graph.EntryPoint,
+	uniqueKeyPrefix string,
+	reachableFiles []uint32,
+	dataForSourceMaps func() []DataForSourceMap,
+) []graph.OutputFile
+
+func (b *Bundle) Compile(log logger.Log, timer *helpers.Timer, mangleCache map[string]interface{}, link Linker) ([]graph.OutputFile, string) {
+	timer.Begin("Compile phase")
+	defer timer.End("Compile phase")
+
+	if b.options.CancelFlag.DidCancel() {
+		return nil, ""
+	}
+
+	options := b.options
+
+	// In most cases we don't need synchronized access to the mangle cache
+	cssUsedLocalNames := make(map[string]bool)
+	options.ExclusiveMangleCacheUpdate = func(cb func(
+		mangleCache map[string]interface{},
+		cssUsedLocalNames map[string]bool,
+	)) {
+		cb(mangleCache, cssUsedLocalNames)
+	}
+
+	files := make([]graph.InputFile, len(b.files))
+	for i, file := range b.files {
+		files[i] = file.inputFile
+	}
+
+	// Get the base path from the options or choose the lowest common ancestor of all entry points
+	allReachableFiles := findReachableFiles(files, b.entryPoints)
+
+	// Compute source map data in parallel with linking
+	timer.Begin("Spawn source map tasks")
+	dataForSourceMaps := b.computeDataForSourceMapsInParallel(&options, allReachableFiles)
+	timer.End("Spawn source map tasks")
+
+	var resultGroups [][]graph.OutputFile
+	if options.CodeSplitting || len(b.entryPoints) == 1 {
+		// If code splitting is enabled or if there's only one entry point, link all entry points together
+		resultGroups = [][]graph.OutputFile{link(&options, timer, log, b.fs, b.res,
+			files, b.entryPoints, b.uniqueKeyPrefix, allReachableFiles, dataForSourceMaps)}
+	} else {
+		// Otherwise, link each entry point with the runtime file separately
+		waitGroup := sync.WaitGroup{}
+		resultGroups = make([][]graph.OutputFile, len(b.entryPoints))
+		serializer := helpers.MakeSerializer(len(b.entryPoints))
+		for i, entryPoint := range b.entryPoints {
+			waitGroup.Add(1)
+			go func(i int, entryPoint graph.EntryPoint) {
+				entryPoints := []graph.EntryPoint{entryPoint}
+				forked := timer.Fork()
+
+				// Each goroutine needs a separate options object
+				optionsClone := options
+				optionsClone.ExclusiveMangleCacheUpdate = func(cb func(
+					mangleCache map[string]interface{},
+					cssUsedLocalNames map[string]bool,
+				)) {
+					// Serialize all accesses to the mangle cache in entry point order for determinism
+					serializer.Enter(i)
+					defer serializer.Leave(i)
+					cb(mangleCache, cssUsedLocalNames)
+				}
+
+				resultGroups[i] = link(&optionsClone, forked, log, b.fs, b.res, files, entryPoints,
+					b.uniqueKeyPrefix, findReachableFiles(files, entryPoints), dataForSourceMaps)
+				timer.Join(forked)
+				waitGroup.Done()
+			}(i, entryPoint)
+		}
+		waitGroup.Wait()
+	}
+
+	// Join the results in entry point order for determinism
+	var outputFiles []graph.OutputFile
+	for _, group := range resultGroups {
+		outputFiles = append(outputFiles, group...)
+	}
+
+	// Also generate the metadata file if necessary
+	var metafileJSON string
+	if options.NeedsMetafile {
+		timer.Begin("Generate metadata JSON")
+		metafileJSON = b.generateMetadataJSON(outputFiles, allReachableFiles, options.ASCIIOnly)
+		timer.End("Generate metadata JSON")
+	}
+
+	if !options.WriteToStdout {
+		// Make sure an output file never overwrites an input file
+		if !options.AllowOverwrite {
+			sourceAbsPaths := make(map[string]uint32)
+			for _, sourceIndex := range allReachableFiles {
+				keyPath := b.files[sourceIndex].inputFile.Source.KeyPath
+				if keyPath.Namespace == "file" {
+					absPathKey := canonicalFileSystemPathForWindows(keyPath.Text)
+					sourceAbsPaths[absPathKey] = sourceIndex
+				}
+			}
+			for _, outputFile := range outputFiles {
+				absPathKey := canonicalFileSystemPathForWindows(outputFile.AbsPath)
+				if sourceIndex, ok := sourceAbsPaths[absPathKey]; ok {
+					hint := ""
+					switch logger.API {
+					case logger.CLIAPI:
+						hint = " (use \"--allow-overwrite\" to allow this)"
+					case logger.JSAPI:
+						hint = " (use \"allowOverwrite: true\" to allow this)"
+					case logger.GoAPI:
+						hint = " (use \"AllowOverwrite: true\" to allow this)"
+					}
+					log.AddError(nil, logger.Range{},
+						fmt.Sprintf("Refusing to overwrite input file %q%s",
+							b.files[sourceIndex].inputFile.Source.PrettyPath, hint))
+				}
+			}
+		}
+
+		// Make sure an output file never overwrites another output file. This
+		// is almost certainly unintentional and would otherwise happen silently.
+		//
+		// Make an exception for files that have identical contents. In that case
+		// the duplicate is just silently filtered out. This can happen with the
+		// "file" loader, for example.
+		outputFileMap := make(map[string][]byte)
+		end := 0
+		for _, outputFile := range outputFiles {
+			absPathKey := canonicalFileSystemPathForWindows(outputFile.AbsPath)
+			contents, ok := outputFileMap[absPathKey]
+
+			// If this isn't a duplicate, keep the output file
+			if !ok {
+				outputFileMap[absPathKey] = outputFile.Contents
+				outputFiles[end] = outputFile
+				end++
+				continue
+			}
+
+			// If the names and contents are both the same, only keep the first one
+			if bytes.Equal(contents, outputFile.Contents) {
+				continue
+			}
+
+			// Otherwise, generate an error
+			outputPath := outputFile.AbsPath
+			if relPath, ok := b.fs.Rel(b.fs.Cwd(), outputPath); ok {
+				outputPath = relPath
+			}
+			log.AddError(nil, logger.Range{}, "Two output files share the same path but have different contents: "+outputPath)
+		}
+		outputFiles = outputFiles[:end]
+	}
+
+	return outputFiles, metafileJSON
+}
+
+// Find all files reachable from all entry points. This order should be
+// deterministic given that the entry point order is deterministic, since the
+// returned order is the postorder of the graph traversal and import record
+// order within a given file is deterministic.
+func findReachableFiles(files []graph.InputFile, entryPoints []graph.EntryPoint) []uint32 {
+	visited := make(map[uint32]bool)
+	var order []uint32
+	var visit func(uint32)
+
+	// Include this file and all files it imports
+	visit = func(sourceIndex uint32) {
+		if !visited[sourceIndex] {
+			visited[sourceIndex] = true
+			file := &files[sourceIndex]
+			if repr, ok := file.Repr.(*graph.JSRepr); ok && repr.CSSSourceIndex.IsValid() {
+				visit(repr.CSSSourceIndex.GetIndex())
+			}
+			if recordsPtr := file.Repr.ImportRecords(); recordsPtr != nil {
+				for _, record := range *recordsPtr {
+					if record.SourceIndex.IsValid() {
+						visit(record.SourceIndex.GetIndex())
+					} else if record.CopySourceIndex.IsValid() {
+						visit(record.CopySourceIndex.GetIndex())
+					}
+				}
+			}
+
+			// Each file must come after its dependencies
+			order = append(order, sourceIndex)
+		}
+	}
+
+	// The runtime is always included in case it's needed
+	visit(runtime.SourceIndex)
+
+	// Include all files reachable from any entry point
+	for _, entryPoint := range entryPoints {
+		visit(entryPoint.SourceIndex)
+	}
+
+	return order
+}
+
+// This is done in parallel with linking because linking is a mostly serial
+// phase and there are extra resources for parallelism. This could also be done
+// during parsing but that would slow down parsing and delay the start of the
+// linking phase, which then delays the whole bundling process.
+//
+// However, doing this during parsing would allow it to be cached along with
+// the parsed ASTs which would then speed up incremental builds. In the future
+// it could be good to optionally have this be computed during the parsing
+// phase when incremental builds are active but otherwise still have it be
+// computed during linking for optimal speed during non-incremental builds.
+func (b *Bundle) computeDataForSourceMapsInParallel(options *config.Options, reachableFiles []uint32) func() []DataForSourceMap {
+	if options.SourceMap == config.SourceMapNone {
+		return func() []DataForSourceMap {
+			return nil
+		}
+	}
+
+	var waitGroup sync.WaitGroup
+	results := make([]DataForSourceMap, len(b.files))
+
+	for _, sourceIndex := range reachableFiles {
+		if f := &b.files[sourceIndex]; f.inputFile.Loader.CanHaveSourceMap() {
+			var approximateLineCount int32
+			switch repr := f.inputFile.Repr.(type) {
+			case *graph.JSRepr:
+				approximateLineCount = repr.AST.ApproximateLineCount
+			case *graph.CSSRepr:
+				approximateLineCount = repr.AST.ApproximateLineCount
+			}
+			waitGroup.Add(1)
+			go func(sourceIndex uint32, f *scannerFile, approximateLineCount int32) {
+				result := &results[sourceIndex]
+				result.LineOffsetTables = sourcemap.GenerateLineOffsetTables(f.inputFile.Source.Contents, approximateLineCount)
+				sm := f.inputFile.InputSourceMap
+				if !options.ExcludeSourcesContent {
+					if sm == nil {
+						// Simple case: no nested source map
+						result.QuotedContents = [][]byte{helpers.QuoteForJSON(f.inputFile.Source.Contents, options.ASCIIOnly)}
+					} else {
+						// Complex case: nested source map
+						result.QuotedContents = make([][]byte, len(sm.Sources))
+						nullContents := []byte("null")
+						for i := range sm.Sources {
+							// Missing contents become a "null" literal
+							quotedContents := nullContents
+							if i < len(sm.SourcesContent) {
+								if value := sm.SourcesContent[i]; value.Quoted != "" && (!options.ASCIIOnly || !isASCIIOnly(value.Quoted)) {
+									// Just use the value directly from the input file
+									quotedContents = []byte(value.Quoted)
+								} else if value.Value != nil {
+									// Re-quote non-ASCII values if output is ASCII-only.
+									// Also quote values that haven't been quoted yet
+									// (happens when the entire "sourcesContent" array is
+									// absent and the source has been found on the file
+									// system using the "sources" array).
+									quotedContents = helpers.QuoteForJSON(helpers.UTF16ToString(value.Value), options.ASCIIOnly)
+								}
+							}
+							result.QuotedContents[i] = quotedContents
+						}
+					}
+				}
+				waitGroup.Done()
+			}(sourceIndex, f, approximateLineCount)
+		}
+	}
+
+	return func() []DataForSourceMap {
+		waitGroup.Wait()
+		return results
+	}
+}
+
+func (b *Bundle) generateMetadataJSON(results []graph.OutputFile, allReachableFiles []uint32, asciiOnly bool) string {
+	sb := strings.Builder{}
+	sb.WriteString("{\n  \"inputs\": {")
+
+	// Write inputs
+	isFirst := true
+	for _, sourceIndex := range allReachableFiles {
+		if b.files[sourceIndex].inputFile.OmitFromSourceMapsAndMetafile {
+			continue
+		}
+		if file := &b.files[sourceIndex]; len(file.jsonMetadataChunk) > 0 {
+			if isFirst {
+				isFirst = false
+				sb.WriteString("\n    ")
+			} else {
+				sb.WriteString(",\n    ")
+			}
+			sb.WriteString(file.jsonMetadataChunk)
+		}
+	}
+
+	sb.WriteString("\n  },\n  \"outputs\": {")
+
+	// Write outputs
+	isFirst = true
+	paths := make(map[string]bool)
+	for _, result := range results {
+		if len(result.JSONMetadataChunk) > 0 {
+			path := resolver.PrettyPath(b.fs, logger.Path{Text: result.AbsPath, Namespace: "file"})
+			if paths[path] {
+				// Don't write out the same path twice (can happen with the "file" loader)
+				continue
+			}
+			if isFirst {
+				isFirst = false
+				sb.WriteString("\n    ")
+			} else {
+				sb.WriteString(",\n    ")
+			}
+			paths[path] = true
+			sb.WriteString(fmt.Sprintf("%s: ", helpers.QuoteForJSON(path, asciiOnly)))
+			sb.WriteString(result.JSONMetadataChunk)
+		}
+	}
+
+	sb.WriteString("\n  }\n}\n")
+	return sb.String()
+}
+
+type runtimeCacheKey struct {
+	unsupportedJSFeatures compat.JSFeature
+	minifySyntax          bool
+	minifyIdentifiers     bool
+}
+
+type runtimeCache struct {
+	astMap   map[runtimeCacheKey]js_ast.AST
+	astMutex sync.Mutex
+}
+
+var globalRuntimeCache runtimeCache
+
+func (cache *runtimeCache) parseRuntime(options *config.Options) (source logger.Source, runtimeAST js_ast.AST, ok bool) {
+	key := runtimeCacheKey{
+		// All configuration options that the runtime code depends on must go here
+		unsupportedJSFeatures: options.UnsupportedJSFeatures,
+		minifySyntax:          options.MinifySyntax,
+		minifyIdentifiers:     options.MinifyIdentifiers,
+	}
+
+	// Determine which source to use
+	source = runtime.Source(key.unsupportedJSFeatures)
+
+	// Cache hit?
+	(func() {
+		cache.astMutex.Lock()
+		defer cache.astMutex.Unlock()
+		if cache.astMap != nil {
+			runtimeAST, ok = cache.astMap[key]
+		}
+	})()
+	if ok {
+		return
+	}
+
+	// Cache miss
+	log := logger.NewDeferLog(logger.DeferLogAll, nil)
+	runtimeAST, ok = js_parser.Parse(log, source, js_parser.OptionsFromConfig(&config.Options{
+		// These configuration options must only depend on the key
+		UnsupportedJSFeatures: key.unsupportedJSFeatures,
+		MinifySyntax:          key.minifySyntax,
+		MinifyIdentifiers:     key.minifyIdentifiers,
+
+		// Always do tree shaking for the runtime because we never want to
+		// include unnecessary runtime code
+		TreeShaking: true,
+	}))
+	if log.HasErrors() {
+		msgs := "Internal error: failed to parse runtime:\n"
+		for _, msg := range log.Done() {
+			msgs += msg.String(logger.OutputOptions{IncludeSource: true}, logger.TerminalInfo{})
+		}
+		panic(msgs[:len(msgs)-1])
+	}
+
+	// Cache for next time
+	if ok {
+		cache.astMutex.Lock()
+		defer cache.astMutex.Unlock()
+		if cache.astMap == nil {
+			cache.astMap = make(map[runtimeCacheKey]js_ast.AST)
+		}
+		cache.astMap[key] = runtimeAST
+	}
+	return
+}
+
+// Returns the path of this file relative to "outbase", which is then ready to
+// be joined with the absolute output directory path. The directory and name
+// components are returned separately for convenience.
+func PathRelativeToOutbase(
+	inputFile *graph.InputFile,
+	options *config.Options,
+	fs fs.FS,
+	avoidIndex bool,
+	customFilePath string,
+) (relDir string, baseName string) {
+	relDir = "/"
+	absPath := inputFile.Source.KeyPath.Text
+
+	if customFilePath != "" {
+		// Use the configured output path if present
+		absPath = customFilePath
+		if !fs.IsAbs(absPath) {
+			absPath = fs.Join(options.AbsOutputBase, absPath)
+		}
+	} else if inputFile.Source.KeyPath.Namespace != "file" {
+		// Come up with a path for virtual paths (i.e. non-file-system paths)
+		dir, base, _ := logger.PlatformIndependentPathDirBaseExt(absPath)
+		if avoidIndex && base == "index" {
+			_, base, _ = logger.PlatformIndependentPathDirBaseExt(dir)
+		}
+		baseName = sanitizeFilePathForVirtualModulePath(base)
+		return
+	} else {
+		// Heuristic: If the file is named something like "index.js", then use
+		// the name of the parent directory instead. This helps avoid the
+		// situation where many chunks are named "index" because of people
+		// dynamically-importing npm packages that make use of node's implicit
+		// "index" file name feature.
+		if avoidIndex {
+			base := fs.Base(absPath)
+			base = base[:len(base)-len(fs.Ext(base))]
+			if base == "index" {
+				absPath = fs.Dir(absPath)
+			}
+		}
+	}
+
+	// Try to get a relative path to the base directory
+	relPath, ok := fs.Rel(options.AbsOutputBase, absPath)
+	if !ok {
+		// This can fail in some situations such as on different drives on
+		// Windows. In that case we just use the file name.
+		baseName = fs.Base(absPath)
+	} else {
+		// Now we finally have a relative path
+		relDir = fs.Dir(relPath) + "/"
+		baseName = fs.Base(relPath)
+
+		// Use platform-independent slashes
+		relDir = strings.ReplaceAll(relDir, "\\", "/")
+
+		// Replace leading "../" so we don't try to write outside of the output
+		// directory. This normally can't happen because "AbsOutputBase" is
+		// automatically computed to contain all entry point files, but it can
+		// happen if someone sets it manually via the "outbase" API option.
+		//
+		// Note that we can't just strip any leading "../" because that could
+		// cause two separate entry point paths to collide. For example, there
+		// could be both "src/index.js" and "../src/index.js" as entry points.
+		dotDotCount := 0
+		for strings.HasPrefix(relDir[dotDotCount*3:], "../") {
+			dotDotCount++
+		}
+		if dotDotCount > 0 {
+			// The use of "_.._" here is somewhat arbitrary but it is unlikely to
+			// collide with a folder named by a human and it works on Windows
+			// (Windows doesn't like names that end with a "."). And not starting
+			// with a "." means that it will not be hidden on Unix.
+			relDir = strings.Repeat("_.._/", dotDotCount) + relDir[dotDotCount*3:]
+		}
+		for strings.HasSuffix(relDir, "/") {
+			relDir = relDir[:len(relDir)-1]
+		}
+		relDir = "/" + relDir
+		if strings.HasSuffix(relDir, "/.") {
+			relDir = relDir[:len(relDir)-1]
+		}
+	}
+
+	// Strip the file extension if the output path is an input file
+	if customFilePath == "" {
+		ext := fs.Ext(baseName)
+		baseName = baseName[:len(baseName)-len(ext)]
+	}
+	return
+}
+
+func sanitizeFilePathForVirtualModulePath(path string) string {
+	// Convert it to a safe file path. See: https://stackoverflow.com/a/31976060
+	sb := strings.Builder{}
+	needsGap := false
+	for _, c := range path {
+		switch c {
+		case 0:
+			// These characters are forbidden on Unix and Windows
+
+		case '<', '>', ':', '"', '|', '?', '*':
+			// These characters are forbidden on Windows
+
+		default:
+			if c < 0x20 {
+				// These characters are forbidden on Windows
+				break
+			}
+
+			// Turn runs of invalid characters into a '_'
+			if needsGap {
+				sb.WriteByte('_')
+				needsGap = false
+			}
+
+			sb.WriteRune(c)
+			continue
+		}
+
+		if sb.Len() > 0 {
+			needsGap = true
+		}
+	}
+
+	// Make sure the name isn't empty
+	if sb.Len() == 0 {
+		return "_"
+	}
+
+	// Note: An extension will be added to this base name, so there is no need to
+	// avoid forbidden file names such as ".." since ".js" is a valid file name.
+	return sb.String()
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/cache/cache.go b/source/vendor/github.com/evanw/esbuild/internal/cache/cache.go
new file mode 100644
index 0000000..8b1dd8c
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/cache/cache.go
@@ -0,0 +1,115 @@
+package cache
+
+import (
+	"sync"
+
+	"github.com/evanw/esbuild/internal/logger"
+	"github.com/evanw/esbuild/internal/runtime"
+)
+
+// This is a cache of the parsed contents of a set of files. The idea is to be
+// able to reuse the results of parsing between builds and make subsequent
+// builds faster by avoiding redundant parsing work. This only works if:
+//
+//   - The AST information in the cache must be considered immutable. There is
+//     no way to enforce this in Go, but please be disciplined about this. The
+//     ASTs are shared in between builds. Any information that must be mutated
+//     in the AST during a build must be done on a shallow clone of the data if
+//     the mutation happens after parsing (i.e. a clone that clones everything
+//     that will be mutated and shares only the parts that won't be mutated).
+//
+//   - The information in the cache must not depend at all on the contents of
+//     any file other than the file being cached. Invalidating an entry in the
+//     cache does not also invalidate any entries that depend on that file, so
+//     caching information that depends on other files can result in incorrect
+//     results due to reusing stale data. For example, do not "bake in" some
+//     value imported from another file.
+//
+//   - Cached ASTs must only be reused if the parsing options are identical
+//     between builds. For example, it would be bad if the AST parser depended
+//     on options inherited from a nearby "package.json" file but those options
+//     were not part of the cache key. Then the cached AST could incorrectly be
+//     reused even if the contents of that "package.json" file have changed.
+type CacheSet struct {
+	FSCache          FSCache
+	CSSCache         CSSCache
+	JSONCache        JSONCache
+	JSCache          JSCache
+	SourceIndexCache SourceIndexCache
+}
+
+func MakeCacheSet() *CacheSet {
+	return &CacheSet{
+		SourceIndexCache: SourceIndexCache{
+			globEntries:     make(map[uint64]uint32),
+			entries:         make(map[sourceIndexKey]uint32),
+			nextSourceIndex: runtime.SourceIndex + 1,
+		},
+		FSCache: FSCache{
+			entries: make(map[string]*fsEntry),
+		},
+		CSSCache: CSSCache{
+			entries: make(map[logger.Path]*cssCacheEntry),
+		},
+		JSONCache: JSONCache{
+			entries: make(map[logger.Path]*jsonCacheEntry),
+		},
+		JSCache: JSCache{
+			entries: make(map[logger.Path]*jsCacheEntry),
+		},
+	}
+}
+
+type SourceIndexCache struct {
+	globEntries     map[uint64]uint32
+	entries         map[sourceIndexKey]uint32
+	mutex           sync.Mutex
+	nextSourceIndex uint32
+}
+
+type SourceIndexKind uint8
+
+const (
+	SourceIndexNormal SourceIndexKind = iota
+	SourceIndexJSStubForCSS
+)
+
+type sourceIndexKey struct {
+	path logger.Path
+	kind SourceIndexKind
+}
+
+func (c *SourceIndexCache) LenHint() uint32 {
+	c.mutex.Lock()
+	defer c.mutex.Unlock()
+
+	// Add some extra room at the end for a new file or two without reallocating
+	const someExtraRoom = 16
+	return c.nextSourceIndex + someExtraRoom
+}
+
+func (c *SourceIndexCache) Get(path logger.Path, kind SourceIndexKind) uint32 {
+	key := sourceIndexKey{path: path, kind: kind}
+	c.mutex.Lock()
+	defer c.mutex.Unlock()
+	if sourceIndex, ok := c.entries[key]; ok {
+		return sourceIndex
+	}
+	sourceIndex := c.nextSourceIndex
+	c.nextSourceIndex++
+	c.entries[key] = sourceIndex
+	return sourceIndex
+}
+
+func (c *SourceIndexCache) GetGlob(parentSourceIndex uint32, globIndex uint32) uint32 {
+	key := (uint64(parentSourceIndex) << 32) | uint64(globIndex)
+	c.mutex.Lock()
+	defer c.mutex.Unlock()
+	if sourceIndex, ok := c.globEntries[key]; ok {
+		return sourceIndex
+	}
+	sourceIndex := c.nextSourceIndex
+	c.nextSourceIndex++
+	c.globEntries[key] = sourceIndex
+	return sourceIndex
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/cache/cache_ast.go b/source/vendor/github.com/evanw/esbuild/internal/cache/cache_ast.go
new file mode 100644
index 0000000..c976f89
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/cache/cache_ast.go
@@ -0,0 +1,190 @@
+package cache
+
+import (
+	"sync"
+
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_parser"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/js_parser"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+// This cache intends to avoid unnecessarily re-parsing files in subsequent
+// builds. For a given path, parsing can be avoided if the contents of the file
+// and the options for the parser are the same as last time. Even if the
+// contents of the file are the same, the options for the parser may have
+// changed if they depend on some other file ("package.json" for example).
+//
+// This cache checks if the file contents have changed even though we have
+// the ability to detect if a file has changed on the file system by reading
+// its metadata. First of all, if the file contents are cached then they should
+// be the same pointer, which makes the comparison trivial. Also we want to
+// cache the AST for plugins in the common case that the plugin output stays
+// the same.
+
+////////////////////////////////////////////////////////////////////////////////
+// CSS
+
+type CSSCache struct {
+	entries map[logger.Path]*cssCacheEntry
+	mutex   sync.Mutex
+}
+
+type cssCacheEntry struct {
+	source  logger.Source
+	msgs    []logger.Msg
+	ast     css_ast.AST
+	options css_parser.Options
+}
+
+func (c *CSSCache) Parse(log logger.Log, source logger.Source, options css_parser.Options) css_ast.AST {
+	// Check the cache
+	entry := func() *cssCacheEntry {
+		c.mutex.Lock()
+		defer c.mutex.Unlock()
+		return c.entries[source.KeyPath]
+	}()
+
+	// Cache hit
+	if entry != nil && entry.source == source && entry.options.Equal(&options) {
+		for _, msg := range entry.msgs {
+			log.AddMsg(msg)
+		}
+		return entry.ast
+	}
+
+	// Cache miss
+	tempLog := logger.NewDeferLog(logger.DeferLogAll, log.Overrides)
+	ast := css_parser.Parse(tempLog, source, options)
+	msgs := tempLog.Done()
+	for _, msg := range msgs {
+		log.AddMsg(msg)
+	}
+
+	// Create the cache entry
+	entry = &cssCacheEntry{
+		source:  source,
+		options: options,
+		ast:     ast,
+		msgs:    msgs,
+	}
+
+	// Save for next time
+	c.mutex.Lock()
+	defer c.mutex.Unlock()
+	c.entries[source.KeyPath] = entry
+	return ast
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// JSON
+
+type JSONCache struct {
+	entries map[logger.Path]*jsonCacheEntry
+	mutex   sync.Mutex
+}
+
+type jsonCacheEntry struct {
+	expr    js_ast.Expr
+	msgs    []logger.Msg
+	source  logger.Source
+	options js_parser.JSONOptions
+	ok      bool
+}
+
+func (c *JSONCache) Parse(log logger.Log, source logger.Source, options js_parser.JSONOptions) (js_ast.Expr, bool) {
+	// Check the cache
+	entry := func() *jsonCacheEntry {
+		c.mutex.Lock()
+		defer c.mutex.Unlock()
+		return c.entries[source.KeyPath]
+	}()
+
+	// Cache hit
+	if entry != nil && entry.source == source && entry.options == options {
+		for _, msg := range entry.msgs {
+			log.AddMsg(msg)
+		}
+		return entry.expr, entry.ok
+	}
+
+	// Cache miss
+	tempLog := logger.NewDeferLog(logger.DeferLogAll, log.Overrides)
+	expr, ok := js_parser.ParseJSON(tempLog, source, options)
+	msgs := tempLog.Done()
+	for _, msg := range msgs {
+		log.AddMsg(msg)
+	}
+
+	// Create the cache entry
+	entry = &jsonCacheEntry{
+		source:  source,
+		options: options,
+		expr:    expr,
+		ok:      ok,
+		msgs:    msgs,
+	}
+
+	// Save for next time
+	c.mutex.Lock()
+	defer c.mutex.Unlock()
+	c.entries[source.KeyPath] = entry
+	return expr, ok
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// JS
+
+type JSCache struct {
+	entries map[logger.Path]*jsCacheEntry
+	mutex   sync.Mutex
+}
+
+type jsCacheEntry struct {
+	source  logger.Source
+	msgs    []logger.Msg
+	options js_parser.Options
+	ast     js_ast.AST
+	ok      bool
+}
+
+func (c *JSCache) Parse(log logger.Log, source logger.Source, options js_parser.Options) (js_ast.AST, bool) {
+	// Check the cache
+	entry := func() *jsCacheEntry {
+		c.mutex.Lock()
+		defer c.mutex.Unlock()
+		return c.entries[source.KeyPath]
+	}()
+
+	// Cache hit
+	if entry != nil && entry.source == source && entry.options.Equal(&options) {
+		for _, msg := range entry.msgs {
+			log.AddMsg(msg)
+		}
+		return entry.ast, entry.ok
+	}
+
+	// Cache miss
+	tempLog := logger.NewDeferLog(logger.DeferLogAll, log.Overrides)
+	ast, ok := js_parser.Parse(tempLog, source, options)
+	msgs := tempLog.Done()
+	for _, msg := range msgs {
+		log.AddMsg(msg)
+	}
+
+	// Create the cache entry
+	entry = &jsCacheEntry{
+		source:  source,
+		options: options,
+		ast:     ast,
+		ok:      ok,
+		msgs:    msgs,
+	}
+
+	// Save for next time
+	c.mutex.Lock()
+	defer c.mutex.Unlock()
+	c.entries[source.KeyPath] = entry
+	return ast, ok
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/cache/cache_fs.go b/source/vendor/github.com/evanw/esbuild/internal/cache/cache_fs.go
new file mode 100644
index 0000000..ab4d08e
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/cache/cache_fs.go
@@ -0,0 +1,52 @@
+package cache
+
+import (
+	"sync"
+
+	"github.com/evanw/esbuild/internal/fs"
+)
+
+// This cache uses information from the "stat" syscall to try to avoid re-
+// reading files from the file system during subsequent builds if the file
+// hasn't changed. The assumption is reading the file metadata is faster than
+// reading the file contents.
+
+type FSCache struct {
+	entries map[string]*fsEntry
+	mutex   sync.Mutex
+}
+
+type fsEntry struct {
+	contents       string
+	modKey         fs.ModKey
+	isModKeyUsable bool
+}
+
+func (c *FSCache) ReadFile(fs fs.FS, path string) (contents string, canonicalError error, originalError error) {
+	entry := func() *fsEntry {
+		c.mutex.Lock()
+		defer c.mutex.Unlock()
+		return c.entries[path]
+	}()
+
+	// If the file's modification key hasn't changed since it was cached, assume
+	// the contents of the file are also the same and skip reading the file.
+	modKey, modKeyErr := fs.ModKey(path)
+	if entry != nil && entry.isModKeyUsable && modKeyErr == nil && entry.modKey == modKey {
+		return entry.contents, nil, nil
+	}
+
+	contents, err, originalError := fs.ReadFile(path)
+	if err != nil {
+		return "", err, originalError
+	}
+
+	c.mutex.Lock()
+	defer c.mutex.Unlock()
+	c.entries[path] = &fsEntry{
+		contents:       contents,
+		modKey:         modKey,
+		isModKeyUsable: modKeyErr == nil,
+	}
+	return contents, nil, nil
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/compat/compat.go b/source/vendor/github.com/evanw/esbuild/internal/compat/compat.go
new file mode 100644
index 0000000..bd2d0ff
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/compat/compat.go
@@ -0,0 +1,92 @@
+package compat
+
+import (
+	"strconv"
+	"strings"
+
+	"github.com/evanw/esbuild/internal/ast"
+)
+
+type v struct {
+	major uint16
+	minor uint8
+	patch uint8
+}
+
+type Semver struct {
+	// "1.2.3-alpha" => { Parts: {1, 2, 3}, PreRelease: "-alpha" }
+	Parts      []int
+	PreRelease string
+}
+
+func (v Semver) String() string {
+	b := strings.Builder{}
+	for _, part := range v.Parts {
+		if b.Len() > 0 {
+			b.WriteRune('.')
+		}
+		b.WriteString(strconv.Itoa(part))
+	}
+	b.WriteString(v.PreRelease)
+	return b.String()
+}
+
+// Returns <0 if "a < b"
+// Returns 0 if "a == b"
+// Returns >0 if "a > b"
+func compareVersions(a v, b Semver) int {
+	diff := int(a.major)
+	if len(b.Parts) > 0 {
+		diff -= b.Parts[0]
+	}
+	if diff == 0 {
+		diff = int(a.minor)
+		if len(b.Parts) > 1 {
+			diff -= b.Parts[1]
+		}
+	}
+	if diff == 0 {
+		diff = int(a.patch)
+		if len(b.Parts) > 2 {
+			diff -= b.Parts[2]
+		}
+	}
+	if diff == 0 && len(b.PreRelease) != 0 {
+		return 1 // "1.0.0" > "1.0.0-alpha"
+	}
+	return diff
+}
+
+// The start is inclusive and the end is exclusive
+type versionRange struct {
+	start v
+	end   v // Use 0.0.0 for "no end"
+}
+
+func isVersionSupported(ranges []versionRange, version Semver) bool {
+	for _, r := range ranges {
+		if compareVersions(r.start, version) <= 0 && (r.end == (v{}) || compareVersions(r.end, version) > 0) {
+			return true
+		}
+	}
+	return false
+}
+
+func SymbolFeature(kind ast.SymbolKind) JSFeature {
+	switch kind {
+	case ast.SymbolPrivateField:
+		return ClassPrivateField
+	case ast.SymbolPrivateMethod:
+		return ClassPrivateMethod
+	case ast.SymbolPrivateGet, ast.SymbolPrivateSet, ast.SymbolPrivateGetSetPair:
+		return ClassPrivateAccessor
+	case ast.SymbolPrivateStaticField:
+		return ClassPrivateStaticField
+	case ast.SymbolPrivateStaticMethod:
+		return ClassPrivateStaticMethod
+	case ast.SymbolPrivateStaticGet, ast.SymbolPrivateStaticSet, ast.SymbolPrivateStaticGetSetPair:
+		return ClassPrivateStaticAccessor
+	default:
+		return 0
+	}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/compat/css_table.go b/source/vendor/github.com/evanw/esbuild/internal/compat/css_table.go
new file mode 100644
index 0000000..1cd717e
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/compat/css_table.go
@@ -0,0 +1,361 @@
+// This file was automatically generated by "css_table.ts"
+
+package compat
+
+import (
+	"github.com/evanw/esbuild/internal/css_ast"
+)
+
+type CSSFeature uint16
+
+const (
+	ColorFunctions CSSFeature = 1 << iota
+	GradientDoublePosition
+	GradientInterpolation
+	GradientMidpoints
+	HWB
+	HexRGBA
+	InlineStyle
+	InsetProperty
+	IsPseudoClass
+	Modern_RGB_HSL
+	Nesting
+	RebeccaPurple
+)
+
+var StringToCSSFeature = map[string]CSSFeature{
+	"color-functions":          ColorFunctions,
+	"gradient-double-position": GradientDoublePosition,
+	"gradient-interpolation":   GradientInterpolation,
+	"gradient-midpoints":       GradientMidpoints,
+	"hwb":                      HWB,
+	"hex-rgba":                 HexRGBA,
+	"inline-style":             InlineStyle,
+	"inset-property":           InsetProperty,
+	"is-pseudo-class":          IsPseudoClass,
+	"modern-rgb-hsl":           Modern_RGB_HSL,
+	"nesting":                  Nesting,
+	"rebecca-purple":           RebeccaPurple,
+}
+
+func (features CSSFeature) Has(feature CSSFeature) bool {
+	return (features & feature) != 0
+}
+
+func (features CSSFeature) ApplyOverrides(overrides CSSFeature, mask CSSFeature) CSSFeature {
+	return (features & ^mask) | (overrides & mask)
+}
+
+var cssTable = map[CSSFeature]map[Engine][]versionRange{
+	ColorFunctions: {
+		Chrome:  {{start: v{111, 0, 0}}},
+		Edge:    {{start: v{111, 0, 0}}},
+		Firefox: {{start: v{113, 0, 0}}},
+		IOS:     {{start: v{15, 4, 0}}},
+		Opera:   {{start: v{97, 0, 0}}},
+		Safari:  {{start: v{15, 4, 0}}},
+	},
+	GradientDoublePosition: {
+		Chrome:  {{start: v{72, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		Firefox: {{start: v{83, 0, 0}}},
+		IOS:     {{start: v{12, 2, 0}}},
+		Opera:   {{start: v{60, 0, 0}}},
+		Safari:  {{start: v{12, 1, 0}}},
+	},
+	GradientInterpolation: {
+		Chrome: {{start: v{111, 0, 0}}},
+		Edge:   {{start: v{111, 0, 0}}},
+		IOS:    {{start: v{16, 2, 0}}},
+		Opera:  {{start: v{97, 0, 0}}},
+		Safari: {{start: v{16, 2, 0}}},
+	},
+	GradientMidpoints: {
+		Chrome:  {{start: v{40, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		Firefox: {{start: v{36, 0, 0}}},
+		IOS:     {{start: v{7, 0, 0}}},
+		Opera:   {{start: v{27, 0, 0}}},
+		Safari:  {{start: v{7, 0, 0}}},
+	},
+	HWB: {
+		Chrome:  {{start: v{101, 0, 0}}},
+		Edge:    {{start: v{101, 0, 0}}},
+		Firefox: {{start: v{96, 0, 0}}},
+		IOS:     {{start: v{15, 0, 0}}},
+		Opera:   {{start: v{87, 0, 0}}},
+		Safari:  {{start: v{15, 0, 0}}},
+	},
+	HexRGBA: {
+		Chrome:  {{start: v{62, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		Firefox: {{start: v{49, 0, 0}}},
+		IOS:     {{start: v{9, 3, 0}}},
+		Opera:   {{start: v{49, 0, 0}}},
+		Safari:  {{start: v{10, 0, 0}}},
+	},
+	InlineStyle: {},
+	InsetProperty: {
+		Chrome:  {{start: v{87, 0, 0}}},
+		Edge:    {{start: v{87, 0, 0}}},
+		Firefox: {{start: v{66, 0, 0}}},
+		IOS:     {{start: v{14, 5, 0}}},
+		Opera:   {{start: v{73, 0, 0}}},
+		Safari:  {{start: v{14, 1, 0}}},
+	},
+	IsPseudoClass: {
+		Chrome:  {{start: v{88, 0, 0}}},
+		Edge:    {{start: v{88, 0, 0}}},
+		Firefox: {{start: v{78, 0, 0}}},
+		IOS:     {{start: v{14, 0, 0}}},
+		Opera:   {{start: v{75, 0, 0}}},
+		Safari:  {{start: v{14, 0, 0}}},
+	},
+	Modern_RGB_HSL: {
+		Chrome:  {{start: v{66, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		Firefox: {{start: v{52, 0, 0}}},
+		IOS:     {{start: v{12, 2, 0}}},
+		Opera:   {{start: v{53, 0, 0}}},
+		Safari:  {{start: v{12, 1, 0}}},
+	},
+	Nesting: {
+		Chrome:  {{start: v{120, 0, 0}}},
+		Edge:    {{start: v{120, 0, 0}}},
+		Firefox: {{start: v{117, 0, 0}}},
+		IOS:     {{start: v{17, 2, 0}}},
+		Opera:   {{start: v{106, 0, 0}}},
+		Safari:  {{start: v{17, 2, 0}}},
+	},
+	RebeccaPurple: {
+		Chrome:  {{start: v{38, 0, 0}}},
+		Edge:    {{start: v{12, 0, 0}}},
+		Firefox: {{start: v{33, 0, 0}}},
+		IE:      {{start: v{11, 0, 0}}},
+		IOS:     {{start: v{8, 0, 0}}},
+		Opera:   {{start: v{25, 0, 0}}},
+		Safari:  {{start: v{9, 0, 0}}},
+	},
+}
+
+// Return all features that are not available in at least one environment
+func UnsupportedCSSFeatures(constraints map[Engine]Semver) (unsupported CSSFeature) {
+	for feature, engines := range cssTable {
+		if feature == InlineStyle {
+			continue // This is purely user-specified
+		}
+		for engine, version := range constraints {
+			if !engine.IsBrowser() {
+				// Specifying "--target=es2020" shouldn't affect CSS
+				continue
+			}
+			if versionRanges, ok := engines[engine]; !ok || !isVersionSupported(versionRanges, version) {
+				unsupported |= feature
+			}
+		}
+	}
+	return
+}
+
+type CSSPrefix uint8
+
+const (
+	KhtmlPrefix CSSPrefix = 1 << iota
+	MozPrefix
+	MsPrefix
+	OPrefix
+	WebkitPrefix
+
+	NoPrefix CSSPrefix = 0
+)
+
+type prefixData struct {
+	// Note: In some cases, earlier versions did not require a prefix but later
+	// ones do. This is the case for Microsoft Edge for example, which switched
+	// the underlying browser engine from a custom one to the one from Chrome.
+	// However, we assume that users specifying a browser version for CSS mean
+	// "works in this version or newer", so we still add a prefix when a target
+	// is an old Edge version.
+	engine        Engine
+	withoutPrefix v
+	prefix        CSSPrefix
+}
+
+var cssPrefixTable = map[css_ast.D][]prefixData{
+	css_ast.DAppearance: {
+		{engine: Chrome, prefix: WebkitPrefix, withoutPrefix: v{84, 0, 0}},
+		{engine: Edge, prefix: WebkitPrefix, withoutPrefix: v{84, 0, 0}},
+		{engine: Firefox, prefix: MozPrefix, withoutPrefix: v{80, 0, 0}},
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+		{engine: Opera, prefix: WebkitPrefix, withoutPrefix: v{73, 0, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+	},
+	css_ast.DBackdropFilter: {
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{18, 0, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{18, 0, 0}},
+	},
+	css_ast.DBackgroundClip: {
+		{engine: Chrome, prefix: WebkitPrefix, withoutPrefix: v{120, 0, 0}},
+		{engine: Edge, prefix: MsPrefix, withoutPrefix: v{15, 0, 0}},
+		{engine: Edge, prefix: WebkitPrefix, withoutPrefix: v{120, 0, 0}},
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{14, 0, 0}},
+		{engine: Opera, prefix: WebkitPrefix, withoutPrefix: v{106, 0, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{14, 0, 0}},
+	},
+	css_ast.DBoxDecorationBreak: {
+		{engine: Chrome, prefix: WebkitPrefix},
+		{engine: Edge, prefix: WebkitPrefix},
+		{engine: IOS, prefix: WebkitPrefix},
+		{engine: Opera, prefix: WebkitPrefix},
+		{engine: Safari, prefix: WebkitPrefix},
+	},
+	css_ast.DClipPath: {
+		{engine: Chrome, prefix: WebkitPrefix, withoutPrefix: v{55, 0, 0}},
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{13, 0, 0}},
+		{engine: Opera, prefix: WebkitPrefix, withoutPrefix: v{42, 0, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{13, 1, 0}},
+	},
+	css_ast.DFontKerning: {
+		{engine: Chrome, prefix: WebkitPrefix, withoutPrefix: v{33, 0, 0}},
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{12, 0, 0}},
+		{engine: Opera, prefix: WebkitPrefix, withoutPrefix: v{20, 0, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{9, 1, 0}},
+	},
+	css_ast.DHyphens: {
+		{engine: Edge, prefix: MsPrefix, withoutPrefix: v{79, 0, 0}},
+		{engine: Firefox, prefix: MozPrefix, withoutPrefix: v{43, 0, 0}},
+		{engine: IE, prefix: MsPrefix},
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{17, 0, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{17, 0, 0}},
+	},
+	css_ast.DInitialLetter: {
+		{engine: IOS, prefix: WebkitPrefix},
+		{engine: Safari, prefix: WebkitPrefix},
+	},
+	css_ast.DMaskComposite: {
+		{engine: Chrome, prefix: WebkitPrefix, withoutPrefix: v{120, 0, 0}},
+		{engine: Edge, prefix: WebkitPrefix, withoutPrefix: v{120, 0, 0}},
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+		{engine: Opera, prefix: WebkitPrefix, withoutPrefix: v{106, 0, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+	},
+	css_ast.DMaskImage: {
+		{engine: Chrome, prefix: WebkitPrefix, withoutPrefix: v{120, 0, 0}},
+		{engine: Edge, prefix: WebkitPrefix, withoutPrefix: v{120, 0, 0}},
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+		{engine: Opera, prefix: WebkitPrefix},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+	},
+	css_ast.DMaskOrigin: {
+		{engine: Chrome, prefix: WebkitPrefix, withoutPrefix: v{120, 0, 0}},
+		{engine: Edge, prefix: WebkitPrefix, withoutPrefix: v{120, 0, 0}},
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+		{engine: Opera, prefix: WebkitPrefix, withoutPrefix: v{106, 0, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+	},
+	css_ast.DMaskPosition: {
+		{engine: Chrome, prefix: WebkitPrefix, withoutPrefix: v{120, 0, 0}},
+		{engine: Edge, prefix: WebkitPrefix, withoutPrefix: v{120, 0, 0}},
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+		{engine: Opera, prefix: WebkitPrefix, withoutPrefix: v{106, 0, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+	},
+	css_ast.DMaskRepeat: {
+		{engine: Chrome, prefix: WebkitPrefix, withoutPrefix: v{120, 0, 0}},
+		{engine: Edge, prefix: WebkitPrefix, withoutPrefix: v{120, 0, 0}},
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+		{engine: Opera, prefix: WebkitPrefix, withoutPrefix: v{106, 0, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+	},
+	css_ast.DMaskSize: {
+		{engine: Chrome, prefix: WebkitPrefix, withoutPrefix: v{120, 0, 0}},
+		{engine: Edge, prefix: WebkitPrefix, withoutPrefix: v{120, 0, 0}},
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+		{engine: Opera, prefix: WebkitPrefix, withoutPrefix: v{106, 0, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+	},
+	css_ast.DPosition: {
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{13, 0, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{13, 0, 0}},
+	},
+	css_ast.DPrintColorAdjust: {
+		{engine: Chrome, prefix: WebkitPrefix},
+		{engine: Edge, prefix: WebkitPrefix},
+		{engine: Opera, prefix: WebkitPrefix},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{15, 4, 0}},
+	},
+	css_ast.DTabSize: {
+		{engine: Firefox, prefix: MozPrefix, withoutPrefix: v{91, 0, 0}},
+		{engine: Opera, prefix: OPrefix, withoutPrefix: v{15, 0, 0}},
+	},
+	css_ast.DTextDecorationColor: {
+		{engine: Firefox, prefix: MozPrefix, withoutPrefix: v{36, 0, 0}},
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{12, 2, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{12, 1, 0}},
+	},
+	css_ast.DTextDecorationLine: {
+		{engine: Firefox, prefix: MozPrefix, withoutPrefix: v{36, 0, 0}},
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{12, 2, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{12, 1, 0}},
+	},
+	css_ast.DTextDecorationSkip: {
+		{engine: IOS, prefix: WebkitPrefix, withoutPrefix: v{12, 2, 0}},
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{12, 1, 0}},
+	},
+	css_ast.DTextEmphasisColor: {
+		{engine: Chrome, prefix: WebkitPrefix, withoutPrefix: v{99, 0, 0}},
+		{engine: Edge, prefix: WebkitPrefix, withoutPrefix: v{99, 0, 0}},
+		{engine: Opera, prefix: WebkitPrefix, withoutPrefix: v{85, 0, 0}},
+	},
+	css_ast.DTextEmphasisPosition: {
+		{engine: Chrome, prefix: WebkitPrefix, withoutPrefix: v{99, 0, 0}},
+		{engine: Edge, prefix: WebkitPrefix, withoutPrefix: v{99, 0, 0}},
+		{engine: Opera, prefix: WebkitPrefix, withoutPrefix: v{85, 0, 0}},
+	},
+	css_ast.DTextEmphasisStyle: {
+		{engine: Chrome, prefix: WebkitPrefix, withoutPrefix: v{99, 0, 0}},
+		{engine: Edge, prefix: WebkitPrefix, withoutPrefix: v{99, 0, 0}},
+		{engine: Opera, prefix: WebkitPrefix, withoutPrefix: v{85, 0, 0}},
+	},
+	css_ast.DTextOrientation: {
+		{engine: Safari, prefix: WebkitPrefix, withoutPrefix: v{14, 0, 0}},
+	},
+	css_ast.DTextSizeAdjust: {
+		{engine: Edge, prefix: MsPrefix, withoutPrefix: v{79, 0, 0}},
+		{engine: IOS, prefix: WebkitPrefix},
+	},
+	css_ast.DUserSelect: {
+		{engine: Chrome, prefix: WebkitPrefix, withoutPrefix: v{54, 0, 0}},
+		{engine: Edge, prefix: MsPrefix, withoutPrefix: v{79, 0, 0}},
+		{engine: Firefox, prefix: MozPrefix, withoutPrefix: v{69, 0, 0}},
+		{engine: IE, prefix: MsPrefix},
+		{engine: IOS, prefix: WebkitPrefix},
+		{engine: Opera, prefix: WebkitPrefix, withoutPrefix: v{41, 0, 0}},
+		{engine: Safari, prefix: KhtmlPrefix, withoutPrefix: v{3, 0, 0}},
+		{engine: Safari, prefix: WebkitPrefix},
+	},
+}
+
+func CSSPrefixData(constraints map[Engine]Semver) (entries map[css_ast.D]CSSPrefix) {
+	for property, items := range cssPrefixTable {
+		prefixes := NoPrefix
+		for engine, version := range constraints {
+			if !engine.IsBrowser() {
+				// Specifying "--target=es2020" shouldn't affect CSS
+				continue
+			}
+			for _, item := range items {
+				if item.engine == engine && (item.withoutPrefix == v{} || compareVersions(item.withoutPrefix, version) > 0) {
+					prefixes |= item.prefix
+				}
+			}
+		}
+		if prefixes != NoPrefix {
+			if entries == nil {
+				entries = make(map[css_ast.D]CSSPrefix)
+			}
+			entries[property] = prefixes
+		}
+	}
+	return
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/compat/js_table.go b/source/vendor/github.com/evanw/esbuild/internal/compat/js_table.go
new file mode 100644
index 0000000..89180c3
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/compat/js_table.go
@@ -0,0 +1,910 @@
+// This file was automatically generated by "js_table.ts"
+
+package compat
+
+type Engine uint8
+
+const (
+	Chrome Engine = iota
+	Deno
+	Edge
+	ES
+	Firefox
+	Hermes
+	IE
+	IOS
+	Node
+	Opera
+	Rhino
+	Safari
+)
+
+func (e Engine) String() string {
+	switch e {
+	case Chrome:
+		return "chrome"
+	case Deno:
+		return "deno"
+	case Edge:
+		return "edge"
+	case ES:
+		return "es"
+	case Firefox:
+		return "firefox"
+	case Hermes:
+		return "hermes"
+	case IE:
+		return "ie"
+	case IOS:
+		return "ios"
+	case Node:
+		return "node"
+	case Opera:
+		return "opera"
+	case Rhino:
+		return "rhino"
+	case Safari:
+		return "safari"
+	}
+	return ""
+}
+
+func (e Engine) IsBrowser() bool {
+	switch e {
+	case Chrome, Edge, Firefox, IE, IOS, Opera, Safari:
+		return true
+	}
+	return false
+}
+
+type JSFeature uint64
+
+const (
+	ArbitraryModuleNamespaceNames JSFeature = 1 << iota
+	ArraySpread
+	Arrow
+	AsyncAwait
+	AsyncGenerator
+	Bigint
+	Class
+	ClassField
+	ClassPrivateAccessor
+	ClassPrivateBrandCheck
+	ClassPrivateField
+	ClassPrivateMethod
+	ClassPrivateStaticAccessor
+	ClassPrivateStaticField
+	ClassPrivateStaticMethod
+	ClassStaticBlocks
+	ClassStaticField
+	ConstAndLet
+	Decorators
+	DefaultArgument
+	Destructuring
+	DynamicImport
+	ExponentOperator
+	ExportStarAs
+	ForAwait
+	ForOf
+	FunctionNameConfigurable
+	FunctionOrClassPropertyAccess
+	Generator
+	Hashbang
+	ImportAssertions
+	ImportAttributes
+	ImportMeta
+	InlineScript
+	LogicalAssignment
+	NestedRestBinding
+	NewTarget
+	NodeColonPrefixImport
+	NodeColonPrefixRequire
+	NullishCoalescing
+	ObjectAccessors
+	ObjectExtensions
+	ObjectRestSpread
+	OptionalCatchBinding
+	OptionalChain
+	RegexpDotAllFlag
+	RegexpLookbehindAssertions
+	RegexpMatchIndices
+	RegexpNamedCaptureGroups
+	RegexpSetNotation
+	RegexpStickyAndUnicodeFlags
+	RegexpUnicodePropertyEscapes
+	RestArgument
+	TemplateLiteral
+	TopLevelAwait
+	TypeofExoticObjectIsObject
+	UnicodeEscapes
+	Using
+)
+
+var StringToJSFeature = map[string]JSFeature{
+	"arbitrary-module-namespace-names":  ArbitraryModuleNamespaceNames,
+	"array-spread":                      ArraySpread,
+	"arrow":                             Arrow,
+	"async-await":                       AsyncAwait,
+	"async-generator":                   AsyncGenerator,
+	"bigint":                            Bigint,
+	"class":                             Class,
+	"class-field":                       ClassField,
+	"class-private-accessor":            ClassPrivateAccessor,
+	"class-private-brand-check":         ClassPrivateBrandCheck,
+	"class-private-field":               ClassPrivateField,
+	"class-private-method":              ClassPrivateMethod,
+	"class-private-static-accessor":     ClassPrivateStaticAccessor,
+	"class-private-static-field":        ClassPrivateStaticField,
+	"class-private-static-method":       ClassPrivateStaticMethod,
+	"class-static-blocks":               ClassStaticBlocks,
+	"class-static-field":                ClassStaticField,
+	"const-and-let":                     ConstAndLet,
+	"decorators":                        Decorators,
+	"default-argument":                  DefaultArgument,
+	"destructuring":                     Destructuring,
+	"dynamic-import":                    DynamicImport,
+	"exponent-operator":                 ExponentOperator,
+	"export-star-as":                    ExportStarAs,
+	"for-await":                         ForAwait,
+	"for-of":                            ForOf,
+	"function-name-configurable":        FunctionNameConfigurable,
+	"function-or-class-property-access": FunctionOrClassPropertyAccess,
+	"generator":                         Generator,
+	"hashbang":                          Hashbang,
+	"import-assertions":                 ImportAssertions,
+	"import-attributes":                 ImportAttributes,
+	"import-meta":                       ImportMeta,
+	"inline-script":                     InlineScript,
+	"logical-assignment":                LogicalAssignment,
+	"nested-rest-binding":               NestedRestBinding,
+	"new-target":                        NewTarget,
+	"node-colon-prefix-import":          NodeColonPrefixImport,
+	"node-colon-prefix-require":         NodeColonPrefixRequire,
+	"nullish-coalescing":                NullishCoalescing,
+	"object-accessors":                  ObjectAccessors,
+	"object-extensions":                 ObjectExtensions,
+	"object-rest-spread":                ObjectRestSpread,
+	"optional-catch-binding":            OptionalCatchBinding,
+	"optional-chain":                    OptionalChain,
+	"regexp-dot-all-flag":               RegexpDotAllFlag,
+	"regexp-lookbehind-assertions":      RegexpLookbehindAssertions,
+	"regexp-match-indices":              RegexpMatchIndices,
+	"regexp-named-capture-groups":       RegexpNamedCaptureGroups,
+	"regexp-set-notation":               RegexpSetNotation,
+	"regexp-sticky-and-unicode-flags":   RegexpStickyAndUnicodeFlags,
+	"regexp-unicode-property-escapes":   RegexpUnicodePropertyEscapes,
+	"rest-argument":                     RestArgument,
+	"template-literal":                  TemplateLiteral,
+	"top-level-await":                   TopLevelAwait,
+	"typeof-exotic-object-is-object":    TypeofExoticObjectIsObject,
+	"unicode-escapes":                   UnicodeEscapes,
+	"using":                             Using,
+}
+
+func (features JSFeature) Has(feature JSFeature) bool {
+	return (features & feature) != 0
+}
+
+func (features JSFeature) ApplyOverrides(overrides JSFeature, mask JSFeature) JSFeature {
+	return (features & ^mask) | (overrides & mask)
+}
+
+var jsTable = map[JSFeature]map[Engine][]versionRange{
+	ArbitraryModuleNamespaceNames: {
+		Chrome:  {{start: v{90, 0, 0}}},
+		ES:      {{start: v{2022, 0, 0}}},
+		Firefox: {{start: v{87, 0, 0}}},
+		IOS:     {{start: v{14, 5, 0}}},
+		Node:    {{start: v{16, 0, 0}}},
+		Safari:  {{start: v{14, 1, 0}}},
+	},
+	ArraySpread: {
+		// Note: The latest version of "IE" failed 15 tests including: spread syntax for iterable objects: spreading non-iterables is a runtime error
+		// Note: The latest version of "Rhino" failed 15 tests including: spread syntax for iterable objects: spreading non-iterables is a runtime error
+		Chrome:  {{start: v{46, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{13, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{36, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{10, 0, 0}}},
+		Node:    {{start: v{5, 0, 0}}},
+		Opera:   {{start: v{33, 0, 0}}},
+		Safari:  {{start: v{10, 0, 0}}},
+	},
+	Arrow: {
+		// Note: The latest version of "Hermes" failed 3 tests including: arrow functions: lexical "super" binding in constructors
+		// Note: The latest version of "IE" failed 13 tests including: arrow functions: "this" unchanged by call or apply
+		// Note: The latest version of "Rhino" failed 3 tests including: arrow functions: lexical "new.target" binding
+		Chrome:  {{start: v{49, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{13, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{45, 0, 0}}},
+		IOS:     {{start: v{10, 0, 0}}},
+		Node:    {{start: v{6, 0, 0}}},
+		Opera:   {{start: v{36, 0, 0}}},
+		Safari:  {{start: v{10, 0, 0}}},
+	},
+	AsyncAwait: {
+		// Note: The latest version of "Hermes" failed 4 tests including: async functions: async arrow functions
+		// Note: The latest version of "IE" failed 16 tests including: async functions: async arrow functions
+		// Note: The latest version of "Rhino" failed 16 tests including: async functions: async arrow functions
+		Chrome:  {{start: v{55, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{15, 0, 0}}},
+		ES:      {{start: v{2017, 0, 0}}},
+		Firefox: {{start: v{52, 0, 0}}},
+		IOS:     {{start: v{11, 0, 0}}},
+		Node:    {{start: v{7, 6, 0}}},
+		Opera:   {{start: v{42, 0, 0}}},
+		Safari:  {{start: v{11, 0, 0}}},
+	},
+	AsyncGenerator: {
+		// Note: The latest version of "Hermes" failed this test: Asynchronous Iterators: async generators
+		// Note: The latest version of "IE" failed this test: Asynchronous Iterators: async generators
+		// Note: The latest version of "Rhino" failed this test: Asynchronous Iterators: async generators
+		Chrome:  {{start: v{63, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2018, 0, 0}}},
+		Firefox: {{start: v{57, 0, 0}}},
+		IOS:     {{start: v{12, 0, 0}}},
+		Node:    {{start: v{10, 0, 0}}},
+		Opera:   {{start: v{50, 0, 0}}},
+		Safari:  {{start: v{12, 0, 0}}},
+	},
+	Bigint: {
+		// Note: The latest version of "IE" failed this test: BigInt: basic functionality
+		Chrome:  {{start: v{67, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2020, 0, 0}}},
+		Firefox: {{start: v{68, 0, 0}}},
+		Hermes:  {{start: v{0, 12, 0}}},
+		IOS:     {{start: v{14, 0, 0}}},
+		Node:    {{start: v{10, 4, 0}}},
+		Opera:   {{start: v{54, 0, 0}}},
+		Rhino:   {{start: v{1, 7, 14}}},
+		Safari:  {{start: v{14, 0, 0}}},
+	},
+	Class: {
+		// Note: The latest version of "Hermes" failed 24 tests including: class: accessor properties
+		// Note: The latest version of "IE" failed 24 tests including: class: accessor properties
+		// Note: The latest version of "Rhino" failed 24 tests including: class: accessor properties
+		Chrome:  {{start: v{49, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{13, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{45, 0, 0}}},
+		IOS:     {{start: v{10, 0, 0}}},
+		Node:    {{start: v{6, 0, 0}}},
+		Opera:   {{start: v{36, 0, 0}}},
+		Safari:  {{start: v{10, 0, 0}}},
+	},
+	ClassField: {
+		// Note: The latest version of "Hermes" failed 2 tests including: instance class fields: computed instance class fields
+		// Note: The latest version of "IE" failed 2 tests including: instance class fields: computed instance class fields
+		// Note: The latest version of "Rhino" failed 2 tests including: instance class fields: computed instance class fields
+		Chrome:  {{start: v{73, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2022, 0, 0}}},
+		Firefox: {{start: v{69, 0, 0}}},
+		IOS:     {{start: v{14, 0, 0}}},
+		Node:    {{start: v{12, 0, 0}}},
+		Opera:   {{start: v{60, 0, 0}}},
+		Safari:  {{start: v{14, 0, 0}}},
+	},
+	ClassPrivateAccessor: {
+		// Note: The latest version of "Hermes" failed this test: private class methods: private accessor properties
+		// Note: The latest version of "IE" failed this test: private class methods: private accessor properties
+		// Note: The latest version of "Rhino" failed this test: private class methods: private accessor properties
+		Chrome:  {{start: v{84, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{84, 0, 0}}},
+		ES:      {{start: v{2022, 0, 0}}},
+		Firefox: {{start: v{90, 0, 0}}},
+		IOS:     {{start: v{15, 0, 0}}},
+		Node:    {{start: v{14, 6, 0}}},
+		Opera:   {{start: v{70, 0, 0}}},
+		Safari:  {{start: v{15, 0, 0}}},
+	},
+	ClassPrivateBrandCheck: {
+		// Note: The latest version of "Hermes" failed this test: Ergonomic brand checks for private fields
+		// Note: The latest version of "IE" failed this test: Ergonomic brand checks for private fields
+		// Note: The latest version of "Rhino" failed this test: Ergonomic brand checks for private fields
+		Chrome:  {{start: v{91, 0, 0}}},
+		Deno:    {{start: v{1, 9, 0}}},
+		Edge:    {{start: v{91, 0, 0}}},
+		ES:      {{start: v{2022, 0, 0}}},
+		Firefox: {{start: v{90, 0, 0}}},
+		IOS:     {{start: v{15, 0, 0}}},
+		Node:    {{start: v{16, 4, 0}}},
+		Opera:   {{start: v{77, 0, 0}}},
+		Safari:  {{start: v{15, 0, 0}}},
+	},
+	ClassPrivateField: {
+		// Note: The latest version of "Hermes" failed 4 tests including: instance class fields: optional deep private instance class fields access
+		// Note: The latest version of "IE" failed 4 tests including: instance class fields: optional deep private instance class fields access
+		// Note: The latest version of "Rhino" failed 4 tests including: instance class fields: optional deep private instance class fields access
+		Chrome:  {{start: v{84, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{84, 0, 0}}},
+		ES:      {{start: v{2022, 0, 0}}},
+		Firefox: {{start: v{90, 0, 0}}},
+		IOS:     {{start: v{14, 5, 0}}},
+		Node:    {{start: v{14, 6, 0}}},
+		Opera:   {{start: v{70, 0, 0}}},
+		Safari:  {{start: v{14, 1, 0}}},
+	},
+	ClassPrivateMethod: {
+		// Note: The latest version of "Hermes" failed this test: private class methods: private instance methods
+		// Note: The latest version of "IE" failed this test: private class methods: private instance methods
+		// Note: The latest version of "Rhino" failed this test: private class methods: private instance methods
+		Chrome:  {{start: v{84, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{84, 0, 0}}},
+		ES:      {{start: v{2022, 0, 0}}},
+		Firefox: {{start: v{90, 0, 0}}},
+		IOS:     {{start: v{15, 0, 0}}},
+		Node:    {{start: v{14, 6, 0}}},
+		Opera:   {{start: v{70, 0, 0}}},
+		Safari:  {{start: v{15, 0, 0}}},
+	},
+	ClassPrivateStaticAccessor: {
+		// Note: The latest version of "Hermes" failed this test: private class methods: private static accessor properties
+		// Note: The latest version of "IE" failed this test: private class methods: private static accessor properties
+		// Note: The latest version of "Rhino" failed this test: private class methods: private static accessor properties
+		Chrome:  {{start: v{84, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{84, 0, 0}}},
+		ES:      {{start: v{2022, 0, 0}}},
+		Firefox: {{start: v{90, 0, 0}}},
+		IOS:     {{start: v{15, 0, 0}}},
+		Node:    {{start: v{14, 6, 0}}},
+		Opera:   {{start: v{70, 0, 0}}},
+		Safari:  {{start: v{15, 0, 0}}},
+	},
+	ClassPrivateStaticField: {
+		// Note: The latest version of "Hermes" failed this test: static class fields: private static class fields
+		// Note: The latest version of "IE" failed this test: static class fields: private static class fields
+		// Note: The latest version of "Rhino" failed this test: static class fields: private static class fields
+		Chrome:  {{start: v{74, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2022, 0, 0}}},
+		Firefox: {{start: v{90, 0, 0}}},
+		IOS:     {{start: v{14, 5, 0}}},
+		Node:    {{start: v{12, 0, 0}}},
+		Opera:   {{start: v{62, 0, 0}}},
+		Safari:  {{start: v{14, 1, 0}}},
+	},
+	ClassPrivateStaticMethod: {
+		// Note: The latest version of "Hermes" failed this test: private class methods: private static methods
+		// Note: The latest version of "IE" failed this test: private class methods: private static methods
+		// Note: The latest version of "Rhino" failed this test: private class methods: private static methods
+		Chrome:  {{start: v{84, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{84, 0, 0}}},
+		ES:      {{start: v{2022, 0, 0}}},
+		Firefox: {{start: v{90, 0, 0}}},
+		IOS:     {{start: v{15, 0, 0}}},
+		Node:    {{start: v{14, 6, 0}}},
+		Opera:   {{start: v{70, 0, 0}}},
+		Safari:  {{start: v{15, 0, 0}}},
+	},
+	ClassStaticBlocks: {
+		Chrome:  {{start: v{91, 0, 0}}},
+		Deno:    {{start: v{1, 14, 0}}},
+		Edge:    {{start: v{94, 0, 0}}},
+		ES:      {{start: v{2022, 0, 0}}},
+		Firefox: {{start: v{93, 0, 0}}},
+		IOS:     {{start: v{16, 4, 0}}},
+		Node:    {{start: v{16, 11, 0}}},
+		Opera:   {{start: v{80, 0, 0}}},
+		Safari:  {{start: v{16, 4, 0}}},
+	},
+	ClassStaticField: {
+		// Note: The latest version of "Hermes" failed 2 tests including: static class fields: computed static class fields
+		// Note: The latest version of "IE" failed 2 tests including: static class fields: computed static class fields
+		// Note: The latest version of "Rhino" failed 2 tests including: static class fields: computed static class fields
+		Chrome:  {{start: v{73, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2022, 0, 0}}},
+		Firefox: {{start: v{75, 0, 0}}},
+		IOS:     {{start: v{14, 5, 0}}},
+		Node:    {{start: v{12, 0, 0}}},
+		Opera:   {{start: v{60, 0, 0}}},
+		Safari:  {{start: v{14, 1, 0}}},
+	},
+	ConstAndLet: {
+		// Note: The latest version of "Hermes" failed 20 tests including: const: for loop statement scope
+		// Note: The latest version of "IE" failed 6 tests including: const: for-in loop iteration scope
+		// Note: The latest version of "Rhino" failed 22 tests including: const: cannot be in statements
+		Chrome:  {{start: v{49, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{14, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{51, 0, 0}}},
+		IOS:     {{start: v{11, 0, 0}}},
+		Node:    {{start: v{6, 0, 0}}},
+		Opera:   {{start: v{36, 0, 0}}},
+		Safari:  {{start: v{11, 0, 0}}},
+	},
+	Decorators: {},
+	DefaultArgument: {
+		// Note: The latest version of "Hermes" failed 2 tests including: default function parameters: separate scope
+		// Note: The latest version of "IE" failed 7 tests including: default function parameters: arguments object interaction
+		// Note: The latest version of "Rhino" failed 7 tests including: default function parameters: arguments object interaction
+		Chrome:  {{start: v{49, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{14, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{53, 0, 0}}},
+		IOS:     {{start: v{10, 0, 0}}},
+		Node:    {{start: v{6, 0, 0}}},
+		Opera:   {{start: v{36, 0, 0}}},
+		Safari:  {{start: v{10, 0, 0}}},
+	},
+	Destructuring: {
+		// Note: The latest version of "Hermes" failed 3 tests including: destructuring, declarations: defaults, let temporal dead zone
+		// Note: The latest version of "IE" failed 71 tests including: destructuring, assignment: chained iterable destructuring
+		// Note: The latest version of "Rhino" failed 33 tests including: destructuring, assignment: computed properties
+		Chrome:  {{start: v{51, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{18, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{53, 0, 0}}},
+		IOS:     {{start: v{10, 0, 0}}},
+		Node:    {{start: v{6, 5, 0}}},
+		Opera:   {{start: v{38, 0, 0}}},
+		Safari:  {{start: v{10, 0, 0}}},
+	},
+	DynamicImport: {
+		Chrome:  {{start: v{63, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{67, 0, 0}}},
+		IOS:     {{start: v{11, 0, 0}}},
+		Node:    {{start: v{12, 20, 0}, end: v{13, 0, 0}}, {start: v{13, 2, 0}}},
+		Opera:   {{start: v{50, 0, 0}}},
+		Safari:  {{start: v{11, 1, 0}}},
+	},
+	ExponentOperator: {
+		// Note: The latest version of "IE" failed 3 tests including: exponentiation (**) operator: assignment
+		Chrome:  {{start: v{52, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{14, 0, 0}}},
+		ES:      {{start: v{2016, 0, 0}}},
+		Firefox: {{start: v{52, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{10, 3, 0}}},
+		Node:    {{start: v{7, 0, 0}}},
+		Opera:   {{start: v{39, 0, 0}}},
+		Rhino:   {{start: v{1, 7, 14}}},
+		Safari:  {{start: v{10, 1, 0}}},
+	},
+	ExportStarAs: {
+		Chrome:  {{start: v{72, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2020, 0, 0}}},
+		Firefox: {{start: v{80, 0, 0}}},
+		IOS:     {{start: v{14, 5, 0}}},
+		Node:    {{start: v{13, 2, 0}}},
+		Opera:   {{start: v{60, 0, 0}}},
+		Safari:  {{start: v{14, 1, 0}}},
+	},
+	ForAwait: {
+		// Note: The latest version of "Hermes" failed this test: Asynchronous Iterators: for-await-of loops
+		// Note: The latest version of "IE" failed this test: Asynchronous Iterators: for-await-of loops
+		// Note: The latest version of "Rhino" failed this test: Asynchronous Iterators: for-await-of loops
+		Chrome:  {{start: v{63, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2018, 0, 0}}},
+		Firefox: {{start: v{57, 0, 0}}},
+		IOS:     {{start: v{12, 0, 0}}},
+		Node:    {{start: v{10, 0, 0}}},
+		Opera:   {{start: v{50, 0, 0}}},
+		Safari:  {{start: v{12, 0, 0}}},
+	},
+	ForOf: {
+		// Note: The latest version of "IE" failed 9 tests including: for..of loops: iterator closing, break
+		// Note: The latest version of "Rhino" failed 2 tests including: for..of loops: iterator closing, break
+		Chrome:  {{start: v{51, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{15, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{53, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{10, 0, 0}}},
+		Node:    {{start: v{6, 5, 0}}},
+		Opera:   {{start: v{38, 0, 0}}},
+		Safari:  {{start: v{10, 0, 0}}},
+	},
+	FunctionNameConfigurable: {
+		// Note: The latest version of "IE" failed this test: function "name" property: isn't writable, is configurable
+		Chrome:  {{start: v{43, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{12, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{38, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{10, 0, 0}}},
+		Node:    {{start: v{4, 0, 0}}},
+		Opera:   {{start: v{30, 0, 0}}},
+		Rhino:   {{start: v{1, 7, 15}}},
+		Safari:  {{start: v{10, 0, 0}}},
+	},
+	FunctionOrClassPropertyAccess: {
+		Chrome:  {{start: v{0, 0, 0}}},
+		Deno:    {{start: v{0, 0, 0}}},
+		Edge:    {{start: v{0, 0, 0}}},
+		ES:      {{start: v{0, 0, 0}}},
+		Firefox: {{start: v{0, 0, 0}}},
+		Hermes:  {{start: v{0, 0, 0}}},
+		IE:      {{start: v{0, 0, 0}}},
+		IOS:     {{start: v{0, 0, 0}}},
+		Node:    {{start: v{0, 0, 0}}},
+		Opera:   {{start: v{0, 0, 0}}},
+		Rhino:   {{start: v{0, 0, 0}}},
+		Safari:  {{start: v{16, 3, 0}}},
+	},
+	Generator: {
+		// Note: The latest version of "Hermes" failed 3 tests including: generators: computed shorthand generators, classes
+		// Note: The latest version of "IE" failed 27 tests including: generators: %GeneratorPrototype%
+		// Note: The latest version of "Rhino" failed 11 tests including: generators: %GeneratorPrototype%
+		Chrome:  {{start: v{50, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{13, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{53, 0, 0}}},
+		IOS:     {{start: v{10, 0, 0}}},
+		Node:    {{start: v{6, 0, 0}}},
+		Opera:   {{start: v{37, 0, 0}}},
+		Safari:  {{start: v{10, 0, 0}}},
+	},
+	Hashbang: {
+		// Note: The latest version of "IE" failed this test: Hashbang Grammar
+		Chrome:  {{start: v{74, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2023, 0, 0}}},
+		Firefox: {{start: v{67, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{13, 4, 0}}},
+		Node:    {{start: v{12, 5, 0}}},
+		Opera:   {{start: v{62, 0, 0}}},
+		Rhino:   {{start: v{1, 7, 15}}},
+		Safari:  {{start: v{13, 1, 0}}},
+	},
+	ImportAssertions: {
+		Chrome: {{start: v{91, 0, 0}}},
+		Deno:   {{start: v{1, 17, 0}}},
+		Edge:   {{start: v{91, 0, 0}}},
+		Node:   {{start: v{16, 14, 0}, end: v{22, 0, 0}}},
+	},
+	ImportAttributes: {
+		Chrome: {{start: v{123, 0, 0}}},
+		Deno:   {{start: v{1, 37, 0}}},
+		Edge:   {{start: v{123, 0, 0}}},
+		IOS:    {{start: v{17, 2, 0}}},
+		Node:   {{start: v{18, 20, 0}, end: v{19, 0, 0}}, {start: v{20, 10, 0}}},
+		Opera:  {{start: v{109, 0, 0}}},
+		Safari: {{start: v{17, 2, 0}}},
+	},
+	ImportMeta: {
+		Chrome:  {{start: v{64, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2020, 0, 0}}},
+		Firefox: {{start: v{62, 0, 0}}},
+		IOS:     {{start: v{12, 0, 0}}},
+		Node:    {{start: v{10, 4, 0}}},
+		Opera:   {{start: v{51, 0, 0}}},
+		Safari:  {{start: v{11, 1, 0}}},
+	},
+	InlineScript: {},
+	LogicalAssignment: {
+		// Note: The latest version of "IE" failed 9 tests including: Logical Assignment: &&= basic support
+		// Note: The latest version of "Rhino" failed 9 tests including: Logical Assignment: &&= basic support
+		Chrome:  {{start: v{85, 0, 0}}},
+		Deno:    {{start: v{1, 2, 0}}},
+		Edge:    {{start: v{85, 0, 0}}},
+		ES:      {{start: v{2021, 0, 0}}},
+		Firefox: {{start: v{79, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{14, 0, 0}}},
+		Node:    {{start: v{15, 0, 0}}},
+		Opera:   {{start: v{71, 0, 0}}},
+		Safari:  {{start: v{14, 0, 0}}},
+	},
+	NestedRestBinding: {
+		// Note: The latest version of "IE" failed 2 tests including: nested rest destructuring, declarations
+		// Note: The latest version of "Rhino" failed 2 tests including: nested rest destructuring, declarations
+		Chrome:  {{start: v{49, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{14, 0, 0}}},
+		ES:      {{start: v{2016, 0, 0}}},
+		Firefox: {{start: v{47, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{10, 3, 0}}},
+		Node:    {{start: v{6, 0, 0}}},
+		Opera:   {{start: v{36, 0, 0}}},
+		Safari:  {{start: v{10, 1, 0}}},
+	},
+	NewTarget: {
+		// Note: The latest version of "IE" failed 2 tests including: new.target: assignment is an early error
+		// Note: The latest version of "Rhino" failed 2 tests including: new.target: assignment is an early error
+		Chrome:  {{start: v{46, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{14, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{41, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{10, 0, 0}}},
+		Node:    {{start: v{5, 0, 0}}},
+		Opera:   {{start: v{33, 0, 0}}},
+		Safari:  {{start: v{10, 0, 0}}},
+	},
+	NodeColonPrefixImport: {
+		ES:   {{start: v{0, 0, 0}}},
+		Node: {{start: v{12, 20, 0}, end: v{13, 0, 0}}, {start: v{14, 13, 1}}},
+	},
+	NodeColonPrefixRequire: {
+		ES:   {{start: v{0, 0, 0}}},
+		Node: {{start: v{14, 18, 0}, end: v{15, 0, 0}}, {start: v{16, 0, 0}}},
+	},
+	NullishCoalescing: {
+		// Note: The latest version of "IE" failed this test: nullish coalescing operator (??)
+		// Note: The latest version of "Rhino" failed this test: nullish coalescing operator (??)
+		Chrome:  {{start: v{80, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{80, 0, 0}}},
+		ES:      {{start: v{2020, 0, 0}}},
+		Firefox: {{start: v{72, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{13, 4, 0}}},
+		Node:    {{start: v{14, 0, 0}}},
+		Opera:   {{start: v{67, 0, 0}}},
+		Safari:  {{start: v{13, 1, 0}}},
+	},
+	ObjectAccessors: {
+		Chrome:  {{start: v{5, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{12, 0, 0}}},
+		ES:      {{start: v{5, 0, 0}}},
+		Firefox: {{start: v{2, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IE:      {{start: v{9, 0, 0}}},
+		IOS:     {{start: v{6, 0, 0}}},
+		Node:    {{start: v{0, 4, 0}}},
+		Opera:   {{start: v{10, 10, 0}}},
+		Rhino:   {{start: v{1, 7, 13}}},
+		Safari:  {{start: v{3, 1, 0}}},
+	},
+	ObjectExtensions: {
+		// Note: The latest version of "IE" failed 6 tests including: object literal extensions: computed accessors
+		// Note: The latest version of "Rhino" failed 3 tests including: object literal extensions: computed accessors
+		Chrome:  {{start: v{44, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{12, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{34, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{10, 0, 0}}},
+		Node:    {{start: v{4, 0, 0}}},
+		Opera:   {{start: v{31, 0, 0}}},
+		Safari:  {{start: v{10, 0, 0}}},
+	},
+	ObjectRestSpread: {
+		// Note: The latest version of "IE" failed 2 tests including: object rest/spread properties: object rest properties
+		// Note: The latest version of "Rhino" failed 2 tests including: object rest/spread properties: object rest properties
+		Chrome:  {{start: v{60, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2018, 0, 0}}},
+		Firefox: {{start: v{55, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{11, 3, 0}}},
+		Node:    {{start: v{8, 3, 0}}},
+		Opera:   {{start: v{47, 0, 0}}},
+		Safari:  {{start: v{11, 1, 0}}},
+	},
+	OptionalCatchBinding: {
+		// Note: The latest version of "IE" failed 3 tests including: optional catch binding: await
+		// Note: The latest version of "Rhino" failed this test: optional catch binding: await
+		Chrome:  {{start: v{66, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2019, 0, 0}}},
+		Firefox: {{start: v{58, 0, 0}}},
+		Hermes:  {{start: v{0, 12, 0}}},
+		IOS:     {{start: v{11, 3, 0}}},
+		Node:    {{start: v{10, 0, 0}}},
+		Opera:   {{start: v{53, 0, 0}}},
+		Safari:  {{start: v{11, 1, 0}}},
+	},
+	OptionalChain: {
+		// Note: The latest version of "IE" failed 5 tests including: optional chaining operator (?.): optional bracket access
+		// Note: The latest version of "Rhino" failed 5 tests including: optional chaining operator (?.): optional bracket access
+		Chrome:  {{start: v{91, 0, 0}}},
+		Deno:    {{start: v{1, 9, 0}}},
+		Edge:    {{start: v{91, 0, 0}}},
+		ES:      {{start: v{2020, 0, 0}}},
+		Firefox: {{start: v{74, 0, 0}}},
+		Hermes:  {{start: v{0, 12, 0}}},
+		IOS:     {{start: v{13, 4, 0}}},
+		Node:    {{start: v{16, 1, 0}}},
+		Opera:   {{start: v{77, 0, 0}}},
+		Safari:  {{start: v{13, 1, 0}}},
+	},
+	RegexpDotAllFlag: {
+		// Note: The latest version of "IE" failed this test: s (dotAll) flag for regular expressions
+		Chrome:  {{start: v{62, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2018, 0, 0}}},
+		Firefox: {{start: v{78, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{11, 3, 0}}},
+		Node:    {{start: v{8, 10, 0}}},
+		Opera:   {{start: v{49, 0, 0}}},
+		Rhino:   {{start: v{1, 7, 15}}},
+		Safari:  {{start: v{11, 1, 0}}},
+	},
+	RegexpLookbehindAssertions: {
+		// Note: The latest version of "IE" failed this test: RegExp Lookbehind Assertions
+		// Note: The latest version of "Rhino" failed this test: RegExp Lookbehind Assertions
+		Chrome:  {{start: v{62, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2018, 0, 0}}},
+		Firefox: {{start: v{78, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{16, 4, 0}}},
+		Node:    {{start: v{8, 10, 0}}},
+		Opera:   {{start: v{49, 0, 0}}},
+		Safari:  {{start: v{16, 4, 0}}},
+	},
+	RegexpMatchIndices: {
+		Chrome:  {{start: v{90, 0, 0}}},
+		Deno:    {{start: v{1, 8, 0}}},
+		Edge:    {{start: v{90, 0, 0}}},
+		ES:      {{start: v{2022, 0, 0}}},
+		Firefox: {{start: v{88, 0, 0}}},
+		IOS:     {{start: v{15, 0, 0}}},
+		Node:    {{start: v{16, 0, 0}}},
+		Opera:   {{start: v{76, 0, 0}}},
+		Safari:  {{start: v{15, 0, 0}}},
+	},
+	RegexpNamedCaptureGroups: {
+		// Note: The latest version of "Hermes" failed this test: RegExp named capture groups
+		// Note: The latest version of "IE" failed this test: RegExp named capture groups
+		// Note: The latest version of "Rhino" failed this test: RegExp named capture groups
+		Chrome:  {{start: v{64, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{79, 0, 0}}},
+		ES:      {{start: v{2018, 0, 0}}},
+		Firefox: {{start: v{78, 0, 0}}},
+		IOS:     {{start: v{11, 3, 0}}},
+		Node:    {{start: v{10, 0, 0}}},
+		Opera:   {{start: v{51, 0, 0}}},
+		Safari:  {{start: v{11, 1, 0}}},
+	},
+	RegexpSetNotation: {
+		ES: {{start: v{2024, 0, 0}}},
+	},
+	RegexpStickyAndUnicodeFlags: {
+		// Note: The latest version of "IE" failed 6 tests including: RegExp "y" and "u" flags: "u" flag
+		// Note: The latest version of "Rhino" failed 4 tests including: RegExp "y" and "u" flags: "u" flag
+		Chrome:  {{start: v{50, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{13, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{46, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{12, 0, 0}}},
+		Node:    {{start: v{6, 0, 0}}},
+		Opera:   {{start: v{37, 0, 0}}},
+		Safari:  {{start: v{12, 0, 0}}},
+	},
+	RegexpUnicodePropertyEscapes: {
+		// Note: The latest version of "Chrome" failed this test: RegExp Unicode Property Escapes: Unicode 16.0
+		// Note: The latest version of "Edge" failed this test: RegExp Unicode Property Escapes: Unicode 16.0
+		// Note: The latest version of "Firefox" failed 2 tests including: RegExp Unicode Property Escapes: Unicode 15.1
+		// Note: The latest version of "Hermes" failed 8 tests including: RegExp Unicode Property Escapes: Unicode 11
+		// Note: The latest version of "IE" failed 8 tests including: RegExp Unicode Property Escapes: Unicode 11
+		// Note: The latest version of "IOS" failed this test: RegExp Unicode Property Escapes: Unicode 16.0
+		// Note: The latest version of "Node" failed this test: RegExp Unicode Property Escapes: Unicode 16.0
+		// Note: The latest version of "Rhino" failed 8 tests including: RegExp Unicode Property Escapes: Unicode 11
+		// Note: The latest version of "Safari" failed this test: RegExp Unicode Property Escapes: Unicode 16.0
+		ES:    {{start: v{2018, 0, 0}}},
+		Opera: {{start: v{111, 0, 0}}},
+	},
+	RestArgument: {
+		// Note: The latest version of "Hermes" failed this test: rest parameters: function 'length' property
+		// Note: The latest version of "IE" failed 5 tests including: rest parameters: arguments object interaction
+		// Note: The latest version of "Rhino" failed 2 tests including: rest parameters: arguments object interaction
+		Chrome:  {{start: v{47, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{12, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{43, 0, 0}}},
+		IOS:     {{start: v{10, 0, 0}}},
+		Node:    {{start: v{6, 0, 0}}},
+		Opera:   {{start: v{34, 0, 0}}},
+		Safari:  {{start: v{10, 0, 0}}},
+	},
+	TemplateLiteral: {
+		// Note: The latest version of "Hermes" failed this test: template literals: TemplateStrings call site caching
+		// Note: The latest version of "IE" failed 7 tests including: template literals: TemplateStrings call site caching
+		// Note: The latest version of "Rhino" failed this test: template literals: toString conversion
+		Chrome:  {{start: v{41, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{13, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{34, 0, 0}}},
+		IOS:     {{start: v{13, 0, 0}}},
+		Node:    {{start: v{10, 0, 0}}},
+		Opera:   {{start: v{28, 0, 0}}},
+		Safari:  {{start: v{13, 0, 0}}},
+	},
+	TopLevelAwait: {
+		Chrome:  {{start: v{89, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{89, 0, 0}}},
+		ES:      {{start: v{2022, 0, 0}}},
+		Firefox: {{start: v{89, 0, 0}}},
+		IOS:     {{start: v{15, 0, 0}}},
+		Node:    {{start: v{14, 8, 0}}},
+		Opera:   {{start: v{75, 0, 0}}},
+		Safari:  {{start: v{15, 0, 0}}},
+	},
+	TypeofExoticObjectIsObject: {
+		Chrome:  {{start: v{0, 0, 0}}},
+		Deno:    {{start: v{0, 0, 0}}},
+		Edge:    {{start: v{0, 0, 0}}},
+		ES:      {{start: v{2020, 0, 0}}},
+		Firefox: {{start: v{0, 0, 0}}},
+		Hermes:  {{start: v{0, 0, 0}}},
+		IOS:     {{start: v{0, 0, 0}}},
+		Node:    {{start: v{0, 0, 0}}},
+		Opera:   {{start: v{0, 0, 0}}},
+		Rhino:   {{start: v{0, 0, 0}}},
+		Safari:  {{start: v{0, 0, 0}}},
+	},
+	UnicodeEscapes: {
+		// Note: The latest version of "IE" failed 2 tests including: Unicode code point escapes: in identifiers
+		Chrome:  {{start: v{44, 0, 0}}},
+		Deno:    {{start: v{1, 0, 0}}},
+		Edge:    {{start: v{12, 0, 0}}},
+		ES:      {{start: v{2015, 0, 0}}},
+		Firefox: {{start: v{53, 0, 0}}},
+		Hermes:  {{start: v{0, 7, 0}}},
+		IOS:     {{start: v{9, 0, 0}}},
+		Node:    {{start: v{4, 0, 0}}},
+		Opera:   {{start: v{31, 0, 0}}},
+		Rhino:   {{start: v{1, 7, 15}}},
+		Safari:  {{start: v{9, 0, 0}}},
+	},
+	Using: {},
+}
+
+// Return all features that are not available in at least one environment
+func UnsupportedJSFeatures(constraints map[Engine]Semver) (unsupported JSFeature) {
+	for feature, engines := range jsTable {
+		if feature == InlineScript {
+			continue // This is purely user-specified
+		}
+		for engine, version := range constraints {
+			if versionRanges, ok := engines[engine]; !ok || !isVersionSupported(versionRanges, version) {
+				unsupported |= feature
+			}
+		}
+	}
+	return
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/config/config.go b/source/vendor/github.com/evanw/esbuild/internal/config/config.go
new file mode 100644
index 0000000..615d688
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/config/config.go
@@ -0,0 +1,842 @@
+package config
+
+import (
+	"fmt"
+	"regexp"
+	"strings"
+	"sync"
+	"sync/atomic"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type JSXOptions struct {
+	Factory          DefineExpr
+	Fragment         DefineExpr
+	Parse            bool
+	Preserve         bool
+	AutomaticRuntime bool
+	ImportSource     string
+	Development      bool
+	SideEffects      bool
+}
+
+type TSJSX uint8
+
+const (
+	TSJSXNone TSJSX = iota
+	TSJSXPreserve
+	TSJSXReactNative
+	TSJSXReact
+	TSJSXReactJSX
+	TSJSXReactJSXDev
+)
+
+type TSOptions struct {
+	Config              TSConfig
+	Parse               bool
+	NoAmbiguousLessThan bool
+}
+
+type TSConfigJSX struct {
+	// If not empty, these should override the default values
+	JSXFactory         []string // Default if empty: "React.createElement"
+	JSXFragmentFactory []string // Default if empty: "React.Fragment"
+	JSXImportSource    *string  // Default if empty: "react"
+	JSX                TSJSX
+}
+
+// This is used for "extends" in "tsconfig.json"
+func (derived *TSConfigJSX) ApplyExtendedConfig(base TSConfigJSX) {
+	if base.JSXFactory != nil {
+		derived.JSXFactory = base.JSXFactory
+	}
+	if base.JSXFragmentFactory != nil {
+		derived.JSXFragmentFactory = base.JSXFragmentFactory
+	}
+	if base.JSXImportSource != nil {
+		derived.JSXImportSource = base.JSXImportSource
+	}
+	if base.JSX != TSJSXNone {
+		derived.JSX = base.JSX
+	}
+}
+
+func (tsConfig *TSConfigJSX) ApplyTo(jsxOptions *JSXOptions) {
+	switch tsConfig.JSX {
+	case TSJSXPreserve, TSJSXReactNative:
+		// Deliberately don't set "Preserve = true" here. Some tools from Vercel
+		// apparently automatically set "jsx": "preserve" in "tsconfig.json" and
+		// people are then confused when esbuild preserves their JSX. Ignoring this
+		// value means you now have to explicitly pass "--jsx=preserve" to esbuild
+		// to get this behavior.
+
+	case TSJSXReact:
+		jsxOptions.AutomaticRuntime = false
+		jsxOptions.Development = false
+
+	case TSJSXReactJSX:
+		jsxOptions.AutomaticRuntime = true
+		// Deliberately don't set "Development = false" here. People want to be
+		// able to have "react-jsx" in their "tsconfig.json" file and then swap
+		// that to "react-jsxdev" by passing "--jsx-dev" to esbuild.
+
+	case TSJSXReactJSXDev:
+		jsxOptions.AutomaticRuntime = true
+		jsxOptions.Development = true
+	}
+
+	if len(tsConfig.JSXFactory) > 0 {
+		jsxOptions.Factory = DefineExpr{Parts: tsConfig.JSXFactory}
+	}
+
+	if len(tsConfig.JSXFragmentFactory) > 0 {
+		jsxOptions.Fragment = DefineExpr{Parts: tsConfig.JSXFragmentFactory}
+	}
+
+	if tsConfig.JSXImportSource != nil {
+		jsxOptions.ImportSource = *tsConfig.JSXImportSource
+	}
+}
+
+// Note: This can currently only contain primitive values. It's compared
+// for equality using a structural equality comparison by the JS parser.
+type TSConfig struct {
+	ExperimentalDecorators  MaybeBool
+	ImportsNotUsedAsValues  TSImportsNotUsedAsValues
+	PreserveValueImports    MaybeBool
+	Target                  TSTarget
+	UseDefineForClassFields MaybeBool
+	VerbatimModuleSyntax    MaybeBool
+}
+
+// This is used for "extends" in "tsconfig.json"
+func (derived *TSConfig) ApplyExtendedConfig(base TSConfig) {
+	if base.ExperimentalDecorators != Unspecified {
+		derived.ExperimentalDecorators = base.ExperimentalDecorators
+	}
+	if base.ImportsNotUsedAsValues != TSImportsNotUsedAsValues_None {
+		derived.ImportsNotUsedAsValues = base.ImportsNotUsedAsValues
+	}
+	if base.PreserveValueImports != Unspecified {
+		derived.PreserveValueImports = base.PreserveValueImports
+	}
+	if base.Target != TSTargetUnspecified {
+		derived.Target = base.Target
+	}
+	if base.UseDefineForClassFields != Unspecified {
+		derived.UseDefineForClassFields = base.UseDefineForClassFields
+	}
+	if base.VerbatimModuleSyntax != Unspecified {
+		derived.VerbatimModuleSyntax = base.VerbatimModuleSyntax
+	}
+}
+
+func (cfg *TSConfig) UnusedImportFlags() (flags TSUnusedImportFlags) {
+	if cfg.VerbatimModuleSyntax == True {
+		return TSUnusedImport_KeepStmt | TSUnusedImport_KeepValues
+	}
+	if cfg.PreserveValueImports == True {
+		flags |= TSUnusedImport_KeepValues
+	}
+	if cfg.ImportsNotUsedAsValues == TSImportsNotUsedAsValues_Preserve || cfg.ImportsNotUsedAsValues == TSImportsNotUsedAsValues_Error {
+		flags |= TSUnusedImport_KeepStmt
+	}
+	return
+}
+
+type Platform uint8
+
+const (
+	PlatformBrowser Platform = iota
+	PlatformNode
+	PlatformNeutral
+)
+
+type SourceMap uint8
+
+const (
+	SourceMapNone SourceMap = iota
+	SourceMapInline
+	SourceMapLinkedWithComment
+	SourceMapExternalWithoutComment
+	SourceMapInlineAndExternal
+)
+
+type LegalComments uint8
+
+const (
+	LegalCommentsInline LegalComments = iota
+	LegalCommentsNone
+	LegalCommentsEndOfFile
+	LegalCommentsLinkedWithComment
+	LegalCommentsExternalWithoutComment
+)
+
+func (lc LegalComments) HasExternalFile() bool {
+	return lc == LegalCommentsLinkedWithComment || lc == LegalCommentsExternalWithoutComment
+}
+
+type Loader uint8
+
+const (
+	LoaderNone Loader = iota
+	LoaderBase64
+	LoaderBinary
+	LoaderCopy
+	LoaderCSS
+	LoaderDataURL
+	LoaderDefault
+	LoaderEmpty
+	LoaderFile
+	LoaderGlobalCSS
+	LoaderJS
+	LoaderJSON
+	LoaderWithTypeJSON // Has a "with { type: 'json' }" attribute
+	LoaderJSX
+	LoaderLocalCSS
+	LoaderText
+	LoaderTS
+	LoaderTSNoAmbiguousLessThan // Used with ".mts" and ".cts"
+	LoaderTSX
+)
+
+var LoaderToString = []string{
+	"none",
+	"base64",
+	"binary",
+	"copy",
+	"css",
+	"dataurl",
+	"default",
+	"empty",
+	"file",
+	"global-css",
+	"js",
+	"json",
+	"json",
+	"jsx",
+	"local-css",
+	"text",
+	"ts",
+	"ts",
+	"tsx",
+}
+
+func (loader Loader) IsTypeScript() bool {
+	switch loader {
+	case LoaderTS, LoaderTSNoAmbiguousLessThan, LoaderTSX:
+		return true
+	}
+	return false
+}
+
+func (loader Loader) IsCSS() bool {
+	switch loader {
+	case
+		LoaderCSS, LoaderGlobalCSS, LoaderLocalCSS:
+		return true
+	}
+	return false
+}
+
+func (loader Loader) CanHaveSourceMap() bool {
+	switch loader {
+	case
+		LoaderJS, LoaderJSX,
+		LoaderTS, LoaderTSNoAmbiguousLessThan, LoaderTSX,
+		LoaderCSS, LoaderGlobalCSS, LoaderLocalCSS,
+		LoaderJSON, LoaderWithTypeJSON, LoaderText:
+		return true
+	}
+	return false
+}
+
+type Format uint8
+
+const (
+	// This is used when not bundling. It means to preserve whatever form the
+	// import or export was originally in. ES6 syntax stays ES6 syntax and
+	// CommonJS syntax stays CommonJS syntax.
+	FormatPreserve Format = iota
+
+	// IIFE stands for immediately-invoked function expression. That looks like
+	// this:
+	//
+	//   (() => {
+	//     ... bundled code ...
+	//   })();
+	//
+	// If the optional GlobalName is configured, then we'll write out this:
+	//
+	//   let globalName = (() => {
+	//     ... bundled code ...
+	//     return exports;
+	//   })();
+	//
+	FormatIIFE
+
+	// The CommonJS format looks like this:
+	//
+	//   ... bundled code ...
+	//   module.exports = exports;
+	//
+	FormatCommonJS
+
+	// The ES module format looks like this:
+	//
+	//   ... bundled code ...
+	//   export {...};
+	//
+	FormatESModule
+)
+
+func (f Format) KeepESMImportExportSyntax() bool {
+	return f == FormatPreserve || f == FormatESModule
+}
+
+func (f Format) String() string {
+	switch f {
+	case FormatIIFE:
+		return "iife"
+	case FormatCommonJS:
+		return "cjs"
+	case FormatESModule:
+		return "esm"
+	}
+	return ""
+}
+
+type StdinInfo struct {
+	Contents      string
+	SourceFile    string
+	AbsResolveDir string
+	Loader        Loader
+}
+
+type WildcardPattern struct {
+	Prefix string
+	Suffix string
+}
+
+type ExternalMatchers struct {
+	Exact    map[string]bool
+	Patterns []WildcardPattern
+}
+
+func (matchers ExternalMatchers) HasMatchers() bool {
+	return len(matchers.Exact) > 0 || len(matchers.Patterns) > 0
+}
+
+type ExternalSettings struct {
+	PreResolve  ExternalMatchers
+	PostResolve ExternalMatchers
+}
+
+type APICall uint8
+
+const (
+	BuildCall APICall = iota
+	TransformCall
+)
+
+type Mode uint8
+
+const (
+	ModePassThrough Mode = iota
+	ModeConvertFormat
+	ModeBundle
+)
+
+type MaybeBool uint8
+
+const (
+	Unspecified MaybeBool = iota
+	True
+	False
+)
+
+type CancelFlag struct {
+	uint32
+}
+
+func (flag *CancelFlag) Cancel() {
+	atomic.StoreUint32(&flag.uint32, 1)
+}
+
+// This checks for nil in one place so we don't have to do that everywhere
+func (flag *CancelFlag) DidCancel() bool {
+	return flag != nil && atomic.LoadUint32(&flag.uint32) != 0
+}
+
+type Options struct {
+	ModuleTypeData js_ast.ModuleTypeData
+	Defines        *ProcessedDefines
+	TSAlwaysStrict *TSAlwaysStrict
+	MangleProps    *regexp.Regexp
+	ReserveProps   *regexp.Regexp
+	CancelFlag     *CancelFlag
+
+	// When mangling property names, call this function with a callback and do
+	// the property name mangling inside the callback. The callback takes an
+	// argument which is the mangle cache map to mutate. These callbacks are
+	// serialized so mutating the map does not require extra synchronization.
+	//
+	// This is a callback for determinism reasons. We may be building multiple
+	// entry points in parallel that are supposed to share a single cache. We
+	// don't want the order that each entry point mangles properties in to cause
+	// the output to change, so we serialize the property mangling over all entry
+	// points in entry point order. However, we still want to link everything in
+	// parallel so only property mangling is serialized, which is implemented by
+	// this function blocking until the previous entry point's property mangling
+	// has finished.
+	ExclusiveMangleCacheUpdate func(cb func(
+		mangleCache map[string]interface{},
+		cssUsedLocalNames map[string]bool,
+	))
+
+	// This is the original information that was used to generate the
+	// unsupported feature sets above. It's used for error messages.
+	OriginalTargetEnv string
+
+	DropLabels       []string
+	ExtensionOrder   []string
+	MainFields       []string
+	Conditions       []string
+	AbsNodePaths     []string // The "NODE_PATH" variable from Node.js
+	ExternalSettings ExternalSettings
+	ExternalPackages bool
+	PackageAliases   map[string]string
+
+	AbsOutputFile      string
+	AbsOutputDir       string
+	AbsOutputBase      string
+	OutputExtensionJS  string
+	OutputExtensionCSS string
+	GlobalName         []string
+	TSConfigPath       string
+	TSConfigRaw        string
+	ExtensionToLoader  map[string]Loader
+
+	PublicPath      string
+	InjectPaths     []string
+	InjectedDefines []InjectedDefine
+	InjectedFiles   []InjectedFile
+
+	JSBanner  string
+	JSFooter  string
+	CSSBanner string
+	CSSFooter string
+
+	EntryPathTemplate []PathTemplate
+	ChunkPathTemplate []PathTemplate
+	AssetPathTemplate []PathTemplate
+
+	Plugins    []Plugin
+	SourceRoot string
+	Stdin      *StdinInfo
+	JSX        JSXOptions
+	LineLimit  int
+
+	CSSPrefixData          map[css_ast.D]compat.CSSPrefix
+	UnsupportedJSFeatures  compat.JSFeature
+	UnsupportedCSSFeatures compat.CSSFeature
+
+	UnsupportedJSFeatureOverrides      compat.JSFeature
+	UnsupportedJSFeatureOverridesMask  compat.JSFeature
+	UnsupportedCSSFeatureOverrides     compat.CSSFeature
+	UnsupportedCSSFeatureOverridesMask compat.CSSFeature
+
+	TS                TSOptions
+	Mode              Mode
+	PreserveSymlinks  bool
+	MinifyWhitespace  bool
+	MinifyIdentifiers bool
+	MinifySyntax      bool
+	ProfilerNames     bool
+	CodeSplitting     bool
+	WatchMode         bool
+	AllowOverwrite    bool
+	LegalComments     LegalComments
+
+	// If true, make sure to generate a single file that can be written to stdout
+	WriteToStdout bool
+
+	OmitRuntimeForTests    bool
+	OmitJSXRuntimeForTests bool
+	ASCIIOnly              bool
+	KeepNames              bool
+	IgnoreDCEAnnotations   bool
+	TreeShaking            bool
+	DropDebugger           bool
+	MangleQuoted           bool
+	Platform               Platform
+	OutputFormat           Format
+	NeedsMetafile          bool
+	SourceMap              SourceMap
+	ExcludeSourcesContent  bool
+}
+
+type TSImportsNotUsedAsValues uint8
+
+const (
+	TSImportsNotUsedAsValues_None TSImportsNotUsedAsValues = iota
+	TSImportsNotUsedAsValues_Remove
+	TSImportsNotUsedAsValues_Preserve
+	TSImportsNotUsedAsValues_Error
+)
+
+// These flags represent the following separate "tsconfig.json" settings:
+//
+// - importsNotUsedAsValues
+// - preserveValueImports
+// - verbatimModuleSyntax
+//
+// TypeScript prefers for people to use "verbatimModuleSyntax" and has
+// deprecated the other two settings, but we must still support them.
+// All settings are combined into these two behavioral flags for us.
+type TSUnusedImportFlags uint8
+
+// With !TSUnusedImport_KeepStmt && !TSUnusedImport_KeepValues:
+//
+//	"import 'foo'"                      => "import 'foo'"
+//	"import * as unused from 'foo'"     => ""
+//	"import { unused } from 'foo'"      => ""
+//	"import { type unused } from 'foo'" => ""
+//
+// With TSUnusedImport_KeepStmt && !TSUnusedImport_KeepValues:
+//
+//	"import 'foo'"                      => "import 'foo'"
+//	"import * as unused from 'foo'"     => "import 'foo'"
+//	"import { unused } from 'foo'"      => "import 'foo'"
+//	"import { type unused } from 'foo'" => "import 'foo'"
+//
+// With !TSUnusedImport_KeepStmt && TSUnusedImport_KeepValues:
+//
+//	"import 'foo'"                      => "import 'foo'"
+//	"import * as unused from 'foo'"     => "import * as unused from 'foo'"
+//	"import { unused } from 'foo'"      => "import { unused } from 'foo'"
+//	"import { type unused } from 'foo'" => ""
+//
+// With TSUnusedImport_KeepStmt && TSUnusedImport_KeepValues:
+//
+//	"import 'foo'"                      => "import 'foo'"
+//	"import * as unused from 'foo'"     => "import * as unused from 'foo'"
+//	"import { unused } from 'foo'"      => "import { unused } from 'foo'"
+//	"import { type unused } from 'foo'" => "import {} from 'foo'"
+const (
+	TSUnusedImport_KeepStmt   TSUnusedImportFlags = 1 << iota // "importsNotUsedAsValues" != "remove"
+	TSUnusedImport_KeepValues                                 // "preserveValueImports" == true
+)
+
+type TSTarget uint8
+
+const (
+	TSTargetUnspecified     TSTarget = iota
+	TSTargetBelowES2022              // "useDefineForClassFields" defaults to false
+	TSTargetAtOrAboveES2022          // "useDefineForClassFields" defaults to true
+)
+
+type TSAlwaysStrict struct {
+	// This information is only used for error messages
+	Name   string
+	Source logger.Source
+	Range  logger.Range
+
+	// This information can affect code transformation
+	Value bool
+}
+
+type PathPlaceholder uint8
+
+const (
+	NoPlaceholder PathPlaceholder = iota
+
+	// The relative path from the original parent directory to the configured
+	// "outbase" directory, or to the lowest common ancestor directory
+	DirPlaceholder
+
+	// The original name of the file, or the manual chunk name, or the name of
+	// the type of output file ("entry" or "chunk" or "asset")
+	NamePlaceholder
+
+	// A hash of the contents of this file, and the contents and output paths of
+	// all dependencies (except for their hash placeholders)
+	HashPlaceholder
+
+	// The original extension of the file, or the name of the output file
+	// (e.g. "css", "svg", "png")
+	ExtPlaceholder
+)
+
+type PathTemplate struct {
+	Data        string
+	Placeholder PathPlaceholder
+}
+
+type PathPlaceholders struct {
+	Dir  *string
+	Name *string
+	Hash *string
+	Ext  *string
+}
+
+func (placeholders PathPlaceholders) Get(placeholder PathPlaceholder) *string {
+	switch placeholder {
+	case DirPlaceholder:
+		return placeholders.Dir
+	case NamePlaceholder:
+		return placeholders.Name
+	case HashPlaceholder:
+		return placeholders.Hash
+	case ExtPlaceholder:
+		return placeholders.Ext
+	}
+	return nil
+}
+
+func TemplateToString(template []PathTemplate) string {
+	if len(template) == 1 && template[0].Placeholder == NoPlaceholder {
+		// Avoid allocations in this case
+		return template[0].Data
+	}
+	sb := strings.Builder{}
+	for _, part := range template {
+		sb.WriteString(part.Data)
+		switch part.Placeholder {
+		case DirPlaceholder:
+			sb.WriteString("[dir]")
+		case NamePlaceholder:
+			sb.WriteString("[name]")
+		case HashPlaceholder:
+			sb.WriteString("[hash]")
+		case ExtPlaceholder:
+			sb.WriteString("[ext]")
+		}
+	}
+	return sb.String()
+}
+
+func HasPlaceholder(template []PathTemplate, placeholder PathPlaceholder) bool {
+	for _, part := range template {
+		if part.Placeholder == placeholder {
+			return true
+		}
+	}
+	return false
+}
+
+func SubstituteTemplate(template []PathTemplate, placeholders PathPlaceholders) []PathTemplate {
+	// Don't allocate if no substitution is possible and the template is already minimal
+	shouldSubstitute := false
+	for i, part := range template {
+		if placeholders.Get(part.Placeholder) != nil || (part.Placeholder == NoPlaceholder && i+1 < len(template)) {
+			shouldSubstitute = true
+			break
+		}
+	}
+	if !shouldSubstitute {
+		return template
+	}
+
+	// Otherwise, substitute and merge as appropriate
+	result := make([]PathTemplate, 0, len(template))
+	for _, part := range template {
+		if sub := placeholders.Get(part.Placeholder); sub != nil {
+			part.Data += *sub
+			part.Placeholder = NoPlaceholder
+		}
+		if last := len(result) - 1; last >= 0 && result[last].Placeholder == NoPlaceholder {
+			last := &result[last]
+			last.Data += part.Data
+			last.Placeholder = part.Placeholder
+		} else {
+			result = append(result, part)
+		}
+	}
+	return result
+}
+
+func ShouldCallRuntimeRequire(mode Mode, outputFormat Format) bool {
+	return mode == ModeBundle && outputFormat != FormatCommonJS
+}
+
+type InjectedDefine struct {
+	Data   js_ast.E
+	Name   string
+	Source logger.Source
+}
+
+type InjectedFile struct {
+	Exports      []InjectableExport
+	DefineName   string // For injected files generated when you "--define" a non-literal
+	Source       logger.Source
+	IsCopyLoader bool // If you set the loader to "copy" (see https://github.com/evanw/esbuild/issues/3041)
+}
+
+type InjectableExport struct {
+	Alias string
+	Loc   logger.Loc
+}
+
+var filterMutex sync.Mutex
+var filterCache map[string]*regexp.Regexp
+
+func compileFilter(filter string) (result *regexp.Regexp) {
+	if filter == "" {
+		// Must provide a filter
+		return nil
+	}
+	ok := false
+
+	// Cache hit?
+	(func() {
+		filterMutex.Lock()
+		defer filterMutex.Unlock()
+		if filterCache != nil {
+			result, ok = filterCache[filter]
+		}
+	})()
+	if ok {
+		return
+	}
+
+	// Cache miss
+	result, err := regexp.Compile(filter)
+	if err != nil {
+		return nil
+	}
+
+	// Cache for next time
+	filterMutex.Lock()
+	defer filterMutex.Unlock()
+	if filterCache == nil {
+		filterCache = make(map[string]*regexp.Regexp)
+	}
+	filterCache[filter] = result
+	return
+}
+
+func CompileFilterForPlugin(pluginName string, kind string, filter string) (*regexp.Regexp, error) {
+	if filter == "" {
+		return nil, fmt.Errorf("[%s] %q is missing a filter", pluginName, kind)
+	}
+
+	result := compileFilter(filter)
+	if result == nil {
+		return nil, fmt.Errorf("[%s] %q filter is not a valid Go regular expression: %q", pluginName, kind, filter)
+	}
+
+	return result, nil
+}
+
+func PluginAppliesToPath(path logger.Path, filter *regexp.Regexp, namespace string) bool {
+	return (namespace == "" || path.Namespace == namespace) && filter.MatchString(path.Text)
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// Plugin API
+
+type Plugin struct {
+	Name      string
+	OnStart   []OnStart
+	OnResolve []OnResolve
+	OnLoad    []OnLoad
+}
+
+type OnStart struct {
+	Callback func() OnStartResult
+	Name     string
+}
+
+type OnStartResult struct {
+	ThrownError error
+	Msgs        []logger.Msg
+}
+
+type OnResolve struct {
+	Filter    *regexp.Regexp
+	Callback  func(OnResolveArgs) OnResolveResult
+	Name      string
+	Namespace string
+}
+
+type OnResolveArgs struct {
+	Path       string
+	ResolveDir string
+	PluginData interface{}
+	Importer   logger.Path
+	Kind       ast.ImportKind
+	With       logger.ImportAttributes
+}
+
+type OnResolveResult struct {
+	PluginName string
+
+	Msgs        []logger.Msg
+	ThrownError error
+
+	AbsWatchFiles []string
+	AbsWatchDirs  []string
+
+	PluginData       interface{}
+	Path             logger.Path
+	External         bool
+	IsSideEffectFree bool
+}
+
+type OnLoad struct {
+	Filter    *regexp.Regexp
+	Callback  func(OnLoadArgs) OnLoadResult
+	Name      string
+	Namespace string
+}
+
+type OnLoadArgs struct {
+	PluginData interface{}
+	Path       logger.Path
+}
+
+type OnLoadResult struct {
+	PluginName string
+
+	Contents      *string
+	AbsResolveDir string
+	PluginData    interface{}
+
+	Msgs        []logger.Msg
+	ThrownError error
+
+	AbsWatchFiles []string
+	AbsWatchDirs  []string
+
+	Loader Loader
+}
+
+func PrettyPrintTargetEnvironment(originalTargetEnv string, unsupportedJSFeatureOverridesMask compat.JSFeature) (where string) {
+	where = "the configured target environment"
+	overrides := ""
+	if unsupportedJSFeatureOverridesMask != 0 {
+		count := 0
+		mask := unsupportedJSFeatureOverridesMask
+		for mask != 0 {
+			if (mask & 1) != 0 {
+				count++
+			}
+			mask >>= 1
+		}
+		s := "s"
+		if count == 1 {
+			s = ""
+		}
+		overrides = fmt.Sprintf(" + %d override%s", count, s)
+	}
+	if originalTargetEnv != "" {
+		where = fmt.Sprintf("%s (%s%s)", where, originalTargetEnv, overrides)
+	}
+	return
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/config/globals.go b/source/vendor/github.com/evanw/esbuild/internal/config/globals.go
new file mode 100644
index 0000000..4a77c02
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/config/globals.go
@@ -0,0 +1,1014 @@
+package config
+
+import (
+	"math"
+	"strings"
+	"sync"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+)
+
+var processedGlobalsMutex sync.Mutex
+var processedGlobals *ProcessedDefines
+
+// If something is in this list, then a direct identifier expression or property
+// access chain matching this will be assumed to have no side effects and will
+// be removed.
+//
+// This also means code is allowed to be reordered past things in this list. For
+// example, if "console.log" is in this list, permitting reordering allows for
+// "if (a) console.log(b); else console.log(c)" to be reordered and transformed
+// into "console.log(a ? b : c)". Notice that "a" and "console.log" are in a
+// different order, which can only happen if evaluating the "console.log"
+// property access can be assumed to not change the value of "a".
+//
+// Note that membership in this list says nothing about whether calling any of
+// these functions has any side effects. It only says something about
+// referencing these function without calling them.
+var knownGlobals = [][]string{
+	// These global identifiers should exist in all JavaScript environments. This
+	// deliberately omits "NaN", "Infinity", and "undefined" because these are
+	// treated as automatically-inlined constants instead of identifiers.
+	{"Array"},
+	{"Boolean"},
+	{"Function"},
+	{"Math"},
+	{"Number"},
+	{"Object"},
+	{"RegExp"},
+	{"String"},
+
+	// Object: Static methods
+	// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object#Static_methods
+	{"Object", "assign"},
+	{"Object", "create"},
+	{"Object", "defineProperties"},
+	{"Object", "defineProperty"},
+	{"Object", "entries"},
+	{"Object", "freeze"},
+	{"Object", "fromEntries"},
+	{"Object", "getOwnPropertyDescriptor"},
+	{"Object", "getOwnPropertyDescriptors"},
+	{"Object", "getOwnPropertyNames"},
+	{"Object", "getOwnPropertySymbols"},
+	{"Object", "getPrototypeOf"},
+	{"Object", "is"},
+	{"Object", "isExtensible"},
+	{"Object", "isFrozen"},
+	{"Object", "isSealed"},
+	{"Object", "keys"},
+	{"Object", "preventExtensions"},
+	{"Object", "seal"},
+	{"Object", "setPrototypeOf"},
+	{"Object", "values"},
+
+	// Object: Instance methods
+	// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object#Instance_methods
+	{"Object", "prototype", "__defineGetter__"},
+	{"Object", "prototype", "__defineSetter__"},
+	{"Object", "prototype", "__lookupGetter__"},
+	{"Object", "prototype", "__lookupSetter__"},
+	{"Object", "prototype", "hasOwnProperty"},
+	{"Object", "prototype", "isPrototypeOf"},
+	{"Object", "prototype", "propertyIsEnumerable"},
+	{"Object", "prototype", "toLocaleString"},
+	{"Object", "prototype", "toString"},
+	{"Object", "prototype", "unwatch"},
+	{"Object", "prototype", "valueOf"},
+	{"Object", "prototype", "watch"},
+
+	// Symbol: Static properties
+	// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Symbol#static_properties
+	{"Symbol", "asyncDispose"},
+	{"Symbol", "asyncIterator"},
+	{"Symbol", "dispose"},
+	{"Symbol", "hasInstance"},
+	{"Symbol", "isConcatSpreadable"},
+	{"Symbol", "iterator"},
+	{"Symbol", "match"},
+	{"Symbol", "matchAll"},
+	{"Symbol", "replace"},
+	{"Symbol", "search"},
+	{"Symbol", "species"},
+	{"Symbol", "split"},
+	{"Symbol", "toPrimitive"},
+	{"Symbol", "toStringTag"},
+	{"Symbol", "unscopables"},
+
+	// Math: Static properties
+	// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Math#Static_properties
+	{"Math", "E"},
+	{"Math", "LN10"},
+	{"Math", "LN2"},
+	{"Math", "LOG10E"},
+	{"Math", "LOG2E"},
+	{"Math", "PI"},
+	{"Math", "SQRT1_2"},
+	{"Math", "SQRT2"},
+
+	// Math: Static methods
+	// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Math#Static_methods
+	{"Math", "abs"},
+	{"Math", "acos"},
+	{"Math", "acosh"},
+	{"Math", "asin"},
+	{"Math", "asinh"},
+	{"Math", "atan"},
+	{"Math", "atan2"},
+	{"Math", "atanh"},
+	{"Math", "cbrt"},
+	{"Math", "ceil"},
+	{"Math", "clz32"},
+	{"Math", "cos"},
+	{"Math", "cosh"},
+	{"Math", "exp"},
+	{"Math", "expm1"},
+	{"Math", "floor"},
+	{"Math", "fround"},
+	{"Math", "hypot"},
+	{"Math", "imul"},
+	{"Math", "log"},
+	{"Math", "log10"},
+	{"Math", "log1p"},
+	{"Math", "log2"},
+	{"Math", "max"},
+	{"Math", "min"},
+	{"Math", "pow"},
+	{"Math", "random"},
+	{"Math", "round"},
+	{"Math", "sign"},
+	{"Math", "sin"},
+	{"Math", "sinh"},
+	{"Math", "sqrt"},
+	{"Math", "tan"},
+	{"Math", "tanh"},
+	{"Math", "trunc"},
+
+	// Reflect: Static methods
+	// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Reflect#static_methods
+	{"Reflect", "apply"},
+	{"Reflect", "construct"},
+	{"Reflect", "defineProperty"},
+	{"Reflect", "deleteProperty"},
+	{"Reflect", "get"},
+	{"Reflect", "getOwnPropertyDescriptor"},
+	{"Reflect", "getPrototypeOf"},
+	{"Reflect", "has"},
+	{"Reflect", "isExtensible"},
+	{"Reflect", "ownKeys"},
+	{"Reflect", "preventExtensions"},
+	{"Reflect", "set"},
+	{"Reflect", "setPrototypeOf"},
+
+	// JSON: Static Methods
+	// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/JSON#static_methods
+	{"JSON", "parse"},
+	{"JSON", "stringify"},
+
+	// Other globals present in both the browser and node (except "eval" because
+	// it has special behavior)
+	{"AbortController"},
+	{"AbortSignal"},
+	{"AggregateError"},
+	{"ArrayBuffer"},
+	{"BigInt"},
+	{"DataView"},
+	{"Date"},
+	{"Error"},
+	{"EvalError"},
+	{"Event"},
+	{"EventTarget"},
+	{"Float32Array"},
+	{"Float64Array"},
+	{"Int16Array"},
+	{"Int32Array"},
+	{"Int8Array"},
+	{"Intl"},
+	{"JSON"},
+	{"Map"},
+	{"MessageChannel"},
+	{"MessageEvent"},
+	{"MessagePort"},
+	{"Promise"},
+	{"Proxy"},
+	{"RangeError"},
+	{"ReferenceError"},
+	{"Reflect"},
+	{"Set"},
+	{"Symbol"},
+	{"SyntaxError"},
+	{"TextDecoder"},
+	{"TextEncoder"},
+	{"TypeError"},
+	{"URIError"},
+	{"URL"},
+	{"URLSearchParams"},
+	{"Uint16Array"},
+	{"Uint32Array"},
+	{"Uint8Array"},
+	{"Uint8ClampedArray"},
+	{"WeakMap"},
+	{"WeakSet"},
+	{"WebAssembly"},
+	{"clearInterval"},
+	{"clearTimeout"},
+	{"console"},
+	{"decodeURI"},
+	{"decodeURIComponent"},
+	{"encodeURI"},
+	{"encodeURIComponent"},
+	{"escape"},
+	{"globalThis"},
+	{"isFinite"},
+	{"isNaN"},
+	{"parseFloat"},
+	{"parseInt"},
+	{"queueMicrotask"},
+	{"setInterval"},
+	{"setTimeout"},
+	{"unescape"},
+
+	// Console method references are assumed to have no side effects
+	// https://developer.mozilla.org/en-US/docs/Web/API/console
+	{"console", "assert"},
+	{"console", "clear"},
+	{"console", "count"},
+	{"console", "countReset"},
+	{"console", "debug"},
+	{"console", "dir"},
+	{"console", "dirxml"},
+	{"console", "error"},
+	{"console", "group"},
+	{"console", "groupCollapsed"},
+	{"console", "groupEnd"},
+	{"console", "info"},
+	{"console", "log"},
+	{"console", "table"},
+	{"console", "time"},
+	{"console", "timeEnd"},
+	{"console", "timeLog"},
+	{"console", "trace"},
+	{"console", "warn"},
+
+	// CSSOM APIs
+	{"CSSAnimation"},
+	{"CSSFontFaceRule"},
+	{"CSSImportRule"},
+	{"CSSKeyframeRule"},
+	{"CSSKeyframesRule"},
+	{"CSSMediaRule"},
+	{"CSSNamespaceRule"},
+	{"CSSPageRule"},
+	{"CSSRule"},
+	{"CSSRuleList"},
+	{"CSSStyleDeclaration"},
+	{"CSSStyleRule"},
+	{"CSSStyleSheet"},
+	{"CSSSupportsRule"},
+	{"CSSTransition"},
+
+	// SVG DOM
+	{"SVGAElement"},
+	{"SVGAngle"},
+	{"SVGAnimateElement"},
+	{"SVGAnimateMotionElement"},
+	{"SVGAnimateTransformElement"},
+	{"SVGAnimatedAngle"},
+	{"SVGAnimatedBoolean"},
+	{"SVGAnimatedEnumeration"},
+	{"SVGAnimatedInteger"},
+	{"SVGAnimatedLength"},
+	{"SVGAnimatedLengthList"},
+	{"SVGAnimatedNumber"},
+	{"SVGAnimatedNumberList"},
+	{"SVGAnimatedPreserveAspectRatio"},
+	{"SVGAnimatedRect"},
+	{"SVGAnimatedString"},
+	{"SVGAnimatedTransformList"},
+	{"SVGAnimationElement"},
+	{"SVGCircleElement"},
+	{"SVGClipPathElement"},
+	{"SVGComponentTransferFunctionElement"},
+	{"SVGDefsElement"},
+	{"SVGDescElement"},
+	{"SVGElement"},
+	{"SVGEllipseElement"},
+	{"SVGFEBlendElement"},
+	{"SVGFEColorMatrixElement"},
+	{"SVGFEComponentTransferElement"},
+	{"SVGFECompositeElement"},
+	{"SVGFEConvolveMatrixElement"},
+	{"SVGFEDiffuseLightingElement"},
+	{"SVGFEDisplacementMapElement"},
+	{"SVGFEDistantLightElement"},
+	{"SVGFEDropShadowElement"},
+	{"SVGFEFloodElement"},
+	{"SVGFEFuncAElement"},
+	{"SVGFEFuncBElement"},
+	{"SVGFEFuncGElement"},
+	{"SVGFEFuncRElement"},
+	{"SVGFEGaussianBlurElement"},
+	{"SVGFEImageElement"},
+	{"SVGFEMergeElement"},
+	{"SVGFEMergeNodeElement"},
+	{"SVGFEMorphologyElement"},
+	{"SVGFEOffsetElement"},
+	{"SVGFEPointLightElement"},
+	{"SVGFESpecularLightingElement"},
+	{"SVGFESpotLightElement"},
+	{"SVGFETileElement"},
+	{"SVGFETurbulenceElement"},
+	{"SVGFilterElement"},
+	{"SVGForeignObjectElement"},
+	{"SVGGElement"},
+	{"SVGGeometryElement"},
+	{"SVGGradientElement"},
+	{"SVGGraphicsElement"},
+	{"SVGImageElement"},
+	{"SVGLength"},
+	{"SVGLengthList"},
+	{"SVGLineElement"},
+	{"SVGLinearGradientElement"},
+	{"SVGMPathElement"},
+	{"SVGMarkerElement"},
+	{"SVGMaskElement"},
+	{"SVGMatrix"},
+	{"SVGMetadataElement"},
+	{"SVGNumber"},
+	{"SVGNumberList"},
+	{"SVGPathElement"},
+	{"SVGPatternElement"},
+	{"SVGPoint"},
+	{"SVGPointList"},
+	{"SVGPolygonElement"},
+	{"SVGPolylineElement"},
+	{"SVGPreserveAspectRatio"},
+	{"SVGRadialGradientElement"},
+	{"SVGRect"},
+	{"SVGRectElement"},
+	{"SVGSVGElement"},
+	{"SVGScriptElement"},
+	{"SVGSetElement"},
+	{"SVGStopElement"},
+	{"SVGStringList"},
+	{"SVGStyleElement"},
+	{"SVGSwitchElement"},
+	{"SVGSymbolElement"},
+	{"SVGTSpanElement"},
+	{"SVGTextContentElement"},
+	{"SVGTextElement"},
+	{"SVGTextPathElement"},
+	{"SVGTextPositioningElement"},
+	{"SVGTitleElement"},
+	{"SVGTransform"},
+	{"SVGTransformList"},
+	{"SVGUnitTypes"},
+	{"SVGUseElement"},
+	{"SVGViewElement"},
+
+	// Other browser APIs
+	//
+	// This list contains all globals present in modern versions of Chrome, Safari,
+	// and Firefox except for the following properties, since they have a side effect
+	// of triggering layout (https://gist.github.com/paulirish/5d52fb081b3570c81e3a):
+	//
+	//   - scrollX
+	//   - scrollY
+	//   - innerWidth
+	//   - innerHeight
+	//   - pageXOffset
+	//   - pageYOffset
+	//
+	// The following globals have also been removed since they sometimes throw an
+	// exception when accessed, which is a side effect (for more information see
+	// https://stackoverflow.com/a/33047477):
+	//
+	//   - localStorage
+	//   - sessionStorage
+	//
+	{"AnalyserNode"},
+	{"Animation"},
+	{"AnimationEffect"},
+	{"AnimationEvent"},
+	{"AnimationPlaybackEvent"},
+	{"AnimationTimeline"},
+	{"Attr"},
+	{"Audio"},
+	{"AudioBuffer"},
+	{"AudioBufferSourceNode"},
+	{"AudioDestinationNode"},
+	{"AudioListener"},
+	{"AudioNode"},
+	{"AudioParam"},
+	{"AudioProcessingEvent"},
+	{"AudioScheduledSourceNode"},
+	{"BarProp"},
+	{"BeforeUnloadEvent"},
+	{"BiquadFilterNode"},
+	{"Blob"},
+	{"BlobEvent"},
+	{"ByteLengthQueuingStrategy"},
+	{"CDATASection"},
+	{"CSS"},
+	{"CanvasGradient"},
+	{"CanvasPattern"},
+	{"CanvasRenderingContext2D"},
+	{"ChannelMergerNode"},
+	{"ChannelSplitterNode"},
+	{"CharacterData"},
+	{"ClipboardEvent"},
+	{"CloseEvent"},
+	{"Comment"},
+	{"CompositionEvent"},
+	{"ConvolverNode"},
+	{"CountQueuingStrategy"},
+	{"Crypto"},
+	{"CustomElementRegistry"},
+	{"CustomEvent"},
+	{"DOMException"},
+	{"DOMImplementation"},
+	{"DOMMatrix"},
+	{"DOMMatrixReadOnly"},
+	{"DOMParser"},
+	{"DOMPoint"},
+	{"DOMPointReadOnly"},
+	{"DOMQuad"},
+	{"DOMRect"},
+	{"DOMRectList"},
+	{"DOMRectReadOnly"},
+	{"DOMStringList"},
+	{"DOMStringMap"},
+	{"DOMTokenList"},
+	{"DataTransfer"},
+	{"DataTransferItem"},
+	{"DataTransferItemList"},
+	{"DelayNode"},
+	{"Document"},
+	{"DocumentFragment"},
+	{"DocumentTimeline"},
+	{"DocumentType"},
+	{"DragEvent"},
+	{"DynamicsCompressorNode"},
+	{"Element"},
+	{"ErrorEvent"},
+	{"EventSource"},
+	{"File"},
+	{"FileList"},
+	{"FileReader"},
+	{"FocusEvent"},
+	{"FontFace"},
+	{"FormData"},
+	{"GainNode"},
+	{"Gamepad"},
+	{"GamepadButton"},
+	{"GamepadEvent"},
+	{"Geolocation"},
+	{"GeolocationPositionError"},
+	{"HTMLAllCollection"},
+	{"HTMLAnchorElement"},
+	{"HTMLAreaElement"},
+	{"HTMLAudioElement"},
+	{"HTMLBRElement"},
+	{"HTMLBaseElement"},
+	{"HTMLBodyElement"},
+	{"HTMLButtonElement"},
+	{"HTMLCanvasElement"},
+	{"HTMLCollection"},
+	{"HTMLDListElement"},
+	{"HTMLDataElement"},
+	{"HTMLDataListElement"},
+	{"HTMLDetailsElement"},
+	{"HTMLDirectoryElement"},
+	{"HTMLDivElement"},
+	{"HTMLDocument"},
+	{"HTMLElement"},
+	{"HTMLEmbedElement"},
+	{"HTMLFieldSetElement"},
+	{"HTMLFontElement"},
+	{"HTMLFormControlsCollection"},
+	{"HTMLFormElement"},
+	{"HTMLFrameElement"},
+	{"HTMLFrameSetElement"},
+	{"HTMLHRElement"},
+	{"HTMLHeadElement"},
+	{"HTMLHeadingElement"},
+	{"HTMLHtmlElement"},
+	{"HTMLIFrameElement"},
+	{"HTMLImageElement"},
+	{"HTMLInputElement"},
+	{"HTMLLIElement"},
+	{"HTMLLabelElement"},
+	{"HTMLLegendElement"},
+	{"HTMLLinkElement"},
+	{"HTMLMapElement"},
+	{"HTMLMarqueeElement"},
+	{"HTMLMediaElement"},
+	{"HTMLMenuElement"},
+	{"HTMLMetaElement"},
+	{"HTMLMeterElement"},
+	{"HTMLModElement"},
+	{"HTMLOListElement"},
+	{"HTMLObjectElement"},
+	{"HTMLOptGroupElement"},
+	{"HTMLOptionElement"},
+	{"HTMLOptionsCollection"},
+	{"HTMLOutputElement"},
+	{"HTMLParagraphElement"},
+	{"HTMLParamElement"},
+	{"HTMLPictureElement"},
+	{"HTMLPreElement"},
+	{"HTMLProgressElement"},
+	{"HTMLQuoteElement"},
+	{"HTMLScriptElement"},
+	{"HTMLSelectElement"},
+	{"HTMLSlotElement"},
+	{"HTMLSourceElement"},
+	{"HTMLSpanElement"},
+	{"HTMLStyleElement"},
+	{"HTMLTableCaptionElement"},
+	{"HTMLTableCellElement"},
+	{"HTMLTableColElement"},
+	{"HTMLTableElement"},
+	{"HTMLTableRowElement"},
+	{"HTMLTableSectionElement"},
+	{"HTMLTemplateElement"},
+	{"HTMLTextAreaElement"},
+	{"HTMLTimeElement"},
+	{"HTMLTitleElement"},
+	{"HTMLTrackElement"},
+	{"HTMLUListElement"},
+	{"HTMLUnknownElement"},
+	{"HTMLVideoElement"},
+	{"HashChangeEvent"},
+	{"Headers"},
+	{"History"},
+	{"IDBCursor"},
+	{"IDBCursorWithValue"},
+	{"IDBDatabase"},
+	{"IDBFactory"},
+	{"IDBIndex"},
+	{"IDBKeyRange"},
+	{"IDBObjectStore"},
+	{"IDBOpenDBRequest"},
+	{"IDBRequest"},
+	{"IDBTransaction"},
+	{"IDBVersionChangeEvent"},
+	{"Image"},
+	{"ImageData"},
+	{"InputEvent"},
+	{"IntersectionObserver"},
+	{"IntersectionObserverEntry"},
+	{"KeyboardEvent"},
+	{"KeyframeEffect"},
+	{"Location"},
+	{"MediaCapabilities"},
+	{"MediaElementAudioSourceNode"},
+	{"MediaEncryptedEvent"},
+	{"MediaError"},
+	{"MediaList"},
+	{"MediaQueryList"},
+	{"MediaQueryListEvent"},
+	{"MediaRecorder"},
+	{"MediaSource"},
+	{"MediaStream"},
+	{"MediaStreamAudioDestinationNode"},
+	{"MediaStreamAudioSourceNode"},
+	{"MediaStreamTrack"},
+	{"MediaStreamTrackEvent"},
+	{"MimeType"},
+	{"MimeTypeArray"},
+	{"MouseEvent"},
+	{"MutationEvent"},
+	{"MutationObserver"},
+	{"MutationRecord"},
+	{"NamedNodeMap"},
+	{"Navigator"},
+	{"Node"},
+	{"NodeFilter"},
+	{"NodeIterator"},
+	{"NodeList"},
+	{"Notification"},
+	{"OfflineAudioCompletionEvent"},
+	{"Option"},
+	{"OscillatorNode"},
+	{"PageTransitionEvent"},
+	{"Path2D"},
+	{"Performance"},
+	{"PerformanceEntry"},
+	{"PerformanceMark"},
+	{"PerformanceMeasure"},
+	{"PerformanceNavigation"},
+	{"PerformanceObserver"},
+	{"PerformanceObserverEntryList"},
+	{"PerformanceResourceTiming"},
+	{"PerformanceTiming"},
+	{"PeriodicWave"},
+	{"Plugin"},
+	{"PluginArray"},
+	{"PointerEvent"},
+	{"PopStateEvent"},
+	{"ProcessingInstruction"},
+	{"ProgressEvent"},
+	{"PromiseRejectionEvent"},
+	{"RTCCertificate"},
+	{"RTCDTMFSender"},
+	{"RTCDTMFToneChangeEvent"},
+	{"RTCDataChannel"},
+	{"RTCDataChannelEvent"},
+	{"RTCIceCandidate"},
+	{"RTCPeerConnection"},
+	{"RTCPeerConnectionIceEvent"},
+	{"RTCRtpReceiver"},
+	{"RTCRtpSender"},
+	{"RTCRtpTransceiver"},
+	{"RTCSessionDescription"},
+	{"RTCStatsReport"},
+	{"RTCTrackEvent"},
+	{"RadioNodeList"},
+	{"Range"},
+	{"ReadableStream"},
+	{"Request"},
+	{"ResizeObserver"},
+	{"ResizeObserverEntry"},
+	{"Response"},
+	{"Screen"},
+	{"ScriptProcessorNode"},
+	{"SecurityPolicyViolationEvent"},
+	{"Selection"},
+	{"ShadowRoot"},
+	{"SourceBuffer"},
+	{"SourceBufferList"},
+	{"SpeechSynthesisEvent"},
+	{"SpeechSynthesisUtterance"},
+	{"StaticRange"},
+	{"Storage"},
+	{"StorageEvent"},
+	{"StyleSheet"},
+	{"StyleSheetList"},
+	{"Text"},
+	{"TextMetrics"},
+	{"TextTrack"},
+	{"TextTrackCue"},
+	{"TextTrackCueList"},
+	{"TextTrackList"},
+	{"TimeRanges"},
+	{"TrackEvent"},
+	{"TransitionEvent"},
+	{"TreeWalker"},
+	{"UIEvent"},
+	{"VTTCue"},
+	{"ValidityState"},
+	{"VisualViewport"},
+	{"WaveShaperNode"},
+	{"WebGLActiveInfo"},
+	{"WebGLBuffer"},
+	{"WebGLContextEvent"},
+	{"WebGLFramebuffer"},
+	{"WebGLProgram"},
+	{"WebGLQuery"},
+	{"WebGLRenderbuffer"},
+	{"WebGLRenderingContext"},
+	{"WebGLSampler"},
+	{"WebGLShader"},
+	{"WebGLShaderPrecisionFormat"},
+	{"WebGLSync"},
+	{"WebGLTexture"},
+	{"WebGLUniformLocation"},
+	{"WebKitCSSMatrix"},
+	{"WebSocket"},
+	{"WheelEvent"},
+	{"Window"},
+	{"Worker"},
+	{"XMLDocument"},
+	{"XMLHttpRequest"},
+	{"XMLHttpRequestEventTarget"},
+	{"XMLHttpRequestUpload"},
+	{"XMLSerializer"},
+	{"XPathEvaluator"},
+	{"XPathExpression"},
+	{"XPathResult"},
+	{"XSLTProcessor"},
+	{"alert"},
+	{"atob"},
+	{"blur"},
+	{"btoa"},
+	{"cancelAnimationFrame"},
+	{"captureEvents"},
+	{"close"},
+	{"closed"},
+	{"confirm"},
+	{"customElements"},
+	{"devicePixelRatio"},
+	{"document"},
+	{"event"},
+	{"fetch"},
+	{"find"},
+	{"focus"},
+	{"frameElement"},
+	{"frames"},
+	{"getComputedStyle"},
+	{"getSelection"},
+	{"history"},
+	{"indexedDB"},
+	{"isSecureContext"},
+	{"length"},
+	{"location"},
+	{"locationbar"},
+	{"matchMedia"},
+	{"menubar"},
+	{"moveBy"},
+	{"moveTo"},
+	{"name"},
+	{"navigator"},
+	{"onabort"},
+	{"onafterprint"},
+	{"onanimationend"},
+	{"onanimationiteration"},
+	{"onanimationstart"},
+	{"onbeforeprint"},
+	{"onbeforeunload"},
+	{"onblur"},
+	{"oncanplay"},
+	{"oncanplaythrough"},
+	{"onchange"},
+	{"onclick"},
+	{"oncontextmenu"},
+	{"oncuechange"},
+	{"ondblclick"},
+	{"ondrag"},
+	{"ondragend"},
+	{"ondragenter"},
+	{"ondragleave"},
+	{"ondragover"},
+	{"ondragstart"},
+	{"ondrop"},
+	{"ondurationchange"},
+	{"onemptied"},
+	{"onended"},
+	{"onerror"},
+	{"onfocus"},
+	{"ongotpointercapture"},
+	{"onhashchange"},
+	{"oninput"},
+	{"oninvalid"},
+	{"onkeydown"},
+	{"onkeypress"},
+	{"onkeyup"},
+	{"onlanguagechange"},
+	{"onload"},
+	{"onloadeddata"},
+	{"onloadedmetadata"},
+	{"onloadstart"},
+	{"onlostpointercapture"},
+	{"onmessage"},
+	{"onmousedown"},
+	{"onmouseenter"},
+	{"onmouseleave"},
+	{"onmousemove"},
+	{"onmouseout"},
+	{"onmouseover"},
+	{"onmouseup"},
+	{"onoffline"},
+	{"ononline"},
+	{"onpagehide"},
+	{"onpageshow"},
+	{"onpause"},
+	{"onplay"},
+	{"onplaying"},
+	{"onpointercancel"},
+	{"onpointerdown"},
+	{"onpointerenter"},
+	{"onpointerleave"},
+	{"onpointermove"},
+	{"onpointerout"},
+	{"onpointerover"},
+	{"onpointerup"},
+	{"onpopstate"},
+	{"onprogress"},
+	{"onratechange"},
+	{"onrejectionhandled"},
+	{"onreset"},
+	{"onresize"},
+	{"onscroll"},
+	{"onseeked"},
+	{"onseeking"},
+	{"onselect"},
+	{"onstalled"},
+	{"onstorage"},
+	{"onsubmit"},
+	{"onsuspend"},
+	{"ontimeupdate"},
+	{"ontoggle"},
+	{"ontransitioncancel"},
+	{"ontransitionend"},
+	{"ontransitionrun"},
+	{"ontransitionstart"},
+	{"onunhandledrejection"},
+	{"onunload"},
+	{"onvolumechange"},
+	{"onwaiting"},
+	{"onwebkitanimationend"},
+	{"onwebkitanimationiteration"},
+	{"onwebkitanimationstart"},
+	{"onwebkittransitionend"},
+	{"onwheel"},
+	{"open"},
+	{"opener"},
+	{"origin"},
+	{"outerHeight"},
+	{"outerWidth"},
+	{"parent"},
+	{"performance"},
+	{"personalbar"},
+	{"postMessage"},
+	{"print"},
+	{"prompt"},
+	{"releaseEvents"},
+	{"requestAnimationFrame"},
+	{"resizeBy"},
+	{"resizeTo"},
+	{"screen"},
+	{"screenLeft"},
+	{"screenTop"},
+	{"screenX"},
+	{"screenY"},
+	{"scroll"},
+	{"scrollBy"},
+	{"scrollTo"},
+	{"scrollbars"},
+	{"self"},
+	{"speechSynthesis"},
+	{"status"},
+	{"statusbar"},
+	{"stop"},
+	{"toolbar"},
+	{"top"},
+	{"webkitURL"},
+	{"window"},
+}
+
+// We currently only support compile-time replacement with certain expressions:
+//
+//   - Primitive literals
+//   - Identifiers
+//   - "Entity names" which are identifiers followed by property accesses
+//
+// We don't support arbitrary expressions because arbitrary expressions may
+// require the full AST. For example, there could be "import()" or "require()"
+// expressions that need an import record. We also need to re-generate some
+// nodes such as identifiers within the injected context so that they can
+// bind to symbols in that context. Other expressions such as "this" may
+// also be contextual.
+type DefineExpr struct {
+	Constant            js_ast.E
+	Parts               []string
+	InjectedDefineIndex ast.Index32
+}
+
+type DefineData struct {
+	DefineExpr *DefineExpr
+	Flags      DefineFlags
+}
+
+type DefineFlags uint8
+
+const (
+	// True if accessing this value is known to not have any side effects. For
+	// example, a bare reference to "Object.create" can be removed because it
+	// does not have any observable side effects.
+	CanBeRemovedIfUnused DefineFlags = 1 << iota
+
+	// True if a call to this value is known to not have any side effects. For
+	// example, a bare call to "Object()" can be removed because it does not
+	// have any observable side effects.
+	CallCanBeUnwrappedIfUnused
+
+	// If true, the user has indicated that every direct calls to a property on
+	// this object and all of that call's arguments are to be removed from the
+	// output, even when the arguments have side effects. This is used to
+	// implement the "--drop:console" flag.
+	MethodCallsMustBeReplacedWithUndefined
+
+	// Symbol values are known to not have side effects when used as property
+	// names in class declarations and object literals.
+	IsSymbolInstance
+)
+
+func (flags DefineFlags) Has(flag DefineFlags) bool {
+	return (flags & flag) != 0
+}
+
+func mergeDefineData(old DefineData, new DefineData) DefineData {
+	new.Flags |= old.Flags
+	return new
+}
+
+type DotDefine struct {
+	Data  DefineData
+	Parts []string
+}
+
+type ProcessedDefines struct {
+	IdentifierDefines map[string]DefineData
+	DotDefines        map[string][]DotDefine
+}
+
+// This transformation is expensive, so we only want to do it once. Make sure
+// to only call processDefines() once per compilation. Unfortunately Golang
+// doesn't have an efficient way to copy a map and the overhead of copying
+// all of the properties into a new map once for every new parser noticeably
+// slows down our benchmarks.
+func ProcessDefines(userDefines map[string]DefineData) ProcessedDefines {
+	// Optimization: reuse known globals if there are no user-specified defines
+	hasUserDefines := len(userDefines) != 0
+	if !hasUserDefines {
+		processedGlobalsMutex.Lock()
+		if processedGlobals != nil {
+			defer processedGlobalsMutex.Unlock()
+			return *processedGlobals
+		}
+		processedGlobalsMutex.Unlock()
+	}
+
+	result := ProcessedDefines{
+		IdentifierDefines: make(map[string]DefineData),
+		DotDefines:        make(map[string][]DotDefine),
+	}
+
+	// Mark these property accesses as free of side effects. That means they can
+	// be removed if their result is unused. We can't just remove all unused
+	// property accesses since property accesses can have side effects. For
+	// example, the property access "a.b.c" has the side effect of throwing an
+	// exception if "a.b" is undefined.
+	for _, parts := range knownGlobals {
+		tail := parts[len(parts)-1]
+		if len(parts) == 1 {
+			result.IdentifierDefines[tail] = DefineData{Flags: CanBeRemovedIfUnused}
+		} else {
+			flags := CanBeRemovedIfUnused
+
+			// All properties on the "Symbol" global are currently symbol instances
+			// (i.e. "typeof Symbol.iterator === 'symbol'"). This is used to avoid
+			// treating properties with these names as having side effects.
+			if parts[0] == "Symbol" {
+				flags |= IsSymbolInstance
+			}
+
+			result.DotDefines[tail] = append(result.DotDefines[tail], DotDefine{Parts: parts, Data: DefineData{Flags: flags}})
+		}
+	}
+
+	// Swap in certain literal values because those can be constant folded
+	result.IdentifierDefines["undefined"] = DefineData{
+		DefineExpr: &DefineExpr{Constant: js_ast.EUndefinedShared},
+	}
+	result.IdentifierDefines["NaN"] = DefineData{
+		DefineExpr: &DefineExpr{Constant: &js_ast.ENumber{Value: math.NaN()}},
+	}
+	result.IdentifierDefines["Infinity"] = DefineData{
+		DefineExpr: &DefineExpr{Constant: &js_ast.ENumber{Value: math.Inf(1)}},
+	}
+
+	// Then copy the user-specified defines in afterwards, which will overwrite
+	// any known globals above.
+	for key, data := range userDefines {
+		parts := strings.Split(key, ".")
+
+		// Identifier defines are special-cased
+		if len(parts) == 1 {
+			result.IdentifierDefines[key] = mergeDefineData(result.IdentifierDefines[key], data)
+			continue
+		}
+
+		tail := parts[len(parts)-1]
+		dotDefines := result.DotDefines[tail]
+		found := false
+
+		// Try to merge with existing dot defines first
+		for i, define := range dotDefines {
+			if helpers.StringArraysEqual(parts, define.Parts) {
+				define := &dotDefines[i]
+				define.Data = mergeDefineData(define.Data, data)
+				found = true
+				break
+			}
+		}
+
+		if !found {
+			dotDefines = append(dotDefines, DotDefine{Parts: parts, Data: data})
+		}
+		result.DotDefines[tail] = dotDefines
+	}
+
+	// Potentially cache the result for next time
+	if !hasUserDefines {
+		processedGlobalsMutex.Lock()
+		defer processedGlobalsMutex.Unlock()
+		if processedGlobals == nil {
+			processedGlobals = &result
+		}
+	}
+	return result
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_ast/css_ast.go b/source/vendor/github.com/evanw/esbuild/internal/css_ast/css_ast.go
new file mode 100644
index 0000000..b7a1731
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_ast/css_ast.go
@@ -0,0 +1,1205 @@
+package css_ast
+
+import (
+	"strconv"
+	"strings"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+// CSS syntax comes in two layers: a minimal syntax that generally accepts
+// anything that looks vaguely like CSS, and a large set of built-in rules
+// (the things browsers actually interpret). That way CSS parsers can read
+// unknown rules and skip over them without having to stop due to errors.
+//
+// This AST format is mostly just the minimal syntax. It parses unknown rules
+// into a tree with enough information that it can write them back out again.
+// There are some additional layers of syntax including selectors and @-rules
+// which allow for better pretty-printing and minification.
+//
+// Most of the AST just references ranges of the original file by keeping the
+// original "Token" values around from the lexer. This is a memory-efficient
+// representation that helps provide good parsing and printing performance.
+
+type AST struct {
+	Symbols              []ast.Symbol
+	CharFreq             *ast.CharFreq
+	ImportRecords        []ast.ImportRecord
+	Rules                []Rule
+	SourceMapComment     logger.Span
+	ApproximateLineCount int32
+	LocalSymbols         []ast.LocRef
+	LocalScope           map[string]ast.LocRef
+	GlobalScope          map[string]ast.LocRef
+	Composes             map[ast.Ref]*Composes
+
+	// These contain all layer names in the file. It can be used to replace the
+	// layer-related side effects of importing this file. They are split into two
+	// groups (those before and after "@import" rules) so that the linker can put
+	// them in the right places.
+	LayersPreImport  [][]string
+	LayersPostImport [][]string
+}
+
+type Composes struct {
+	// Note that each of these can be either local or global. Local examples:
+	//
+	//   .foo { composes: bar }
+	//   .bar { color: red }
+	//
+	// Global examples:
+	//
+	//   .foo { composes: bar from global }
+	//   .foo :global { composes: bar }
+	//   .foo { :global { composes: bar } }
+	//   :global .bar { color: red }
+	//
+	Names []ast.LocRef
+
+	// Each of these is local in another file. For example:
+	//
+	//   .foo { composes: bar from "bar.css" }
+	//   .foo { composes: bar from url(bar.css) }
+	//
+	ImportedNames []ImportedComposesName
+
+	// This tracks what CSS properties each class uses so that we can warn when
+	// "composes" is used incorrectly to compose two classes from separate files
+	// that declare the same CSS properties.
+	Properties map[string]logger.Loc
+}
+
+type ImportedComposesName struct {
+	Alias             string
+	AliasLoc          logger.Loc
+	ImportRecordIndex uint32
+}
+
+// We create a lot of tokens, so make sure this layout is memory-efficient.
+// The layout here isn't optimal because it biases for convenience (e.g.
+// "string" could be shorter) but at least the ordering of fields was
+// deliberately chosen to minimize size.
+type Token struct {
+	// Contains the child tokens for component values that are simple blocks.
+	// These are either "(", "{", "[", or function tokens. The closing token is
+	// implicit and is not stored.
+	Children *[]Token // 8 bytes
+
+	// This is the raw contents of the token most of the time. However, it
+	// contains the decoded string contents for "TString" tokens.
+	Text string // 16 bytes
+
+	// The source location at the start of the token
+	Loc logger.Loc // 4 bytes
+
+	// URL tokens have an associated import record at the top-level of the AST.
+	// This index points to that import record.
+	//
+	// Symbol tokens have an associated symbol. This index is the "InnerIndex"
+	// of the "Ref" for this symbol. The "SourceIndex" for the "Ref" is just
+	// the source index of the file for this AST.
+	PayloadIndex uint32 // 4 bytes
+
+	// The division between the number and the unit for "TDimension" tokens.
+	UnitOffset uint16 // 2 bytes
+
+	// This will never be "TWhitespace" because whitespace isn't stored as a
+	// token directly. Instead it is stored in "HasWhitespaceAfter" on the
+	// previous token. This is to make it easier to pattern-match against
+	// tokens when handling CSS rules, since whitespace almost always doesn't
+	// matter. That way you can pattern match against e.g. "rgb(r, g, b)" and
+	// not have to handle all possible combinations of embedded whitespace
+	// tokens.
+	//
+	// There is one exception to this: when in verbatim whitespace mode and
+	// the token list is non-empty and is only whitespace tokens. In that case
+	// a single whitespace token is emitted. This is because otherwise there
+	// would be no tokens to attach the whitespace before/after flags to.
+	Kind css_lexer.T // 1 byte
+
+	// These flags indicate the presence of a "TWhitespace" token before or after
+	// this token. There should be whitespace printed between two tokens if either
+	// token indicates that there should be whitespace. Note that whitespace may
+	// be altered by processing in certain situations (e.g. minification).
+	Whitespace WhitespaceFlags // 1 byte
+}
+
+type WhitespaceFlags uint8
+
+const (
+	WhitespaceBefore WhitespaceFlags = 1 << iota
+	WhitespaceAfter
+)
+
+// This is necessary when comparing tokens between two different files
+type CrossFileEqualityCheck struct {
+	ImportRecordsA []ast.ImportRecord
+	ImportRecordsB []ast.ImportRecord
+	Symbols        ast.SymbolMap
+	SourceIndexA   uint32
+	SourceIndexB   uint32
+}
+
+func (check *CrossFileEqualityCheck) RefsAreEquivalent(a ast.Ref, b ast.Ref) bool {
+	if a == b {
+		return true
+	}
+	if check == nil || check.Symbols.SymbolsForSource == nil {
+		return false
+	}
+	a = ast.FollowSymbols(check.Symbols, a)
+	b = ast.FollowSymbols(check.Symbols, b)
+	if a == b {
+		return true
+	}
+	symbolA := check.Symbols.Get(a)
+	symbolB := check.Symbols.Get(b)
+	return symbolA.Kind == ast.SymbolGlobalCSS && symbolB.Kind == ast.SymbolGlobalCSS && symbolA.OriginalName == symbolB.OriginalName
+}
+
+func (a Token) Equal(b Token, check *CrossFileEqualityCheck) bool {
+	if a.Kind == b.Kind && a.Text == b.Text && a.Whitespace == b.Whitespace {
+		// URLs should be compared based on the text of the associated import record
+		// (which is what will actually be printed) instead of the original text
+		if a.Kind == css_lexer.TURL {
+			if check == nil {
+				// If both tokens are in the same file, just compare the index
+				if a.PayloadIndex != b.PayloadIndex {
+					return false
+				}
+			} else {
+				// If the tokens come from separate files, compare the import records
+				// themselves instead of comparing the indices. This can happen when
+				// the linker runs a "DuplicateRuleRemover" during bundling. This
+				// doesn't compare the source indices because at this point during
+				// linking, paths inside the bundle (e.g. due to the "copy" loader)
+				// should have already been converted into text (e.g. the "unique key"
+				// string).
+				if check.ImportRecordsA[a.PayloadIndex].Path.Text !=
+					check.ImportRecordsB[b.PayloadIndex].Path.Text {
+					return false
+				}
+			}
+		}
+
+		// Symbols should be compared based on the symbol reference instead of the
+		// original text
+		if a.Kind == css_lexer.TSymbol {
+			if check == nil {
+				// If both tokens are in the same file, just compare the index
+				if a.PayloadIndex != b.PayloadIndex {
+					return false
+				}
+			} else {
+				// If the tokens come from separate files, compare the symbols themselves
+				refA := ast.Ref{SourceIndex: check.SourceIndexA, InnerIndex: a.PayloadIndex}
+				refB := ast.Ref{SourceIndex: check.SourceIndexB, InnerIndex: b.PayloadIndex}
+				if !check.RefsAreEquivalent(refA, refB) {
+					return false
+				}
+			}
+		}
+
+		if a.Children == nil && b.Children == nil {
+			return true
+		}
+
+		if a.Children != nil && b.Children != nil && TokensEqual(*a.Children, *b.Children, check) {
+			return true
+		}
+	}
+
+	return false
+}
+
+func TokensEqual(a []Token, b []Token, check *CrossFileEqualityCheck) bool {
+	if len(a) != len(b) {
+		return false
+	}
+	for i, ai := range a {
+		if !ai.Equal(b[i], check) {
+			return false
+		}
+	}
+	return true
+}
+
+func HashTokens(hash uint32, tokens []Token) uint32 {
+	hash = helpers.HashCombine(hash, uint32(len(tokens)))
+
+	for _, t := range tokens {
+		hash = helpers.HashCombine(hash, uint32(t.Kind))
+		if t.Kind != css_lexer.TURL {
+			hash = helpers.HashCombineString(hash, t.Text)
+		}
+		if t.Children != nil {
+			hash = HashTokens(hash, *t.Children)
+		}
+	}
+
+	return hash
+}
+
+func (a Token) EqualIgnoringWhitespace(b Token) bool {
+	if a.Kind == b.Kind && a.Text == b.Text && a.PayloadIndex == b.PayloadIndex {
+		if a.Children == nil && b.Children == nil {
+			return true
+		}
+
+		if a.Children != nil && b.Children != nil && TokensEqualIgnoringWhitespace(*a.Children, *b.Children) {
+			return true
+		}
+	}
+
+	return false
+}
+
+func TokensEqualIgnoringWhitespace(a []Token, b []Token) bool {
+	if len(a) != len(b) {
+		return false
+	}
+	for i, c := range a {
+		if !c.EqualIgnoringWhitespace(b[i]) {
+			return false
+		}
+	}
+	return true
+}
+
+func TokensAreCommaSeparated(tokens []Token) bool {
+	if n := len(tokens); (n & 1) != 0 {
+		for i := 1; i < n; i += 2 {
+			if tokens[i].Kind != css_lexer.TComma {
+				return false
+			}
+		}
+		return true
+	}
+	return false
+}
+
+type PercentageFlags uint8
+
+const (
+	AllowPercentageBelow0 PercentageFlags = 1 << iota
+	AllowPercentageAbove100
+	AllowAnyPercentage = AllowPercentageBelow0 | AllowPercentageAbove100
+)
+
+func (t Token) NumberOrFractionForPercentage(percentReferenceRange float64, flags PercentageFlags) (float64, bool) {
+	switch t.Kind {
+	case css_lexer.TNumber:
+		if f, err := strconv.ParseFloat(t.Text, 64); err == nil {
+			return f, true
+		}
+
+	case css_lexer.TPercentage:
+		if f, err := strconv.ParseFloat(t.PercentageValue(), 64); err == nil {
+			if (flags&AllowPercentageBelow0) == 0 && f < 0 {
+				return 0, true
+			}
+			if (flags&AllowPercentageAbove100) == 0 && f > 100 {
+				return percentReferenceRange, true
+			}
+			return f / 100 * percentReferenceRange, true
+		}
+	}
+
+	return 0, false
+}
+
+func (t Token) ClampedFractionForPercentage() (float64, bool) {
+	if t.Kind == css_lexer.TPercentage {
+		if f, err := strconv.ParseFloat(t.PercentageValue(), 64); err == nil {
+			if f < 0 {
+				return 0, true
+			}
+			if f > 100 {
+				return 1, true
+			}
+			return f / 100, true
+		}
+	}
+
+	return 0, false
+}
+
+// https://drafts.csswg.org/css-values-3/#lengths
+// For zero lengths the unit identifier is optional
+// (i.e. can be syntactically represented as the <number> 0).
+func (t *Token) TurnLengthIntoNumberIfZero() bool {
+	if t.Kind == css_lexer.TDimension && t.DimensionValue() == "0" {
+		t.Kind = css_lexer.TNumber
+		t.Text = "0"
+		return true
+	}
+	return false
+}
+
+func (t *Token) TurnLengthOrPercentageIntoNumberIfZero() bool {
+	if t.Kind == css_lexer.TPercentage && t.PercentageValue() == "0" {
+		t.Kind = css_lexer.TNumber
+		t.Text = "0"
+		return true
+	}
+	return t.TurnLengthIntoNumberIfZero()
+}
+
+func (t Token) PercentageValue() string {
+	return t.Text[:len(t.Text)-1]
+}
+
+func (t Token) DimensionValue() string {
+	return t.Text[:t.UnitOffset]
+}
+
+func (t Token) DimensionUnit() string {
+	return t.Text[t.UnitOffset:]
+}
+
+func (t Token) DimensionUnitIsSafeLength() bool {
+	switch strings.ToLower(t.DimensionUnit()) {
+	// These units can be reasonably expected to be supported everywhere.
+	// Information used: https://developer.mozilla.org/en-US/docs/Web/CSS/length
+	case "cm", "em", "in", "mm", "pc", "pt", "px":
+		return true
+	}
+	return false
+}
+
+func (t Token) IsZero() bool {
+	return t.Kind == css_lexer.TNumber && t.Text == "0"
+}
+
+func (t Token) IsOne() bool {
+	return t.Kind == css_lexer.TNumber && t.Text == "1"
+}
+
+func (t Token) IsAngle() bool {
+	if t.Kind == css_lexer.TDimension {
+		unit := strings.ToLower(t.DimensionUnit())
+		return unit == "deg" || unit == "grad" || unit == "rad" || unit == "turn"
+	}
+	return false
+}
+
+func CloneTokensWithoutImportRecords(tokensIn []Token) (tokensOut []Token) {
+	for _, t := range tokensIn {
+		if t.Children != nil {
+			children := CloneTokensWithoutImportRecords(*t.Children)
+			t.Children = &children
+		}
+		tokensOut = append(tokensOut, t)
+	}
+	return
+}
+
+func CloneTokensWithImportRecords(
+	tokensIn []Token, importRecordsIn []ast.ImportRecord,
+	tokensOut []Token, importRecordsOut []ast.ImportRecord,
+) ([]Token, []ast.ImportRecord) {
+	// Preallocate the output array if we can
+	if tokensOut == nil {
+		tokensOut = make([]Token, 0, len(tokensIn))
+	}
+
+	for _, t := range tokensIn {
+		// Clear the source mapping if this token is being used in another file
+		t.Loc.Start = 0
+
+		// If this is a URL token, also clone the import record
+		if t.Kind == css_lexer.TURL {
+			importRecordIndex := uint32(len(importRecordsOut))
+			importRecordsOut = append(importRecordsOut, importRecordsIn[t.PayloadIndex])
+			t.PayloadIndex = importRecordIndex
+		}
+
+		// Also search for URL tokens in this token's children
+		if t.Children != nil {
+			var children []Token
+			children, importRecordsOut = CloneTokensWithImportRecords(*t.Children, importRecordsIn, children, importRecordsOut)
+			t.Children = &children
+		}
+
+		tokensOut = append(tokensOut, t)
+	}
+
+	return tokensOut, importRecordsOut
+}
+
+type Rule struct {
+	Data R
+	Loc  logger.Loc
+}
+
+type R interface {
+	Equal(rule R, check *CrossFileEqualityCheck) bool
+	Hash() (uint32, bool)
+}
+
+func RulesEqual(a []Rule, b []Rule, check *CrossFileEqualityCheck) bool {
+	if len(a) != len(b) {
+		return false
+	}
+	for i, ai := range a {
+		if !ai.Data.Equal(b[i].Data, check) {
+			return false
+		}
+	}
+	return true
+}
+
+func HashRules(hash uint32, rules []Rule) uint32 {
+	hash = helpers.HashCombine(hash, uint32(len(rules)))
+	for _, child := range rules {
+		if childHash, ok := child.Data.Hash(); ok {
+			hash = helpers.HashCombine(hash, childHash)
+		} else {
+			hash = helpers.HashCombine(hash, 0)
+		}
+	}
+	return hash
+}
+
+type RAtCharset struct {
+	Encoding string
+}
+
+func (a *RAtCharset) Equal(rule R, check *CrossFileEqualityCheck) bool {
+	b, ok := rule.(*RAtCharset)
+	return ok && a.Encoding == b.Encoding
+}
+
+func (r *RAtCharset) Hash() (uint32, bool) {
+	hash := uint32(1)
+	hash = helpers.HashCombineString(hash, r.Encoding)
+	return hash, true
+}
+
+type ImportConditions struct {
+	// The syntax for "@import" has been extended with optional conditions that
+	// behave as if the imported file was wrapped in a "@layer", "@supports",
+	// and/or "@media" rule. The possible syntax combinations are as follows:
+	//
+	//   @import url(...);
+	//   @import url(...) layer;
+	//   @import url(...) layer(layer-name);
+	//   @import url(...) layer(layer-name) supports(supports-condition);
+	//   @import url(...) layer(layer-name) supports(supports-condition) list-of-media-queries;
+	//   @import url(...) layer(layer-name) list-of-media-queries;
+	//   @import url(...) supports(supports-condition);
+	//   @import url(...) supports(supports-condition) list-of-media-queries;
+	//   @import url(...) list-of-media-queries;
+	//
+	// From: https://developer.mozilla.org/en-US/docs/Web/CSS/@import#syntax
+	Media []Token
+
+	// These two fields will only ever have zero or one tokens. However, they are
+	// implemented as arrays for convenience because most of esbuild's helper
+	// functions that operate on tokens take arrays instead of individual tokens.
+	Layers   []Token
+	Supports []Token
+}
+
+func (c *ImportConditions) CloneWithImportRecords(importRecordsIn []ast.ImportRecord, importRecordsOut []ast.ImportRecord) (ImportConditions, []ast.ImportRecord) {
+	result := ImportConditions{}
+	result.Layers, importRecordsOut = CloneTokensWithImportRecords(c.Layers, importRecordsIn, nil, importRecordsOut)
+	result.Supports, importRecordsOut = CloneTokensWithImportRecords(c.Supports, importRecordsIn, nil, importRecordsOut)
+	result.Media, importRecordsOut = CloneTokensWithImportRecords(c.Media, importRecordsIn, nil, importRecordsOut)
+	return result, importRecordsOut
+}
+
+type RAtImport struct {
+	ImportConditions  *ImportConditions
+	ImportRecordIndex uint32
+}
+
+func (*RAtImport) Equal(rule R, check *CrossFileEqualityCheck) bool {
+	return false
+}
+
+func (r *RAtImport) Hash() (uint32, bool) {
+	return 0, false
+}
+
+type RAtKeyframes struct {
+	AtToken       string
+	Name          ast.LocRef
+	Blocks        []KeyframeBlock
+	CloseBraceLoc logger.Loc
+}
+
+type KeyframeBlock struct {
+	Selectors     []string
+	Rules         []Rule
+	Loc           logger.Loc
+	CloseBraceLoc logger.Loc
+}
+
+func (a *RAtKeyframes) Equal(rule R, check *CrossFileEqualityCheck) bool {
+	if b, ok := rule.(*RAtKeyframes); ok && strings.EqualFold(a.AtToken, b.AtToken) && check.RefsAreEquivalent(a.Name.Ref, b.Name.Ref) && len(a.Blocks) == len(b.Blocks) {
+		for i, ai := range a.Blocks {
+			bi := b.Blocks[i]
+			if len(ai.Selectors) != len(bi.Selectors) {
+				return false
+			}
+			for j, aj := range ai.Selectors {
+				if aj != bi.Selectors[j] {
+					return false
+				}
+			}
+			if !RulesEqual(ai.Rules, bi.Rules, check) {
+				return false
+			}
+		}
+		return true
+	}
+	return false
+}
+
+func (r *RAtKeyframes) Hash() (uint32, bool) {
+	hash := uint32(2)
+	hash = helpers.HashCombineString(hash, r.AtToken)
+	hash = helpers.HashCombine(hash, uint32(len(r.Blocks)))
+	for _, block := range r.Blocks {
+		hash = helpers.HashCombine(hash, uint32(len(block.Selectors)))
+		for _, sel := range block.Selectors {
+			hash = helpers.HashCombineString(hash, sel)
+		}
+		hash = HashRules(hash, block.Rules)
+	}
+	return hash, true
+}
+
+type RKnownAt struct {
+	AtToken       string
+	Prelude       []Token
+	Rules         []Rule
+	CloseBraceLoc logger.Loc
+}
+
+func (a *RKnownAt) Equal(rule R, check *CrossFileEqualityCheck) bool {
+	b, ok := rule.(*RKnownAt)
+	return ok && strings.EqualFold(a.AtToken, b.AtToken) && TokensEqual(a.Prelude, b.Prelude, check) && RulesEqual(a.Rules, b.Rules, check)
+}
+
+func (r *RKnownAt) Hash() (uint32, bool) {
+	hash := uint32(3)
+	hash = helpers.HashCombineString(hash, r.AtToken)
+	hash = HashTokens(hash, r.Prelude)
+	hash = HashRules(hash, r.Rules)
+	return hash, true
+}
+
+type RUnknownAt struct {
+	AtToken string
+	Prelude []Token
+	Block   []Token
+}
+
+func (a *RUnknownAt) Equal(rule R, check *CrossFileEqualityCheck) bool {
+	b, ok := rule.(*RUnknownAt)
+	return ok && strings.EqualFold(a.AtToken, b.AtToken) && TokensEqual(a.Prelude, b.Prelude, check) && TokensEqual(a.Block, b.Block, check)
+}
+
+func (r *RUnknownAt) Hash() (uint32, bool) {
+	hash := uint32(4)
+	hash = helpers.HashCombineString(hash, r.AtToken)
+	hash = HashTokens(hash, r.Prelude)
+	hash = HashTokens(hash, r.Block)
+	return hash, true
+}
+
+type RSelector struct {
+	Selectors     []ComplexSelector
+	Rules         []Rule
+	CloseBraceLoc logger.Loc
+}
+
+func (a *RSelector) Equal(rule R, check *CrossFileEqualityCheck) bool {
+	b, ok := rule.(*RSelector)
+	return ok && ComplexSelectorsEqual(a.Selectors, b.Selectors, check) && RulesEqual(a.Rules, b.Rules, check)
+}
+
+func (r *RSelector) Hash() (uint32, bool) {
+	hash := uint32(5)
+	hash = helpers.HashCombine(hash, uint32(len(r.Selectors)))
+	hash = HashComplexSelectors(hash, r.Selectors)
+	hash = HashRules(hash, r.Rules)
+	return hash, true
+}
+
+type RQualified struct {
+	Prelude       []Token
+	Rules         []Rule
+	CloseBraceLoc logger.Loc
+}
+
+func (a *RQualified) Equal(rule R, check *CrossFileEqualityCheck) bool {
+	b, ok := rule.(*RQualified)
+	return ok && TokensEqual(a.Prelude, b.Prelude, check) && RulesEqual(a.Rules, b.Rules, check)
+}
+
+func (r *RQualified) Hash() (uint32, bool) {
+	hash := uint32(6)
+	hash = HashTokens(hash, r.Prelude)
+	hash = HashRules(hash, r.Rules)
+	return hash, true
+}
+
+type RDeclaration struct {
+	KeyText   string
+	Value     []Token
+	KeyRange  logger.Range
+	Key       D // Compare using this instead of "Key" for speed
+	Important bool
+}
+
+func (a *RDeclaration) Equal(rule R, check *CrossFileEqualityCheck) bool {
+	b, ok := rule.(*RDeclaration)
+	return ok && a.KeyText == b.KeyText && TokensEqual(a.Value, b.Value, check) && a.Important == b.Important
+}
+
+func (r *RDeclaration) Hash() (uint32, bool) {
+	var hash uint32
+	if r.Key == DUnknown {
+		if r.Important {
+			hash = uint32(7)
+		} else {
+			hash = uint32(8)
+		}
+		hash = helpers.HashCombineString(hash, r.KeyText)
+	} else {
+		if r.Important {
+			hash = uint32(9)
+		} else {
+			hash = uint32(10)
+		}
+		hash = helpers.HashCombine(hash, uint32(r.Key))
+	}
+	hash = HashTokens(hash, r.Value)
+	return hash, true
+}
+
+type RBadDeclaration struct {
+	Tokens []Token
+}
+
+func (a *RBadDeclaration) Equal(rule R, check *CrossFileEqualityCheck) bool {
+	b, ok := rule.(*RBadDeclaration)
+	return ok && TokensEqual(a.Tokens, b.Tokens, check)
+}
+
+func (r *RBadDeclaration) Hash() (uint32, bool) {
+	hash := uint32(11)
+	hash = HashTokens(hash, r.Tokens)
+	return hash, true
+}
+
+type RComment struct {
+	Text string
+}
+
+func (a *RComment) Equal(rule R, check *CrossFileEqualityCheck) bool {
+	b, ok := rule.(*RComment)
+	return ok && a.Text == b.Text
+}
+
+func (r *RComment) Hash() (uint32, bool) {
+	hash := uint32(12)
+	hash = helpers.HashCombineString(hash, r.Text)
+	return hash, true
+}
+
+type RAtLayer struct {
+	Names         [][]string
+	Rules         []Rule
+	CloseBraceLoc logger.Loc
+}
+
+func (a *RAtLayer) Equal(rule R, check *CrossFileEqualityCheck) bool {
+	if b, ok := rule.(*RAtLayer); ok && len(a.Names) == len(b.Names) && len(a.Rules) == len(b.Rules) {
+		for i, ai := range a.Names {
+			bi := b.Names[i]
+			if len(ai) != len(bi) {
+				return false
+			}
+			for j, aj := range ai {
+				if aj != bi[j] {
+					return false
+				}
+			}
+		}
+		if !RulesEqual(a.Rules, b.Rules, check) {
+			return false
+		}
+	}
+	return false
+}
+
+func (r *RAtLayer) Hash() (uint32, bool) {
+	hash := uint32(13)
+	hash = helpers.HashCombine(hash, uint32(len(r.Names)))
+	for _, parts := range r.Names {
+		hash = helpers.HashCombine(hash, uint32(len(parts)))
+		for _, part := range parts {
+			hash = helpers.HashCombineString(hash, part)
+		}
+	}
+	hash = HashRules(hash, r.Rules)
+	return hash, true
+}
+
+type ComplexSelector struct {
+	Selectors []CompoundSelector
+}
+
+func ComplexSelectorsEqual(a []ComplexSelector, b []ComplexSelector, check *CrossFileEqualityCheck) bool {
+	if len(a) != len(b) {
+		return false
+	}
+	for i, ai := range a {
+		if !ai.Equal(b[i], check) {
+			return false
+		}
+	}
+	return true
+}
+
+func HashComplexSelectors(hash uint32, selectors []ComplexSelector) uint32 {
+	for _, complex := range selectors {
+		hash = helpers.HashCombine(hash, uint32(len(complex.Selectors)))
+		for _, sel := range complex.Selectors {
+			if sel.TypeSelector != nil {
+				hash = helpers.HashCombineString(hash, sel.TypeSelector.Name.Text)
+			} else {
+				hash = helpers.HashCombine(hash, 0)
+			}
+			hash = helpers.HashCombine(hash, uint32(len(sel.SubclassSelectors)))
+			for _, ss := range sel.SubclassSelectors {
+				hash = helpers.HashCombine(hash, ss.Data.Hash())
+			}
+			hash = helpers.HashCombine(hash, uint32(sel.Combinator.Byte))
+		}
+	}
+	return hash
+}
+
+func (s ComplexSelector) CloneWithoutLeadingCombinator() ComplexSelector {
+	clone := ComplexSelector{Selectors: make([]CompoundSelector, len(s.Selectors))}
+	for i, sel := range s.Selectors {
+		if i == 0 {
+			sel.Combinator = Combinator{}
+		}
+		clone.Selectors[i] = sel.Clone()
+	}
+	return clone
+}
+
+func (sel ComplexSelector) IsRelative() bool {
+	if sel.Selectors[0].Combinator.Byte == 0 {
+		for _, inner := range sel.Selectors {
+			if inner.HasNestingSelector() {
+				return false
+			}
+			for _, ss := range inner.SubclassSelectors {
+				if pseudo, ok := ss.Data.(*SSPseudoClassWithSelectorList); ok {
+					for _, nested := range pseudo.Selectors {
+						if !nested.IsRelative() {
+							return false
+						}
+					}
+				}
+			}
+		}
+	}
+	return true
+}
+
+func tokensContainAmpersandRecursive(tokens []Token) bool {
+	for _, t := range tokens {
+		if t.Kind == css_lexer.TDelimAmpersand {
+			return true
+		}
+		if children := t.Children; children != nil && tokensContainAmpersandRecursive(*children) {
+			return true
+		}
+	}
+	return false
+}
+
+func (sel ComplexSelector) UsesPseudoElement() bool {
+	for _, sel := range sel.Selectors {
+		for _, ss := range sel.SubclassSelectors {
+			if class, ok := ss.Data.(*SSPseudoClass); ok {
+				if class.IsElement {
+					return true
+				}
+
+				// https://www.w3.org/TR/selectors-4/#single-colon-pseudos
+				// The four Level 2 pseudo-elements (::before, ::after, ::first-line,
+				// and ::first-letter) may, for legacy reasons, be represented using
+				// the <pseudo-class-selector> grammar, with only a single ":"
+				// character at their start.
+				switch class.Name {
+				case "before", "after", "first-line", "first-letter":
+					return true
+				}
+			}
+		}
+	}
+	return false
+}
+
+func (a ComplexSelector) Equal(b ComplexSelector, check *CrossFileEqualityCheck) bool {
+	if len(a.Selectors) != len(b.Selectors) {
+		return false
+	}
+
+	for i, ai := range a.Selectors {
+		bi := b.Selectors[i]
+		if ai.HasNestingSelector() != bi.HasNestingSelector() || ai.Combinator.Byte != bi.Combinator.Byte {
+			return false
+		}
+
+		if ats, bts := ai.TypeSelector, bi.TypeSelector; (ats == nil) != (bts == nil) {
+			return false
+		} else if ats != nil && bts != nil && !ats.Equal(*bts) {
+			return false
+		}
+
+		if len(ai.SubclassSelectors) != len(bi.SubclassSelectors) {
+			return false
+		}
+		for j, aj := range ai.SubclassSelectors {
+			if !aj.Data.Equal(bi.SubclassSelectors[j].Data, check) {
+				return false
+			}
+		}
+	}
+
+	return true
+}
+
+type Combinator struct {
+	Loc  logger.Loc
+	Byte uint8 // Optional, may be 0 for no combinator
+}
+
+type CompoundSelector struct {
+	TypeSelector       *NamespacedName
+	SubclassSelectors  []SubclassSelector
+	NestingSelectorLoc ast.Index32 // "&"
+	Combinator         Combinator  // Optional, may be 0
+
+	// If this is true, this is a "&" that was generated by a bare ":local" or ":global"
+	WasEmptyFromLocalOrGlobal bool
+}
+
+func (sel *CompoundSelector) HasNestingSelector() bool {
+	return sel.NestingSelectorLoc.IsValid()
+}
+
+func (sel CompoundSelector) IsSingleAmpersand() bool {
+	return sel.HasNestingSelector() && sel.Combinator.Byte == 0 && sel.TypeSelector == nil && len(sel.SubclassSelectors) == 0
+}
+
+func (sel CompoundSelector) IsInvalidBecauseEmpty() bool {
+	return !sel.HasNestingSelector() && sel.TypeSelector == nil && len(sel.SubclassSelectors) == 0
+}
+
+func (sel CompoundSelector) Range() (r logger.Range) {
+	if sel.Combinator.Byte != 0 {
+		r = logger.Range{Loc: sel.Combinator.Loc, Len: 1}
+	}
+	if sel.TypeSelector != nil {
+		r.ExpandBy(sel.TypeSelector.Range())
+	}
+	if sel.NestingSelectorLoc.IsValid() {
+		r.ExpandBy(logger.Range{Loc: logger.Loc{Start: int32(sel.NestingSelectorLoc.GetIndex())}, Len: 1})
+	}
+	if len(sel.SubclassSelectors) > 0 {
+		for _, ss := range sel.SubclassSelectors {
+			r.ExpandBy(ss.Range)
+		}
+	}
+	return
+}
+
+func (sel CompoundSelector) Clone() CompoundSelector {
+	clone := sel
+
+	if sel.TypeSelector != nil {
+		t := sel.TypeSelector.Clone()
+		clone.TypeSelector = &t
+	}
+
+	if sel.SubclassSelectors != nil {
+		selectors := make([]SubclassSelector, len(sel.SubclassSelectors))
+		for i, ss := range sel.SubclassSelectors {
+			ss.Data = ss.Data.Clone()
+			selectors[i] = ss
+		}
+		clone.SubclassSelectors = selectors
+	}
+
+	return clone
+}
+
+type NameToken struct {
+	Text  string
+	Range logger.Range
+	Kind  css_lexer.T
+}
+
+func (a NameToken) Equal(b NameToken) bool {
+	return a.Text == b.Text && a.Kind == b.Kind
+}
+
+type NamespacedName struct {
+	// If present, this is an identifier or "*" and is followed by a "|" character
+	NamespacePrefix *NameToken
+
+	// This is an identifier or "*"
+	Name NameToken
+}
+
+func (n NamespacedName) Range() logger.Range {
+	if n.NamespacePrefix != nil {
+		loc := n.NamespacePrefix.Range.Loc
+		return logger.Range{Loc: loc, Len: n.Name.Range.End() - loc.Start}
+	}
+	return n.Name.Range
+}
+
+func (n NamespacedName) Clone() NamespacedName {
+	clone := n
+	if n.NamespacePrefix != nil {
+		prefix := *n.NamespacePrefix
+		clone.NamespacePrefix = &prefix
+	}
+	return clone
+}
+
+func (a NamespacedName) Equal(b NamespacedName) bool {
+	return a.Name.Equal(b.Name) && (a.NamespacePrefix == nil) == (b.NamespacePrefix == nil) &&
+		(a.NamespacePrefix == nil || b.NamespacePrefix == nil || a.NamespacePrefix.Equal(b.Name))
+}
+
+type SubclassSelector struct {
+	Data  SS
+	Range logger.Range
+}
+
+type SS interface {
+	Equal(ss SS, check *CrossFileEqualityCheck) bool
+	Hash() uint32
+	Clone() SS
+}
+
+type SSHash struct {
+	Name ast.LocRef
+}
+
+func (a *SSHash) Equal(ss SS, check *CrossFileEqualityCheck) bool {
+	b, ok := ss.(*SSHash)
+	return ok && check.RefsAreEquivalent(a.Name.Ref, b.Name.Ref)
+}
+
+func (ss *SSHash) Hash() uint32 {
+	hash := uint32(1)
+	return hash
+}
+
+func (ss *SSHash) Clone() SS {
+	clone := *ss
+	return &clone
+}
+
+type SSClass struct {
+	Name ast.LocRef
+}
+
+func (a *SSClass) Equal(ss SS, check *CrossFileEqualityCheck) bool {
+	b, ok := ss.(*SSClass)
+	return ok && check.RefsAreEquivalent(a.Name.Ref, b.Name.Ref)
+}
+
+func (ss *SSClass) Hash() uint32 {
+	hash := uint32(2)
+	return hash
+}
+
+func (ss *SSClass) Clone() SS {
+	clone := *ss
+	return &clone
+}
+
+type SSAttribute struct {
+	MatcherOp       string // Either "" or one of: "=" "~=" "|=" "^=" "$=" "*="
+	MatcherValue    string
+	NamespacedName  NamespacedName
+	MatcherModifier byte // Either 0 or one of: 'i' 'I' 's' 'S'
+}
+
+func (a *SSAttribute) Equal(ss SS, check *CrossFileEqualityCheck) bool {
+	b, ok := ss.(*SSAttribute)
+	return ok && a.NamespacedName.Equal(b.NamespacedName) && a.MatcherOp == b.MatcherOp &&
+		a.MatcherValue == b.MatcherValue && a.MatcherModifier == b.MatcherModifier
+}
+
+func (ss *SSAttribute) Hash() uint32 {
+	hash := uint32(3)
+	hash = helpers.HashCombineString(hash, ss.NamespacedName.Name.Text)
+	hash = helpers.HashCombineString(hash, ss.MatcherOp)
+	hash = helpers.HashCombineString(hash, ss.MatcherValue)
+	return hash
+}
+
+func (ss *SSAttribute) Clone() SS {
+	clone := *ss
+	clone.NamespacedName = ss.NamespacedName.Clone()
+	return &clone
+}
+
+type SSPseudoClass struct {
+	Name      string
+	Args      []Token
+	IsElement bool // If true, this is prefixed by "::" instead of ":"
+}
+
+func (a *SSPseudoClass) Equal(ss SS, check *CrossFileEqualityCheck) bool {
+	b, ok := ss.(*SSPseudoClass)
+	return ok && a.Name == b.Name && TokensEqual(a.Args, b.Args, check) && a.IsElement == b.IsElement
+}
+
+func (ss *SSPseudoClass) Hash() uint32 {
+	hash := uint32(4)
+	hash = helpers.HashCombineString(hash, ss.Name)
+	hash = HashTokens(hash, ss.Args)
+	return hash
+}
+
+func (ss *SSPseudoClass) Clone() SS {
+	clone := *ss
+	if ss.Args != nil {
+		ss.Args = CloneTokensWithoutImportRecords(ss.Args)
+	}
+	return &clone
+}
+
+type PseudoClassKind uint8
+
+const (
+	PseudoClassGlobal PseudoClassKind = iota
+	PseudoClassHas
+	PseudoClassIs
+	PseudoClassLocal
+	PseudoClassNot
+	PseudoClassNthChild
+	PseudoClassNthLastChild
+	PseudoClassNthLastOfType
+	PseudoClassNthOfType
+	PseudoClassWhere
+)
+
+func (kind PseudoClassKind) HasNthIndex() bool {
+	return kind >= PseudoClassNthChild && kind <= PseudoClassNthOfType
+}
+
+func (kind PseudoClassKind) String() string {
+	switch kind {
+	case PseudoClassGlobal:
+		return "global"
+	case PseudoClassHas:
+		return "has"
+	case PseudoClassIs:
+		return "is"
+	case PseudoClassLocal:
+		return "local"
+	case PseudoClassNot:
+		return "not"
+	case PseudoClassNthChild:
+		return "nth-child"
+	case PseudoClassNthLastChild:
+		return "nth-last-child"
+	case PseudoClassNthLastOfType:
+		return "nth-last-of-type"
+	case PseudoClassNthOfType:
+		return "nth-of-type"
+	case PseudoClassWhere:
+		return "where"
+	default:
+		panic("Internal error")
+	}
+}
+
+// This is the "An+B" syntax
+type NthIndex struct {
+	A string
+	B string // May be "even" or "odd"
+}
+
+func (index *NthIndex) Minify() {
+	// "even" => "2n"
+	if index.B == "even" {
+		index.A = "2"
+		index.B = ""
+		return
+	}
+
+	// "2n+1" => "odd"
+	if index.A == "2" && index.B == "1" {
+		index.A = ""
+		index.B = "odd"
+		return
+	}
+
+	// "0n+1" => "1"
+	if index.A == "0" {
+		index.A = ""
+		if index.B == "" {
+			// "0n" => "0"
+			index.B = "0"
+		}
+		return
+	}
+
+	// "1n+0" => "1n"
+	if index.B == "0" && index.A != "" {
+		index.B = ""
+	}
+}
+
+// See https://drafts.csswg.org/selectors/#grouping
+type SSPseudoClassWithSelectorList struct {
+	Selectors []ComplexSelector
+	Index     NthIndex
+	Kind      PseudoClassKind
+}
+
+func (a *SSPseudoClassWithSelectorList) Equal(ss SS, check *CrossFileEqualityCheck) bool {
+	b, ok := ss.(*SSPseudoClassWithSelectorList)
+	return ok && a.Kind == b.Kind && a.Index == b.Index && ComplexSelectorsEqual(a.Selectors, b.Selectors, check)
+}
+
+func (ss *SSPseudoClassWithSelectorList) Hash() uint32 {
+	hash := uint32(5)
+	hash = helpers.HashCombine(hash, uint32(ss.Kind))
+	hash = helpers.HashCombineString(hash, ss.Index.A)
+	hash = helpers.HashCombineString(hash, ss.Index.B)
+	hash = HashComplexSelectors(hash, ss.Selectors)
+	return hash
+}
+
+func (ss *SSPseudoClassWithSelectorList) Clone() SS {
+	clone := *ss
+	clone.Selectors = make([]ComplexSelector, len(ss.Selectors))
+	for i, sel := range ss.Selectors {
+		clone.Selectors[i] = sel.CloneWithoutLeadingCombinator()
+	}
+	return &clone
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_ast/css_decl_table.go b/source/vendor/github.com/evanw/esbuild/internal/css_ast/css_decl_table.go
new file mode 100644
index 0000000..231eac4
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_ast/css_decl_table.go
@@ -0,0 +1,698 @@
+package css_ast
+
+import (
+	"strings"
+	"sync"
+
+	"github.com/evanw/esbuild/internal/helpers"
+)
+
+type D uint16
+
+const (
+	DUnknown D = iota
+	DAlignContent
+	DAlignItems
+	DAlignSelf
+	DAlignmentBaseline
+	DAll
+	DAnimation
+	DAnimationDelay
+	DAnimationDirection
+	DAnimationDuration
+	DAnimationFillMode
+	DAnimationIterationCount
+	DAnimationName
+	DAnimationPlayState
+	DAnimationTimingFunction
+	DAppearance
+	DBackdropFilter
+	DBackfaceVisibility
+	DBackground
+	DBackgroundAttachment
+	DBackgroundClip
+	DBackgroundColor
+	DBackgroundImage
+	DBackgroundOrigin
+	DBackgroundPosition
+	DBackgroundPositionX
+	DBackgroundPositionY
+	DBackgroundRepeat
+	DBackgroundSize
+	DBaselineShift
+	DBlockSize
+	DBorder
+	DBorderBlockEnd
+	DBorderBlockEndColor
+	DBorderBlockEndStyle
+	DBorderBlockEndWidth
+	DBorderBlockStart
+	DBorderBlockStartColor
+	DBorderBlockStartStyle
+	DBorderBlockStartWidth
+	DBorderBottom
+	DBorderBottomColor
+	DBorderBottomLeftRadius
+	DBorderBottomRightRadius
+	DBorderBottomStyle
+	DBorderBottomWidth
+	DBorderCollapse
+	DBorderColor
+	DBorderImage
+	DBorderImageOutset
+	DBorderImageRepeat
+	DBorderImageSlice
+	DBorderImageSource
+	DBorderImageWidth
+	DBorderInlineEnd
+	DBorderInlineEndColor
+	DBorderInlineEndStyle
+	DBorderInlineEndWidth
+	DBorderInlineStart
+	DBorderInlineStartColor
+	DBorderInlineStartStyle
+	DBorderInlineStartWidth
+	DBorderLeft
+	DBorderLeftColor
+	DBorderLeftStyle
+	DBorderLeftWidth
+	DBorderRadius
+	DBorderRight
+	DBorderRightColor
+	DBorderRightStyle
+	DBorderRightWidth
+	DBorderSpacing
+	DBorderStyle
+	DBorderTop
+	DBorderTopColor
+	DBorderTopLeftRadius
+	DBorderTopRightRadius
+	DBorderTopStyle
+	DBorderTopWidth
+	DBorderWidth
+	DBottom
+	DBoxDecorationBreak
+	DBoxShadow
+	DBoxSizing
+	DBreakAfter
+	DBreakBefore
+	DBreakInside
+	DCaptionSide
+	DCaretColor
+	DClear
+	DClip
+	DClipPath
+	DClipRule
+	DColor
+	DColorInterpolation
+	DColorInterpolationFilters
+	DColumnCount
+	DColumnFill
+	DColumnGap
+	DColumnRule
+	DColumnRuleColor
+	DColumnRuleStyle
+	DColumnRuleWidth
+	DColumnSpan
+	DColumnWidth
+	DColumns
+	DComposes
+	DContainer
+	DContainerName
+	DContainerType
+	DContent
+	DCounterIncrement
+	DCounterReset
+	DCssFloat
+	DCssText
+	DCursor
+	DDirection
+	DDisplay
+	DDominantBaseline
+	DEmptyCells
+	DFill
+	DFillOpacity
+	DFillRule
+	DFilter
+	DFlex
+	DFlexBasis
+	DFlexDirection
+	DFlexFlow
+	DFlexGrow
+	DFlexShrink
+	DFlexWrap
+	DFloat
+	DFloodColor
+	DFloodOpacity
+	DFont
+	DFontFamily
+	DFontFeatureSettings
+	DFontKerning
+	DFontSize
+	DFontSizeAdjust
+	DFontStretch
+	DFontStyle
+	DFontSynthesis
+	DFontVariant
+	DFontVariantCaps
+	DFontVariantEastAsian
+	DFontVariantLigatures
+	DFontVariantNumeric
+	DFontVariantPosition
+	DFontWeight
+	DGap
+	DGlyphOrientationVertical
+	DGrid
+	DGridArea
+	DGridAutoColumns
+	DGridAutoFlow
+	DGridAutoRows
+	DGridColumn
+	DGridColumnEnd
+	DGridColumnGap
+	DGridColumnStart
+	DGridGap
+	DGridRow
+	DGridRowEnd
+	DGridRowGap
+	DGridRowStart
+	DGridTemplate
+	DGridTemplateAreas
+	DGridTemplateColumns
+	DGridTemplateRows
+	DHeight
+	DHyphens
+	DImageOrientation
+	DImageRendering
+	DInitialLetter
+	DInlineSize
+	DInset
+	DJustifyContent
+	DJustifyItems
+	DJustifySelf
+	DLeft
+	DLetterSpacing
+	DLightingColor
+	DLineBreak
+	DLineHeight
+	DListStyle
+	DListStyleImage
+	DListStylePosition
+	DListStyleType
+	DMargin
+	DMarginBlockEnd
+	DMarginBlockStart
+	DMarginBottom
+	DMarginInlineEnd
+	DMarginInlineStart
+	DMarginLeft
+	DMarginRight
+	DMarginTop
+	DMarker
+	DMarkerEnd
+	DMarkerMid
+	DMarkerStart
+	DMask
+	DMaskComposite
+	DMaskImage
+	DMaskOrigin
+	DMaskPosition
+	DMaskRepeat
+	DMaskSize
+	DMaskType
+	DMaxBlockSize
+	DMaxHeight
+	DMaxInlineSize
+	DMaxWidth
+	DMinBlockSize
+	DMinHeight
+	DMinInlineSize
+	DMinWidth
+	DObjectFit
+	DObjectPosition
+	DOpacity
+	DOrder
+	DOrphans
+	DOutline
+	DOutlineColor
+	DOutlineOffset
+	DOutlineStyle
+	DOutlineWidth
+	DOverflow
+	DOverflowAnchor
+	DOverflowWrap
+	DOverflowX
+	DOverflowY
+	DOverscrollBehavior
+	DOverscrollBehaviorBlock
+	DOverscrollBehaviorInline
+	DOverscrollBehaviorX
+	DOverscrollBehaviorY
+	DPadding
+	DPaddingBlockEnd
+	DPaddingBlockStart
+	DPaddingBottom
+	DPaddingInlineEnd
+	DPaddingInlineStart
+	DPaddingLeft
+	DPaddingRight
+	DPaddingTop
+	DPageBreakAfter
+	DPageBreakBefore
+	DPageBreakInside
+	DPaintOrder
+	DPerspective
+	DPerspectiveOrigin
+	DPlaceContent
+	DPlaceItems
+	DPlaceSelf
+	DPointerEvents
+	DPosition
+	DPrintColorAdjust
+	DQuotes
+	DResize
+	DRight
+	DRotate
+	DRowGap
+	DRubyAlign
+	DRubyPosition
+	DScale
+	DScrollBehavior
+	DShapeRendering
+	DStopColor
+	DStopOpacity
+	DStroke
+	DStrokeDasharray
+	DStrokeDashoffset
+	DStrokeLinecap
+	DStrokeLinejoin
+	DStrokeMiterlimit
+	DStrokeOpacity
+	DStrokeWidth
+	DTabSize
+	DTableLayout
+	DTextAlign
+	DTextAlignLast
+	DTextAnchor
+	DTextCombineUpright
+	DTextDecoration
+	DTextDecorationColor
+	DTextDecorationLine
+	DTextDecorationSkip
+	DTextDecorationStyle
+	DTextEmphasis
+	DTextEmphasisColor
+	DTextEmphasisPosition
+	DTextEmphasisStyle
+	DTextIndent
+	DTextJustify
+	DTextOrientation
+	DTextOverflow
+	DTextRendering
+	DTextShadow
+	DTextSizeAdjust
+	DTextTransform
+	DTextUnderlinePosition
+	DTop
+	DTouchAction
+	DTransform
+	DTransformBox
+	DTransformOrigin
+	DTransformStyle
+	DTransition
+	DTransitionDelay
+	DTransitionDuration
+	DTransitionProperty
+	DTransitionTimingFunction
+	DTranslate
+	DUnicodeBidi
+	DUserSelect
+	DVerticalAlign
+	DVisibility
+	DWhiteSpace
+	DWidows
+	DWidth
+	DWillChange
+	DWordBreak
+	DWordSpacing
+	DWordWrap
+	DWritingMode
+	DZIndex
+	DZoom
+)
+
+var KnownDeclarations = map[string]D{
+	"align-content":               DAlignContent,
+	"align-items":                 DAlignItems,
+	"align-self":                  DAlignSelf,
+	"alignment-baseline":          DAlignmentBaseline,
+	"all":                         DAll,
+	"animation":                   DAnimation,
+	"animation-delay":             DAnimationDelay,
+	"animation-direction":         DAnimationDirection,
+	"animation-duration":          DAnimationDuration,
+	"animation-fill-mode":         DAnimationFillMode,
+	"animation-iteration-count":   DAnimationIterationCount,
+	"animation-name":              DAnimationName,
+	"animation-play-state":        DAnimationPlayState,
+	"animation-timing-function":   DAnimationTimingFunction,
+	"appearance":                  DAppearance,
+	"backdrop-filter":             DBackdropFilter,
+	"backface-visibility":         DBackfaceVisibility,
+	"background":                  DBackground,
+	"background-attachment":       DBackgroundAttachment,
+	"background-clip":             DBackgroundClip,
+	"background-color":            DBackgroundColor,
+	"background-image":            DBackgroundImage,
+	"background-origin":           DBackgroundOrigin,
+	"background-position":         DBackgroundPosition,
+	"background-position-x":       DBackgroundPositionX,
+	"background-position-y":       DBackgroundPositionY,
+	"background-repeat":           DBackgroundRepeat,
+	"background-size":             DBackgroundSize,
+	"baseline-shift":              DBaselineShift,
+	"block-size":                  DBlockSize,
+	"border":                      DBorder,
+	"border-block-end":            DBorderBlockEnd,
+	"border-block-end-color":      DBorderBlockEndColor,
+	"border-block-end-style":      DBorderBlockEndStyle,
+	"border-block-end-width":      DBorderBlockEndWidth,
+	"border-block-start":          DBorderBlockStart,
+	"border-block-start-color":    DBorderBlockStartColor,
+	"border-block-start-style":    DBorderBlockStartStyle,
+	"border-block-start-width":    DBorderBlockStartWidth,
+	"border-bottom":               DBorderBottom,
+	"border-bottom-color":         DBorderBottomColor,
+	"border-bottom-left-radius":   DBorderBottomLeftRadius,
+	"border-bottom-right-radius":  DBorderBottomRightRadius,
+	"border-bottom-style":         DBorderBottomStyle,
+	"border-bottom-width":         DBorderBottomWidth,
+	"border-collapse":             DBorderCollapse,
+	"border-color":                DBorderColor,
+	"border-image":                DBorderImage,
+	"border-image-outset":         DBorderImageOutset,
+	"border-image-repeat":         DBorderImageRepeat,
+	"border-image-slice":          DBorderImageSlice,
+	"border-image-source":         DBorderImageSource,
+	"border-image-width":          DBorderImageWidth,
+	"border-inline-end":           DBorderInlineEnd,
+	"border-inline-end-color":     DBorderInlineEndColor,
+	"border-inline-end-style":     DBorderInlineEndStyle,
+	"border-inline-end-width":     DBorderInlineEndWidth,
+	"border-inline-start":         DBorderInlineStart,
+	"border-inline-start-color":   DBorderInlineStartColor,
+	"border-inline-start-style":   DBorderInlineStartStyle,
+	"border-inline-start-width":   DBorderInlineStartWidth,
+	"border-left":                 DBorderLeft,
+	"border-left-color":           DBorderLeftColor,
+	"border-left-style":           DBorderLeftStyle,
+	"border-left-width":           DBorderLeftWidth,
+	"border-radius":               DBorderRadius,
+	"border-right":                DBorderRight,
+	"border-right-color":          DBorderRightColor,
+	"border-right-style":          DBorderRightStyle,
+	"border-right-width":          DBorderRightWidth,
+	"border-spacing":              DBorderSpacing,
+	"border-style":                DBorderStyle,
+	"border-top":                  DBorderTop,
+	"border-top-color":            DBorderTopColor,
+	"border-top-left-radius":      DBorderTopLeftRadius,
+	"border-top-right-radius":     DBorderTopRightRadius,
+	"border-top-style":            DBorderTopStyle,
+	"border-top-width":            DBorderTopWidth,
+	"border-width":                DBorderWidth,
+	"bottom":                      DBottom,
+	"box-decoration-break":        DBoxDecorationBreak,
+	"box-shadow":                  DBoxShadow,
+	"box-sizing":                  DBoxSizing,
+	"break-after":                 DBreakAfter,
+	"break-before":                DBreakBefore,
+	"break-inside":                DBreakInside,
+	"caption-side":                DCaptionSide,
+	"caret-color":                 DCaretColor,
+	"clear":                       DClear,
+	"clip":                        DClip,
+	"clip-path":                   DClipPath,
+	"clip-rule":                   DClipRule,
+	"color":                       DColor,
+	"color-interpolation":         DColorInterpolation,
+	"color-interpolation-filters": DColorInterpolationFilters,
+	"column-count":                DColumnCount,
+	"column-fill":                 DColumnFill,
+	"column-gap":                  DColumnGap,
+	"column-rule":                 DColumnRule,
+	"column-rule-color":           DColumnRuleColor,
+	"column-rule-style":           DColumnRuleStyle,
+	"column-rule-width":           DColumnRuleWidth,
+	"column-span":                 DColumnSpan,
+	"column-width":                DColumnWidth,
+	"columns":                     DColumns,
+	"composes":                    DComposes,
+	"container":                   DContainer,
+	"container-name":              DContainerName,
+	"container-type":              DContainerType,
+	"content":                     DContent,
+	"counter-increment":           DCounterIncrement,
+	"counter-reset":               DCounterReset,
+	"css-float":                   DCssFloat,
+	"css-text":                    DCssText,
+	"cursor":                      DCursor,
+	"direction":                   DDirection,
+	"display":                     DDisplay,
+	"dominant-baseline":           DDominantBaseline,
+	"empty-cells":                 DEmptyCells,
+	"fill":                        DFill,
+	"fill-opacity":                DFillOpacity,
+	"fill-rule":                   DFillRule,
+	"filter":                      DFilter,
+	"flex":                        DFlex,
+	"flex-basis":                  DFlexBasis,
+	"flex-direction":              DFlexDirection,
+	"flex-flow":                   DFlexFlow,
+	"flex-grow":                   DFlexGrow,
+	"flex-shrink":                 DFlexShrink,
+	"flex-wrap":                   DFlexWrap,
+	"float":                       DFloat,
+	"flood-color":                 DFloodColor,
+	"flood-opacity":               DFloodOpacity,
+	"font":                        DFont,
+	"font-family":                 DFontFamily,
+	"font-feature-settings":       DFontFeatureSettings,
+	"font-kerning":                DFontKerning,
+	"font-size":                   DFontSize,
+	"font-size-adjust":            DFontSizeAdjust,
+	"font-stretch":                DFontStretch,
+	"font-style":                  DFontStyle,
+	"font-synthesis":              DFontSynthesis,
+	"font-variant":                DFontVariant,
+	"font-variant-caps":           DFontVariantCaps,
+	"font-variant-east-asian":     DFontVariantEastAsian,
+	"font-variant-ligatures":      DFontVariantLigatures,
+	"font-variant-numeric":        DFontVariantNumeric,
+	"font-variant-position":       DFontVariantPosition,
+	"font-weight":                 DFontWeight,
+	"gap":                         DGap,
+	"glyph-orientation-vertical":  DGlyphOrientationVertical,
+	"grid":                        DGrid,
+	"grid-area":                   DGridArea,
+	"grid-auto-columns":           DGridAutoColumns,
+	"grid-auto-flow":              DGridAutoFlow,
+	"grid-auto-rows":              DGridAutoRows,
+	"grid-column":                 DGridColumn,
+	"grid-column-end":             DGridColumnEnd,
+	"grid-column-gap":             DGridColumnGap,
+	"grid-column-start":           DGridColumnStart,
+	"grid-gap":                    DGridGap,
+	"grid-row":                    DGridRow,
+	"grid-row-end":                DGridRowEnd,
+	"grid-row-gap":                DGridRowGap,
+	"grid-row-start":              DGridRowStart,
+	"grid-template":               DGridTemplate,
+	"grid-template-areas":         DGridTemplateAreas,
+	"grid-template-columns":       DGridTemplateColumns,
+	"grid-template-rows":          DGridTemplateRows,
+	"height":                      DHeight,
+	"hyphens":                     DHyphens,
+	"image-orientation":           DImageOrientation,
+	"image-rendering":             DImageRendering,
+	"initial-letter":              DInitialLetter,
+	"inline-size":                 DInlineSize,
+	"inset":                       DInset,
+	"justify-content":             DJustifyContent,
+	"justify-items":               DJustifyItems,
+	"justify-self":                DJustifySelf,
+	"left":                        DLeft,
+	"letter-spacing":              DLetterSpacing,
+	"lighting-color":              DLightingColor,
+	"line-break":                  DLineBreak,
+	"line-height":                 DLineHeight,
+	"list-style":                  DListStyle,
+	"list-style-image":            DListStyleImage,
+	"list-style-position":         DListStylePosition,
+	"list-style-type":             DListStyleType,
+	"margin":                      DMargin,
+	"margin-block-end":            DMarginBlockEnd,
+	"margin-block-start":          DMarginBlockStart,
+	"margin-bottom":               DMarginBottom,
+	"margin-inline-end":           DMarginInlineEnd,
+	"margin-inline-start":         DMarginInlineStart,
+	"margin-left":                 DMarginLeft,
+	"margin-right":                DMarginRight,
+	"margin-top":                  DMarginTop,
+	"marker":                      DMarker,
+	"marker-end":                  DMarkerEnd,
+	"marker-mid":                  DMarkerMid,
+	"marker-start":                DMarkerStart,
+	"mask":                        DMask,
+	"mask-composite":              DMaskComposite,
+	"mask-image":                  DMaskImage,
+	"mask-origin":                 DMaskOrigin,
+	"mask-position":               DMaskPosition,
+	"mask-repeat":                 DMaskRepeat,
+	"mask-size":                   DMaskSize,
+	"mask-type":                   DMaskType,
+	"max-block-size":              DMaxBlockSize,
+	"max-height":                  DMaxHeight,
+	"max-inline-size":             DMaxInlineSize,
+	"max-width":                   DMaxWidth,
+	"min-block-size":              DMinBlockSize,
+	"min-height":                  DMinHeight,
+	"min-inline-size":             DMinInlineSize,
+	"min-width":                   DMinWidth,
+	"object-fit":                  DObjectFit,
+	"object-position":             DObjectPosition,
+	"opacity":                     DOpacity,
+	"order":                       DOrder,
+	"orphans":                     DOrphans,
+	"outline":                     DOutline,
+	"outline-color":               DOutlineColor,
+	"outline-offset":              DOutlineOffset,
+	"outline-style":               DOutlineStyle,
+	"outline-width":               DOutlineWidth,
+	"overflow":                    DOverflow,
+	"overflow-anchor":             DOverflowAnchor,
+	"overflow-wrap":               DOverflowWrap,
+	"overflow-x":                  DOverflowX,
+	"overflow-y":                  DOverflowY,
+	"overscroll-behavior":         DOverscrollBehavior,
+	"overscroll-behavior-block":   DOverscrollBehaviorBlock,
+	"overscroll-behavior-inline":  DOverscrollBehaviorInline,
+	"overscroll-behavior-x":       DOverscrollBehaviorX,
+	"overscroll-behavior-y":       DOverscrollBehaviorY,
+	"padding":                     DPadding,
+	"padding-block-end":           DPaddingBlockEnd,
+	"padding-block-start":         DPaddingBlockStart,
+	"padding-bottom":              DPaddingBottom,
+	"padding-inline-end":          DPaddingInlineEnd,
+	"padding-inline-start":        DPaddingInlineStart,
+	"padding-left":                DPaddingLeft,
+	"padding-right":               DPaddingRight,
+	"padding-top":                 DPaddingTop,
+	"page-break-after":            DPageBreakAfter,
+	"page-break-before":           DPageBreakBefore,
+	"page-break-inside":           DPageBreakInside,
+	"paint-order":                 DPaintOrder,
+	"perspective":                 DPerspective,
+	"perspective-origin":          DPerspectiveOrigin,
+	"place-content":               DPlaceContent,
+	"place-items":                 DPlaceItems,
+	"place-self":                  DPlaceSelf,
+	"pointer-events":              DPointerEvents,
+	"position":                    DPosition,
+	"print-color-adjust":          DPrintColorAdjust,
+	"quotes":                      DQuotes,
+	"resize":                      DResize,
+	"right":                       DRight,
+	"rotate":                      DRotate,
+	"row-gap":                     DRowGap,
+	"ruby-align":                  DRubyAlign,
+	"ruby-position":               DRubyPosition,
+	"scale":                       DScale,
+	"scroll-behavior":             DScrollBehavior,
+	"shape-rendering":             DShapeRendering,
+	"stop-color":                  DStopColor,
+	"stop-opacity":                DStopOpacity,
+	"stroke":                      DStroke,
+	"stroke-dasharray":            DStrokeDasharray,
+	"stroke-dashoffset":           DStrokeDashoffset,
+	"stroke-linecap":              DStrokeLinecap,
+	"stroke-linejoin":             DStrokeLinejoin,
+	"stroke-miterlimit":           DStrokeMiterlimit,
+	"stroke-opacity":              DStrokeOpacity,
+	"stroke-width":                DStrokeWidth,
+	"tab-size":                    DTabSize,
+	"table-layout":                DTableLayout,
+	"text-align":                  DTextAlign,
+	"text-align-last":             DTextAlignLast,
+	"text-anchor":                 DTextAnchor,
+	"text-combine-upright":        DTextCombineUpright,
+	"text-decoration":             DTextDecoration,
+	"text-decoration-color":       DTextDecorationColor,
+	"text-decoration-line":        DTextDecorationLine,
+	"text-decoration-skip":        DTextDecorationSkip,
+	"text-decoration-style":       DTextDecorationStyle,
+	"text-emphasis":               DTextEmphasis,
+	"text-emphasis-color":         DTextEmphasisColor,
+	"text-emphasis-position":      DTextEmphasisPosition,
+	"text-emphasis-style":         DTextEmphasisStyle,
+	"text-indent":                 DTextIndent,
+	"text-justify":                DTextJustify,
+	"text-orientation":            DTextOrientation,
+	"text-overflow":               DTextOverflow,
+	"text-rendering":              DTextRendering,
+	"text-shadow":                 DTextShadow,
+	"text-size-adjust":            DTextSizeAdjust,
+	"text-transform":              DTextTransform,
+	"text-underline-position":     DTextUnderlinePosition,
+	"top":                         DTop,
+	"touch-action":                DTouchAction,
+	"transform":                   DTransform,
+	"transform-box":               DTransformBox,
+	"transform-origin":            DTransformOrigin,
+	"transform-style":             DTransformStyle,
+	"transition":                  DTransition,
+	"transition-delay":            DTransitionDelay,
+	"transition-duration":         DTransitionDuration,
+	"transition-property":         DTransitionProperty,
+	"transition-timing-function":  DTransitionTimingFunction,
+	"translate":                   DTranslate,
+	"unicode-bidi":                DUnicodeBidi,
+	"user-select":                 DUserSelect,
+	"vertical-align":              DVerticalAlign,
+	"visibility":                  DVisibility,
+	"white-space":                 DWhiteSpace,
+	"widows":                      DWidows,
+	"width":                       DWidth,
+	"will-change":                 DWillChange,
+	"word-break":                  DWordBreak,
+	"word-spacing":                DWordSpacing,
+	"word-wrap":                   DWordWrap,
+	"writing-mode":                DWritingMode,
+	"z-index":                     DZIndex,
+	"zoom":                        DZoom,
+}
+
+var typoDetector *helpers.TypoDetector
+var typoDetectorMutex sync.Mutex
+
+func MaybeCorrectDeclarationTypo(text string) (string, bool) {
+	// Ignore CSS variables, which should not be corrected to CSS properties
+	if strings.HasPrefix(text, "--") {
+		return "", false
+	}
+
+	typoDetectorMutex.Lock()
+	defer typoDetectorMutex.Unlock()
+
+	// Lazily-initialize the typo detector for speed when it's not needed
+	if typoDetector == nil {
+		valid := make([]string, 0, len(KnownDeclarations))
+		for key := range KnownDeclarations {
+			valid = append(valid, key)
+		}
+		detector := helpers.MakeTypoDetector(valid)
+		typoDetector = &detector
+	}
+
+	return typoDetector.MaybeCorrectTypo(text)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_lexer/css_lexer.go b/source/vendor/github.com/evanw/esbuild/internal/css_lexer/css_lexer.go
new file mode 100644
index 0000000..9e8c088
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_lexer/css_lexer.go
@@ -0,0 +1,1081 @@
+package css_lexer
+
+import (
+	"strings"
+	"unicode/utf8"
+
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+// The lexer converts a source file to a stream of tokens. Unlike esbuild's
+// JavaScript lexer, this CSS lexer runs to completion before the CSS parser
+// begins, resulting in a single array of all tokens in the file.
+
+type T uint8
+
+const eof = -1
+
+const (
+	TEndOfFile T = iota
+
+	TAtKeyword
+	TUnterminatedString
+	TBadURL
+	TCDC // "-->"
+	TCDO // "<!--"
+	TCloseBrace
+	TCloseBracket
+	TCloseParen
+	TColon
+	TComma
+	TDelim
+	TDelimAmpersand
+	TDelimAsterisk
+	TDelimBar
+	TDelimCaret
+	TDelimDollar
+	TDelimDot
+	TDelimEquals
+	TDelimExclamation
+	TDelimGreaterThan
+	TDelimMinus
+	TDelimPlus
+	TDelimSlash
+	TDelimTilde
+	TDimension
+	TFunction
+	THash
+	TIdent
+	TNumber
+	TOpenBrace
+	TOpenBracket
+	TOpenParen
+	TPercentage
+	TSemicolon
+	TString
+	TURL
+	TWhitespace
+
+	// This is never something that the lexer generates directly. Instead this is
+	// an esbuild-specific token for global/local names that "TIdent" tokens may
+	// be changed into.
+	TSymbol
+)
+
+var tokenToString = []string{
+	"end of file",
+	"@-keyword",
+	"bad string token",
+	"bad URL token",
+	"\"-->\"",
+	"\"<!--\"",
+	"\"}\"",
+	"\"]\"",
+	"\")\"",
+	"\":\"",
+	"\",\"",
+	"delimiter",
+	"\"&\"",
+	"\"*\"",
+	"\"|\"",
+	"\"^\"",
+	"\"$\"",
+	"\".\"",
+	"\"=\"",
+	"\"!\"",
+	"\">\"",
+	"\"-\"",
+	"\"+\"",
+	"\"/\"",
+	"\"~\"",
+	"dimension",
+	"function token",
+	"hash token",
+	"identifier",
+	"number",
+	"\"{\"",
+	"\"[\"",
+	"\"(\"",
+	"percentage",
+	"\";\"",
+	"string token",
+	"URL token",
+	"whitespace",
+
+	"identifier",
+}
+
+func (t T) String() string {
+	return tokenToString[t]
+}
+
+func (t T) IsNumeric() bool {
+	return t == TNumber || t == TPercentage || t == TDimension
+}
+
+type TokenFlags uint8
+
+const (
+	IsID TokenFlags = 1 << iota
+	DidWarnAboutSingleLineComment
+)
+
+// This token struct is designed to be memory-efficient. It just references a
+// range in the input file instead of directly containing the substring of text
+// since a range takes up less memory than a string.
+type Token struct {
+	Range      logger.Range // 8 bytes
+	UnitOffset uint16       // 2 bytes
+	Kind       T            // 1 byte
+	Flags      TokenFlags   // 1 byte
+}
+
+func (token Token) DecodedText(contents string) string {
+	raw := contents[token.Range.Loc.Start:token.Range.End()]
+
+	switch token.Kind {
+	case TIdent, TDimension:
+		return decodeEscapesInToken(raw)
+
+	case TAtKeyword, THash:
+		return decodeEscapesInToken(raw[1:])
+
+	case TFunction:
+		return decodeEscapesInToken(raw[:len(raw)-1])
+
+	case TString:
+		return decodeEscapesInToken(raw[1 : len(raw)-1])
+
+	case TURL:
+		start := 4
+		end := len(raw)
+
+		// Note: URL tokens with syntax errors may not have a trailing ")"
+		if raw[end-1] == ')' {
+			end--
+		}
+
+		// Trim leading and trailing whitespace
+		for start < end && isWhitespace(rune(raw[start])) {
+			start++
+		}
+		for start < end && isWhitespace(rune(raw[end-1])) {
+			end--
+		}
+
+		return decodeEscapesInToken(raw[start:end])
+	}
+
+	return raw
+}
+
+type lexer struct {
+	Options
+	log                     logger.Log
+	source                  logger.Source
+	allComments             []logger.Range
+	legalCommentsBefore     []Comment
+	sourceMappingURL        logger.Span
+	tracker                 logger.LineColumnTracker
+	approximateNewlineCount int
+	current                 int
+	oldSingleLineCommentEnd logger.Loc
+	codePoint               rune
+	Token                   Token
+}
+
+type Comment struct {
+	Text            string
+	Loc             logger.Loc
+	TokenIndexAfter uint32
+}
+
+type TokenizeResult struct {
+	Tokens               []Token
+	AllComments          []logger.Range
+	LegalComments        []Comment
+	SourceMapComment     logger.Span
+	ApproximateLineCount int32
+}
+
+type Options struct {
+	RecordAllComments bool
+}
+
+func Tokenize(log logger.Log, source logger.Source, options Options) TokenizeResult {
+	lexer := lexer{
+		Options: options,
+		log:     log,
+		source:  source,
+		tracker: logger.MakeLineColumnTracker(&source),
+	}
+	lexer.step()
+
+	// The U+FEFF character is usually a zero-width non-breaking space. However,
+	// when it's used at the start of a text stream it is called a BOM (byte order
+	// mark) instead and indicates that the text stream is UTF-8 encoded. This is
+	// problematic for us because CSS does not treat U+FEFF as whitespace. Only
+	// " \t\r\n\f" characters are treated as whitespace. Skip over the BOM if it
+	// is present so it doesn't cause us trouble when we try to parse it.
+	if lexer.codePoint == '\uFEFF' {
+		lexer.step()
+	}
+
+	lexer.next()
+	var tokens []Token
+	var legalComments []Comment
+	for lexer.Token.Kind != TEndOfFile {
+		if lexer.legalCommentsBefore != nil {
+			for _, comment := range lexer.legalCommentsBefore {
+				comment.TokenIndexAfter = uint32(len(tokens))
+				legalComments = append(legalComments, comment)
+			}
+			lexer.legalCommentsBefore = nil
+		}
+		tokens = append(tokens, lexer.Token)
+		lexer.next()
+	}
+	if lexer.legalCommentsBefore != nil {
+		for _, comment := range lexer.legalCommentsBefore {
+			comment.TokenIndexAfter = uint32(len(tokens))
+			legalComments = append(legalComments, comment)
+		}
+		lexer.legalCommentsBefore = nil
+	}
+	return TokenizeResult{
+		Tokens:               tokens,
+		AllComments:          lexer.allComments,
+		LegalComments:        legalComments,
+		ApproximateLineCount: int32(lexer.approximateNewlineCount) + 1,
+		SourceMapComment:     lexer.sourceMappingURL,
+	}
+}
+
+func (lexer *lexer) step() {
+	codePoint, width := utf8.DecodeRuneInString(lexer.source.Contents[lexer.current:])
+
+	// Use -1 to indicate the end of the file
+	if width == 0 {
+		codePoint = eof
+	}
+
+	// Track the approximate number of newlines in the file so we can preallocate
+	// the line offset table in the printer for source maps. The line offset table
+	// is the #1 highest allocation in the heap profile, so this is worth doing.
+	// This count is approximate because it handles "\n" and "\r\n" (the common
+	// cases) but not "\r" or "\u2028" or "\u2029". Getting this wrong is harmless
+	// because it's only a preallocation. The array will just grow if it's too small.
+	if codePoint == '\n' {
+		lexer.approximateNewlineCount++
+	}
+
+	lexer.codePoint = codePoint
+	lexer.Token.Range.Len = int32(lexer.current) - lexer.Token.Range.Loc.Start
+	lexer.current += width
+}
+
+func (lexer *lexer) next() {
+	// Reference: https://www.w3.org/TR/css-syntax-3/
+
+	for {
+		lexer.Token = Token{Range: logger.Range{Loc: logger.Loc{Start: lexer.Token.Range.End()}}}
+
+		switch lexer.codePoint {
+		case eof:
+			lexer.Token.Kind = TEndOfFile
+
+		case '/':
+			lexer.step()
+			switch lexer.codePoint {
+			case '*':
+				lexer.step()
+				lexer.consumeToEndOfMultiLineComment(lexer.Token.Range)
+				continue
+			case '/':
+				// Warn when people use "//" comments, which are invalid in CSS
+				loc := lexer.Token.Range.Loc
+				if loc.Start >= lexer.oldSingleLineCommentEnd.Start {
+					contents := lexer.source.Contents
+					end := lexer.current
+					for end < len(contents) && !isNewline(rune(contents[end])) {
+						end++
+					}
+					lexer.log.AddID(logger.MsgID_CSS_JSCommentInCSS, logger.Warning, &lexer.tracker, logger.Range{Loc: loc, Len: 2},
+						"Comments in CSS use \"/* ... */\" instead of \"//\"")
+					lexer.oldSingleLineCommentEnd.Start = int32(end)
+					lexer.Token.Flags |= DidWarnAboutSingleLineComment
+				}
+			}
+			lexer.Token.Kind = TDelimSlash
+
+		case ' ', '\t', '\n', '\r', '\f':
+			lexer.step()
+			for {
+				if isWhitespace(lexer.codePoint) {
+					lexer.step()
+				} else if lexer.codePoint == '/' && lexer.current < len(lexer.source.Contents) && lexer.source.Contents[lexer.current] == '*' {
+					startRange := logger.Range{Loc: logger.Loc{Start: lexer.Token.Range.End()}, Len: 2}
+					lexer.step()
+					lexer.step()
+					lexer.consumeToEndOfMultiLineComment(startRange)
+				} else {
+					break
+				}
+			}
+			lexer.Token.Kind = TWhitespace
+
+		case '"', '\'':
+			lexer.Token.Kind = lexer.consumeString()
+
+		case '#':
+			lexer.step()
+			if IsNameContinue(lexer.codePoint) || lexer.isValidEscape() {
+				lexer.Token.Kind = THash
+				if lexer.wouldStartIdentifier() {
+					lexer.Token.Flags |= IsID
+				}
+				lexer.consumeName()
+			} else {
+				lexer.Token.Kind = TDelim
+			}
+
+		case '(':
+			lexer.step()
+			lexer.Token.Kind = TOpenParen
+
+		case ')':
+			lexer.step()
+			lexer.Token.Kind = TCloseParen
+
+		case '[':
+			lexer.step()
+			lexer.Token.Kind = TOpenBracket
+
+		case ']':
+			lexer.step()
+			lexer.Token.Kind = TCloseBracket
+
+		case '{':
+			lexer.step()
+			lexer.Token.Kind = TOpenBrace
+
+		case '}':
+			lexer.step()
+			lexer.Token.Kind = TCloseBrace
+
+		case ',':
+			lexer.step()
+			lexer.Token.Kind = TComma
+
+		case ':':
+			lexer.step()
+			lexer.Token.Kind = TColon
+
+		case ';':
+			lexer.step()
+			lexer.Token.Kind = TSemicolon
+
+		case '+':
+			if lexer.wouldStartNumber() {
+				lexer.Token.Kind = lexer.consumeNumeric()
+			} else {
+				lexer.step()
+				lexer.Token.Kind = TDelimPlus
+			}
+
+		case '.':
+			if lexer.wouldStartNumber() {
+				lexer.Token.Kind = lexer.consumeNumeric()
+			} else {
+				lexer.step()
+				lexer.Token.Kind = TDelimDot
+			}
+
+		case '-':
+			if lexer.wouldStartNumber() {
+				lexer.Token.Kind = lexer.consumeNumeric()
+			} else if lexer.current+2 <= len(lexer.source.Contents) && lexer.source.Contents[lexer.current:lexer.current+2] == "->" {
+				lexer.step()
+				lexer.step()
+				lexer.step()
+				lexer.Token.Kind = TCDC
+			} else if lexer.wouldStartIdentifier() {
+				lexer.Token.Kind = lexer.consumeIdentLike()
+			} else {
+				lexer.step()
+				lexer.Token.Kind = TDelimMinus
+			}
+
+		case '<':
+			if lexer.current+3 <= len(lexer.source.Contents) && lexer.source.Contents[lexer.current:lexer.current+3] == "!--" {
+				lexer.step()
+				lexer.step()
+				lexer.step()
+				lexer.step()
+				lexer.Token.Kind = TCDO
+			} else {
+				lexer.step()
+				lexer.Token.Kind = TDelim
+			}
+
+		case '@':
+			lexer.step()
+			if lexer.wouldStartIdentifier() {
+				lexer.consumeName()
+				lexer.Token.Kind = TAtKeyword
+			} else {
+				lexer.Token.Kind = TDelim
+			}
+
+		case '\\':
+			if lexer.isValidEscape() {
+				lexer.Token.Kind = lexer.consumeIdentLike()
+			} else {
+				lexer.step()
+				lexer.log.AddError(&lexer.tracker, lexer.Token.Range, "Invalid escape")
+				lexer.Token.Kind = TDelim
+			}
+
+		case '0', '1', '2', '3', '4', '5', '6', '7', '8', '9':
+			lexer.Token.Kind = lexer.consumeNumeric()
+
+		case '>':
+			lexer.step()
+			lexer.Token.Kind = TDelimGreaterThan
+
+		case '~':
+			lexer.step()
+			lexer.Token.Kind = TDelimTilde
+
+		case '&':
+			lexer.step()
+			lexer.Token.Kind = TDelimAmpersand
+
+		case '*':
+			lexer.step()
+			lexer.Token.Kind = TDelimAsterisk
+
+		case '|':
+			lexer.step()
+			lexer.Token.Kind = TDelimBar
+
+		case '!':
+			lexer.step()
+			lexer.Token.Kind = TDelimExclamation
+
+		case '=':
+			lexer.step()
+			lexer.Token.Kind = TDelimEquals
+
+		case '^':
+			lexer.step()
+			lexer.Token.Kind = TDelimCaret
+
+		case '$':
+			lexer.step()
+			lexer.Token.Kind = TDelimDollar
+
+		default:
+			if IsNameStart(lexer.codePoint) {
+				lexer.Token.Kind = lexer.consumeIdentLike()
+			} else {
+				lexer.step()
+				lexer.Token.Kind = TDelim
+			}
+		}
+
+		return
+	}
+}
+
+func (lexer *lexer) consumeToEndOfMultiLineComment(startRange logger.Range) {
+	startOfSourceMappingURL := 0
+	isLegalComment := false
+
+	switch lexer.codePoint {
+	case '#', '@':
+		// Keep track of the contents of the "sourceMappingURL=" comment
+		if strings.HasPrefix(lexer.source.Contents[lexer.current:], " sourceMappingURL=") {
+			startOfSourceMappingURL = lexer.current + len(" sourceMappingURL=")
+		}
+
+	case '!':
+		// Remember if this is a legal comment
+		isLegalComment = true
+	}
+
+	for {
+		switch lexer.codePoint {
+		case '*':
+			endOfSourceMappingURL := lexer.current - 1
+			lexer.step()
+			if lexer.codePoint == '/' {
+				commentEnd := lexer.current
+				lexer.step()
+
+				// Record the source mapping URL
+				if startOfSourceMappingURL != 0 {
+					r := logger.Range{Loc: logger.Loc{Start: int32(startOfSourceMappingURL)}}
+					text := lexer.source.Contents[startOfSourceMappingURL:endOfSourceMappingURL]
+					for int(r.Len) < len(text) && !isWhitespace(rune(text[r.Len])) {
+						r.Len++
+					}
+					lexer.sourceMappingURL = logger.Span{Text: text[:r.Len], Range: r}
+				}
+
+				// Record all comments
+				commentRange := logger.Range{Loc: startRange.Loc, Len: int32(commentEnd) - startRange.Loc.Start}
+				if lexer.RecordAllComments {
+					lexer.allComments = append(lexer.allComments, commentRange)
+				}
+
+				// Record legal comments
+				if text := lexer.source.Contents[startRange.Loc.Start:commentEnd]; isLegalComment || containsAtPreserveOrAtLicense(text) {
+					text = lexer.source.CommentTextWithoutIndent(commentRange)
+					lexer.legalCommentsBefore = append(lexer.legalCommentsBefore, Comment{Loc: startRange.Loc, Text: text})
+				}
+				return
+			}
+
+		case eof: // This indicates the end of the file
+			lexer.log.AddErrorWithNotes(&lexer.tracker, logger.Range{Loc: logger.Loc{Start: lexer.Token.Range.End()}},
+				"Expected \"*/\" to terminate multi-line comment",
+				[]logger.MsgData{lexer.tracker.MsgData(startRange, "The multi-line comment starts here:")})
+			return
+
+		default:
+			lexer.step()
+		}
+	}
+}
+
+func containsAtPreserveOrAtLicense(text string) bool {
+	for i, c := range text {
+		if c == '@' && (strings.HasPrefix(text[i+1:], "preserve") || strings.HasPrefix(text[i+1:], "license")) {
+			return true
+		}
+	}
+	return false
+}
+
+func (lexer *lexer) isValidEscape() bool {
+	if lexer.codePoint != '\\' {
+		return false
+	}
+	c, _ := utf8.DecodeRuneInString(lexer.source.Contents[lexer.current:])
+	return !isNewline(c)
+}
+
+func (lexer *lexer) wouldStartIdentifier() bool {
+	if IsNameStart(lexer.codePoint) {
+		return true
+	}
+
+	if lexer.codePoint == '-' {
+		c, width := utf8.DecodeRuneInString(lexer.source.Contents[lexer.current:])
+		if c == utf8.RuneError && width <= 1 {
+			return false // Decoding error
+		}
+		if IsNameStart(c) || c == '-' {
+			return true
+		}
+		if c == '\\' {
+			c2, _ := utf8.DecodeRuneInString(lexer.source.Contents[lexer.current+width:])
+			return !isNewline(c2)
+		}
+		return false
+	}
+
+	return lexer.isValidEscape()
+}
+
+func WouldStartIdentifierWithoutEscapes(text string) bool {
+	c, width := utf8.DecodeRuneInString(text)
+	if c == utf8.RuneError && width <= 1 {
+		return false // Decoding error
+	}
+	if IsNameStart(c) {
+		return true
+	}
+
+	if c == '-' {
+		c2, width2 := utf8.DecodeRuneInString(text[width:])
+		if c2 == utf8.RuneError && width2 <= 1 {
+			return false // Decoding error
+		}
+		if IsNameStart(c2) || c2 == '-' {
+			return true
+		}
+	}
+	return false
+}
+
+func RangeOfIdentifier(source logger.Source, loc logger.Loc) logger.Range {
+	text := source.Contents[loc.Start:]
+	if len(text) == 0 {
+		return logger.Range{Loc: loc, Len: 0}
+	}
+
+	i := 0
+	n := len(text)
+
+	for {
+		c, width := utf8.DecodeRuneInString(text[i:])
+		if IsNameContinue(c) {
+			i += width
+			continue
+		}
+
+		// Handle an escape
+		if c == '\\' && i+1 < n && !isNewline(rune(text[i+1])) {
+			i += width // Skip the backslash
+			c, width = utf8.DecodeRuneInString(text[i:])
+			if _, ok := isHex(c); ok {
+				i += width
+				c, width = utf8.DecodeRuneInString(text[i:])
+				for j := 0; j < 5; j++ {
+					if _, ok := isHex(c); !ok {
+						break
+					}
+					i += width
+					c, width = utf8.DecodeRuneInString(text[i:])
+				}
+				if isWhitespace(c) {
+					i += width
+				}
+			}
+			continue
+		}
+
+		break
+	}
+
+	// Don't end with a whitespace
+	if i > 0 && isWhitespace(rune(text[i-1])) {
+		i--
+	}
+
+	return logger.Range{Loc: loc, Len: int32(i)}
+}
+
+func (lexer *lexer) wouldStartNumber() bool {
+	if lexer.codePoint >= '0' && lexer.codePoint <= '9' {
+		return true
+	} else if lexer.codePoint == '.' {
+		contents := lexer.source.Contents
+		if lexer.current < len(contents) {
+			c := contents[lexer.current]
+			return c >= '0' && c <= '9'
+		}
+	} else if lexer.codePoint == '+' || lexer.codePoint == '-' {
+		contents := lexer.source.Contents
+		n := len(contents)
+		if lexer.current < n {
+			c := contents[lexer.current]
+			if c >= '0' && c <= '9' {
+				return true
+			}
+			if c == '.' && lexer.current+1 < n {
+				c = contents[lexer.current+1]
+				return c >= '0' && c <= '9'
+			}
+		}
+	}
+	return false
+}
+
+// Note: This function is hot in profiles
+func (lexer *lexer) consumeName() string {
+	// Common case: no escapes, identifier is a substring of the input. Doing this
+	// in a tight loop that avoids UTF-8 decoding and that increments a single
+	// number instead of doing "step()" is noticeably faster. For example, doing
+	// this sped up end-to-end parsing and printing of a large CSS file from 97ms
+	// to 84ms (around 15% faster).
+	contents := lexer.source.Contents
+	if IsNameContinue(lexer.codePoint) {
+		n := len(contents)
+		i := lexer.current
+		for i < n && IsNameContinue(rune(contents[i])) {
+			i++
+		}
+		lexer.current = i
+		lexer.step()
+	}
+	raw := contents[lexer.Token.Range.Loc.Start:lexer.Token.Range.End()]
+	if !lexer.isValidEscape() {
+		return raw
+	}
+
+	// Uncommon case: escapes, identifier is allocated
+	sb := strings.Builder{}
+	sb.WriteString(raw)
+	sb.WriteRune(lexer.consumeEscape())
+	for {
+		if IsNameContinue(lexer.codePoint) {
+			sb.WriteRune(lexer.codePoint)
+			lexer.step()
+		} else if lexer.isValidEscape() {
+			sb.WriteRune(lexer.consumeEscape())
+		} else {
+			break
+		}
+	}
+	return sb.String()
+}
+
+func (lexer *lexer) consumeEscape() rune {
+	lexer.step() // Skip the backslash
+	c := lexer.codePoint
+
+	if hex, ok := isHex(c); ok {
+		lexer.step()
+		for i := 0; i < 5; i++ {
+			if next, ok := isHex(lexer.codePoint); ok {
+				lexer.step()
+				hex = hex*16 + next
+			} else {
+				break
+			}
+		}
+		if isWhitespace(lexer.codePoint) {
+			lexer.step()
+		}
+		if hex == 0 || (hex >= 0xD800 && hex <= 0xDFFF) || hex > 0x10FFFF {
+			return utf8.RuneError
+		}
+		return rune(hex)
+	}
+
+	if c == eof {
+		return utf8.RuneError
+	}
+
+	lexer.step()
+	return c
+}
+
+func (lexer *lexer) consumeIdentLike() T {
+	name := lexer.consumeName()
+
+	if lexer.codePoint == '(' {
+		matchingLoc := logger.Loc{Start: lexer.Token.Range.End()}
+		lexer.step()
+		if len(name) == 3 {
+			u, r, l := name[0], name[1], name[2]
+			if (u == 'u' || u == 'U') && (r == 'r' || r == 'R') && (l == 'l' || l == 'L') {
+				// Save state
+				approximateNewlineCount := lexer.approximateNewlineCount
+				codePoint := lexer.codePoint
+				tokenRangeLen := lexer.Token.Range.Len
+				current := lexer.current
+
+				// Check to see if this is a URL token instead of a function
+				for isWhitespace(lexer.codePoint) {
+					lexer.step()
+				}
+				if lexer.codePoint != '"' && lexer.codePoint != '\'' {
+					return lexer.consumeURL(matchingLoc)
+				}
+
+				// Restore state (i.e. backtrack)
+				lexer.approximateNewlineCount = approximateNewlineCount
+				lexer.codePoint = codePoint
+				lexer.Token.Range.Len = tokenRangeLen
+				lexer.current = current
+			}
+		}
+		return TFunction
+	}
+
+	return TIdent
+}
+
+func (lexer *lexer) consumeURL(matchingLoc logger.Loc) T {
+validURL:
+	for {
+		switch lexer.codePoint {
+		case ')':
+			lexer.step()
+			return TURL
+
+		case eof:
+			loc := logger.Loc{Start: lexer.Token.Range.End()}
+			lexer.log.AddIDWithNotes(logger.MsgID_CSS_CSSSyntaxError, logger.Warning, &lexer.tracker, logger.Range{Loc: loc}, "Expected \")\" to end URL token",
+				[]logger.MsgData{lexer.tracker.MsgData(logger.Range{Loc: matchingLoc, Len: 1}, "The unbalanced \"(\" is here:")})
+			return TURL
+
+		case ' ', '\t', '\n', '\r', '\f':
+			lexer.step()
+			for isWhitespace(lexer.codePoint) {
+				lexer.step()
+			}
+			if lexer.codePoint != ')' {
+				loc := logger.Loc{Start: lexer.Token.Range.End()}
+				lexer.log.AddIDWithNotes(logger.MsgID_CSS_CSSSyntaxError, logger.Warning, &lexer.tracker, logger.Range{Loc: loc}, "Expected \")\" to end URL token",
+					[]logger.MsgData{lexer.tracker.MsgData(logger.Range{Loc: matchingLoc, Len: 1}, "The unbalanced \"(\" is here:")})
+				if lexer.codePoint == eof {
+					return TURL
+				}
+				break validURL
+			}
+			lexer.step()
+			return TURL
+
+		case '"', '\'', '(':
+			r := logger.Range{Loc: logger.Loc{Start: lexer.Token.Range.End()}, Len: 1}
+			lexer.log.AddIDWithNotes(logger.MsgID_CSS_CSSSyntaxError, logger.Warning, &lexer.tracker, r, "Expected \")\" to end URL token",
+				[]logger.MsgData{lexer.tracker.MsgData(logger.Range{Loc: matchingLoc, Len: 1}, "The unbalanced \"(\" is here:")})
+			break validURL
+
+		case '\\':
+			if !lexer.isValidEscape() {
+				r := logger.Range{Loc: logger.Loc{Start: lexer.Token.Range.End()}, Len: 1}
+				lexer.log.AddID(logger.MsgID_CSS_CSSSyntaxError, logger.Warning, &lexer.tracker, r, "Invalid escape")
+				break validURL
+			}
+			lexer.consumeEscape()
+
+		default:
+			if isNonPrintable(lexer.codePoint) {
+				r := logger.Range{Loc: logger.Loc{Start: lexer.Token.Range.End()}, Len: 1}
+				lexer.log.AddID(logger.MsgID_CSS_CSSSyntaxError, logger.Warning, &lexer.tracker, r, "Unexpected non-printable character in URL token")
+				break validURL
+			}
+			lexer.step()
+		}
+	}
+
+	// Consume the remnants of a bad url
+	for {
+		switch lexer.codePoint {
+		case ')', eof:
+			lexer.step()
+			return TBadURL
+
+		case '\\':
+			if lexer.isValidEscape() {
+				lexer.consumeEscape()
+			}
+		}
+		lexer.step()
+	}
+}
+
+func (lexer *lexer) consumeString() T {
+	quote := lexer.codePoint
+	lexer.step()
+
+	for {
+		switch lexer.codePoint {
+		case '\\':
+			lexer.step()
+
+			// Handle Windows CRLF
+			if lexer.codePoint == '\r' {
+				lexer.step()
+				if lexer.codePoint == '\n' {
+					lexer.step()
+				}
+				continue
+			}
+
+			// Otherwise, fall through to ignore the character after the backslash
+
+		case eof, '\n', '\r', '\f':
+			lexer.log.AddID(logger.MsgID_CSS_CSSSyntaxError, logger.Warning, &lexer.tracker,
+				logger.Range{Loc: logger.Loc{Start: lexer.Token.Range.End()}},
+				"Unterminated string token")
+			return TUnterminatedString
+
+		case quote:
+			lexer.step()
+			return TString
+		}
+		lexer.step()
+	}
+}
+
+func (lexer *lexer) consumeNumeric() T {
+	// Skip over leading sign
+	if lexer.codePoint == '+' || lexer.codePoint == '-' {
+		lexer.step()
+	}
+
+	// Skip over leading digits
+	for lexer.codePoint >= '0' && lexer.codePoint <= '9' {
+		lexer.step()
+	}
+
+	// Skip over digits after dot
+	if lexer.codePoint == '.' {
+		lexer.step()
+		for lexer.codePoint >= '0' && lexer.codePoint <= '9' {
+			lexer.step()
+		}
+	}
+
+	// Skip over exponent
+	if lexer.codePoint == 'e' || lexer.codePoint == 'E' {
+		contents := lexer.source.Contents
+
+		// Look ahead before advancing to make sure this is an exponent, not a unit
+		if lexer.current < len(contents) {
+			c := contents[lexer.current]
+			if (c == '+' || c == '-') && lexer.current+1 < len(contents) {
+				c = contents[lexer.current+1]
+			}
+
+			// Only consume this if it's an exponent
+			if c >= '0' && c <= '9' {
+				lexer.step()
+				if lexer.codePoint == '+' || lexer.codePoint == '-' {
+					lexer.step()
+				}
+				for lexer.codePoint >= '0' && lexer.codePoint <= '9' {
+					lexer.step()
+				}
+			}
+		}
+	}
+
+	// Determine the numeric type
+	if lexer.wouldStartIdentifier() {
+		lexer.Token.UnitOffset = uint16(lexer.Token.Range.Len)
+		lexer.consumeName()
+		return TDimension
+	}
+	if lexer.codePoint == '%' {
+		lexer.step()
+		return TPercentage
+	}
+	return TNumber
+}
+
+func IsNameStart(c rune) bool {
+	return (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || c == '_' || c >= 0x80 || c == '\x00'
+}
+
+func IsNameContinue(c rune) bool {
+	return IsNameStart(c) || (c >= '0' && c <= '9') || c == '-'
+}
+
+func isNewline(c rune) bool {
+	switch c {
+	case '\n', '\r', '\f':
+		return true
+	}
+	return false
+}
+
+func isWhitespace(c rune) bool {
+	switch c {
+	case ' ', '\t', '\n', '\r', '\f':
+		return true
+	}
+	return false
+}
+
+func isHex(c rune) (int, bool) {
+	if c >= '0' && c <= '9' {
+		return int(c - '0'), true
+	}
+	if c >= 'a' && c <= 'f' {
+		return int(c + (10 - 'a')), true
+	}
+	if c >= 'A' && c <= 'F' {
+		return int(c + (10 - 'A')), true
+	}
+	return 0, false
+}
+
+func isNonPrintable(c rune) bool {
+	return c <= 0x08 || c == 0x0B || (c >= 0x0E && c <= 0x1F) || c == 0x7F
+}
+
+func decodeEscapesInToken(inner string) string {
+	i := 0
+
+	for i < len(inner) {
+		if c := inner[i]; c == '\\' || c == '\x00' {
+			break
+		}
+		i++
+	}
+
+	if i == len(inner) {
+		return inner
+	}
+
+	sb := strings.Builder{}
+	sb.WriteString(inner[:i])
+	inner = inner[i:]
+
+	for len(inner) > 0 {
+		c, width := utf8.DecodeRuneInString(inner)
+		inner = inner[width:]
+
+		if c != '\\' {
+			if c == '\x00' {
+				c = utf8.RuneError
+			}
+			sb.WriteRune(c)
+			continue
+		}
+
+		if len(inner) == 0 {
+			sb.WriteRune(utf8.RuneError)
+			continue
+		}
+
+		c, width = utf8.DecodeRuneInString(inner)
+		inner = inner[width:]
+		hex, ok := isHex(c)
+
+		if !ok {
+			if c == '\n' || c == '\f' {
+				continue
+			}
+
+			// Handle Windows CRLF
+			if c == '\r' {
+				c, width = utf8.DecodeRuneInString(inner)
+				if c == '\n' {
+					inner = inner[width:]
+				}
+				continue
+			}
+
+			// If we get here, this is not a valid escape. However, this is still
+			// allowed. In this case the backslash is just ignored.
+			sb.WriteRune(c)
+			continue
+		}
+
+		// Parse up to five additional hex characters (so six in total)
+		for i := 0; i < 5 && len(inner) > 0; i++ {
+			c, width = utf8.DecodeRuneInString(inner)
+			if next, ok := isHex(c); ok {
+				inner = inner[width:]
+				hex = hex*16 + next
+			} else {
+				break
+			}
+		}
+
+		if len(inner) > 0 {
+			c, width = utf8.DecodeRuneInString(inner)
+			if isWhitespace(c) {
+				inner = inner[width:]
+			}
+		}
+
+		if hex == 0 || (hex >= 0xD800 && hex <= 0xDFFF) || hex > 0x10FFFF {
+			sb.WriteRune(utf8.RuneError)
+			continue
+		}
+
+		sb.WriteRune(rune(hex))
+	}
+
+	return sb.String()
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_color_spaces.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_color_spaces.go
new file mode 100644
index 0000000..721ecd3
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_color_spaces.go
@@ -0,0 +1,620 @@
+package css_parser
+
+import (
+	"math"
+
+	"github.com/evanw/esbuild/internal/helpers"
+)
+
+// Wrap float64 math to avoid compiler optimizations that break determinism
+type F64 = helpers.F64
+
+// Reference: https://drafts.csswg.org/css-color/#color-conversion-code
+
+type colorSpace uint8
+
+const (
+	colorSpace_a98_rgb colorSpace = iota
+	colorSpace_display_p3
+	colorSpace_hsl
+	colorSpace_hwb
+	colorSpace_lab
+	colorSpace_lch
+	colorSpace_oklab
+	colorSpace_oklch
+	colorSpace_prophoto_rgb
+	colorSpace_rec2020
+	colorSpace_srgb
+	colorSpace_srgb_linear
+	colorSpace_xyz
+	colorSpace_xyz_d50
+	colorSpace_xyz_d65
+)
+
+func (colorSpace colorSpace) isPolar() bool {
+	switch colorSpace {
+	case colorSpace_hsl, colorSpace_hwb, colorSpace_lch, colorSpace_oklch:
+		return true
+	}
+	return false
+}
+
+type hueMethod uint8
+
+const (
+	shorterHue hueMethod = iota
+	longerHue
+	increasingHue
+	decreasingHue
+)
+
+func lin_srgb(r F64, g F64, b F64) (F64, F64, F64) {
+	f := func(val F64) F64 {
+		if abs := val.Abs(); abs.Value() < 0.04045 {
+			return val.DivConst(12.92)
+		} else {
+			return abs.AddConst(0.055).DivConst(1.055).PowConst(2.4).WithSignFrom(val)
+		}
+	}
+	return f(r), f(g), f(b)
+}
+
+func gam_srgb(r F64, g F64, b F64) (F64, F64, F64) {
+	f := func(val F64) F64 {
+		if abs := val.Abs(); abs.Value() > 0.0031308 {
+			return abs.PowConst(1 / 2.4).MulConst(1.055).SubConst(0.055).WithSignFrom(val)
+		} else {
+			return val.MulConst(12.92)
+		}
+	}
+	return f(r), f(g), f(b)
+}
+
+func lin_srgb_to_xyz(r F64, g F64, b F64) (F64, F64, F64) {
+	M := [9]float64{
+		506752.0 / 1228815, 87881.0 / 245763, 12673.0 / 70218,
+		87098.0 / 409605, 175762.0 / 245763, 12673.0 / 175545,
+		7918.0 / 409605, 87881.0 / 737289, 1001167.0 / 1053270,
+	}
+	return multiplyMatrices(M, r, g, b)
+}
+
+func xyz_to_lin_srgb(x F64, y F64, z F64) (F64, F64, F64) {
+	M := [9]float64{
+		12831.0 / 3959, -329.0 / 214, -1974.0 / 3959,
+		-851781.0 / 878810, 1648619.0 / 878810, 36519.0 / 878810,
+		705.0 / 12673, -2585.0 / 12673, 705.0 / 667,
+	}
+	return multiplyMatrices(M, x, y, z)
+}
+
+func lin_p3(r F64, g F64, b F64) (F64, F64, F64) {
+	return lin_srgb(r, g, b)
+}
+
+func gam_p3(r F64, g F64, b F64) (F64, F64, F64) {
+	return gam_srgb(r, g, b)
+}
+
+func lin_p3_to_xyz(r F64, g F64, b F64) (F64, F64, F64) {
+	M := [9]float64{
+		608311.0 / 1250200, 189793.0 / 714400, 198249.0 / 1000160,
+		35783.0 / 156275, 247089.0 / 357200, 198249.0 / 2500400,
+		0.0 / 1, 32229.0 / 714400, 5220557.0 / 5000800,
+	}
+	return multiplyMatrices(M, r, g, b)
+}
+
+func xyz_to_lin_p3(x F64, y F64, z F64) (F64, F64, F64) {
+	M := [9]float64{
+		446124.0 / 178915, -333277.0 / 357830, -72051.0 / 178915,
+		-14852.0 / 17905, 63121.0 / 35810, 423.0 / 17905,
+		11844.0 / 330415, -50337.0 / 660830, 316169.0 / 330415,
+	}
+	return multiplyMatrices(M, x, y, z)
+}
+
+func lin_prophoto(r F64, g F64, b F64) (F64, F64, F64) {
+	f := func(val F64) F64 {
+		const Et2 = 16.0 / 512
+		if abs := val.Abs(); abs.Value() <= Et2 {
+			return val.DivConst(16)
+		} else {
+			return abs.PowConst(1.8).WithSignFrom(val)
+		}
+	}
+	return f(r), f(g), f(b)
+}
+
+func gam_prophoto(r F64, g F64, b F64) (F64, F64, F64) {
+	f := func(val F64) F64 {
+		const Et = 1.0 / 512
+		if abs := val.Abs(); abs.Value() >= Et {
+			return abs.PowConst(1 / 1.8).WithSignFrom(val)
+		} else {
+			return val.MulConst(16)
+		}
+	}
+	return f(r), f(g), f(b)
+}
+
+func lin_prophoto_to_xyz(r F64, g F64, b F64) (F64, F64, F64) {
+	M := [9]float64{
+		0.7977604896723027, 0.13518583717574031, 0.0313493495815248,
+		0.2880711282292934, 0.7118432178101014, 0.00008565396060525902,
+		0.0, 0.0, 0.8251046025104601,
+	}
+	return multiplyMatrices(M, r, g, b)
+}
+
+func xyz_to_lin_prophoto(x F64, y F64, z F64) (F64, F64, F64) {
+	M := [9]float64{
+		1.3457989731028281, -0.25558010007997534, -0.05110628506753401,
+		-0.5446224939028347, 1.5082327413132781, 0.02053603239147973,
+		0.0, 0.0, 1.2119675456389454,
+	}
+	return multiplyMatrices(M, x, y, z)
+}
+
+func lin_a98rgb(r F64, g F64, b F64) (F64, F64, F64) {
+	f := func(val F64) F64 {
+		return val.Abs().PowConst(563.0 / 256).WithSignFrom(val)
+	}
+	return f(r), f(g), f(b)
+}
+
+func gam_a98rgb(r F64, g F64, b F64) (F64, F64, F64) {
+	f := func(val F64) F64 {
+		return val.Abs().PowConst(256.0 / 563).WithSignFrom(val)
+	}
+	return f(r), f(g), f(b)
+}
+
+func lin_a98rgb_to_xyz(r F64, g F64, b F64) (F64, F64, F64) {
+	M := [9]float64{
+		573536.0 / 994567, 263643.0 / 1420810, 187206.0 / 994567,
+		591459.0 / 1989134, 6239551.0 / 9945670, 374412.0 / 4972835,
+		53769.0 / 1989134, 351524.0 / 4972835, 4929758.0 / 4972835,
+	}
+	return multiplyMatrices(M, r, g, b)
+}
+
+func xyz_to_lin_a98rgb(x F64, y F64, z F64) (F64, F64, F64) {
+	M := [9]float64{
+		1829569.0 / 896150, -506331.0 / 896150, -308931.0 / 896150,
+		-851781.0 / 878810, 1648619.0 / 878810, 36519.0 / 878810,
+		16779.0 / 1248040, -147721.0 / 1248040, 1266979.0 / 1248040,
+	}
+	return multiplyMatrices(M, x, y, z)
+}
+
+func lin_2020(r F64, g F64, b F64) (F64, F64, F64) {
+	f := func(val F64) F64 {
+		const α = 1.09929682680944
+		const β = 0.018053968510807
+		if abs := val.Abs(); abs.Value() < β*4.5 {
+			return val.DivConst(4.5)
+		} else {
+			return abs.AddConst(α - 1).DivConst(α).PowConst(1 / 0.45).WithSignFrom(val)
+		}
+	}
+	return f(r), f(g), f(b)
+}
+
+func gam_2020(r F64, g F64, b F64) (F64, F64, F64) {
+	f := func(val F64) F64 {
+		const α = 1.09929682680944
+		const β = 0.018053968510807
+		if abs := val.Abs(); abs.Value() > β {
+			return abs.PowConst(0.45).MulConst(α).SubConst(α - 1).WithSignFrom(val)
+		} else {
+			return val.MulConst(4.5)
+		}
+	}
+	return f(r), f(g), f(b)
+}
+
+func lin_2020_to_xyz(r F64, g F64, b F64) (F64, F64, F64) {
+	var M = [9]float64{
+		63426534.0 / 99577255, 20160776.0 / 139408157, 47086771.0 / 278816314,
+		26158966.0 / 99577255, 472592308.0 / 697040785, 8267143.0 / 139408157,
+		0.0 / 1, 19567812.0 / 697040785, 295819943.0 / 278816314,
+	}
+	return multiplyMatrices(M, r, g, b)
+}
+
+func xyz_to_lin_2020(x F64, y F64, z F64) (F64, F64, F64) {
+	M := [9]float64{
+		30757411.0 / 17917100, -6372589.0 / 17917100, -4539589.0 / 17917100,
+		-19765991.0 / 29648200, 47925759.0 / 29648200, 467509.0 / 29648200,
+		792561.0 / 44930125, -1921689.0 / 44930125, 42328811.0 / 44930125,
+	}
+	return multiplyMatrices(M, x, y, z)
+}
+
+func d65_to_d50(x F64, y F64, z F64) (F64, F64, F64) {
+	M := [9]float64{
+		1.0479297925449969, 0.022946870601609652, -0.05019226628920524,
+		0.02962780877005599, 0.9904344267538799, -0.017073799063418826,
+		-0.009243040646204504, 0.015055191490298152, 0.7518742814281371,
+	}
+	return multiplyMatrices(M, x, y, z)
+}
+
+func d50_to_d65(x F64, y F64, z F64) (F64, F64, F64) {
+	M := [9]float64{
+		0.955473421488075, -0.02309845494876471, 0.06325924320057072,
+		-0.0283697093338637, 1.0099953980813041, 0.021041441191917323,
+		0.012314014864481998, -0.020507649298898964, 1.330365926242124,
+	}
+	return multiplyMatrices(M, x, y, z)
+}
+
+const d50_x = 0.3457 / 0.3585
+const d50_z = (1.0 - 0.3457 - 0.3585) / 0.3585
+
+func xyz_to_lab(x F64, y F64, z F64) (F64, F64, F64) {
+	const ε = 216.0 / 24389
+	const κ = 24389.0 / 27
+
+	x = x.DivConst(d50_x)
+	z = z.DivConst(d50_z)
+
+	var f0, f1, f2 F64
+	if x.Value() > ε {
+		f0 = x.Cbrt()
+	} else {
+		f0 = x.MulConst(κ).AddConst(16).DivConst(116)
+	}
+	if y.Value() > ε {
+		f1 = y.Cbrt()
+	} else {
+		f1 = y.MulConst(κ).AddConst(16).DivConst(116)
+	}
+	if z.Value() > ε {
+		f2 = z.Cbrt()
+	} else {
+		f2 = z.MulConst(κ).AddConst(16).DivConst(116)
+	}
+
+	return f1.MulConst(116).SubConst(16),
+		f0.Sub(f1).MulConst(500),
+		f1.Sub(f2).MulConst(200)
+}
+
+func lab_to_xyz(l F64, a F64, b F64) (x F64, y F64, z F64) {
+	const κ = 24389.0 / 27
+	const ε = 216.0 / 24389
+
+	f1 := l.AddConst(16).DivConst(116)
+	f0 := a.DivConst(500).Add(f1)
+	f2 := f1.Sub(b.DivConst(200))
+
+	f0_3 := f0.Cubed()
+	f2_3 := f2.Cubed()
+
+	if f0_3.Value() > ε {
+		x = f0_3
+	} else {
+		x = f0.MulConst(116).SubConst(16).DivConst(κ)
+	}
+	if l.Value() > κ*ε {
+		y = l.AddConst(16).DivConst(116)
+		y = y.Cubed()
+	} else {
+		y = l.DivConst(κ)
+	}
+	if f2_3.Value() > ε {
+		z = f2_3
+	} else {
+		z = f2.MulConst(116).SubConst(16).DivConst(κ)
+	}
+
+	return x.MulConst(d50_x), y, z.MulConst(d50_z)
+}
+
+func lab_to_lch(l F64, a F64, b F64) (F64, F64, F64) {
+	hue := b.Atan2(a).MulConst(180 / math.Pi)
+	if hue.Value() < 0 {
+		hue = hue.AddConst(360)
+	}
+	return l,
+		a.Squared().Add(b.Squared()).Sqrt(),
+		hue
+}
+
+func lch_to_lab(l F64, c F64, h F64) (F64, F64, F64) {
+	return l,
+		h.MulConst(math.Pi / 180).Cos().Mul(c),
+		h.MulConst(math.Pi / 180).Sin().Mul(c)
+}
+
+func xyz_to_oklab(x F64, y F64, z F64) (F64, F64, F64) {
+	XYZtoLMS := [9]float64{
+		0.8190224432164319, 0.3619062562801221, -0.12887378261216414,
+		0.0329836671980271, 0.9292868468965546, 0.03614466816999844,
+		0.048177199566046255, 0.26423952494422764, 0.6335478258136937,
+	}
+	LMStoOKLab := [9]float64{
+		0.2104542553, 0.7936177850, -0.0040720468,
+		1.9779984951, -2.4285922050, 0.4505937099,
+		0.0259040371, 0.7827717662, -0.8086757660,
+	}
+	l, m, s := multiplyMatrices(XYZtoLMS, x, y, z)
+	return multiplyMatrices(LMStoOKLab, l.Cbrt(), m.Cbrt(), s.Cbrt())
+}
+
+func oklab_to_xyz(l F64, a F64, b F64) (F64, F64, F64) {
+	LMStoXYZ := [9]float64{
+		1.2268798733741557, -0.5578149965554813, 0.28139105017721583,
+		-0.04057576262431372, 1.1122868293970594, -0.07171106666151701,
+		-0.07637294974672142, -0.4214933239627914, 1.5869240244272418,
+	}
+	OKLabtoLMS := [9]float64{
+		0.99999999845051981432, 0.39633779217376785678, 0.21580375806075880339,
+		1.0000000088817607767, -0.1055613423236563494, -0.063854174771705903402,
+		1.0000000546724109177, -0.089484182094965759684, -1.2914855378640917399,
+	}
+	l, m, s := multiplyMatrices(OKLabtoLMS, l, a, b)
+	return multiplyMatrices(LMStoXYZ, l.Cubed(), m.Cubed(), s.Cubed())
+}
+
+func oklab_to_oklch(l F64, a F64, b F64) (F64, F64, F64) {
+	return lab_to_lch(l, a, b)
+}
+
+func oklch_to_oklab(l F64, c F64, h F64) (F64, F64, F64) {
+	return lch_to_lab(l, c, h)
+}
+
+func multiplyMatrices(A [9]float64, b0 F64, b1 F64, b2 F64) (F64, F64, F64) {
+	return b0.MulConst(A[0]).Add(b1.MulConst(A[1])).Add(b2.MulConst(A[2])),
+		b0.MulConst(A[3]).Add(b1.MulConst(A[4])).Add(b2.MulConst(A[5])),
+		b0.MulConst(A[6]).Add(b1.MulConst(A[7])).Add(b2.MulConst(A[8]))
+}
+
+func delta_eok(L1 F64, a1 F64, b1 F64, L2 F64, a2 F64, b2 F64) F64 {
+	ΔL_sq := L1.Sub(L2).Squared()
+	Δa_sq := a1.Sub(a2).Squared()
+	Δb_sq := b1.Sub(b2).Squared()
+	return ΔL_sq.Add(Δa_sq).Add(Δb_sq).Sqrt()
+}
+
+func gamut_mapping_xyz_to_srgb(x F64, y F64, z F64) (F64, F64, F64) {
+	origin_l, origin_c, origin_h := oklab_to_oklch(xyz_to_oklab(x, y, z))
+
+	if origin_l.Value() >= 1 || origin_l.Value() <= 0 {
+		return origin_l, origin_l, origin_l
+	}
+
+	oklch_to_srgb := func(l F64, c F64, h F64) (F64, F64, F64) {
+		l, a, b := oklch_to_oklab(l, c, h)
+		x, y, z := oklab_to_xyz(l, a, b)
+		r, g, b := xyz_to_lin_srgb(x, y, z)
+		return gam_srgb(r, g, b)
+	}
+
+	srgb_to_oklab := func(r F64, g F64, b F64) (F64, F64, F64) {
+		r, g, b = lin_srgb(r, g, b)
+		x, y, z := lin_srgb_to_xyz(r, g, b)
+		return xyz_to_oklab(x, y, z)
+	}
+
+	inGamut := func(r F64, g F64, b F64) bool {
+		return r.Value() >= 0 && r.Value() <= 1 &&
+			g.Value() >= 0 && g.Value() <= 1 &&
+			b.Value() >= 0 && b.Value() <= 1
+	}
+
+	r, g, b := oklch_to_srgb(origin_l, origin_c, origin_h)
+	if inGamut(r, g, b) {
+		return r, g, b
+	}
+
+	const JND = 0.02
+	const epsilon = 0.0001
+	min := helpers.NewF64(0.0)
+	max := origin_c
+
+	clip := func(x F64) F64 {
+		if x.Value() < 0 {
+			return helpers.NewF64(0)
+		}
+		if x.Value() > 1 {
+			return helpers.NewF64(1)
+		}
+		return x
+	}
+
+	for max.Sub(min).Value() > epsilon {
+		chroma := min.Add(max).DivConst(2)
+		origin_c = chroma
+
+		r, g, b = oklch_to_srgb(origin_l, origin_c, origin_h)
+		if inGamut(r, g, b) {
+			min = chroma
+			continue
+		}
+
+		clipped_r, clipped_g, clipped_b := clip(r), clip(g), clip(b)
+		L1, a1, b1 := srgb_to_oklab(clipped_r, clipped_b, clipped_g)
+		L2, a2, b2 := srgb_to_oklab(r, g, b)
+		E := delta_eok(L1, a1, b1, L2, a2, b2)
+		if E.Value() < JND {
+			return clipped_r, clipped_g, clipped_b
+		}
+
+		max = chroma
+	}
+
+	return r, g, b
+}
+
+func hsl_to_rgb(hue F64, sat F64, light F64) (F64, F64, F64) {
+	hue = hue.DivConst(360)
+	hue = hue.Sub(hue.Floor())
+	hue = hue.MulConst(360)
+
+	sat = sat.DivConst(100)
+	light = light.DivConst(100)
+
+	f := func(n float64) F64 {
+		k := hue.DivConst(30).AddConst(n)
+		k = k.DivConst(12)
+		k = k.Sub(k.Floor())
+		k = k.MulConst(12)
+		a := helpers.Min2(light, light.Neg().AddConst(1)).Mul(sat)
+		return light.Sub(helpers.Max2(helpers.NewF64(-1), helpers.Min3(k.SubConst(3), k.Neg().AddConst(9), helpers.NewF64(1))).Mul(a))
+	}
+
+	return f(0), f(8), f(4)
+}
+
+func rgb_to_hsl(red F64, green F64, blue F64) (F64, F64, F64) {
+	max := helpers.Max3(red, green, blue)
+	min := helpers.Min3(red, green, blue)
+	hue, sat, light := helpers.NewF64(math.NaN()), helpers.NewF64(0.0), min.Add(max).DivConst(2)
+	d := max.Sub(min)
+
+	if d.Value() != 0 {
+		if div := helpers.Min2(light, light.Neg().AddConst(1)); div.Value() != 0 {
+			sat = max.Sub(light).Div(div)
+		}
+
+		switch max {
+		case red:
+			hue = green.Sub(blue).Div(d)
+			if green.Value() < blue.Value() {
+				hue = hue.AddConst(6)
+			}
+		case green:
+			hue = blue.Sub(red).Div(d).AddConst(2)
+		case blue:
+			hue = red.Sub(green).Div(d).AddConst(4)
+		}
+
+		hue = hue.MulConst(60)
+	}
+
+	return hue, sat.MulConst(100), light.MulConst(100)
+}
+
+func hwb_to_rgb(hue F64, white F64, black F64) (F64, F64, F64) {
+	white = white.DivConst(100)
+	black = black.DivConst(100)
+	if white.Add(black).Value() >= 1 {
+		gray := white.Div(white.Add(black))
+		return gray, gray, gray
+	}
+	delta := white.Add(black).Neg().AddConst(1)
+	r, g, b := hsl_to_rgb(hue, helpers.NewF64(100), helpers.NewF64(50))
+	r = delta.Mul(r).Add(white)
+	g = delta.Mul(g).Add(white)
+	b = delta.Mul(b).Add(white)
+	return r, g, b
+}
+
+func rgb_to_hwb(red F64, green F64, blue F64) (F64, F64, F64) {
+	h, _, _ := rgb_to_hsl(red, green, blue)
+	white := helpers.Min3(red, green, blue)
+	black := helpers.Max3(red, green, blue).Neg().AddConst(1)
+	return h, white.MulConst(100), black.MulConst(100)
+}
+
+func xyz_to_colorSpace(x F64, y F64, z F64, colorSpace colorSpace) (F64, F64, F64) {
+	switch colorSpace {
+	case colorSpace_a98_rgb:
+		return gam_a98rgb(xyz_to_lin_a98rgb(x, y, z))
+
+	case colorSpace_display_p3:
+		return gam_p3(xyz_to_lin_p3(x, y, z))
+
+	case colorSpace_hsl:
+		return rgb_to_hsl(gam_srgb(xyz_to_lin_srgb(x, y, z)))
+
+	case colorSpace_hwb:
+		return rgb_to_hwb(gam_srgb(xyz_to_lin_srgb(x, y, z)))
+
+	case colorSpace_lab:
+		return xyz_to_lab(d65_to_d50(x, y, z))
+
+	case colorSpace_lch:
+		return lab_to_lch(xyz_to_lab(d65_to_d50(x, y, z)))
+
+	case colorSpace_oklab:
+		return xyz_to_oklab(x, y, z)
+
+	case colorSpace_oklch:
+		return oklab_to_oklch(xyz_to_oklab(x, y, z))
+
+	case colorSpace_prophoto_rgb:
+		return gam_prophoto(xyz_to_lin_prophoto(d65_to_d50(x, y, z)))
+
+	case colorSpace_rec2020:
+		return gam_2020(xyz_to_lin_2020(x, y, z))
+
+	case colorSpace_srgb:
+		return gam_srgb(xyz_to_lin_srgb(x, y, z))
+
+	case colorSpace_srgb_linear:
+		return xyz_to_lin_srgb(x, y, z)
+
+	case colorSpace_xyz, colorSpace_xyz_d65:
+		return x, y, z
+
+	case colorSpace_xyz_d50:
+		return d65_to_d50(x, y, z)
+
+	default:
+		panic("Internal error")
+	}
+}
+
+func colorSpace_to_xyz(v0 F64, v1 F64, v2 F64, colorSpace colorSpace) (F64, F64, F64) {
+	switch colorSpace {
+	case colorSpace_a98_rgb:
+		return lin_a98rgb_to_xyz(lin_a98rgb(v0, v1, v2))
+
+	case colorSpace_display_p3:
+		return lin_p3_to_xyz(lin_p3(v0, v1, v2))
+
+	case colorSpace_hsl:
+		return lin_srgb_to_xyz(lin_srgb(hsl_to_rgb(v0, v1, v2)))
+
+	case colorSpace_hwb:
+		return lin_srgb_to_xyz(lin_srgb(hwb_to_rgb(v0, v1, v2)))
+
+	case colorSpace_lab:
+		return d50_to_d65(lab_to_xyz(v0, v1, v2))
+
+	case colorSpace_lch:
+		return d50_to_d65(lab_to_xyz(lch_to_lab(v0, v1, v2)))
+
+	case colorSpace_oklab:
+		return oklab_to_xyz(v0, v1, v2)
+
+	case colorSpace_oklch:
+		return oklab_to_xyz(oklch_to_oklab(v0, v1, v2))
+
+	case colorSpace_prophoto_rgb:
+		return d50_to_d65(lin_prophoto_to_xyz(lin_prophoto(v0, v1, v2)))
+
+	case colorSpace_rec2020:
+		return lin_2020_to_xyz(lin_2020(v0, v1, v2))
+
+	case colorSpace_srgb:
+		return lin_srgb_to_xyz(lin_srgb(v0, v1, v2))
+
+	case colorSpace_srgb_linear:
+		return lin_srgb_to_xyz(v0, v1, v2)
+
+	case colorSpace_xyz, colorSpace_xyz_d65:
+		return v0, v1, v2
+
+	case colorSpace_xyz_d50:
+		return d50_to_d65(v0, v1, v2)
+
+	default:
+		panic("Internal error")
+	}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls.go
new file mode 100644
index 0000000..eaca876
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls.go
@@ -0,0 +1,538 @@
+package css_parser
+
+import (
+	"strings"
+
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+func (p *parser) commaToken(loc logger.Loc) css_ast.Token {
+	t := css_ast.Token{
+		Loc:  loc,
+		Kind: css_lexer.TComma,
+		Text: ",",
+	}
+	if !p.options.minifyWhitespace {
+		t.Whitespace = css_ast.WhitespaceAfter
+	}
+	return t
+}
+
+func expandTokenQuad(tokens []css_ast.Token, allowedIdent string) (result [4]css_ast.Token, ok bool) {
+	n := len(tokens)
+	if n < 1 || n > 4 {
+		return
+	}
+
+	// Don't do this if we encounter any unexpected tokens such as "var()"
+	for i := 0; i < n; i++ {
+		if t := tokens[i]; !t.Kind.IsNumeric() && (t.Kind != css_lexer.TIdent || allowedIdent == "" || t.Text != allowedIdent) {
+			return
+		}
+	}
+
+	result[0] = tokens[0]
+	if n > 1 {
+		result[1] = tokens[1]
+	} else {
+		result[1] = result[0]
+	}
+	if n > 2 {
+		result[2] = tokens[2]
+	} else {
+		result[2] = result[0]
+	}
+	if n > 3 {
+		result[3] = tokens[3]
+	} else {
+		result[3] = result[1]
+	}
+
+	ok = true
+	return
+}
+
+func compactTokenQuad(a css_ast.Token, b css_ast.Token, c css_ast.Token, d css_ast.Token, minifyWhitespace bool) []css_ast.Token {
+	tokens := []css_ast.Token{a, b, c, d}
+	if tokens[3].EqualIgnoringWhitespace(tokens[1]) {
+		if tokens[2].EqualIgnoringWhitespace(tokens[0]) {
+			if tokens[1].EqualIgnoringWhitespace(tokens[0]) {
+				tokens = tokens[:1]
+			} else {
+				tokens = tokens[:2]
+			}
+		} else {
+			tokens = tokens[:3]
+		}
+	}
+	for i := range tokens {
+		var whitespace css_ast.WhitespaceFlags
+		if !minifyWhitespace || i > 0 {
+			whitespace |= css_ast.WhitespaceBefore
+		}
+		if i+1 < len(tokens) {
+			whitespace |= css_ast.WhitespaceAfter
+		}
+		tokens[i].Whitespace = whitespace
+	}
+	return tokens
+}
+
+func (p *parser) processDeclarations(rules []css_ast.Rule, composesContext *composesContext) (rewrittenRules []css_ast.Rule) {
+	margin := boxTracker{key: css_ast.DMargin, keyText: "margin", allowAuto: true}
+	padding := boxTracker{key: css_ast.DPadding, keyText: "padding", allowAuto: false}
+	inset := boxTracker{key: css_ast.DInset, keyText: "inset", allowAuto: true}
+	borderRadius := borderRadiusTracker{}
+	rewrittenRules = make([]css_ast.Rule, 0, len(rules))
+	didWarnAboutComposes := false
+	wouldClipColorFlag := false
+	var declarationKeys map[string]struct{}
+
+	// Don't automatically generate the "inset" property if it's not supported
+	if p.options.unsupportedCSSFeatures.Has(compat.InsetProperty) {
+		inset.key = css_ast.DUnknown
+		inset.keyText = ""
+	}
+
+	// If this is a local class selector, track which CSS properties it declares.
+	// This is used to warn when CSS "composes" is used incorrectly.
+	if composesContext != nil {
+		for _, ref := range composesContext.parentRefs {
+			composes, ok := p.composes[ref]
+			if !ok {
+				composes = &css_ast.Composes{}
+				p.composes[ref] = composes
+			}
+			properties := composes.Properties
+			if properties == nil {
+				properties = make(map[string]logger.Loc)
+				composes.Properties = properties
+			}
+			for _, rule := range rules {
+				if decl, ok := rule.Data.(*css_ast.RDeclaration); ok && decl.Key != css_ast.DComposes {
+					properties[decl.KeyText] = decl.KeyRange.Loc
+				}
+			}
+		}
+	}
+
+	for i := 0; i < len(rules); i++ {
+		rule := rules[i]
+		rewrittenRules = append(rewrittenRules, rule)
+		decl, ok := rule.Data.(*css_ast.RDeclaration)
+		if !ok {
+			continue
+		}
+
+		// If the previous loop iteration would have clipped a color, we will
+		// duplicate it and insert the clipped copy before the unclipped copy
+		var wouldClipColor *bool
+		if wouldClipColorFlag {
+			wouldClipColorFlag = false
+			clone := *decl
+			clone.Value = css_ast.CloneTokensWithoutImportRecords(clone.Value)
+			decl = &clone
+			rule.Data = decl
+			n := len(rewrittenRules) - 2
+			rewrittenRules = append(rewrittenRules[:n], rule, rewrittenRules[n])
+		} else {
+			wouldClipColor = &wouldClipColorFlag
+		}
+
+		switch decl.Key {
+		case css_ast.DComposes:
+			// Only process "composes" directives if we're in "local-css" or
+			// "global-css" mode. In these cases, "composes" directives will always
+			// be removed (because they are being processed) even if they contain
+			// errors. Otherwise we leave "composes" directives there untouched and
+			// don't check them for errors.
+			if p.options.symbolMode != symbolModeDisabled {
+				if composesContext == nil {
+					if !didWarnAboutComposes {
+						didWarnAboutComposes = true
+						p.log.AddID(logger.MsgID_CSS_CSSSyntaxError, logger.Warning, &p.tracker, decl.KeyRange, "\"composes\" is not valid here")
+					}
+				} else if composesContext.problemRange.Len > 0 {
+					if !didWarnAboutComposes {
+						didWarnAboutComposes = true
+						p.log.AddIDWithNotes(logger.MsgID_CSS_CSSSyntaxError, logger.Warning, &p.tracker, decl.KeyRange, "\"composes\" only works inside single class selectors",
+							[]logger.MsgData{p.tracker.MsgData(composesContext.problemRange, "The parent selector is not a single class selector because of the syntax here:")})
+					}
+				} else {
+					p.handleComposesPragma(*composesContext, decl.Value)
+				}
+				rewrittenRules = rewrittenRules[:len(rewrittenRules)-1]
+			}
+
+		case css_ast.DBackground:
+			for i, t := range decl.Value {
+				t = p.lowerAndMinifyColor(t, wouldClipColor)
+				t = p.lowerAndMinifyGradient(t, wouldClipColor)
+				decl.Value[i] = t
+			}
+
+		case css_ast.DBackgroundImage,
+			css_ast.DBorderImage,
+			css_ast.DMaskImage:
+
+			for i, t := range decl.Value {
+				t = p.lowerAndMinifyGradient(t, wouldClipColor)
+				decl.Value[i] = t
+			}
+
+		case css_ast.DBackgroundColor,
+			css_ast.DBorderBlockEndColor,
+			css_ast.DBorderBlockStartColor,
+			css_ast.DBorderBottomColor,
+			css_ast.DBorderColor,
+			css_ast.DBorderInlineEndColor,
+			css_ast.DBorderInlineStartColor,
+			css_ast.DBorderLeftColor,
+			css_ast.DBorderRightColor,
+			css_ast.DBorderTopColor,
+			css_ast.DCaretColor,
+			css_ast.DColor,
+			css_ast.DColumnRuleColor,
+			css_ast.DFill,
+			css_ast.DFloodColor,
+			css_ast.DLightingColor,
+			css_ast.DOutlineColor,
+			css_ast.DStopColor,
+			css_ast.DStroke,
+			css_ast.DTextDecorationColor,
+			css_ast.DTextEmphasisColor:
+
+			if len(decl.Value) == 1 {
+				decl.Value[0] = p.lowerAndMinifyColor(decl.Value[0], wouldClipColor)
+			}
+
+		case css_ast.DTransform:
+			if p.options.minifySyntax {
+				decl.Value = p.mangleTransforms(decl.Value)
+			}
+
+		case css_ast.DBoxShadow:
+			decl.Value = p.lowerAndMangleBoxShadows(decl.Value, wouldClipColor)
+
+		// Container name
+		case css_ast.DContainer:
+			p.processContainerShorthand(decl.Value)
+		case css_ast.DContainerName:
+			p.processContainerName(decl.Value)
+
+			// Animation name
+		case css_ast.DAnimation:
+			p.processAnimationShorthand(decl.Value)
+		case css_ast.DAnimationName:
+			p.processAnimationName(decl.Value)
+
+		// List style
+		case css_ast.DListStyle:
+			p.processListStyleShorthand(decl.Value)
+		case css_ast.DListStyleType:
+			if len(decl.Value) == 1 {
+				p.processListStyleType(&decl.Value[0])
+			}
+
+			// Font
+		case css_ast.DFont:
+			if p.options.minifySyntax {
+				decl.Value = p.mangleFont(decl.Value)
+			}
+		case css_ast.DFontFamily:
+			if p.options.minifySyntax {
+				if value, ok := p.mangleFontFamily(decl.Value); ok {
+					decl.Value = value
+				}
+			}
+		case css_ast.DFontWeight:
+			if len(decl.Value) == 1 && p.options.minifySyntax {
+				decl.Value[0] = p.mangleFontWeight(decl.Value[0])
+			}
+
+			// Margin
+		case css_ast.DMargin:
+			if p.options.minifySyntax {
+				margin.mangleSides(rewrittenRules, decl, p.options.minifyWhitespace)
+			}
+		case css_ast.DMarginTop:
+			if p.options.minifySyntax {
+				margin.mangleSide(rewrittenRules, decl, p.options.minifyWhitespace, boxTop)
+			}
+		case css_ast.DMarginRight:
+			if p.options.minifySyntax {
+				margin.mangleSide(rewrittenRules, decl, p.options.minifyWhitespace, boxRight)
+			}
+		case css_ast.DMarginBottom:
+			if p.options.minifySyntax {
+				margin.mangleSide(rewrittenRules, decl, p.options.minifyWhitespace, boxBottom)
+			}
+		case css_ast.DMarginLeft:
+			if p.options.minifySyntax {
+				margin.mangleSide(rewrittenRules, decl, p.options.minifyWhitespace, boxLeft)
+			}
+
+		// Padding
+		case css_ast.DPadding:
+			if p.options.minifySyntax {
+				padding.mangleSides(rewrittenRules, decl, p.options.minifyWhitespace)
+			}
+		case css_ast.DPaddingTop:
+			if p.options.minifySyntax {
+				padding.mangleSide(rewrittenRules, decl, p.options.minifyWhitespace, boxTop)
+			}
+		case css_ast.DPaddingRight:
+			if p.options.minifySyntax {
+				padding.mangleSide(rewrittenRules, decl, p.options.minifyWhitespace, boxRight)
+			}
+		case css_ast.DPaddingBottom:
+			if p.options.minifySyntax {
+				padding.mangleSide(rewrittenRules, decl, p.options.minifyWhitespace, boxBottom)
+			}
+		case css_ast.DPaddingLeft:
+			if p.options.minifySyntax {
+				padding.mangleSide(rewrittenRules, decl, p.options.minifyWhitespace, boxLeft)
+			}
+
+		// Inset
+		case css_ast.DInset:
+			if p.options.unsupportedCSSFeatures.Has(compat.InsetProperty) {
+				if decls, ok := p.lowerInset(rule.Loc, decl); ok {
+					rewrittenRules = rewrittenRules[:len(rewrittenRules)-1]
+					for i := range decls {
+						rewrittenRules = append(rewrittenRules, decls[i])
+						if p.options.minifySyntax {
+							inset.mangleSide(rewrittenRules, decls[i].Data.(*css_ast.RDeclaration), p.options.minifyWhitespace, i)
+						}
+					}
+					break
+				}
+			}
+			if p.options.minifySyntax {
+				inset.mangleSides(rewrittenRules, decl, p.options.minifyWhitespace)
+			}
+		case css_ast.DTop:
+			if p.options.minifySyntax {
+				inset.mangleSide(rewrittenRules, decl, p.options.minifyWhitespace, boxTop)
+			}
+		case css_ast.DRight:
+			if p.options.minifySyntax {
+				inset.mangleSide(rewrittenRules, decl, p.options.minifyWhitespace, boxRight)
+			}
+		case css_ast.DBottom:
+			if p.options.minifySyntax {
+				inset.mangleSide(rewrittenRules, decl, p.options.minifyWhitespace, boxBottom)
+			}
+		case css_ast.DLeft:
+			if p.options.minifySyntax {
+				inset.mangleSide(rewrittenRules, decl, p.options.minifyWhitespace, boxLeft)
+			}
+
+		// Border radius
+		case css_ast.DBorderRadius:
+			if p.options.minifySyntax {
+				borderRadius.mangleCorners(rewrittenRules, decl, p.options.minifyWhitespace)
+			}
+		case css_ast.DBorderTopLeftRadius:
+			if p.options.minifySyntax {
+				borderRadius.mangleCorner(rewrittenRules, decl, p.options.minifyWhitespace, borderRadiusTopLeft)
+			}
+		case css_ast.DBorderTopRightRadius:
+			if p.options.minifySyntax {
+				borderRadius.mangleCorner(rewrittenRules, decl, p.options.minifyWhitespace, borderRadiusTopRight)
+			}
+		case css_ast.DBorderBottomRightRadius:
+			if p.options.minifySyntax {
+				borderRadius.mangleCorner(rewrittenRules, decl, p.options.minifyWhitespace, borderRadiusBottomRight)
+			}
+		case css_ast.DBorderBottomLeftRadius:
+			if p.options.minifySyntax {
+				borderRadius.mangleCorner(rewrittenRules, decl, p.options.minifyWhitespace, borderRadiusBottomLeft)
+			}
+		}
+
+		if prefixes, ok := p.options.cssPrefixData[decl.Key]; ok {
+			if declarationKeys == nil {
+				// Only generate this map if it's needed
+				declarationKeys = make(map[string]struct{})
+				for _, rule := range rules {
+					if decl, ok := rule.Data.(*css_ast.RDeclaration); ok {
+						declarationKeys[decl.KeyText] = struct{}{}
+					}
+				}
+			}
+			if (prefixes & compat.WebkitPrefix) != 0 {
+				rewrittenRules = p.insertPrefixedDeclaration(rewrittenRules, "-webkit-", rule.Loc, decl, declarationKeys)
+			}
+			if (prefixes & compat.KhtmlPrefix) != 0 {
+				rewrittenRules = p.insertPrefixedDeclaration(rewrittenRules, "-khtml-", rule.Loc, decl, declarationKeys)
+			}
+			if (prefixes & compat.MozPrefix) != 0 {
+				rewrittenRules = p.insertPrefixedDeclaration(rewrittenRules, "-moz-", rule.Loc, decl, declarationKeys)
+			}
+			if (prefixes & compat.MsPrefix) != 0 {
+				rewrittenRules = p.insertPrefixedDeclaration(rewrittenRules, "-ms-", rule.Loc, decl, declarationKeys)
+			}
+			if (prefixes & compat.OPrefix) != 0 {
+				rewrittenRules = p.insertPrefixedDeclaration(rewrittenRules, "-o-", rule.Loc, decl, declarationKeys)
+			}
+		}
+
+		// If this loop iteration would have clipped a color, the out-of-gamut
+		// colors will not be clipped and this flag will be set. We then set up the
+		// next iteration of the loop to duplicate this rule and process it again
+		// with color clipping enabled.
+		if wouldClipColorFlag {
+			if p.options.unsupportedCSSFeatures.Has(compat.ColorFunctions) {
+				// Only do this if there was no previous instance of that property so
+				// we avoid overwriting any manually-specified fallback values
+				for j := len(rewrittenRules) - 2; j >= 0; j-- {
+					if prev, ok := rewrittenRules[j].Data.(*css_ast.RDeclaration); ok && prev.Key == decl.Key {
+						wouldClipColorFlag = false
+						break
+					}
+				}
+				if wouldClipColorFlag {
+					// If the code above would have clipped a color outside of the sRGB gamut,
+					// process this rule again so we can generate the clipped version next time
+					i -= 1
+					continue
+				}
+			}
+			wouldClipColorFlag = false
+		}
+	}
+
+	// Compact removed rules
+	if p.options.minifySyntax {
+		end := 0
+		for _, rule := range rewrittenRules {
+			if rule.Data != nil {
+				rewrittenRules[end] = rule
+				end++
+			}
+		}
+		rewrittenRules = rewrittenRules[:end]
+	}
+
+	return
+}
+
+func (p *parser) insertPrefixedDeclaration(rules []css_ast.Rule, prefix string, loc logger.Loc, decl *css_ast.RDeclaration, declarationKeys map[string]struct{}) []css_ast.Rule {
+	keyText := prefix + decl.KeyText
+
+	// Don't insert a prefixed declaration if there already is one
+	if _, ok := declarationKeys[keyText]; ok {
+		// We found a previous declaration with a matching prefixed property.
+		// The value is ignored, which matches the behavior of "autoprefixer".
+		return rules
+	}
+
+	// Additional special cases for when the prefix applies
+	switch decl.Key {
+	case css_ast.DBackgroundClip:
+		// The prefix is only needed for "background-clip: text"
+		if len(decl.Value) != 1 || decl.Value[0].Kind != css_lexer.TIdent || !strings.EqualFold(decl.Value[0].Text, "text") {
+			return rules
+		}
+
+	case css_ast.DPosition:
+		// The prefix is only needed for "position: sticky"
+		if len(decl.Value) != 1 || decl.Value[0].Kind != css_lexer.TIdent || !strings.EqualFold(decl.Value[0].Text, "sticky") {
+			return rules
+		}
+	}
+
+	value := css_ast.CloneTokensWithoutImportRecords(decl.Value)
+
+	// Additional special cases for how to transform the contents
+	switch decl.Key {
+	case css_ast.DPosition:
+		// The prefix applies to the value, not the property
+		keyText = decl.KeyText
+		value[0].Text = "-webkit-sticky"
+
+	case css_ast.DUserSelect:
+		// The prefix applies to the value as well as the property
+		if prefix == "-moz-" && len(value) == 1 && value[0].Kind == css_lexer.TIdent && strings.EqualFold(value[0].Text, "none") {
+			value[0].Text = "-moz-none"
+		}
+
+	case css_ast.DMaskComposite:
+		// WebKit uses different names for these values
+		if prefix == "-webkit-" {
+			for i, token := range value {
+				if token.Kind == css_lexer.TIdent {
+					switch token.Text {
+					case "add":
+						value[i].Text = "source-over"
+					case "subtract":
+						value[i].Text = "source-out"
+					case "intersect":
+						value[i].Text = "source-in"
+					case "exclude":
+						value[i].Text = "xor"
+					}
+				}
+			}
+		}
+	}
+
+	// Overwrite the latest declaration with the prefixed declaration
+	rules[len(rules)-1] = css_ast.Rule{Loc: loc, Data: &css_ast.RDeclaration{
+		KeyText:   keyText,
+		KeyRange:  decl.KeyRange,
+		Value:     value,
+		Important: decl.Important,
+	}}
+
+	// Re-add the latest declaration after the inserted declaration
+	rules = append(rules, css_ast.Rule{Loc: loc, Data: decl})
+	return rules
+}
+
+func (p *parser) lowerInset(loc logger.Loc, decl *css_ast.RDeclaration) ([]css_ast.Rule, bool) {
+	if tokens, ok := expandTokenQuad(decl.Value, ""); ok {
+		mask := ^css_ast.WhitespaceAfter
+		if p.options.minifyWhitespace {
+			mask = 0
+		}
+		for i := range tokens {
+			tokens[i].Whitespace &= mask
+		}
+		return []css_ast.Rule{
+			{Loc: loc, Data: &css_ast.RDeclaration{
+				KeyText:   "top",
+				KeyRange:  decl.KeyRange,
+				Key:       css_ast.DTop,
+				Value:     tokens[0:1],
+				Important: decl.Important,
+			}},
+			{Loc: loc, Data: &css_ast.RDeclaration{
+				KeyText:   "right",
+				KeyRange:  decl.KeyRange,
+				Key:       css_ast.DRight,
+				Value:     tokens[1:2],
+				Important: decl.Important,
+			}},
+			{Loc: loc, Data: &css_ast.RDeclaration{
+				KeyText:   "bottom",
+				KeyRange:  decl.KeyRange,
+				Key:       css_ast.DBottom,
+				Value:     tokens[2:3],
+				Important: decl.Important,
+			}},
+			{Loc: loc, Data: &css_ast.RDeclaration{
+				KeyText:   "left",
+				KeyRange:  decl.KeyRange,
+				Key:       css_ast.DLeft,
+				Value:     tokens[3:4],
+				Important: decl.Important,
+			}},
+		}, true
+	}
+	return nil, false
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_animation.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_animation.go
new file mode 100644
index 0000000..e4368d5
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_animation.go
@@ -0,0 +1,119 @@
+package css_parser
+
+import (
+	"strings"
+
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+)
+
+// Scan for animation names in the "animation" shorthand property
+func (p *parser) processAnimationShorthand(tokens []css_ast.Token) {
+	type foundFlags struct {
+		timingFunction bool
+		iterationCount bool
+		direction      bool
+		fillMode       bool
+		playState      bool
+		name           bool
+	}
+
+	found := foundFlags{}
+
+	for i, t := range tokens {
+		switch t.Kind {
+		case css_lexer.TComma:
+			// Reset the flags when we encounter a comma
+			found = foundFlags{}
+
+		case css_lexer.TNumber:
+			if !found.iterationCount {
+				found.iterationCount = true
+				continue
+			}
+
+		case css_lexer.TIdent:
+			if !found.timingFunction {
+				switch strings.ToLower(t.Text) {
+				case "linear", "ease", "ease-in", "ease-out", "ease-in-out", "step-start", "step-end":
+					found.timingFunction = true
+					continue
+				}
+			}
+
+			if !found.iterationCount && strings.ToLower(t.Text) == "infinite" {
+				found.iterationCount = true
+				continue
+			}
+
+			if !found.direction {
+				switch strings.ToLower(t.Text) {
+				case "normal", "reverse", "alternate", "alternate-reverse":
+					found.direction = true
+					continue
+				}
+			}
+
+			if !found.fillMode {
+				switch strings.ToLower(t.Text) {
+				case "none", "forwards", "backwards", "both":
+					found.fillMode = true
+					continue
+				}
+			}
+
+			if !found.playState {
+				switch strings.ToLower(t.Text) {
+				case "running", "paused":
+					found.playState = true
+					continue
+				}
+			}
+
+			if !found.name {
+				p.handleSingleAnimationName(&tokens[i])
+				found.name = true
+				continue
+			}
+
+		case css_lexer.TString:
+			if !found.name {
+				p.handleSingleAnimationName(&tokens[i])
+				found.name = true
+				continue
+			}
+		}
+	}
+}
+
+func (p *parser) processAnimationName(tokens []css_ast.Token) {
+	for i, t := range tokens {
+		if t.Kind == css_lexer.TIdent || t.Kind == css_lexer.TString {
+			p.handleSingleAnimationName(&tokens[i])
+		}
+	}
+}
+
+func (p *parser) handleSingleAnimationName(token *css_ast.Token) {
+	// Do not transform CSS keywords into symbols because they have special
+	// meaning in declarations. For example, "animation-name: none" clears
+	// the animation name. It does not set it to the animation named "none".
+	// You need to use "animation-name: 'none'" to do that.
+	//
+	// Also don't transform strings containing CSS keywords into global symbols
+	// because global symbols are passed through without being renamed, which
+	// will print them as keywords. However, we still want to unconditionally
+	// transform strings into local symbols because local symbols are always
+	// renamed, so they will never be printed as keywords.
+	if (token.Kind == css_lexer.TIdent || (token.Kind == css_lexer.TString && !p.makeLocalSymbols)) && isInvalidAnimationName(token.Text) {
+		return
+	}
+
+	token.Kind = css_lexer.TSymbol
+	token.PayloadIndex = p.symbolForName(token.Loc, token.Text).Ref.InnerIndex
+}
+
+func isInvalidAnimationName(text string) bool {
+	lower := strings.ToLower(text)
+	return lower == "none" || cssWideAndReservedKeywords[lower]
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_border_radius.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_border_radius.go
new file mode 100644
index 0000000..cddc66e
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_border_radius.go
@@ -0,0 +1,217 @@
+package css_parser
+
+import (
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+const (
+	borderRadiusTopLeft = iota
+	borderRadiusTopRight
+	borderRadiusBottomRight
+	borderRadiusBottomLeft
+)
+
+type borderRadiusCorner struct {
+	firstToken    css_ast.Token
+	secondToken   css_ast.Token
+	unitSafety    unitSafetyTracker
+	ruleIndex     uint32 // The index of the originating rule in the rules array
+	wasSingleRule bool   // True if the originating rule was just for this side
+}
+
+type borderRadiusTracker struct {
+	corners   [4]borderRadiusCorner
+	important bool // True if all active rules were flagged as "!important"
+}
+
+func (borderRadius *borderRadiusTracker) updateCorner(rules []css_ast.Rule, corner int, new borderRadiusCorner) {
+	if old := borderRadius.corners[corner]; old.firstToken.Kind != css_lexer.TEndOfFile &&
+		(!new.wasSingleRule || old.wasSingleRule) &&
+		old.unitSafety.status == unitSafe && new.unitSafety.status == unitSafe {
+		rules[old.ruleIndex] = css_ast.Rule{}
+	}
+	borderRadius.corners[corner] = new
+}
+
+func (borderRadius *borderRadiusTracker) mangleCorners(rules []css_ast.Rule, decl *css_ast.RDeclaration, minifyWhitespace bool) {
+	// Reset if we see a change in the "!important" flag
+	if borderRadius.important != decl.Important {
+		borderRadius.corners = [4]borderRadiusCorner{}
+		borderRadius.important = decl.Important
+	}
+
+	tokens := decl.Value
+	beforeSplit := len(tokens)
+	afterSplit := len(tokens)
+
+	// Search for the single slash if present
+	for i, t := range tokens {
+		if t.Kind == css_lexer.TDelimSlash {
+			if beforeSplit == len(tokens) {
+				beforeSplit = i
+				afterSplit = i + 1
+			} else {
+				// Multiple slashes are an error
+				borderRadius.corners = [4]borderRadiusCorner{}
+				return
+			}
+		}
+	}
+
+	// Use a single tracker for the whole rule
+	unitSafety := unitSafetyTracker{}
+	for _, t := range tokens[:beforeSplit] {
+		unitSafety.includeUnitOf(t)
+	}
+	for _, t := range tokens[afterSplit:] {
+		unitSafety.includeUnitOf(t)
+	}
+
+	firstRadii, firstRadiiOk := expandTokenQuad(tokens[:beforeSplit], "")
+	lastRadii, lastRadiiOk := expandTokenQuad(tokens[afterSplit:], "")
+
+	// Stop now if the pattern wasn't matched
+	if !firstRadiiOk || (beforeSplit < afterSplit && !lastRadiiOk) {
+		borderRadius.corners = [4]borderRadiusCorner{}
+		return
+	}
+
+	// Handle the first radii
+	for corner, t := range firstRadii {
+		if unitSafety.status == unitSafe {
+			t.TurnLengthIntoNumberIfZero()
+		}
+		borderRadius.updateCorner(rules, corner, borderRadiusCorner{
+			firstToken:  t,
+			secondToken: t,
+			unitSafety:  unitSafety,
+			ruleIndex:   uint32(len(rules) - 1),
+		})
+	}
+
+	// Handle the last radii
+	if lastRadiiOk {
+		for corner, t := range lastRadii {
+			if unitSafety.status == unitSafe {
+				t.TurnLengthIntoNumberIfZero()
+			}
+			borderRadius.corners[corner].secondToken = t
+		}
+	}
+
+	// Success
+	borderRadius.compactRules(rules, decl.KeyRange, minifyWhitespace)
+}
+
+func (borderRadius *borderRadiusTracker) mangleCorner(rules []css_ast.Rule, decl *css_ast.RDeclaration, minifyWhitespace bool, corner int) {
+	// Reset if we see a change in the "!important" flag
+	if borderRadius.important != decl.Important {
+		borderRadius.corners = [4]borderRadiusCorner{}
+		borderRadius.important = decl.Important
+	}
+
+	if tokens := decl.Value; (len(tokens) == 1 && tokens[0].Kind.IsNumeric()) ||
+		(len(tokens) == 2 && tokens[0].Kind.IsNumeric() && tokens[1].Kind.IsNumeric()) {
+		firstToken := tokens[0]
+		secondToken := firstToken
+		if len(tokens) == 2 {
+			secondToken = tokens[1]
+		}
+
+		// Check to see if these units are safe to use in every browser
+		unitSafety := unitSafetyTracker{}
+		unitSafety.includeUnitOf(firstToken)
+		unitSafety.includeUnitOf(secondToken)
+
+		// Only collapse "0unit" into "0" if the unit is safe
+		if unitSafety.status == unitSafe && firstToken.TurnLengthIntoNumberIfZero() {
+			tokens[0] = firstToken
+		}
+		if len(tokens) == 2 {
+			if unitSafety.status == unitSafe && secondToken.TurnLengthIntoNumberIfZero() {
+				tokens[1] = secondToken
+			}
+
+			// If both tokens are equal, merge them into one
+			if firstToken.EqualIgnoringWhitespace(secondToken) {
+				tokens[0].Whitespace &= ^css_ast.WhitespaceAfter
+				decl.Value = tokens[:1]
+			}
+		}
+
+		borderRadius.updateCorner(rules, corner, borderRadiusCorner{
+			firstToken:    firstToken,
+			secondToken:   secondToken,
+			unitSafety:    unitSafety,
+			ruleIndex:     uint32(len(rules) - 1),
+			wasSingleRule: true,
+		})
+		borderRadius.compactRules(rules, decl.KeyRange, minifyWhitespace)
+	} else {
+		borderRadius.corners = [4]borderRadiusCorner{}
+	}
+}
+
+func (borderRadius *borderRadiusTracker) compactRules(rules []css_ast.Rule, keyRange logger.Range, minifyWhitespace bool) {
+	// All tokens must be present
+	if eof := css_lexer.TEndOfFile; borderRadius.corners[0].firstToken.Kind == eof || borderRadius.corners[1].firstToken.Kind == eof ||
+		borderRadius.corners[2].firstToken.Kind == eof || borderRadius.corners[3].firstToken.Kind == eof {
+		return
+	}
+
+	// All tokens must have the same unit
+	for _, side := range borderRadius.corners[1:] {
+		if !side.unitSafety.isSafeWith(borderRadius.corners[0].unitSafety) {
+			return
+		}
+	}
+
+	// Generate the most minimal representation
+	tokens := compactTokenQuad(
+		borderRadius.corners[0].firstToken,
+		borderRadius.corners[1].firstToken,
+		borderRadius.corners[2].firstToken,
+		borderRadius.corners[3].firstToken,
+		minifyWhitespace,
+	)
+	secondTokens := compactTokenQuad(
+		borderRadius.corners[0].secondToken,
+		borderRadius.corners[1].secondToken,
+		borderRadius.corners[2].secondToken,
+		borderRadius.corners[3].secondToken,
+		minifyWhitespace,
+	)
+	if !css_ast.TokensEqualIgnoringWhitespace(tokens, secondTokens) {
+		var whitespace css_ast.WhitespaceFlags
+		if !minifyWhitespace {
+			whitespace = css_ast.WhitespaceBefore | css_ast.WhitespaceAfter
+		}
+		tokens = append(tokens, css_ast.Token{
+			Loc:        tokens[len(tokens)-1].Loc,
+			Kind:       css_lexer.TDelimSlash,
+			Text:       "/",
+			Whitespace: whitespace,
+		})
+		tokens = append(tokens, secondTokens...)
+	}
+
+	// Remove all of the existing declarations
+	var minLoc logger.Loc
+	for i, corner := range borderRadius.corners {
+		if loc := rules[corner.ruleIndex].Loc; i == 0 || loc.Start < minLoc.Start {
+			minLoc = loc
+		}
+		rules[corner.ruleIndex] = css_ast.Rule{}
+	}
+
+	// Insert the combined declaration where the last rule was
+	rules[borderRadius.corners[3].ruleIndex] = css_ast.Rule{Loc: minLoc, Data: &css_ast.RDeclaration{
+		Key:       css_ast.DBorderRadius,
+		KeyText:   "border-radius",
+		Value:     tokens,
+		KeyRange:  keyRange,
+		Important: borderRadius.important,
+	}}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_box.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_box.go
new file mode 100644
index 0000000..9f7d7ec
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_box.go
@@ -0,0 +1,206 @@
+package css_parser
+
+import (
+	"strings"
+
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+const (
+	boxTop = iota
+	boxRight
+	boxBottom
+	boxLeft
+)
+
+type boxSide struct {
+	token         css_ast.Token
+	unitSafety    unitSafetyTracker
+	ruleIndex     uint32 // The index of the originating rule in the rules array
+	wasSingleRule bool   // True if the originating rule was just for this side
+}
+
+type boxTracker struct {
+	keyText   string
+	sides     [4]boxSide
+	allowAuto bool // If true, allow the "auto" keyword
+	important bool // True if all active rules were flagged as "!important"
+	key       css_ast.D
+}
+
+type unitSafetyStatus uint8
+
+const (
+	unitSafe         unitSafetyStatus = iota // "margin: 0 1px 2cm 3%;"
+	unitUnsafeSingle                         // "margin: 0 1vw 2vw 3vw;"
+	unitUnsafeMixed                          // "margin: 0 1vw 2vh 3ch;"
+)
+
+// We can only compact rules together if they have the same unit safety level.
+// We want to avoid a situation where the browser treats some of the original
+// rules as valid and others as invalid.
+//
+//	Safe:
+//	  top: 1px; left: 0; bottom: 1px; right: 0;
+//	  top: 1Q; left: 2Q; bottom: 3Q; right: 4Q;
+//
+//	Unsafe:
+//	  top: 1vh; left: 2vw; bottom: 3vh; right: 4vw;
+//	  top: 1Q; left: 2Q; bottom: 3Q; right: 0;
+//	  inset: 1Q 0 0 0; top: 0;
+type unitSafetyTracker struct {
+	unit   string
+	status unitSafetyStatus
+}
+
+func (a unitSafetyTracker) isSafeWith(b unitSafetyTracker) bool {
+	return a.status == b.status && a.status != unitUnsafeMixed && (a.status != unitUnsafeSingle || a.unit == b.unit)
+}
+
+func (t *unitSafetyTracker) includeUnitOf(token css_ast.Token) {
+	switch token.Kind {
+	case css_lexer.TNumber:
+		if token.Text == "0" {
+			return
+		}
+
+	case css_lexer.TPercentage:
+		return
+
+	case css_lexer.TDimension:
+		if token.DimensionUnitIsSafeLength() {
+			return
+		} else if unit := token.DimensionUnit(); t.status == unitSafe {
+			t.status = unitUnsafeSingle
+			t.unit = unit
+			return
+		} else if t.status == unitUnsafeSingle && t.unit == unit {
+			return
+		}
+	}
+
+	t.status = unitUnsafeMixed
+}
+
+func (box *boxTracker) updateSide(rules []css_ast.Rule, side int, new boxSide) {
+	if old := box.sides[side]; old.token.Kind != css_lexer.TEndOfFile &&
+		(!new.wasSingleRule || old.wasSingleRule) &&
+		old.unitSafety.status == unitSafe && new.unitSafety.status == unitSafe {
+		rules[old.ruleIndex] = css_ast.Rule{}
+	}
+	box.sides[side] = new
+}
+
+func (box *boxTracker) mangleSides(rules []css_ast.Rule, decl *css_ast.RDeclaration, minifyWhitespace bool) {
+	// Reset if we see a change in the "!important" flag
+	if box.important != decl.Important {
+		box.sides = [4]boxSide{}
+		box.important = decl.Important
+	}
+
+	allowedIdent := ""
+	if box.allowAuto {
+		allowedIdent = "auto"
+	}
+	if quad, ok := expandTokenQuad(decl.Value, allowedIdent); ok {
+		// Use a single tracker for the whole rule
+		unitSafety := unitSafetyTracker{}
+		for _, t := range quad {
+			if !box.allowAuto || t.Kind.IsNumeric() {
+				unitSafety.includeUnitOf(t)
+			}
+		}
+		for side, t := range quad {
+			if unitSafety.status == unitSafe {
+				t.TurnLengthIntoNumberIfZero()
+			}
+			box.updateSide(rules, side, boxSide{
+				token:      t,
+				ruleIndex:  uint32(len(rules) - 1),
+				unitSafety: unitSafety,
+			})
+		}
+		box.compactRules(rules, decl.KeyRange, minifyWhitespace)
+	} else {
+		box.sides = [4]boxSide{}
+	}
+}
+
+func (box *boxTracker) mangleSide(rules []css_ast.Rule, decl *css_ast.RDeclaration, minifyWhitespace bool, side int) {
+	// Reset if we see a change in the "!important" flag
+	if box.important != decl.Important {
+		box.sides = [4]boxSide{}
+		box.important = decl.Important
+	}
+
+	if tokens := decl.Value; len(tokens) == 1 {
+		if t := tokens[0]; t.Kind.IsNumeric() || (t.Kind == css_lexer.TIdent && box.allowAuto && strings.EqualFold(t.Text, "auto")) {
+			unitSafety := unitSafetyTracker{}
+			if !box.allowAuto || t.Kind.IsNumeric() {
+				unitSafety.includeUnitOf(t)
+			}
+			if unitSafety.status == unitSafe && t.TurnLengthIntoNumberIfZero() {
+				tokens[0] = t
+			}
+			box.updateSide(rules, side, boxSide{
+				token:         t,
+				ruleIndex:     uint32(len(rules) - 1),
+				wasSingleRule: true,
+				unitSafety:    unitSafety,
+			})
+			box.compactRules(rules, decl.KeyRange, minifyWhitespace)
+			return
+		}
+	}
+
+	box.sides = [4]boxSide{}
+}
+
+func (box *boxTracker) compactRules(rules []css_ast.Rule, keyRange logger.Range, minifyWhitespace bool) {
+	// Don't compact if the shorthand form is unsupported
+	if box.key == css_ast.DUnknown {
+		return
+	}
+
+	// All tokens must be present
+	if eof := css_lexer.TEndOfFile; box.sides[0].token.Kind == eof || box.sides[1].token.Kind == eof ||
+		box.sides[2].token.Kind == eof || box.sides[3].token.Kind == eof {
+		return
+	}
+
+	// All tokens must have the same unit
+	for _, side := range box.sides[1:] {
+		if !side.unitSafety.isSafeWith(box.sides[0].unitSafety) {
+			return
+		}
+	}
+
+	// Generate the most minimal representation
+	tokens := compactTokenQuad(
+		box.sides[0].token,
+		box.sides[1].token,
+		box.sides[2].token,
+		box.sides[3].token,
+		minifyWhitespace,
+	)
+
+	// Remove all of the existing declarations
+	var minLoc logger.Loc
+	for i, side := range box.sides {
+		if loc := rules[side.ruleIndex].Loc; i == 0 || loc.Start < minLoc.Start {
+			minLoc = loc
+		}
+		rules[side.ruleIndex] = css_ast.Rule{}
+	}
+
+	// Insert the combined declaration where the last rule was
+	rules[box.sides[3].ruleIndex] = css_ast.Rule{Loc: minLoc, Data: &css_ast.RDeclaration{
+		Key:       box.key,
+		KeyText:   box.keyText,
+		Value:     tokens,
+		KeyRange:  keyRange,
+		Important: box.important,
+	}}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_box_shadow.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_box_shadow.go
new file mode 100644
index 0000000..5bc730f
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_box_shadow.go
@@ -0,0 +1,106 @@
+package css_parser
+
+import (
+	"strings"
+
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+)
+
+func (p *parser) lowerAndMangleBoxShadow(tokens []css_ast.Token, wouldClipColor *bool) []css_ast.Token {
+	insetCount := 0
+	colorCount := 0
+	numbersBegin := 0
+	numbersCount := 0
+	numbersDone := false
+	foundUnexpectedToken := false
+
+	for i, t := range tokens {
+		if t.Kind == css_lexer.TNumber || t.Kind == css_lexer.TDimension {
+			if numbersDone {
+				// Track if we found a non-number in between two numbers
+				foundUnexpectedToken = true
+			}
+			if p.options.minifySyntax && t.TurnLengthIntoNumberIfZero() {
+				// "0px" => "0"
+				tokens[i] = t
+			}
+			if numbersCount == 0 {
+				// Track the index of the first number
+				numbersBegin = i
+			}
+			numbersCount++
+		} else {
+			if numbersCount != 0 {
+				// Track when we find a non-number after a number
+				numbersDone = true
+			}
+
+			if looksLikeColor(t) {
+				colorCount++
+				tokens[i] = p.lowerAndMinifyColor(t, wouldClipColor)
+			} else if t.Kind == css_lexer.TIdent && strings.EqualFold(t.Text, "inset") {
+				insetCount++
+			} else {
+				// Track if we found a token other than a number, a color, or "inset"
+				foundUnexpectedToken = true
+			}
+		}
+	}
+
+	// If everything looks like a valid rule, trim trailing zeros off the numbers.
+	// There are three valid configurations of numbers:
+	//
+	//   offset-x | offset-y
+	//   offset-x | offset-y | blur-radius
+	//   offset-x | offset-y | blur-radius | spread-radius
+	//
+	// If omitted, blur-radius and spread-radius are implied to be zero.
+	if p.options.minifySyntax && insetCount <= 1 && colorCount <= 1 && numbersCount > 2 && numbersCount <= 4 && !foundUnexpectedToken {
+		numbersEnd := numbersBegin + numbersCount
+		for numbersCount > 2 && tokens[numbersBegin+numbersCount-1].IsZero() {
+			numbersCount--
+		}
+		tokens = append(tokens[:numbersBegin+numbersCount], tokens[numbersEnd:]...)
+	}
+
+	// Set the whitespace flags
+	for i := range tokens {
+		var whitespace css_ast.WhitespaceFlags
+		if i > 0 || !p.options.minifyWhitespace {
+			whitespace |= css_ast.WhitespaceBefore
+		}
+		if i+1 < len(tokens) {
+			whitespace |= css_ast.WhitespaceAfter
+		}
+		tokens[i].Whitespace = whitespace
+	}
+	return tokens
+}
+
+func (p *parser) lowerAndMangleBoxShadows(tokens []css_ast.Token, wouldClipColor *bool) []css_ast.Token {
+	n := len(tokens)
+	end := 0
+	i := 0
+
+	for i < n {
+		// Find the comma or the end of the token list
+		comma := i
+		for comma < n && tokens[comma].Kind != css_lexer.TComma {
+			comma++
+		}
+
+		// Mangle this individual shadow
+		end += copy(tokens[end:], p.lowerAndMangleBoxShadow(tokens[i:comma], wouldClipColor))
+
+		// Skip over the comma
+		if comma < n {
+			tokens[end] = tokens[comma]
+			end++
+			comma++
+		}
+		i = comma
+	}
+
+	return tokens[:end]
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_color.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_color.go
new file mode 100644
index 0000000..c6b6691
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_color.go
@@ -0,0 +1,938 @@
+package css_parser
+
+import (
+	"fmt"
+	"math"
+	"strconv"
+	"strings"
+
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+	"github.com/evanw/esbuild/internal/helpers"
+)
+
+// These names are shorter than their hex codes
+var shortColorName = map[uint32]string{
+	0x000080ff: "navy",
+	0x008000ff: "green",
+	0x008080ff: "teal",
+	0x4b0082ff: "indigo",
+	0x800000ff: "maroon",
+	0x800080ff: "purple",
+	0x808000ff: "olive",
+	0x808080ff: "gray",
+	0xa0522dff: "sienna",
+	0xa52a2aff: "brown",
+	0xc0c0c0ff: "silver",
+	0xcd853fff: "peru",
+	0xd2b48cff: "tan",
+	0xda70d6ff: "orchid",
+	0xdda0ddff: "plum",
+	0xee82eeff: "violet",
+	0xf0e68cff: "khaki",
+	0xf0ffffff: "azure",
+	0xf5deb3ff: "wheat",
+	0xf5f5dcff: "beige",
+	0xfa8072ff: "salmon",
+	0xfaf0e6ff: "linen",
+	0xff0000ff: "red",
+	0xff6347ff: "tomato",
+	0xff7f50ff: "coral",
+	0xffa500ff: "orange",
+	0xffc0cbff: "pink",
+	0xffd700ff: "gold",
+	0xffe4c4ff: "bisque",
+	0xfffafaff: "snow",
+	0xfffff0ff: "ivory",
+}
+
+var colorNameToHex = map[string]uint32{
+	"black":                0x000000ff,
+	"silver":               0xc0c0c0ff,
+	"gray":                 0x808080ff,
+	"white":                0xffffffff,
+	"maroon":               0x800000ff,
+	"red":                  0xff0000ff,
+	"purple":               0x800080ff,
+	"fuchsia":              0xff00ffff,
+	"green":                0x008000ff,
+	"lime":                 0x00ff00ff,
+	"olive":                0x808000ff,
+	"yellow":               0xffff00ff,
+	"navy":                 0x000080ff,
+	"blue":                 0x0000ffff,
+	"teal":                 0x008080ff,
+	"aqua":                 0x00ffffff,
+	"orange":               0xffa500ff,
+	"aliceblue":            0xf0f8ffff,
+	"antiquewhite":         0xfaebd7ff,
+	"aquamarine":           0x7fffd4ff,
+	"azure":                0xf0ffffff,
+	"beige":                0xf5f5dcff,
+	"bisque":               0xffe4c4ff,
+	"blanchedalmond":       0xffebcdff,
+	"blueviolet":           0x8a2be2ff,
+	"brown":                0xa52a2aff,
+	"burlywood":            0xdeb887ff,
+	"cadetblue":            0x5f9ea0ff,
+	"chartreuse":           0x7fff00ff,
+	"chocolate":            0xd2691eff,
+	"coral":                0xff7f50ff,
+	"cornflowerblue":       0x6495edff,
+	"cornsilk":             0xfff8dcff,
+	"crimson":              0xdc143cff,
+	"cyan":                 0x00ffffff,
+	"darkblue":             0x00008bff,
+	"darkcyan":             0x008b8bff,
+	"darkgoldenrod":        0xb8860bff,
+	"darkgray":             0xa9a9a9ff,
+	"darkgreen":            0x006400ff,
+	"darkgrey":             0xa9a9a9ff,
+	"darkkhaki":            0xbdb76bff,
+	"darkmagenta":          0x8b008bff,
+	"darkolivegreen":       0x556b2fff,
+	"darkorange":           0xff8c00ff,
+	"darkorchid":           0x9932ccff,
+	"darkred":              0x8b0000ff,
+	"darksalmon":           0xe9967aff,
+	"darkseagreen":         0x8fbc8fff,
+	"darkslateblue":        0x483d8bff,
+	"darkslategray":        0x2f4f4fff,
+	"darkslategrey":        0x2f4f4fff,
+	"darkturquoise":        0x00ced1ff,
+	"darkviolet":           0x9400d3ff,
+	"deeppink":             0xff1493ff,
+	"deepskyblue":          0x00bfffff,
+	"dimgray":              0x696969ff,
+	"dimgrey":              0x696969ff,
+	"dodgerblue":           0x1e90ffff,
+	"firebrick":            0xb22222ff,
+	"floralwhite":          0xfffaf0ff,
+	"forestgreen":          0x228b22ff,
+	"gainsboro":            0xdcdcdcff,
+	"ghostwhite":           0xf8f8ffff,
+	"gold":                 0xffd700ff,
+	"goldenrod":            0xdaa520ff,
+	"greenyellow":          0xadff2fff,
+	"grey":                 0x808080ff,
+	"honeydew":             0xf0fff0ff,
+	"hotpink":              0xff69b4ff,
+	"indianred":            0xcd5c5cff,
+	"indigo":               0x4b0082ff,
+	"ivory":                0xfffff0ff,
+	"khaki":                0xf0e68cff,
+	"lavender":             0xe6e6faff,
+	"lavenderblush":        0xfff0f5ff,
+	"lawngreen":            0x7cfc00ff,
+	"lemonchiffon":         0xfffacdff,
+	"lightblue":            0xadd8e6ff,
+	"lightcoral":           0xf08080ff,
+	"lightcyan":            0xe0ffffff,
+	"lightgoldenrodyellow": 0xfafad2ff,
+	"lightgray":            0xd3d3d3ff,
+	"lightgreen":           0x90ee90ff,
+	"lightgrey":            0xd3d3d3ff,
+	"lightpink":            0xffb6c1ff,
+	"lightsalmon":          0xffa07aff,
+	"lightseagreen":        0x20b2aaff,
+	"lightskyblue":         0x87cefaff,
+	"lightslategray":       0x778899ff,
+	"lightslategrey":       0x778899ff,
+	"lightsteelblue":       0xb0c4deff,
+	"lightyellow":          0xffffe0ff,
+	"limegreen":            0x32cd32ff,
+	"linen":                0xfaf0e6ff,
+	"magenta":              0xff00ffff,
+	"mediumaquamarine":     0x66cdaaff,
+	"mediumblue":           0x0000cdff,
+	"mediumorchid":         0xba55d3ff,
+	"mediumpurple":         0x9370dbff,
+	"mediumseagreen":       0x3cb371ff,
+	"mediumslateblue":      0x7b68eeff,
+	"mediumspringgreen":    0x00fa9aff,
+	"mediumturquoise":      0x48d1ccff,
+	"mediumvioletred":      0xc71585ff,
+	"midnightblue":         0x191970ff,
+	"mintcream":            0xf5fffaff,
+	"mistyrose":            0xffe4e1ff,
+	"moccasin":             0xffe4b5ff,
+	"navajowhite":          0xffdeadff,
+	"oldlace":              0xfdf5e6ff,
+	"olivedrab":            0x6b8e23ff,
+	"orangered":            0xff4500ff,
+	"orchid":               0xda70d6ff,
+	"palegoldenrod":        0xeee8aaff,
+	"palegreen":            0x98fb98ff,
+	"paleturquoise":        0xafeeeeff,
+	"palevioletred":        0xdb7093ff,
+	"papayawhip":           0xffefd5ff,
+	"peachpuff":            0xffdab9ff,
+	"peru":                 0xcd853fff,
+	"pink":                 0xffc0cbff,
+	"plum":                 0xdda0ddff,
+	"powderblue":           0xb0e0e6ff,
+	"rosybrown":            0xbc8f8fff,
+	"royalblue":            0x4169e1ff,
+	"saddlebrown":          0x8b4513ff,
+	"salmon":               0xfa8072ff,
+	"sandybrown":           0xf4a460ff,
+	"seagreen":             0x2e8b57ff,
+	"seashell":             0xfff5eeff,
+	"sienna":               0xa0522dff,
+	"skyblue":              0x87ceebff,
+	"slateblue":            0x6a5acdff,
+	"slategray":            0x708090ff,
+	"slategrey":            0x708090ff,
+	"snow":                 0xfffafaff,
+	"springgreen":          0x00ff7fff,
+	"steelblue":            0x4682b4ff,
+	"tan":                  0xd2b48cff,
+	"thistle":              0xd8bfd8ff,
+	"tomato":               0xff6347ff,
+	"turquoise":            0x40e0d0ff,
+	"violet":               0xee82eeff,
+	"wheat":                0xf5deb3ff,
+	"whitesmoke":           0xf5f5f5ff,
+	"yellowgreen":          0x9acd32ff,
+	"rebeccapurple":        0x663399ff,
+}
+
+func parseHex(text string) (uint32, bool) {
+	hex := uint32(0)
+	for _, c := range text {
+		hex <<= 4
+		switch {
+		case c >= '0' && c <= '9':
+			hex |= uint32(c) - '0'
+		case c >= 'a' && c <= 'f':
+			hex |= uint32(c) - ('a' - 10)
+		case c >= 'A' && c <= 'F':
+			hex |= uint32(c) - ('A' - 10)
+		default:
+			return 0, false
+		}
+	}
+	return hex, true
+}
+
+// 0xAABBCCDD => 0xABCD
+func compactHex(v uint32) uint32 {
+	return ((v & 0x0FF00000) >> 12) | ((v & 0x00000FF0) >> 4)
+}
+
+// 0xABCD => 0xAABBCCDD
+func expandHex(v uint32) uint32 {
+	return ((v & 0xF000) << 16) | ((v & 0xFF00) << 12) | ((v & 0x0FF0) << 8) | ((v & 0x00FF) << 4) | (v & 0x000F)
+}
+
+func hexR(v uint32) int { return int(v >> 24) }
+func hexG(v uint32) int { return int((v >> 16) & 255) }
+func hexB(v uint32) int { return int((v >> 8) & 255) }
+func hexA(v uint32) int { return int(v & 255) }
+
+func floatToStringForColor(a float64) string {
+	text := fmt.Sprintf("%.03f", a)
+	for text[len(text)-1] == '0' {
+		text = text[:len(text)-1]
+	}
+	if text[len(text)-1] == '.' {
+		text = text[:len(text)-1]
+	}
+	return text
+}
+
+func degreesForAngle(token css_ast.Token) (float64, bool) {
+	switch token.Kind {
+	case css_lexer.TNumber:
+		if value, err := strconv.ParseFloat(token.Text, 64); err == nil {
+			return value, true
+		}
+
+	case css_lexer.TDimension:
+		if value, err := strconv.ParseFloat(token.DimensionValue(), 64); err == nil {
+			switch token.DimensionUnit() {
+			case "deg":
+				return value, true
+			case "grad":
+				return value * (360.0 / 400.0), true
+			case "rad":
+				return value * (180.0 / math.Pi), true
+			case "turn":
+				return value * 360.0, true
+			}
+		}
+	}
+	return 0, false
+}
+
+func lowerAlphaPercentageToNumber(token css_ast.Token) css_ast.Token {
+	if token.Kind == css_lexer.TPercentage {
+		if value, err := strconv.ParseFloat(token.Text[:len(token.Text)-1], 64); err == nil {
+			token.Kind = css_lexer.TNumber
+			token.Text = floatToStringForColor(value / 100.0)
+		}
+	}
+	return token
+}
+
+// Convert newer color syntax to older color syntax for older browsers
+func (p *parser) lowerAndMinifyColor(token css_ast.Token, wouldClipColor *bool) css_ast.Token {
+	text := token.Text
+
+	switch token.Kind {
+	case css_lexer.THash:
+		if p.options.unsupportedCSSFeatures.Has(compat.HexRGBA) {
+			switch len(text) {
+			case 4:
+				// "#1234" => "rgba(1, 2, 3, 0.004)"
+				if hex, ok := parseHex(text); ok {
+					hex = expandHex(hex)
+					return p.tryToGenerateColor(token, parsedColor{hex: hex}, nil)
+				}
+
+			case 8:
+				// "#12345678" => "rgba(18, 52, 86, 0.47)"
+				if hex, ok := parseHex(text); ok {
+					return p.tryToGenerateColor(token, parsedColor{hex: hex}, nil)
+				}
+			}
+		}
+
+	case css_lexer.TIdent:
+		if p.options.unsupportedCSSFeatures.Has(compat.RebeccaPurple) && strings.EqualFold(text, "rebeccapurple") {
+			token.Kind = css_lexer.THash
+			token.Text = "663399"
+		}
+
+	case css_lexer.TFunction:
+		switch strings.ToLower(text) {
+		case "rgb", "rgba", "hsl", "hsla":
+			if p.options.unsupportedCSSFeatures.Has(compat.Modern_RGB_HSL) {
+				args := *token.Children
+				removeAlpha := false
+				addAlpha := false
+
+				// "hsl(1deg, 2%, 3%)" => "hsl(1, 2%, 3%)"
+				if (text == "hsl" || text == "hsla") && len(args) > 0 {
+					if degrees, ok := degreesForAngle(args[0]); ok {
+						args[0].Kind = css_lexer.TNumber
+						args[0].Text = floatToStringForColor(degrees)
+					}
+				}
+
+				// These check for "IsNumeric" to reject "var()" since a single "var()"
+				// can substitute for multiple tokens and that messes up pattern matching
+				switch len(args) {
+				case 3:
+					// "rgba(1 2 3)" => "rgb(1, 2, 3)"
+					// "hsla(1 2% 3%)" => "hsl(1, 2%, 3%)"
+					if args[0].Kind.IsNumeric() && args[1].Kind.IsNumeric() && args[2].Kind.IsNumeric() {
+						removeAlpha = true
+						args[0].Whitespace = 0
+						args[1].Whitespace = 0
+						commaToken := p.commaToken(token.Loc)
+						token.Children = &[]css_ast.Token{
+							args[0], commaToken,
+							args[1], commaToken,
+							args[2],
+						}
+					}
+
+				case 5:
+					// "rgba(1, 2, 3)" => "rgb(1, 2, 3)"
+					// "hsla(1, 2%, 3%)" => "hsl(1%, 2%, 3%)"
+					if args[0].Kind.IsNumeric() && args[1].Kind == css_lexer.TComma &&
+						args[2].Kind.IsNumeric() && args[3].Kind == css_lexer.TComma &&
+						args[4].Kind.IsNumeric() {
+						removeAlpha = true
+						break
+					}
+
+					// "rgb(1 2 3 / 4%)" => "rgba(1, 2, 3, 0.04)"
+					// "hsl(1 2% 3% / 4%)" => "hsla(1, 2%, 3%, 0.04)"
+					if args[0].Kind.IsNumeric() && args[1].Kind.IsNumeric() && args[2].Kind.IsNumeric() &&
+						args[3].Kind == css_lexer.TDelimSlash && args[4].Kind.IsNumeric() {
+						addAlpha = true
+						args[0].Whitespace = 0
+						args[1].Whitespace = 0
+						args[2].Whitespace = 0
+						commaToken := p.commaToken(token.Loc)
+						token.Children = &[]css_ast.Token{
+							args[0], commaToken,
+							args[1], commaToken,
+							args[2], commaToken,
+							lowerAlphaPercentageToNumber(args[4]),
+						}
+					}
+
+				case 7:
+					// "rgb(1%, 2%, 3%, 4%)" => "rgba(1%, 2%, 3%, 0.04)"
+					// "hsl(1, 2%, 3%, 4%)" => "hsla(1, 2%, 3%, 0.04)"
+					if args[0].Kind.IsNumeric() && args[1].Kind == css_lexer.TComma &&
+						args[2].Kind.IsNumeric() && args[3].Kind == css_lexer.TComma &&
+						args[4].Kind.IsNumeric() && args[5].Kind == css_lexer.TComma &&
+						args[6].Kind.IsNumeric() {
+						addAlpha = true
+						args[6] = lowerAlphaPercentageToNumber(args[6])
+					}
+				}
+
+				if removeAlpha {
+					if strings.EqualFold(text, "rgba") {
+						token.Text = "rgb"
+					} else if strings.EqualFold(text, "hsla") {
+						token.Text = "hsl"
+					}
+				} else if addAlpha {
+					if strings.EqualFold(text, "rgb") {
+						token.Text = "rgba"
+					} else if strings.EqualFold(text, "hsl") {
+						token.Text = "hsla"
+					}
+				}
+			}
+
+		case "hwb":
+			if p.options.unsupportedCSSFeatures.Has(compat.HWB) {
+				if color, ok := parseColor(token); ok {
+					return p.tryToGenerateColor(token, color, wouldClipColor)
+				}
+			}
+
+		case "color", "lab", "lch", "oklab", "oklch":
+			if p.options.unsupportedCSSFeatures.Has(compat.ColorFunctions) {
+				if color, ok := parseColor(token); ok {
+					return p.tryToGenerateColor(token, color, wouldClipColor)
+				}
+			}
+		}
+	}
+
+	// When minifying, try to parse the color and print it back out. This minifies
+	// the color because we always print it out using the shortest encoding.
+	if p.options.minifySyntax {
+		if hex, ok := parseColor(token); ok {
+			token = p.tryToGenerateColor(token, hex, wouldClipColor)
+		}
+	}
+
+	return token
+}
+
+type parsedColor struct {
+	x, y, z       F64    // color if hasColorSpace == true
+	hex           uint32 // color and alpha if hasColorSpace == false, alpha if hasColorSpace == true
+	hasColorSpace bool
+}
+
+func looksLikeColor(token css_ast.Token) bool {
+	switch token.Kind {
+	case css_lexer.TIdent:
+		if _, ok := colorNameToHex[strings.ToLower(token.Text)]; ok {
+			return true
+		}
+
+	case css_lexer.THash:
+		switch len(token.Text) {
+		case 3, 4, 6, 8:
+			if _, ok := parseHex(token.Text); ok {
+				return true
+			}
+		}
+
+	case css_lexer.TFunction:
+		switch strings.ToLower(token.Text) {
+		case
+			"color-mix",
+			"color",
+			"hsl",
+			"hsla",
+			"hwb",
+			"lab",
+			"lch",
+			"oklab",
+			"oklch",
+			"rgb",
+			"rgba":
+			return true
+		}
+	}
+
+	return false
+}
+
+func parseColor(token css_ast.Token) (parsedColor, bool) {
+	text := token.Text
+
+	switch token.Kind {
+	case css_lexer.TIdent:
+		if hex, ok := colorNameToHex[strings.ToLower(text)]; ok {
+			return parsedColor{hex: hex}, true
+		}
+
+	case css_lexer.THash:
+		switch len(text) {
+		case 3:
+			// "#123"
+			if hex, ok := parseHex(text); ok {
+				return parsedColor{hex: (expandHex(hex) << 8) | 0xFF}, true
+			}
+
+		case 4:
+			// "#1234"
+			if hex, ok := parseHex(text); ok {
+				return parsedColor{hex: expandHex(hex)}, true
+			}
+
+		case 6:
+			// "#112233"
+			if hex, ok := parseHex(text); ok {
+				return parsedColor{hex: (hex << 8) | 0xFF}, true
+			}
+
+		case 8:
+			// "#11223344"
+			if hex, ok := parseHex(text); ok {
+				return parsedColor{hex: hex}, true
+			}
+		}
+
+	case css_lexer.TFunction:
+		lowerText := strings.ToLower(text)
+		switch lowerText {
+		case "rgb", "rgba":
+			args := *token.Children
+			var r, g, b, a css_ast.Token
+
+			switch len(args) {
+			case 3:
+				// "rgb(1 2 3)"
+				r, g, b = args[0], args[1], args[2]
+
+			case 5:
+				// "rgba(1, 2, 3)"
+				if args[1].Kind == css_lexer.TComma && args[3].Kind == css_lexer.TComma {
+					r, g, b = args[0], args[2], args[4]
+					break
+				}
+
+				// "rgb(1 2 3 / 4%)"
+				if args[3].Kind == css_lexer.TDelimSlash {
+					r, g, b, a = args[0], args[1], args[2], args[4]
+				}
+
+			case 7:
+				// "rgb(1%, 2%, 3%, 4%)"
+				if args[1].Kind == css_lexer.TComma && args[3].Kind == css_lexer.TComma && args[5].Kind == css_lexer.TComma {
+					r, g, b, a = args[0], args[2], args[4], args[6]
+				}
+			}
+
+			if r, ok := parseColorByte(r, 1); ok {
+				if g, ok := parseColorByte(g, 1); ok {
+					if b, ok := parseColorByte(b, 1); ok {
+						if a, ok := parseAlphaByte(a); ok {
+							return parsedColor{hex: (r << 24) | (g << 16) | (b << 8) | a}, true
+						}
+					}
+				}
+			}
+
+		case "hsl", "hsla":
+			args := *token.Children
+			var h, s, l, a css_ast.Token
+
+			switch len(args) {
+			case 3:
+				// "hsl(1 2 3)"
+				h, s, l = args[0], args[1], args[2]
+
+			case 5:
+				// "hsla(1, 2, 3)"
+				if args[1].Kind == css_lexer.TComma && args[3].Kind == css_lexer.TComma {
+					h, s, l = args[0], args[2], args[4]
+					break
+				}
+
+				// "hsl(1 2 3 / 4%)"
+				if args[3].Kind == css_lexer.TDelimSlash {
+					h, s, l, a = args[0], args[1], args[2], args[4]
+				}
+
+			case 7:
+				// "hsl(1%, 2%, 3%, 4%)"
+				if args[1].Kind == css_lexer.TComma && args[3].Kind == css_lexer.TComma && args[5].Kind == css_lexer.TComma {
+					h, s, l, a = args[0], args[2], args[4], args[6]
+				}
+			}
+
+			// HSL => RGB
+			if h, ok := degreesForAngle(h); ok {
+				if s, ok := s.ClampedFractionForPercentage(); ok {
+					if l, ok := l.ClampedFractionForPercentage(); ok {
+						if a, ok := parseAlphaByte(a); ok {
+							r, g, b := hslToRgb(helpers.NewF64(h), helpers.NewF64(s), helpers.NewF64(l))
+							return parsedColor{hex: packRGBA(r, g, b, a)}, true
+						}
+					}
+				}
+			}
+
+		case "hwb":
+			args := *token.Children
+			var h, s, l, a css_ast.Token
+
+			switch len(args) {
+			case 3:
+				// "hwb(1 2 3)"
+				h, s, l = args[0], args[1], args[2]
+
+			case 5:
+				// "hwb(1 2 3 / 4%)"
+				if args[3].Kind == css_lexer.TDelimSlash {
+					h, s, l, a = args[0], args[1], args[2], args[4]
+				}
+			}
+
+			// HWB => RGB
+			if h, ok := degreesForAngle(h); ok {
+				if white, ok := s.ClampedFractionForPercentage(); ok {
+					if black, ok := l.ClampedFractionForPercentage(); ok {
+						if a, ok := parseAlphaByte(a); ok {
+							r, g, b := hwbToRgb(helpers.NewF64(h), helpers.NewF64(white), helpers.NewF64(black))
+							return parsedColor{hex: packRGBA(r, g, b, a)}, true
+						}
+					}
+				}
+			}
+
+		case "color":
+			args := *token.Children
+			var colorSpace, alpha css_ast.Token
+
+			switch len(args) {
+			case 4:
+				// "color(xyz 1 2 3)"
+				colorSpace = args[0]
+
+			case 6:
+				// "color(xyz 1 2 3 / 50%)"
+				if args[4].Kind == css_lexer.TDelimSlash {
+					colorSpace, alpha = args[0], args[5]
+				}
+			}
+
+			if colorSpace.Kind == css_lexer.TIdent {
+				if v0, ok := args[1].NumberOrFractionForPercentage(1, 0); ok {
+					if v1, ok := args[2].NumberOrFractionForPercentage(1, 0); ok {
+						if v2, ok := args[3].NumberOrFractionForPercentage(1, 0); ok {
+							if a, ok := parseAlphaByte(alpha); ok {
+								v0, v1, v2 := helpers.NewF64(v0), helpers.NewF64(v1), helpers.NewF64(v2)
+								switch strings.ToLower(colorSpace.Text) {
+								case "a98-rgb":
+									r, g, b := lin_a98rgb(v0, v1, v2)
+									x, y, z := lin_a98rgb_to_xyz(r, g, b)
+									return parsedColor{hasColorSpace: true, x: x, y: y, z: z, hex: a}, true
+
+								case "display-p3":
+									r, g, b := lin_p3(v0, v1, v2)
+									x, y, z := lin_p3_to_xyz(r, g, b)
+									return parsedColor{hasColorSpace: true, x: x, y: y, z: z, hex: a}, true
+
+								case "prophoto-rgb":
+									r, g, b := lin_prophoto(v0, v1, v2)
+									x, y, z := lin_prophoto_to_xyz(r, g, b)
+									x, y, z = d50_to_d65(x, y, z)
+									return parsedColor{hasColorSpace: true, x: x, y: y, z: z, hex: a}, true
+
+								case "rec2020":
+									r, g, b := lin_2020(v0, v1, v2)
+									x, y, z := lin_2020_to_xyz(r, g, b)
+									return parsedColor{hasColorSpace: true, x: x, y: y, z: z, hex: a}, true
+
+								case "srgb":
+									r, g, b := lin_srgb(v0, v1, v2)
+									x, y, z := lin_srgb_to_xyz(r, g, b)
+									return parsedColor{hasColorSpace: true, x: x, y: y, z: z, hex: a}, true
+
+								case "srgb-linear":
+									x, y, z := lin_srgb_to_xyz(v0, v1, v2)
+									return parsedColor{hasColorSpace: true, x: x, y: y, z: z, hex: a}, true
+
+								case "xyz", "xyz-d65":
+									return parsedColor{hasColorSpace: true, x: v0, y: v1, z: v2, hex: a}, true
+
+								case "xyz-d50":
+									x, y, z := d50_to_d65(v0, v1, v2)
+									return parsedColor{hasColorSpace: true, x: x, y: y, z: z, hex: a}, true
+								}
+							}
+						}
+					}
+				}
+			}
+
+		case "lab", "lch", "oklab", "oklch":
+			args := *token.Children
+			var v0, v1, v2, alpha css_ast.Token
+
+			switch len(args) {
+			case 3:
+				// "lab(1 2 3)"
+				v0, v1, v2 = args[0], args[1], args[2]
+
+			case 5:
+				// "lab(1 2 3 / 50%)"
+				if args[3].Kind == css_lexer.TDelimSlash {
+					v0, v1, v2, alpha = args[0], args[1], args[2], args[4]
+				}
+			}
+
+			if v0.Kind != css_lexer.T(0) {
+				if alpha, ok := parseAlphaByte(alpha); ok {
+					switch lowerText {
+					case "lab":
+						if v0, ok := v0.NumberOrFractionForPercentage(100, 0); ok {
+							if v1, ok := v1.NumberOrFractionForPercentage(125, css_ast.AllowAnyPercentage); ok {
+								if v2, ok := v2.NumberOrFractionForPercentage(125, css_ast.AllowAnyPercentage); ok {
+									v0, v1, v2 := helpers.NewF64(v0), helpers.NewF64(v1), helpers.NewF64(v2)
+									x, y, z := lab_to_xyz(v0, v1, v2)
+									x, y, z = d50_to_d65(x, y, z)
+									return parsedColor{hasColorSpace: true, x: x, y: y, z: z, hex: alpha}, true
+								}
+							}
+						}
+
+					case "lch":
+						if v0, ok := v0.NumberOrFractionForPercentage(100, 0); ok {
+							if v1, ok := v1.NumberOrFractionForPercentage(125, css_ast.AllowPercentageAbove100); ok {
+								if v2, ok := degreesForAngle(v2); ok {
+									v0, v1, v2 := helpers.NewF64(v0), helpers.NewF64(v1), helpers.NewF64(v2)
+									l, a, b := lch_to_lab(v0, v1, v2)
+									x, y, z := lab_to_xyz(l, a, b)
+									x, y, z = d50_to_d65(x, y, z)
+									return parsedColor{hasColorSpace: true, x: x, y: y, z: z, hex: alpha}, true
+								}
+							}
+						}
+
+					case "oklab":
+						if v0, ok := v0.NumberOrFractionForPercentage(1, 0); ok {
+							if v1, ok := v1.NumberOrFractionForPercentage(0.4, css_ast.AllowAnyPercentage); ok {
+								if v2, ok := v2.NumberOrFractionForPercentage(0.4, css_ast.AllowAnyPercentage); ok {
+									v0, v1, v2 := helpers.NewF64(v0), helpers.NewF64(v1), helpers.NewF64(v2)
+									x, y, z := oklab_to_xyz(v0, v1, v2)
+									return parsedColor{hasColorSpace: true, x: x, y: y, z: z, hex: alpha}, true
+								}
+							}
+						}
+
+					case "oklch":
+						if v0, ok := v0.NumberOrFractionForPercentage(1, 0); ok {
+							if v1, ok := v1.NumberOrFractionForPercentage(0.4, css_ast.AllowPercentageAbove100); ok {
+								if v2, ok := degreesForAngle(v2); ok {
+									v0, v1, v2 := helpers.NewF64(v0), helpers.NewF64(v1), helpers.NewF64(v2)
+									l, a, b := oklch_to_oklab(v0, v1, v2)
+									x, y, z := oklab_to_xyz(l, a, b)
+									return parsedColor{hasColorSpace: true, x: x, y: y, z: z, hex: alpha}, true
+								}
+							}
+						}
+					}
+				}
+			}
+		}
+	}
+
+	return parsedColor{}, false
+}
+
+// Reference: https://drafts.csswg.org/css-color/#hwb-to-rgb
+func hwbToRgb(hue F64, white F64, black F64) (r F64, g F64, b F64) {
+	if white.Add(black).Value() >= 1 {
+		gray := white.Div(white.Add(black))
+		return gray, gray, gray
+	}
+	delta := white.Add(black).Neg().AddConst(1)
+	r, g, b = hslToRgb(hue, helpers.NewF64(1), helpers.NewF64(0.5))
+	r = delta.Mul(r).Add(white)
+	g = delta.Mul(g).Add(white)
+	b = delta.Mul(b).Add(white)
+	return
+}
+
+// Reference https://drafts.csswg.org/css-color/#hsl-to-rgb
+func hslToRgb(hue F64, sat F64, light F64) (r F64, g F64, b F64) {
+	hue = hue.DivConst(360.0)
+	var t2 F64
+	if light.Value() <= 0.5 {
+		t2 = sat.AddConst(1).Mul(light)
+	} else {
+		t2 = light.Add(sat).Sub(light.Mul(sat))
+	}
+	t1 := light.MulConst(2).Sub(t2)
+	r = hueToRgb(t1, t2, hue.AddConst(1.0/3.0))
+	g = hueToRgb(t1, t2, hue)
+	b = hueToRgb(t1, t2, hue.SubConst(1.0/3.0))
+	return
+}
+
+func hueToRgb(t1 F64, t2 F64, hue F64) F64 {
+	hue = hue.Sub(hue.Floor())
+	hue = hue.MulConst(6)
+	var f F64
+	if hue.Value() < 1 {
+		f = helpers.Lerp(t1, t2, hue)
+	} else if hue.Value() < 3 {
+		f = t2
+	} else if hue.Value() < 4 {
+		f = helpers.Lerp(t1, t2, hue.Neg().AddConst(4))
+	} else {
+		f = t1
+	}
+	return f
+}
+
+func packRGBA(rf F64, gf F64, bf F64, a uint32) uint32 {
+	r := floatToByte(rf.Value())
+	g := floatToByte(gf.Value())
+	b := floatToByte(bf.Value())
+	return (r << 24) | (g << 16) | (b << 8) | a
+}
+
+func floatToByte(f float64) uint32 {
+	i := int(math.Round(f * 255))
+	if i < 0 {
+		i = 0
+	} else if i > 255 {
+		i = 255
+	}
+	return uint32(i)
+}
+
+func parseAlphaByte(token css_ast.Token) (uint32, bool) {
+	if token.Kind == css_lexer.T(0) {
+		return 255, true
+	}
+	return parseColorByte(token, 255)
+}
+
+func parseColorByte(token css_ast.Token, scale float64) (uint32, bool) {
+	var i int
+	var ok bool
+
+	switch token.Kind {
+	case css_lexer.TNumber:
+		if f, err := strconv.ParseFloat(token.Text, 64); err == nil {
+			i = int(math.Round(f * scale))
+			ok = true
+		}
+
+	case css_lexer.TPercentage:
+		if f, err := strconv.ParseFloat(token.PercentageValue(), 64); err == nil {
+			i = int(math.Round(f * (255.0 / 100.0)))
+			ok = true
+		}
+	}
+
+	if i < 0 {
+		i = 0
+	} else if i > 255 {
+		i = 255
+	}
+	return uint32(i), ok
+}
+
+func tryToConvertToHexWithoutClipping(x F64, y F64, z F64, a uint32) (uint32, bool) {
+	r, g, b := gam_srgb(xyz_to_lin_srgb(x, y, z))
+	if r.Value() < -0.5/255 || r.Value() > 255.5/255 ||
+		g.Value() < -0.5/255 || g.Value() > 255.5/255 ||
+		b.Value() < -0.5/255 || b.Value() > 255.5/255 {
+		return 0, false
+	}
+	return packRGBA(r, g, b, a), true
+}
+
+func (p *parser) tryToGenerateColor(token css_ast.Token, color parsedColor, wouldClipColor *bool) css_ast.Token {
+	// Note: Do NOT remove color information from fully transparent colors.
+	// Safari behaves differently than other browsers for color interpolation:
+	// https://css-tricks.com/thing-know-gradients-transparent-black/
+
+	// Attempt to convert other color spaces to sRGB, and only continue if the
+	// result (rounded to the nearest byte) will be in the 0-to-1 sRGB range
+	var hex uint32
+	if !color.hasColorSpace {
+		hex = color.hex
+	} else if result, ok := tryToConvertToHexWithoutClipping(color.x, color.y, color.z, color.hex); ok {
+		hex = result
+	} else if wouldClipColor != nil {
+		*wouldClipColor = true
+		return token
+	} else {
+		r, g, b := gamut_mapping_xyz_to_srgb(color.x, color.y, color.z)
+		hex = packRGBA(r, g, b, color.hex)
+	}
+
+	if hexA(hex) == 255 {
+		token.Children = nil
+		if name, ok := shortColorName[hex]; ok && p.options.minifySyntax {
+			token.Kind = css_lexer.TIdent
+			token.Text = name
+		} else {
+			token.Kind = css_lexer.THash
+			hex >>= 8
+			compact := compactHex(hex)
+			if p.options.minifySyntax && hex == expandHex(compact) {
+				token.Text = fmt.Sprintf("%03x", compact)
+			} else {
+				token.Text = fmt.Sprintf("%06x", hex)
+			}
+		}
+	} else if !p.options.unsupportedCSSFeatures.Has(compat.HexRGBA) {
+		token.Children = nil
+		token.Kind = css_lexer.THash
+		compact := compactHex(hex)
+		if p.options.minifySyntax && hex == expandHex(compact) {
+			token.Text = fmt.Sprintf("%04x", compact)
+		} else {
+			token.Text = fmt.Sprintf("%08x", hex)
+		}
+	} else {
+		token.Kind = css_lexer.TFunction
+		token.Text = "rgba"
+		commaToken := p.commaToken(token.Loc)
+		index := hexA(hex) * 4
+		alpha := alphaFractionTable[index : index+4]
+		if space := strings.IndexByte(alpha, ' '); space != -1 {
+			alpha = alpha[:space]
+		}
+		token.Children = &[]css_ast.Token{
+			{Loc: token.Loc, Kind: css_lexer.TNumber, Text: strconv.Itoa(hexR(hex))}, commaToken,
+			{Loc: token.Loc, Kind: css_lexer.TNumber, Text: strconv.Itoa(hexG(hex))}, commaToken,
+			{Loc: token.Loc, Kind: css_lexer.TNumber, Text: strconv.Itoa(hexB(hex))}, commaToken,
+			{Loc: token.Loc, Kind: css_lexer.TNumber, Text: alpha},
+		}
+	}
+
+	return token
+}
+
+// Every four characters in this table is the fraction for that index
+const alphaFractionTable string = "" +
+	"0   .004.008.01 .016.02 .024.027.03 .035.04 .043.047.05 .055.06 " +
+	".063.067.07 .075.08 .082.086.09 .094.098.1  .106.11 .114.118.12 " +
+	".125.13 .133.137.14 .145.15 .153.157.16 .165.17 .173.176.18 .184" +
+	".19 .192.196.2  .204.208.21 .216.22 .224.227.23 .235.24 .243.247" +
+	".25 .255.26 .263.267.27 .275.28 .282.286.29 .294.298.3  .306.31 " +
+	".314.318.32 .325.33 .333.337.34 .345.35 .353.357.36 .365.37 .373" +
+	".376.38 .384.39 .392.396.4  .404.408.41 .416.42 .424.427.43 .435" +
+	".44 .443.447.45 .455.46 .463.467.47 .475.48 .482.486.49 .494.498" +
+	".5  .506.51 .514.518.52 .525.53 .533.537.54 .545.55 .553.557.56 " +
+	".565.57 .573.576.58 .584.59 .592.596.6  .604.608.61 .616.62 .624" +
+	".627.63 .635.64 .643.647.65 .655.66 .663.667.67 .675.68 .682.686" +
+	".69 .694.698.7  .706.71 .714.718.72 .725.73 .733.737.74 .745.75 " +
+	".753.757.76 .765.77 .773.776.78 .784.79 .792.796.8  .804.808.81 " +
+	".816.82 .824.827.83 .835.84 .843.847.85 .855.86 .863.867.87 .875" +
+	".88 .882.886.89 .894.898.9  .906.91 .914.918.92 .925.93 .933.937" +
+	".94 .945.95 .953.957.96 .965.97 .973.976.98 .984.99 .992.9961   "
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_composes.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_composes.go
new file mode 100644
index 0000000..2b8aa0a
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_composes.go
@@ -0,0 +1,103 @@
+package css_parser
+
+import (
+	"fmt"
+	"strings"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type composesContext struct {
+	parentRefs   []ast.Ref
+	parentRange  logger.Range
+	problemRange logger.Range
+}
+
+func (p *parser) handleComposesPragma(context composesContext, tokens []css_ast.Token) {
+	type nameWithLoc struct {
+		loc  logger.Loc
+		text string
+	}
+	var names []nameWithLoc
+	fromGlobal := false
+
+	for i, t := range tokens {
+		if t.Kind == css_lexer.TIdent {
+			// Check for a "from" clause at the end
+			if strings.EqualFold(t.Text, "from") && i+2 == len(tokens) {
+				last := tokens[i+1]
+
+				// A string or a URL is an external file
+				if last.Kind == css_lexer.TString || last.Kind == css_lexer.TURL {
+					var importRecordIndex uint32
+					if last.Kind == css_lexer.TString {
+						importRecordIndex = uint32(len(p.importRecords))
+						p.importRecords = append(p.importRecords, ast.ImportRecord{
+							Kind:  ast.ImportComposesFrom,
+							Path:  logger.Path{Text: last.Text},
+							Range: p.source.RangeOfString(last.Loc),
+						})
+					} else {
+						importRecordIndex = last.PayloadIndex
+						p.importRecords[importRecordIndex].Kind = ast.ImportComposesFrom
+					}
+					for _, parentRef := range context.parentRefs {
+						composes := p.composes[parentRef]
+						for _, name := range names {
+							composes.ImportedNames = append(composes.ImportedNames, css_ast.ImportedComposesName{
+								ImportRecordIndex: importRecordIndex,
+								Alias:             name.text,
+								AliasLoc:          name.loc,
+							})
+						}
+					}
+					return
+				}
+
+				// An identifier must be "global"
+				if last.Kind == css_lexer.TIdent {
+					if strings.EqualFold(last.Text, "global") {
+						fromGlobal = true
+						break
+					}
+
+					p.log.AddID(logger.MsgID_CSS_CSSSyntaxError, logger.Warning, &p.tracker, css_lexer.RangeOfIdentifier(p.source, last.Loc),
+						fmt.Sprintf("\"composes\" declaration uses invalid location %q", last.Text))
+					p.prevError = t.Loc
+					return
+				}
+			}
+
+			names = append(names, nameWithLoc{t.Loc, t.Text})
+			continue
+		}
+
+		// Any unexpected tokens are a syntax error
+		var text string
+		switch t.Kind {
+		case css_lexer.TURL, css_lexer.TBadURL, css_lexer.TString, css_lexer.TUnterminatedString:
+			text = fmt.Sprintf("Unexpected %s", t.Kind.String())
+		default:
+			text = fmt.Sprintf("Unexpected %q", t.Text)
+		}
+		p.log.AddID(logger.MsgID_CSS_CSSSyntaxError, logger.Warning, &p.tracker, logger.Range{Loc: t.Loc}, text)
+		p.prevError = t.Loc
+		return
+	}
+
+	// If we get here, all of these names are not references to another file
+	old := p.makeLocalSymbols
+	if fromGlobal {
+		p.makeLocalSymbols = false
+	}
+	for _, parentRef := range context.parentRefs {
+		composes := p.composes[parentRef]
+		for _, name := range names {
+			composes.Names = append(composes.Names, p.symbolForName(name.loc, name.text))
+		}
+	}
+	p.makeLocalSymbols = old
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_container.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_container.go
new file mode 100644
index 0000000..6c7ca3a
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_container.go
@@ -0,0 +1,53 @@
+package css_parser
+
+import (
+	"strings"
+
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+)
+
+// Scan for container names in the "container" shorthand property
+func (p *parser) processContainerShorthand(tokens []css_ast.Token) {
+	// Validate the syntax
+	for i, t := range tokens {
+		if t.Kind == css_lexer.TIdent {
+			continue
+		}
+		if t.Kind == css_lexer.TDelimSlash && i+2 == len(tokens) && tokens[i+1].Kind == css_lexer.TIdent {
+			break
+		}
+		return
+	}
+
+	// Convert any local names
+	for i, t := range tokens {
+		if t.Kind != css_lexer.TIdent {
+			break
+		}
+		p.handleSingleContainerName(&tokens[i])
+	}
+}
+
+func (p *parser) processContainerName(tokens []css_ast.Token) {
+	// Validate the syntax
+	for _, t := range tokens {
+		if t.Kind != css_lexer.TIdent {
+			return
+		}
+	}
+
+	// Convert any local names
+	for i := range tokens {
+		p.handleSingleContainerName(&tokens[i])
+	}
+}
+
+func (p *parser) handleSingleContainerName(token *css_ast.Token) {
+	if lower := strings.ToLower(token.Text); lower == "none" || cssWideAndReservedKeywords[lower] {
+		return
+	}
+
+	token.Kind = css_lexer.TSymbol
+	token.PayloadIndex = p.symbolForName(token.Loc, token.Text).Ref.InnerIndex
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_font.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_font.go
new file mode 100644
index 0000000..c1644b4
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_font.go
@@ -0,0 +1,138 @@
+package css_parser
+
+import (
+	"strconv"
+	"strings"
+
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+)
+
+// Specification: https://drafts.csswg.org/css-fonts/#font-prop
+// [ <font-style> || <font-variant-css2> || <font-weight> || <font-stretch-css3> ]? <font-size> [ / <line-height> ]? <font-family>
+func (p *parser) mangleFont(tokens []css_ast.Token) []css_ast.Token {
+	var result []css_ast.Token
+
+	// Scan up to the font size
+	pos := 0
+	for ; pos < len(tokens); pos++ {
+		token := tokens[pos]
+		if isFontSize(token) {
+			break
+		}
+
+		switch token.Kind {
+		case css_lexer.TIdent:
+			switch strings.ToLower(token.Text) {
+			case "normal":
+				// "All subproperties of the font property are first reset to their initial values"
+				// This implies that "normal" doesn't do anything. Also all of the optional values
+				// contain "normal" as an option and they are unordered so it's impossible to say
+				// what property "normal" corresponds to. Just drop these tokens to save space.
+				continue
+
+			// <font-style>
+			case "italic":
+			case "oblique":
+				if pos+1 < len(tokens) && tokens[pos+1].IsAngle() {
+					result = append(result, token, tokens[pos+1])
+					pos++
+					continue
+				}
+
+			// <font-variant-css2>
+			case "small-caps":
+
+			// <font-weight>
+			case "bold", "bolder", "lighter":
+				result = append(result, p.mangleFontWeight(token))
+				continue
+
+			// <font-stretch-css3>
+			case "ultra-condensed", "extra-condensed", "condensed", "semi-condensed",
+				"semi-expanded", "expanded", "extra-expanded", "ultra-expanded":
+
+			default:
+				// All other tokens are unrecognized, so we bail if we hit one
+				return tokens
+			}
+			result = append(result, token)
+
+		case css_lexer.TNumber:
+			// "Only values greater than or equal to 1, and less than or equal to
+			// 1000, are valid, and all other values are invalid."
+			if value, err := strconv.ParseFloat(token.Text, 64); err != nil || value < 1 || value > 1000 {
+				return tokens
+			}
+			result = append(result, token)
+
+		default:
+			// All other tokens are unrecognized, so we bail if we hit one
+			return tokens
+		}
+	}
+
+	// <font-size>
+	if pos == len(tokens) {
+		return tokens
+	}
+	result = append(result, tokens[pos])
+	pos++
+
+	// / <line-height>
+	if pos < len(tokens) && tokens[pos].Kind == css_lexer.TDelimSlash {
+		if pos+1 == len(tokens) {
+			return tokens
+		}
+		result = append(result, tokens[pos], tokens[pos+1])
+		pos += 2
+
+		// Remove the whitespace around the "/" character
+		if p.options.minifyWhitespace {
+			result[len(result)-3].Whitespace &= ^css_ast.WhitespaceAfter
+			result[len(result)-2].Whitespace = 0
+			result[len(result)-1].Whitespace &= ^css_ast.WhitespaceBefore
+		}
+	}
+
+	// <font-family>
+	if family, ok := p.mangleFontFamily(tokens[pos:]); ok {
+		if len(result) > 0 && len(family) > 0 && family[0].Kind != css_lexer.TString {
+			family[0].Whitespace |= css_ast.WhitespaceBefore
+		}
+		return append(result, family...)
+	}
+	return tokens
+}
+
+var fontSizeKeywords = map[string]bool{
+	// <absolute-size>: https://drafts.csswg.org/css-fonts/#valdef-font-size-absolute-size
+	"xx-small":  true,
+	"x-small":   true,
+	"small":     true,
+	"medium":    true,
+	"large":     true,
+	"x-large":   true,
+	"xx-large":  true,
+	"xxx-large": true,
+
+	// <relative-size>: https://drafts.csswg.org/css-fonts/#valdef-font-size-relative-size
+	"larger":  true,
+	"smaller": true,
+}
+
+// Specification: https://drafts.csswg.org/css-fonts/#font-size-prop
+func isFontSize(token css_ast.Token) bool {
+	// <length-percentage>
+	if token.Kind == css_lexer.TDimension || token.Kind == css_lexer.TPercentage {
+		return true
+	}
+
+	// <absolute-size> or <relative-size>
+	if token.Kind == css_lexer.TIdent {
+		_, ok := fontSizeKeywords[strings.ToLower(token.Text)]
+		return ok
+	}
+
+	return false
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_font_family.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_font_family.go
new file mode 100644
index 0000000..8c40382
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_font_family.go
@@ -0,0 +1,162 @@
+package css_parser
+
+import (
+	"strings"
+
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+)
+
+// These keywords usually require special handling when parsing.
+
+// Declaring a property to have these values explicitly specifies a particular
+// defaulting behavior instead of setting the property to that identifier value.
+// As specified in CSS Values and Units Level 3, all CSS properties can accept
+// these values.
+//
+// For example, "font-family: 'inherit'" sets the font family to the font named
+// "inherit" while "font-family: inherit" sets the font family to the inherited
+// value.
+//
+// Note that other CSS specifications can define additional CSS-wide keywords,
+// which we should copy here whenever new ones are created so we can quote those
+// identifiers to avoid collisions with any newly-created CSS-wide keywords.
+var cssWideAndReservedKeywords = map[string]bool{
+	// CSS Values and Units Level 3: https://drafts.csswg.org/css-values-3/#common-keywords
+	"initial": true, // CSS-wide keyword
+	"inherit": true, // CSS-wide keyword
+	"unset":   true, // CSS-wide keyword
+	"default": true, // CSS reserved keyword
+
+	// CSS Cascading and Inheritance Level 5: https://drafts.csswg.org/css-cascade-5/#defaulting-keywords
+	"revert":       true, // Cascade-dependent keyword
+	"revert-layer": true, // Cascade-dependent keyword
+}
+
+// Font family names that happen to be the same as a keyword value must be
+// quoted to prevent confusion with the keywords with the same names. UAs must
+// not consider these keywords as matching the <family-name> type.
+// Specification: https://drafts.csswg.org/css-fonts/#generic-font-families
+var genericFamilyNames = map[string]bool{
+	"serif":         true,
+	"sans-serif":    true,
+	"cursive":       true,
+	"fantasy":       true,
+	"monospace":     true,
+	"system-ui":     true,
+	"emoji":         true,
+	"math":          true,
+	"fangsong":      true,
+	"ui-serif":      true,
+	"ui-sans-serif": true,
+	"ui-monospace":  true,
+	"ui-rounded":    true,
+}
+
+// Specification: https://drafts.csswg.org/css-fonts/#font-family-prop
+func (p *parser) mangleFontFamily(tokens []css_ast.Token) ([]css_ast.Token, bool) {
+	result, rest, ok := p.mangleFamilyNameOrGenericName(nil, tokens)
+	if !ok {
+		return nil, false
+	}
+
+	for len(rest) > 0 && rest[0].Kind == css_lexer.TComma {
+		result, rest, ok = p.mangleFamilyNameOrGenericName(append(result, rest[0]), rest[1:])
+		if !ok {
+			return nil, false
+		}
+	}
+
+	if len(rest) > 0 {
+		return nil, false
+	}
+
+	return result, true
+}
+
+func (p *parser) mangleFamilyNameOrGenericName(result []css_ast.Token, tokens []css_ast.Token) ([]css_ast.Token, []css_ast.Token, bool) {
+	if len(tokens) > 0 {
+		t := tokens[0]
+
+		// Handle <generic-family>
+		if t.Kind == css_lexer.TIdent && genericFamilyNames[t.Text] {
+			return append(result, t), tokens[1:], true
+		}
+
+		// Handle <family-name>
+		if t.Kind == css_lexer.TString {
+			// "If a sequence of identifiers is given as a <family-name>, the computed
+			// value is the name converted to a string by joining all the identifiers
+			// in the sequence by single spaces."
+			//
+			// More information: https://mathiasbynens.be/notes/unquoted-font-family
+			names := strings.Split(t.Text, " ")
+			for _, name := range names {
+				if !isValidCustomIdent(name, genericFamilyNames) {
+					return append(result, t), tokens[1:], true
+				}
+			}
+			for i, name := range names {
+				var whitespace css_ast.WhitespaceFlags
+				if i != 0 || !p.options.minifyWhitespace {
+					whitespace = css_ast.WhitespaceBefore
+				}
+				result = append(result, css_ast.Token{
+					Loc:        t.Loc,
+					Kind:       css_lexer.TIdent,
+					Text:       name,
+					Whitespace: whitespace,
+				})
+			}
+			return result, tokens[1:], true
+		}
+
+		// "Font family names other than generic families must either be given
+		// quoted as <string>s, or unquoted as a sequence of one or more
+		// <custom-ident>."
+		if t.Kind == css_lexer.TIdent {
+			for {
+				if !isValidCustomIdent(t.Text, genericFamilyNames) {
+					return nil, nil, false
+				}
+				result = append(result, t)
+				tokens = tokens[1:]
+				if len(tokens) == 0 || tokens[0].Kind != css_lexer.TIdent {
+					break
+				}
+				t = tokens[0]
+			}
+			return result, tokens, true
+		}
+	}
+
+	// Anything other than the cases listed above causes us to bail
+	return nil, nil, false
+}
+
+// Specification: https://drafts.csswg.org/css-values-4/#custom-idents
+func isValidCustomIdent(text string, predefinedKeywords map[string]bool) bool {
+	loweredText := strings.ToLower(text)
+
+	if predefinedKeywords[loweredText] {
+		return false
+	}
+	if cssWideAndReservedKeywords[loweredText] {
+		return false
+	}
+	if loweredText == "" {
+		return false
+	}
+
+	// validate if it contains characters which needs to be escaped
+	if !css_lexer.WouldStartIdentifierWithoutEscapes(text) {
+		return false
+	}
+	for _, c := range text {
+		if !css_lexer.IsNameContinue(c) {
+			return false
+		}
+	}
+
+	return true
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_font_weight.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_font_weight.go
new file mode 100644
index 0000000..6b8fe90
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_font_weight.go
@@ -0,0 +1,25 @@
+package css_parser
+
+import (
+	"strings"
+
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+)
+
+func (p *parser) mangleFontWeight(token css_ast.Token) css_ast.Token {
+	if token.Kind != css_lexer.TIdent {
+		return token
+	}
+
+	switch strings.ToLower(token.Text) {
+	case "normal":
+		token.Text = "400"
+		token.Kind = css_lexer.TNumber
+	case "bold":
+		token.Text = "700"
+		token.Kind = css_lexer.TNumber
+	}
+
+	return token
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_gradient.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_gradient.go
new file mode 100644
index 0000000..4edec47
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_gradient.go
@@ -0,0 +1,1057 @@
+package css_parser
+
+import (
+	"fmt"
+	"math"
+	"strconv"
+	"strings"
+
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type gradientKind uint8
+
+const (
+	linearGradient gradientKind = iota
+	radialGradient
+	conicGradient
+)
+
+type parsedGradient struct {
+	leadingTokens []css_ast.Token
+	colorStops    []colorStop
+	kind          gradientKind
+	repeating     bool
+}
+
+type colorStop struct {
+	positions []css_ast.Token
+	color     css_ast.Token
+	midpoint  css_ast.Token // Absent if "midpoint.Kind == css_lexer.T(0)"
+}
+
+func parseGradient(token css_ast.Token) (gradient parsedGradient, success bool) {
+	if token.Kind != css_lexer.TFunction {
+		return
+	}
+
+	switch strings.ToLower(token.Text) {
+	case "linear-gradient":
+		gradient.kind = linearGradient
+
+	case "radial-gradient":
+		gradient.kind = radialGradient
+
+	case "conic-gradient":
+		gradient.kind = conicGradient
+
+	case "repeating-linear-gradient":
+		gradient.kind = linearGradient
+		gradient.repeating = true
+
+	case "repeating-radial-gradient":
+		gradient.kind = radialGradient
+		gradient.repeating = true
+
+	case "repeating-conic-gradient":
+		gradient.kind = conicGradient
+		gradient.repeating = true
+
+	default:
+		return
+	}
+
+	// Bail if any token is a "var()" since it may introduce commas
+	tokens := *token.Children
+	for _, t := range tokens {
+		if t.Kind == css_lexer.TFunction && strings.EqualFold(t.Text, "var") {
+			return
+		}
+	}
+
+	// Try to strip the initial tokens
+	if len(tokens) > 0 && !looksLikeColor(tokens[0]) {
+		i := 0
+		for i < len(tokens) && tokens[i].Kind != css_lexer.TComma {
+			i++
+		}
+		gradient.leadingTokens = tokens[:i]
+		if i < len(tokens) {
+			tokens = tokens[i+1:]
+		} else {
+			tokens = nil
+		}
+	}
+
+	// Try to parse the color stops
+	for len(tokens) > 0 {
+		// Parse the color
+		color := tokens[0]
+		if !looksLikeColor(color) {
+			return
+		}
+		tokens = tokens[1:]
+
+		// Parse up to two positions
+		var positions []css_ast.Token
+		for len(positions) < 2 && len(tokens) > 0 {
+			position := tokens[0]
+			if position.Kind.IsNumeric() || (position.Kind == css_lexer.TFunction && strings.EqualFold(position.Text, "calc")) {
+				positions = append(positions, position)
+			} else {
+				break
+			}
+			tokens = tokens[1:]
+		}
+
+		// Parse the comma
+		var midpoint css_ast.Token
+		if len(tokens) > 0 {
+			if tokens[0].Kind != css_lexer.TComma {
+				return
+			}
+			tokens = tokens[1:]
+			if len(tokens) == 0 {
+				return
+			}
+
+			// Parse the midpoint, if any
+			if len(tokens) > 0 && tokens[0].Kind.IsNumeric() {
+				midpoint = tokens[0]
+				tokens = tokens[1:]
+
+				// Followed by a mandatory comma
+				if len(tokens) == 0 || tokens[0].Kind != css_lexer.TComma {
+					return
+				}
+				tokens = tokens[1:]
+			}
+		}
+
+		// Add the color stop
+		gradient.colorStops = append(gradient.colorStops, colorStop{
+			color:     color,
+			positions: positions,
+			midpoint:  midpoint,
+		})
+	}
+
+	success = true
+	return
+}
+
+func (p *parser) generateGradient(token css_ast.Token, gradient parsedGradient) css_ast.Token {
+	var children []css_ast.Token
+	commaToken := p.commaToken(token.Loc)
+
+	children = append(children, gradient.leadingTokens...)
+	for _, stop := range gradient.colorStops {
+		if len(children) > 0 {
+			children = append(children, commaToken)
+		}
+		if len(stop.positions) == 0 && stop.midpoint.Kind == css_lexer.T(0) {
+			stop.color.Whitespace &= ^css_ast.WhitespaceAfter
+		}
+		children = append(children, stop.color)
+		children = append(children, stop.positions...)
+		if stop.midpoint.Kind != css_lexer.T(0) {
+			children = append(children, commaToken, stop.midpoint)
+		}
+	}
+
+	token.Children = &children
+	return token
+}
+
+func (p *parser) lowerAndMinifyGradient(token css_ast.Token, wouldClipColor *bool) css_ast.Token {
+	gradient, ok := parseGradient(token)
+	if !ok {
+		return token
+	}
+
+	lowerMidpoints := p.options.unsupportedCSSFeatures.Has(compat.GradientMidpoints)
+	lowerColorSpaces := p.options.unsupportedCSSFeatures.Has(compat.ColorFunctions)
+	lowerInterpolation := p.options.unsupportedCSSFeatures.Has(compat.GradientInterpolation)
+
+	// Assume that if the browser doesn't support color spaces in gradients, then
+	// it doesn't correctly interpolate non-sRGB colors even when a color space
+	// is not specified. This is the case for Firefox 120, for example, which has
+	// support for the "color()" syntax but not for color spaces in gradients.
+	// There is no entry in our feature support matrix for this edge case so we
+	// make this assumption instead.
+	//
+	// Note that this edge case means we have to _replace_ the original gradient
+	// with the expanded one instead of inserting a fallback before it. Otherwise
+	// Firefox 120 would use the original gradient instead of the fallback because
+	// it supports the syntax, but just renders it incorrectly.
+	if lowerInterpolation {
+		lowerColorSpaces = true
+	}
+
+	// Potentially expand the gradient to handle unsupported features
+	didExpand := false
+	if lowerMidpoints || lowerColorSpaces || lowerInterpolation {
+		if colorStops, ok := tryToParseColorStops(gradient); ok {
+			hasColorSpace := false
+			hasMidpoint := false
+			for _, stop := range colorStops {
+				if stop.hasColorSpace {
+					hasColorSpace = true
+				}
+				if stop.midpoint != nil {
+					hasMidpoint = true
+				}
+			}
+			remaining, colorSpace, hueMethod, hasInterpolation := removeColorInterpolation(gradient.leadingTokens)
+			if (hasInterpolation && lowerInterpolation) || (hasColorSpace && lowerColorSpaces) || (hasMidpoint && lowerMidpoints) {
+				if hasInterpolation {
+					tryToExpandGradient(token.Loc, &gradient, colorStops, remaining, colorSpace, hueMethod)
+				} else {
+					if hasColorSpace {
+						colorSpace = colorSpace_oklab
+					} else {
+						colorSpace = colorSpace_srgb
+					}
+					tryToExpandGradient(token.Loc, &gradient, colorStops, gradient.leadingTokens, colorSpace, shorterHue)
+				}
+				didExpand = true
+			}
+		}
+	}
+
+	// Lower all colors in the gradient stop
+	for i, stop := range gradient.colorStops {
+		gradient.colorStops[i].color = p.lowerAndMinifyColor(stop.color, wouldClipColor)
+	}
+
+	if p.options.unsupportedCSSFeatures.Has(compat.GradientDoublePosition) {
+		// Replace double positions with duplicated single positions
+		for _, stop := range gradient.colorStops {
+			if len(stop.positions) > 1 {
+				gradient.colorStops = switchToSinglePositions(gradient.colorStops)
+				break
+			}
+		}
+	} else if p.options.minifySyntax {
+		// Replace duplicated single positions with double positions
+		for i, stop := range gradient.colorStops {
+			if i > 0 && len(stop.positions) == 1 {
+				if prev := gradient.colorStops[i-1]; len(prev.positions) == 1 && prev.midpoint.Kind == css_lexer.T(0) &&
+					css_ast.TokensEqual([]css_ast.Token{prev.color}, []css_ast.Token{stop.color}, nil) {
+					gradient.colorStops = switchToDoublePositions(gradient.colorStops)
+					break
+				}
+			}
+		}
+	}
+
+	if p.options.minifySyntax || didExpand {
+		gradient.colorStops = removeImpliedPositions(gradient.kind, gradient.colorStops)
+	}
+
+	return p.generateGradient(token, gradient)
+}
+
+func removeImpliedPositions(kind gradientKind, colorStops []colorStop) []colorStop {
+	if len(colorStops) == 0 {
+		return colorStops
+	}
+
+	positions := make([]valueWithUnit, len(colorStops))
+	for i, stop := range colorStops {
+		if len(stop.positions) == 1 {
+			if pos, ok := tryToParseValue(stop.positions[0], kind); ok {
+				positions[i] = pos
+				continue
+			}
+		}
+		positions[i].value = helpers.NewF64(math.NaN())
+	}
+
+	start := 0
+	for start < len(colorStops) {
+		if startPos := positions[start]; !startPos.value.IsNaN() {
+			end := start + 1
+		run:
+			for colorStops[end-1].midpoint.Kind == css_lexer.T(0) && end < len(colorStops) {
+				endPos := positions[end]
+				if endPos.value.IsNaN() || endPos.unit != startPos.unit {
+					break
+				}
+
+				// Check that all values in this run are implied. Interpolation is done
+				// using the start and end positions instead of the first and second
+				// positions because it's more accurate.
+				for i := start + 1; i < end; i++ {
+					t := helpers.NewF64(float64(i - start)).DivConst(float64(end - start))
+					impliedValue := helpers.Lerp(startPos.value, endPos.value, t)
+					if positions[i].value.Sub(impliedValue).Abs().Value() > 0.01 {
+						break run
+					}
+				}
+				end++
+			}
+
+			// Clear out all implied values
+			if end-start > 1 {
+				for i := start + 1; i+1 < end; i++ {
+					colorStops[i].positions = nil
+				}
+				start = end - 1
+				continue
+			}
+		}
+		start++
+	}
+
+	if first := colorStops[0].positions; len(first) == 1 &&
+		((first[0].Kind == css_lexer.TPercentage && first[0].PercentageValue() == "0") ||
+			(first[0].Kind == css_lexer.TDimension && first[0].DimensionValue() == "0")) {
+		colorStops[0].positions = nil
+	}
+
+	if last := colorStops[len(colorStops)-1].positions; len(last) == 1 &&
+		last[0].Kind == css_lexer.TPercentage && last[0].PercentageValue() == "100" {
+		colorStops[len(colorStops)-1].positions = nil
+	}
+
+	return colorStops
+}
+
+func switchToSinglePositions(double []colorStop) (single []colorStop) {
+	for _, stop := range double {
+		for i := range stop.positions {
+			stop.positions[i].Whitespace = css_ast.WhitespaceBefore
+		}
+		for len(stop.positions) > 1 {
+			clone := stop
+			clone.positions = stop.positions[:1]
+			clone.midpoint = css_ast.Token{}
+			single = append(single, clone)
+			stop.positions = stop.positions[1:]
+		}
+		single = append(single, stop)
+	}
+	return
+}
+
+func switchToDoublePositions(single []colorStop) (double []colorStop) {
+	for i := 0; i < len(single); i++ {
+		stop := single[i]
+		if i+1 < len(single) && len(stop.positions) == 1 && stop.midpoint.Kind == css_lexer.T(0) {
+			if next := single[i+1]; len(next.positions) == 1 &&
+				css_ast.TokensEqual([]css_ast.Token{stop.color}, []css_ast.Token{next.color}, nil) {
+				double = append(double, colorStop{
+					color:     stop.color,
+					positions: []css_ast.Token{stop.positions[0], next.positions[0]},
+					midpoint:  next.midpoint,
+				})
+				i++
+				continue
+			}
+		}
+		double = append(double, stop)
+	}
+	return
+}
+
+func removeColorInterpolation(tokens []css_ast.Token) ([]css_ast.Token, colorSpace, hueMethod, bool) {
+	for i := 0; i+1 < len(tokens); i++ {
+		if in := tokens[i]; in.Kind == css_lexer.TIdent && strings.EqualFold(in.Text, "in") {
+			if space := tokens[i+1]; space.Kind == css_lexer.TIdent {
+				var colorSpace colorSpace
+				hueMethod := shorterHue
+				start := i
+				end := i + 2
+
+				// Parse the color space
+				switch strings.ToLower(space.Text) {
+				case "a98-rgb":
+					colorSpace = colorSpace_a98_rgb
+				case "display-p3":
+					colorSpace = colorSpace_display_p3
+				case "hsl":
+					colorSpace = colorSpace_hsl
+				case "hwb":
+					colorSpace = colorSpace_hwb
+				case "lab":
+					colorSpace = colorSpace_lab
+				case "lch":
+					colorSpace = colorSpace_lch
+				case "oklab":
+					colorSpace = colorSpace_oklab
+				case "oklch":
+					colorSpace = colorSpace_oklch
+				case "prophoto-rgb":
+					colorSpace = colorSpace_prophoto_rgb
+				case "rec2020":
+					colorSpace = colorSpace_rec2020
+				case "srgb":
+					colorSpace = colorSpace_srgb
+				case "srgb-linear":
+					colorSpace = colorSpace_srgb_linear
+				case "xyz":
+					colorSpace = colorSpace_xyz
+				case "xyz-d50":
+					colorSpace = colorSpace_xyz_d50
+				case "xyz-d65":
+					colorSpace = colorSpace_xyz_d65
+				default:
+					return nil, 0, 0, false
+				}
+
+				// Parse the optional hue mode for polar color spaces
+				if colorSpace.isPolar() && i+3 < len(tokens) {
+					if hue := tokens[i+3]; hue.Kind == css_lexer.TIdent && strings.EqualFold(hue.Text, "hue") {
+						if method := tokens[i+2]; method.Kind == css_lexer.TIdent {
+							switch strings.ToLower(method.Text) {
+							case "shorter":
+								hueMethod = shorterHue
+							case "longer":
+								hueMethod = longerHue
+							case "increasing":
+								hueMethod = increasingHue
+							case "decreasing":
+								hueMethod = decreasingHue
+							default:
+								return nil, 0, 0, false
+							}
+							end = i + 4
+						}
+					}
+				}
+
+				// Remove all parsed tokens
+				remaining := append(append([]css_ast.Token{}, tokens[:start]...), tokens[end:]...)
+				if n := len(remaining); n > 0 {
+					remaining[0].Whitespace &= ^css_ast.WhitespaceBefore
+					remaining[n-1].Whitespace &= ^css_ast.WhitespaceAfter
+				}
+				return remaining, colorSpace, hueMethod, true
+			}
+		}
+	}
+
+	return nil, 0, 0, false
+}
+
+type valueWithUnit struct {
+	unit  string
+	value F64
+}
+
+type parsedColorStop struct {
+	// Position information (may be a sum of two different units)
+	positionTerms []valueWithUnit
+
+	// Color midpoint (a.k.a. transition hint) information
+	midpoint *valueWithUnit
+
+	// Non-premultiplied color information in XYZ space
+	x, y, z, alpha F64
+
+	// Non-premultiplied color information in sRGB space
+	r, g, b F64
+
+	// Premultiplied color information in the interpolation color space
+	v0, v1, v2 F64
+
+	// True if the original color has a color space
+	hasColorSpace bool
+}
+
+func tryToParseColorStops(gradient parsedGradient) ([]parsedColorStop, bool) {
+	var colorStops []parsedColorStop
+
+	for _, stop := range gradient.colorStops {
+		color, ok := parseColor(stop.color)
+		if !ok {
+			return nil, false
+		}
+		var r, g, b F64
+		if !color.hasColorSpace {
+			r = helpers.NewF64(float64(hexR(color.hex))).DivConst(255)
+			g = helpers.NewF64(float64(hexG(color.hex))).DivConst(255)
+			b = helpers.NewF64(float64(hexB(color.hex))).DivConst(255)
+			color.x, color.y, color.z = lin_srgb_to_xyz(lin_srgb(r, g, b))
+		} else {
+			r, g, b = gam_srgb(xyz_to_lin_srgb(color.x, color.y, color.z))
+		}
+		parsedStop := parsedColorStop{
+			x:             color.x,
+			y:             color.y,
+			z:             color.z,
+			r:             r,
+			g:             g,
+			b:             b,
+			alpha:         helpers.NewF64(float64(hexA(color.hex))).DivConst(255),
+			hasColorSpace: color.hasColorSpace,
+		}
+
+		for i, position := range stop.positions {
+			if position, ok := tryToParseValue(position, gradient.kind); ok {
+				parsedStop.positionTerms = []valueWithUnit{position}
+			} else {
+				return nil, false
+			}
+
+			// Expand double positions
+			if i+1 < len(stop.positions) {
+				colorStops = append(colorStops, parsedStop)
+			}
+		}
+
+		if stop.midpoint.Kind != css_lexer.T(0) {
+			if midpoint, ok := tryToParseValue(stop.midpoint, gradient.kind); ok {
+				parsedStop.midpoint = &midpoint
+			} else {
+				return nil, false
+			}
+		}
+
+		colorStops = append(colorStops, parsedStop)
+	}
+
+	// Automatically fill in missing positions
+	if len(colorStops) > 0 {
+		type stopInfo struct {
+			fromPos   valueWithUnit
+			toPos     valueWithUnit
+			fromCount int32
+			toCount   int32
+		}
+
+		// Fill in missing positions for the endpoints first
+		if first := &colorStops[0]; len(first.positionTerms) == 0 {
+			first.positionTerms = []valueWithUnit{{value: helpers.NewF64(0), unit: "%"}}
+		}
+		if last := &colorStops[len(colorStops)-1]; len(last.positionTerms) == 0 {
+			last.positionTerms = []valueWithUnit{{value: helpers.NewF64(100), unit: "%"}}
+		}
+
+		// Set all positions to be greater than the position before them
+		for i, stop := range colorStops {
+			var prevPos valueWithUnit
+			for j := i - 1; j >= 0; j-- {
+				prev := colorStops[j]
+				if prev.midpoint != nil {
+					prevPos = *prev.midpoint
+					break
+				}
+				if len(prev.positionTerms) == 1 {
+					prevPos = prev.positionTerms[0]
+					break
+				}
+			}
+			if len(stop.positionTerms) == 1 {
+				if prevPos.unit == stop.positionTerms[0].unit {
+					stop.positionTerms[0].value = helpers.Max2(prevPos.value, stop.positionTerms[0].value)
+				}
+				prevPos = stop.positionTerms[0]
+			}
+			if stop.midpoint != nil && prevPos.unit == stop.midpoint.unit {
+				stop.midpoint.value = helpers.Max2(prevPos.value, stop.midpoint.value)
+			}
+		}
+
+		// Scan over all other stops with missing positions
+		infos := make([]stopInfo, len(colorStops))
+		for i, stop := range colorStops {
+			if len(stop.positionTerms) == 1 {
+				continue
+			}
+			info := &infos[i]
+
+			// Scan backward
+			for from := i - 1; from >= 0; from-- {
+				fromStop := colorStops[from]
+				info.fromCount++
+				if fromStop.midpoint != nil {
+					info.fromPos = *fromStop.midpoint
+					break
+				}
+				if len(fromStop.positionTerms) == 1 {
+					info.fromPos = fromStop.positionTerms[0]
+					break
+				}
+			}
+
+			// Scan forward
+			for to := i; to < len(colorStops); to++ {
+				info.toCount++
+				if toStop := colorStops[to]; toStop.midpoint != nil {
+					info.toPos = *toStop.midpoint
+					break
+				}
+				if to+1 < len(colorStops) {
+					if toStop := colorStops[to+1]; len(toStop.positionTerms) == 1 {
+						info.toPos = toStop.positionTerms[0]
+						break
+					}
+				}
+			}
+		}
+
+		// Then fill in all other missing positions
+		for i, stop := range colorStops {
+			if len(stop.positionTerms) != 1 {
+				info := infos[i]
+				t := helpers.NewF64(float64(info.fromCount)).DivConst(float64(info.fromCount + info.toCount))
+				if info.fromPos.unit == info.toPos.unit {
+					colorStops[i].positionTerms = []valueWithUnit{{
+						value: helpers.Lerp(info.fromPos.value, info.toPos.value, t),
+						unit:  info.fromPos.unit,
+					}}
+				} else {
+					colorStops[i].positionTerms = []valueWithUnit{{
+						value: t.Neg().AddConst(1).Mul(info.fromPos.value),
+						unit:  info.fromPos.unit,
+					}, {
+						value: t.Mul(info.toPos.value),
+						unit:  info.toPos.unit,
+					}}
+				}
+			}
+		}
+
+		// Midpoints are only supported if they use the same units as their neighbors
+		for i, stop := range colorStops {
+			if stop.midpoint != nil {
+				next := colorStops[i+1]
+				if len(stop.positionTerms) != 1 || stop.midpoint.unit != stop.positionTerms[0].unit ||
+					len(next.positionTerms) != 1 || stop.midpoint.unit != next.positionTerms[0].unit {
+					return nil, false
+				}
+			}
+		}
+	}
+
+	return colorStops, true
+}
+
+func tryToParseValue(token css_ast.Token, kind gradientKind) (result valueWithUnit, success bool) {
+	if kind == conicGradient {
+		// <angle-percentage>
+		switch token.Kind {
+		case css_lexer.TDimension:
+			degrees, ok := degreesForAngle(token)
+			if !ok {
+				return
+			}
+			result.value = helpers.NewF64(degrees).MulConst(100.0 / 360)
+			result.unit = "%"
+
+		case css_lexer.TPercentage:
+			percent, err := strconv.ParseFloat(token.PercentageValue(), 64)
+			if err != nil {
+				return
+			}
+			result.value = helpers.NewF64(percent)
+			result.unit = "%"
+
+		default:
+			return
+		}
+	} else {
+		// <length-percentage>
+		switch token.Kind {
+		case css_lexer.TNumber:
+			zero, err := strconv.ParseFloat(token.Text, 64)
+			if err != nil || zero != 0 {
+				return
+			}
+			result.value = helpers.NewF64(0)
+			result.unit = "%"
+
+		case css_lexer.TDimension:
+			dimensionValue, err := strconv.ParseFloat(token.DimensionValue(), 64)
+			if err != nil {
+				return
+			}
+			result.value = helpers.NewF64(dimensionValue)
+			result.unit = token.DimensionUnit()
+
+		case css_lexer.TPercentage:
+			percentageValue, err := strconv.ParseFloat(token.PercentageValue(), 64)
+			if err != nil {
+				return
+			}
+			result.value = helpers.NewF64(percentageValue)
+			result.unit = "%"
+
+		default:
+			return
+		}
+	}
+
+	success = true
+	return
+}
+
+func tryToExpandGradient(
+	loc logger.Loc,
+	gradient *parsedGradient,
+	colorStops []parsedColorStop,
+	remaining []css_ast.Token,
+	colorSpace colorSpace,
+	hueMethod hueMethod,
+) bool {
+	// Convert color stops into the interpolation color space
+	for i := range colorStops {
+		stop := &colorStops[i]
+		v0, v1, v2 := xyz_to_colorSpace(stop.x, stop.y, stop.z, colorSpace)
+		stop.v0, stop.v1, stop.v2 = premultiply(v0, v1, v2, stop.alpha, colorSpace)
+	}
+
+	// Duplicate the endpoints if they should wrap around to themselves
+	if hueMethod == longerHue && colorSpace.isPolar() && len(colorStops) > 0 {
+		if first := colorStops[0]; len(first.positionTerms) == 1 {
+			if first.positionTerms[0].value.Value() < 0 {
+				colorStops[0].positionTerms[0].value = helpers.NewF64(0)
+			} else if first.positionTerms[0].value.Value() > 0 {
+				first.midpoint = nil
+				first.positionTerms = []valueWithUnit{{value: helpers.NewF64(0), unit: first.positionTerms[0].unit}}
+				colorStops = append([]parsedColorStop{first}, colorStops...)
+			}
+		}
+		if last := colorStops[len(colorStops)-1]; len(last.positionTerms) == 1 {
+			if last.positionTerms[0].unit != "%" || last.positionTerms[0].value.Value() < 100 {
+				last.positionTerms = []valueWithUnit{{value: helpers.NewF64(100), unit: "%"}}
+				colorStops = append(colorStops, last)
+			}
+		}
+	}
+
+	var newColorStops []colorStop
+	var generateColorStops func(
+		int, parsedColorStop, parsedColorStop,
+		F64, F64, F64, F64, F64, F64, F64, F64,
+		F64, F64, F64, F64, F64, F64, F64, F64,
+	)
+
+	generateColorStops = func(
+		depth int,
+		from parsedColorStop, to parsedColorStop,
+		prevX, prevY, prevZ, prevR, prevG, prevB, prevA, prevT F64,
+		nextX, nextY, nextZ, nextR, nextG, nextB, nextA, nextT F64,
+	) {
+		if depth > 4 {
+			return
+		}
+
+		t := prevT.Add(nextT).DivConst(2)
+		positionT := t
+
+		// Handle midpoints (which we have already checked uses the same units)
+		if from.midpoint != nil {
+			fromPos := from.positionTerms[0].value
+			toPos := to.positionTerms[0].value
+			stopPos := helpers.Lerp(fromPos, toPos, t)
+			H := from.midpoint.value.Sub(fromPos).Div(toPos.Sub(fromPos))
+			P := stopPos.Sub(fromPos).Div(toPos.Sub(fromPos))
+			if H.Value() <= 0 {
+				positionT = helpers.NewF64(1)
+			} else if H.Value() >= 1 {
+				positionT = helpers.NewF64(0)
+			} else {
+				positionT = P.Pow(helpers.NewF64(-1).Div(H.Log2()))
+			}
+		}
+
+		v0, v1, v2 := interpolateColors(from.v0, from.v1, from.v2, to.v0, to.v1, to.v2, colorSpace, hueMethod, positionT)
+		a := helpers.Lerp(from.alpha, to.alpha, positionT)
+		v0, v1, v2 = unpremultiply(v0, v1, v2, a, colorSpace)
+		x, y, z := colorSpace_to_xyz(v0, v1, v2, colorSpace)
+
+		// Stop when the color is similar enough to the sRGB midpoint
+		const epsilon = 4.0 / 255
+		r, g, b := gam_srgb(xyz_to_lin_srgb(x, y, z))
+		dr := r.Mul(a).Sub(prevR.Mul(prevA).Add(nextR.Mul(nextA)).DivConst(2))
+		dg := g.Mul(a).Sub(prevG.Mul(prevA).Add(nextG.Mul(nextA)).DivConst(2))
+		db := b.Mul(a).Sub(prevB.Mul(prevA).Add(nextB.Mul(nextA)).DivConst(2))
+		if d := dr.Squared().Add(dg.Squared()).Add(db.Squared()); d.Value() < epsilon*epsilon {
+			return
+		}
+
+		// Recursive split before this stop
+		generateColorStops(depth+1, from, to,
+			prevX, prevY, prevZ, prevR, prevG, prevB, prevA, prevT,
+			x, y, z, r, g, b, a, t)
+
+		// Generate this stop
+		color := makeColorToken(loc, x, y, z, a)
+		positionTerms := interpolatePositions(from.positionTerms, to.positionTerms, t)
+		position := makePositionToken(loc, positionTerms)
+		position.Whitespace = css_ast.WhitespaceBefore
+		newColorStops = append(newColorStops, colorStop{
+			color:     color,
+			positions: []css_ast.Token{position},
+		})
+
+		// Recursive split after this stop
+		generateColorStops(depth+1, from, to,
+			x, y, z, r, g, b, a, t,
+			nextX, nextY, nextZ, nextR, nextG, nextB, nextA, nextT)
+	}
+
+	for i, stop := range colorStops {
+		color := makeColorToken(loc, stop.x, stop.y, stop.z, stop.alpha)
+		position := makePositionToken(loc, stop.positionTerms)
+		position.Whitespace = css_ast.WhitespaceBefore
+		newColorStops = append(newColorStops, colorStop{
+			color:     color,
+			positions: []css_ast.Token{position},
+		})
+
+		// Generate new color stops in between as needed
+		if i+1 < len(colorStops) {
+			next := colorStops[i+1]
+			generateColorStops(0, stop, next,
+				stop.x, stop.y, stop.z, stop.r, stop.g, stop.b, stop.alpha, helpers.NewF64(0),
+				next.x, next.y, next.z, next.r, next.g, next.b, next.alpha, helpers.NewF64(1))
+		}
+	}
+
+	gradient.leadingTokens = remaining
+	gradient.colorStops = newColorStops
+	return true
+}
+
+func formatFloat(value F64, decimals int) string {
+	return strings.TrimSuffix(strings.TrimRight(strconv.FormatFloat(value.Value(), 'f', decimals, 64), "0"), ".")
+}
+
+func makeDimensionOrPercentToken(loc logger.Loc, value F64, unit string) (token css_ast.Token) {
+	token.Loc = loc
+	token.Text = formatFloat(value, 2)
+	if unit == "%" {
+		token.Kind = css_lexer.TPercentage
+	} else {
+		token.Kind = css_lexer.TDimension
+		token.UnitOffset = uint16(len(token.Text))
+	}
+	token.Text += unit
+	return
+}
+
+func makePositionToken(loc logger.Loc, positionTerms []valueWithUnit) css_ast.Token {
+	if len(positionTerms) == 1 {
+		return makeDimensionOrPercentToken(loc, positionTerms[0].value, positionTerms[0].unit)
+	}
+
+	children := make([]css_ast.Token, 0, 1+2*len(positionTerms))
+	for i, term := range positionTerms {
+		if i > 0 {
+			children = append(children, css_ast.Token{
+				Loc:        loc,
+				Kind:       css_lexer.TDelimPlus,
+				Text:       "+",
+				Whitespace: css_ast.WhitespaceBefore | css_ast.WhitespaceAfter,
+			})
+		}
+		children = append(children, makeDimensionOrPercentToken(loc, term.value, term.unit))
+	}
+
+	return css_ast.Token{
+		Loc:      loc,
+		Kind:     css_lexer.TFunction,
+		Text:     "calc",
+		Children: &children,
+	}
+}
+
+func makeColorToken(loc logger.Loc, x F64, y F64, z F64, a F64) (color css_ast.Token) {
+	color.Loc = loc
+	alpha := uint32(a.MulConst(255).Round().Value())
+	if hex, ok := tryToConvertToHexWithoutClipping(x, y, z, alpha); ok {
+		color.Kind = css_lexer.THash
+		if alpha == 255 {
+			color.Text = fmt.Sprintf("%06x", hex>>8)
+		} else {
+			color.Text = fmt.Sprintf("%08x", hex)
+		}
+	} else {
+		children := []css_ast.Token{
+			{
+				Loc:        loc,
+				Kind:       css_lexer.TIdent,
+				Text:       "xyz",
+				Whitespace: css_ast.WhitespaceAfter,
+			},
+			{
+				Loc:        loc,
+				Kind:       css_lexer.TNumber,
+				Text:       formatFloat(x, 3),
+				Whitespace: css_ast.WhitespaceBefore | css_ast.WhitespaceAfter,
+			},
+			{
+				Loc:        loc,
+				Kind:       css_lexer.TNumber,
+				Text:       formatFloat(y, 3),
+				Whitespace: css_ast.WhitespaceBefore | css_ast.WhitespaceAfter,
+			},
+			{
+				Loc:        loc,
+				Kind:       css_lexer.TNumber,
+				Text:       formatFloat(z, 3),
+				Whitespace: css_ast.WhitespaceBefore,
+			},
+		}
+		if a.Value() < 1 {
+			children = append(children,
+				css_ast.Token{
+					Loc:        loc,
+					Kind:       css_lexer.TDelimSlash,
+					Text:       "/",
+					Whitespace: css_ast.WhitespaceBefore | css_ast.WhitespaceAfter,
+				},
+				css_ast.Token{
+					Loc:        loc,
+					Kind:       css_lexer.TNumber,
+					Text:       formatFloat(a, 3),
+					Whitespace: css_ast.WhitespaceBefore,
+				},
+			)
+		}
+		color.Kind = css_lexer.TFunction
+		color.Text = "color"
+		color.Children = &children
+	}
+	return
+}
+
+func interpolateHues(a, b, t F64, hueMethod hueMethod) F64 {
+	a = a.DivConst(360)
+	b = b.DivConst(360)
+	a = a.Sub(a.Floor())
+	b = b.Sub(b.Floor())
+
+	switch hueMethod {
+	case shorterHue:
+		delta := b.Sub(a)
+		if delta.Value() > 0.5 {
+			a = a.AddConst(1)
+		}
+		if delta.Value() < -0.5 {
+			b = b.AddConst(1)
+		}
+
+	case longerHue:
+		delta := b.Sub(a)
+		if delta.Value() > 0 && delta.Value() < 0.5 {
+			a = a.AddConst(1)
+		}
+		if delta.Value() > -0.5 && delta.Value() <= 0 {
+			b = b.AddConst(1)
+		}
+
+	case increasingHue:
+		if b.Value() < a.Value() {
+			b = b.AddConst(1)
+		}
+
+	case decreasingHue:
+		if a.Value() < b.Value() {
+			a = a.AddConst(1)
+		}
+	}
+
+	return helpers.Lerp(a, b, t).MulConst(360)
+}
+
+func interpolateColors(
+	a0, a1, a2 F64, b0, b1, b2 F64,
+	colorSpace colorSpace, hueMethod hueMethod, t F64,
+) (v0 F64, v1 F64, v2 F64) {
+	v1 = helpers.Lerp(a1, b1, t)
+
+	switch colorSpace {
+	case colorSpace_hsl, colorSpace_hwb:
+		v2 = helpers.Lerp(a2, b2, t)
+		v0 = interpolateHues(a0, b0, t, hueMethod)
+
+	case colorSpace_lch, colorSpace_oklch:
+		v0 = helpers.Lerp(a0, b0, t)
+		v2 = interpolateHues(a2, b2, t, hueMethod)
+
+	default:
+		v0 = helpers.Lerp(a0, b0, t)
+		v2 = helpers.Lerp(a2, b2, t)
+	}
+
+	return v0, v1, v2
+}
+
+func interpolatePositions(a []valueWithUnit, b []valueWithUnit, t F64) (result []valueWithUnit) {
+	findUnit := func(unit string) int {
+		for i, x := range result {
+			if x.unit == unit {
+				return i
+			}
+		}
+		result = append(result, valueWithUnit{unit: unit})
+		return len(result) - 1
+	}
+
+	// "result += a * (1 - t)"
+	for _, term := range a {
+		ptr := &result[findUnit(term.unit)]
+		ptr.value = t.Neg().AddConst(1).Mul(term.value).Add(ptr.value)
+	}
+
+	// "result += b * t"
+	for _, term := range b {
+		ptr := &result[findUnit(term.unit)]
+		ptr.value = t.Mul(term.value).Add(ptr.value)
+	}
+
+	// Remove an extra zero value for neatness. We don't remove all
+	// of them because it may be important to retain a single zero.
+	if len(result) > 1 {
+		for i, term := range result {
+			if term.value.Value() == 0 {
+				copy(result[i:], result[i+1:])
+				result = result[:len(result)-1]
+				break
+			}
+		}
+	}
+
+	return
+}
+
+func premultiply(v0, v1, v2, alpha F64, colorSpace colorSpace) (F64, F64, F64) {
+	if alpha.Value() < 1 {
+		switch colorSpace {
+		case colorSpace_hsl, colorSpace_hwb:
+			v2 = v2.Mul(alpha)
+		case colorSpace_lch, colorSpace_oklch:
+			v0 = v0.Mul(alpha)
+		default:
+			v0 = v0.Mul(alpha)
+			v2 = v2.Mul(alpha)
+		}
+		v1 = v1.Mul(alpha)
+	}
+	return v0, v1, v2
+}
+
+func unpremultiply(v0, v1, v2, alpha F64, colorSpace colorSpace) (F64, F64, F64) {
+	if alpha.Value() > 0 && alpha.Value() < 1 {
+		switch colorSpace {
+		case colorSpace_hsl, colorSpace_hwb:
+			v2 = v2.Div(alpha)
+		case colorSpace_lch, colorSpace_oklch:
+			v0 = v0.Div(alpha)
+		default:
+			v0 = v0.Div(alpha)
+			v2 = v2.Div(alpha)
+		}
+		v1 = v1.Div(alpha)
+	}
+	return v0, v1, v2
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_list_style.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_list_style.go
new file mode 100644
index 0000000..113769d
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_list_style.go
@@ -0,0 +1,179 @@
+package css_parser
+
+import (
+	"strings"
+
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+)
+
+// list-style-image: <image> | none
+// <image>: <url> | <gradient>
+// <url>: <url()> | <src()>
+// <gradient>: <linear-gradient()> | <repeating-linear-gradient()> | <radial-gradient()> | <repeating-radial-gradient()>
+//
+// list-style-type: <counter-style> | <string> | none (where the string is a literal bullet marker)
+// <counter-style>: <counter-style-name> | <symbols()>
+// <counter-style-name>: not: decimal | disc | square | circle | disclosure-open | disclosure-closed | <wide keyword>
+// when parsing a <custom-ident> with conflicts, only parse one if no other thing can claim it
+
+func (p *parser) processListStyleShorthand(tokens []css_ast.Token) {
+	if len(tokens) < 1 || len(tokens) > 3 {
+		return
+	}
+
+	foundImage := false
+	foundPosition := false
+	typeIndex := -1
+	noneCount := 0
+
+	for i, t := range tokens {
+		switch t.Kind {
+		case css_lexer.TString:
+			// "list-style-type" is definitely not a <custom-ident>
+			return
+
+		case css_lexer.TURL:
+			if !foundImage {
+				foundImage = true
+				continue
+			}
+
+		case css_lexer.TFunction:
+			if !foundImage {
+				switch strings.ToLower(t.Text) {
+				case "src", "linear-gradient", "repeating-linear-gradient", "radial-gradient", "radial-linear-gradient":
+					foundImage = true
+					continue
+				}
+			}
+
+		case css_lexer.TIdent:
+			lower := strings.ToLower(t.Text)
+
+			// Note: If "none" is present, it's ambiguous whether it applies to
+			// "list-style-image" or "list-style-type". To resolve ambiguity it's
+			// applied at the end to whichever property isn't otherwise set.
+			if lower == "none" {
+				noneCount++
+				continue
+			}
+
+			if !foundPosition && (lower == "inside" || lower == "outside") {
+				foundPosition = true
+				continue
+			}
+
+			if typeIndex == -1 {
+				if cssWideAndReservedKeywords[lower] || predefinedCounterStyles[lower] {
+					// "list-style-type" is definitely not a <custom-ident>
+					return
+				}
+				typeIndex = i
+				continue
+			}
+		}
+
+		// Bail if we hit an unexpected token
+		return
+	}
+
+	if typeIndex != -1 {
+		// The first "none" applies to "list-style-image" if it's missing
+		if !foundImage && noneCount > 0 {
+			noneCount--
+		}
+
+		if noneCount > 0 {
+			// "list-style-type" is "none", not a <custom-ident>
+			return
+		}
+
+		if t := &tokens[typeIndex]; t.Kind == css_lexer.TIdent {
+			t.Kind = css_lexer.TSymbol
+			t.PayloadIndex = p.symbolForName(t.Loc, t.Text).Ref.InnerIndex
+		}
+	}
+}
+
+func (p *parser) processListStyleType(t *css_ast.Token) {
+	if t.Kind == css_lexer.TIdent {
+		if lower := strings.ToLower(t.Text); lower != "none" && !cssWideAndReservedKeywords[lower] && !predefinedCounterStyles[lower] {
+			t.Kind = css_lexer.TSymbol
+			t.PayloadIndex = p.symbolForName(t.Loc, t.Text).Ref.InnerIndex
+		}
+	}
+}
+
+// https://drafts.csswg.org/css-counter-styles-3/#predefined-counters
+var predefinedCounterStyles = map[string]bool{
+	// 6.1. Numeric:
+	"arabic-indic":         true,
+	"armenian":             true,
+	"bengali":              true,
+	"cambodian":            true,
+	"cjk-decimal":          true,
+	"decimal-leading-zero": true,
+	"decimal":              true,
+	"devanagari":           true,
+	"georgian":             true,
+	"gujarati":             true,
+	"gurmukhi":             true,
+	"hebrew":               true,
+	"kannada":              true,
+	"khmer":                true,
+	"lao":                  true,
+	"lower-armenian":       true,
+	"lower-roman":          true,
+	"malayalam":            true,
+	"mongolian":            true,
+	"myanmar":              true,
+	"oriya":                true,
+	"persian":              true,
+	"tamil":                true,
+	"telugu":               true,
+	"thai":                 true,
+	"tibetan":              true,
+	"upper-armenian":       true,
+	"upper-roman":          true,
+
+	// 6.2. Alphabetic:
+	"hiragana-iroha": true,
+	"hiragana":       true,
+	"katakana-iroha": true,
+	"katakana":       true,
+	"lower-alpha":    true,
+	"lower-greek":    true,
+	"lower-latin":    true,
+	"upper-alpha":    true,
+	"upper-latin":    true,
+
+	// 6.3. Symbolic:
+	"circle":            true,
+	"disc":              true,
+	"disclosure-closed": true,
+	"disclosure-open":   true,
+	"square":            true,
+
+	// 6.4. Fixed:
+	"cjk-earthly-branch": true,
+	"cjk-heavenly-stem":  true,
+
+	// 7.1.1. Japanese:
+	"japanese-formal":   true,
+	"japanese-informal": true,
+
+	// 7.1.2. Korean:
+	"korean-hangul-formal":  true,
+	"korean-hanja-formal":   true,
+	"korean-hanja-informal": true,
+
+	// 7.1.3. Chinese:
+	"simp-chinese-formal":   true,
+	"simp-chinese-informal": true,
+	"trad-chinese-formal":   true,
+	"trad-chinese-informal": true,
+
+	// 7.2. Ethiopic Numeric Counter Style:
+	"ethiopic-numeric": true,
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_transform.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_transform.go
new file mode 100644
index 0000000..e5cdbae
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_decls_transform.go
@@ -0,0 +1,347 @@
+package css_parser
+
+import (
+	"strings"
+
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+)
+
+func turnPercentIntoNumberIfShorter(t *css_ast.Token) {
+	if t.Kind == css_lexer.TPercentage {
+		if shifted, ok := shiftDot(t.PercentageValue(), -2); ok && len(shifted) < len(t.Text) {
+			t.Kind = css_lexer.TNumber
+			t.Text = shifted
+		}
+	}
+}
+
+// https://www.w3.org/TR/css-transforms-1/#two-d-transform-functions
+// https://drafts.csswg.org/css-transforms-2/#transform-functions
+func (p *parser) mangleTransforms(tokens []css_ast.Token) []css_ast.Token {
+	for i := range tokens {
+		if token := &tokens[i]; token.Kind == css_lexer.TFunction {
+			if args := *token.Children; css_ast.TokensAreCommaSeparated(args) {
+				n := len(args)
+
+				switch strings.ToLower(token.Text) {
+				////////////////////////////////////////////////////////////////////////////////
+				// 2D transforms
+
+				case "matrix":
+					// specifies a 2D transformation in the form of a transformation
+					// matrix of the six values a, b, c, d, e, f.
+					if n == 11 {
+						// | a c 0 e |
+						// | b d 0 f |
+						// | 0 0 1 0 |
+						// | 0 0 0 1 |
+						a, b, c, d, e, f := args[0], args[2], args[4], args[6], args[8], args[10]
+						if b.IsZero() && c.IsZero() && e.IsZero() && f.IsZero() {
+							// | a 0 0 0 |
+							// | 0 d 0 0 |
+							// | 0 0 1 0 |
+							// | 0 0 0 1 |
+							if a.EqualIgnoringWhitespace(d) {
+								// "matrix(a, 0, 0, a, 0, 0)" => "scale(a)"
+								token.Text = "scale"
+								*token.Children = args[:1]
+							} else if d.IsOne() {
+								// "matrix(a, 0, 0, 1, 0, 0)" => "scaleX(a)"
+								token.Text = "scaleX"
+								*token.Children = args[:1]
+							} else if a.IsOne() {
+								// "matrix(1, 0, 0, d, 0, 0)" => "scaleY(d)"
+								token.Text = "scaleY"
+								*token.Children = args[6:7]
+							} else {
+								// "matrix(a, 0, 0, d, 0, 0)" => "scale(a, d)"
+								token.Text = "scale"
+								*token.Children = append(args[:2], d)
+							}
+
+							// Note: A "matrix" cannot be directly converted into a "translate"
+							// because "translate" requires units while "matrix" requires no
+							// units. I'm not sure exactly what the semantics are so I'm not
+							// sure if you can just add "px" or not. Even if that did work,
+							// you still couldn't substitute values containing "var()" since
+							// units would still not be substituted in that case.
+						}
+					}
+
+				case "translate":
+					// specifies a 2D translation by the vector [tx, ty], where tx is the
+					// first translation-value parameter and ty is the optional second
+					// translation-value parameter. If <ty> is not provided, ty has zero
+					// as a value.
+					if n == 1 {
+						args[0].TurnLengthOrPercentageIntoNumberIfZero()
+					} else if n == 3 {
+						tx, ty := &args[0], &args[2]
+						tx.TurnLengthOrPercentageIntoNumberIfZero()
+						ty.TurnLengthOrPercentageIntoNumberIfZero()
+						if ty.IsZero() {
+							// "translate(tx, 0)" => "translate(tx)"
+							*token.Children = args[:1]
+						} else if tx.IsZero() {
+							// "translate(0, ty)" => "translateY(ty)"
+							token.Text = "translateY"
+							*token.Children = args[2:]
+						}
+					}
+
+				case "translatex":
+					// specifies a translation by the given amount in the X direction.
+					if n == 1 {
+						// "translateX(tx)" => "translate(tx)"
+						token.Text = "translate"
+						args[0].TurnLengthOrPercentageIntoNumberIfZero()
+					}
+
+				case "translatey":
+					// specifies a translation by the given amount in the Y direction.
+					if n == 1 {
+						args[0].TurnLengthOrPercentageIntoNumberIfZero()
+					}
+
+				case "scale":
+					// specifies a 2D scale operation by the [sx,sy] scaling vector
+					// described by the 2 parameters. If the second parameter is not
+					// provided, it takes a value equal to the first. For example,
+					// scale(1, 1) would leave an element unchanged, while scale(2, 2)
+					// would cause it to appear twice as long in both the X and Y axes,
+					// or four times its typical geometric size.
+					if n == 1 {
+						turnPercentIntoNumberIfShorter(&args[0])
+					} else if n == 3 {
+						sx, sy := &args[0], &args[2]
+						turnPercentIntoNumberIfShorter(sx)
+						turnPercentIntoNumberIfShorter(sy)
+						if sx.EqualIgnoringWhitespace(*sy) {
+							// "scale(s, s)" => "scale(s)"
+							*token.Children = args[:1]
+						} else if sy.IsOne() {
+							// "scale(s, 1)" => "scaleX(s)"
+							token.Text = "scaleX"
+							*token.Children = args[:1]
+						} else if sx.IsOne() {
+							// "scale(1, s)" => "scaleY(s)"
+							token.Text = "scaleY"
+							*token.Children = args[2:]
+						}
+					}
+
+				case "scalex":
+					// specifies a 2D scale operation using the [sx,1] scaling vector,
+					// where sx is given as the parameter.
+					if n == 1 {
+						turnPercentIntoNumberIfShorter(&args[0])
+					}
+
+				case "scaley":
+					// specifies a 2D scale operation using the [1,sy] scaling vector,
+					// where sy is given as the parameter.
+					if n == 1 {
+						turnPercentIntoNumberIfShorter(&args[0])
+					}
+
+				case "rotate":
+					// specifies a 2D rotation by the angle specified in the parameter
+					// about the origin of the element, as defined by the
+					// transform-origin property. For example, rotate(90deg) would
+					// cause elements to appear rotated one-quarter of a turn in the
+					// clockwise direction.
+					if n == 1 {
+						args[0].TurnLengthIntoNumberIfZero()
+					}
+
+				// Note: This is considered a 2D transform even though it's specified
+				// in terms of a 3D transform because it doesn't trigger Safari's 3D
+				// transform bugs.
+				case "rotatez":
+					// same as rotate3d(0, 0, 1, <angle>), which is a 3d transform
+					// equivalent to the 2d transform rotate(<angle>).
+					if n == 1 {
+						// "rotateZ(angle)" => "rotate(angle)"
+						token.Text = "rotate"
+						args[0].TurnLengthIntoNumberIfZero()
+					}
+
+				case "skew":
+					// specifies a 2D skew by [ax,ay] for X and Y. If the second
+					// parameter is not provided, it has a zero value.
+					if n == 1 {
+						args[0].TurnLengthIntoNumberIfZero()
+					} else if n == 3 {
+						ax, ay := &args[0], &args[2]
+						ax.TurnLengthIntoNumberIfZero()
+						ay.TurnLengthIntoNumberIfZero()
+						if ay.IsZero() {
+							// "skew(ax, 0)" => "skew(ax)"
+							*token.Children = args[:1]
+						}
+					}
+
+				case "skewx":
+					// specifies a 2D skew transformation along the X axis by the given
+					// angle.
+					if n == 1 {
+						// "skewX(ax)" => "skew(ax)"
+						token.Text = "skew"
+						args[0].TurnLengthIntoNumberIfZero()
+					}
+
+				case "skewy":
+					// specifies a 2D skew transformation along the Y axis by the given
+					// angle.
+					if n == 1 {
+						args[0].TurnLengthIntoNumberIfZero()
+					}
+
+					////////////////////////////////////////////////////////////////////////////////
+					// 3D transforms
+
+					// Note: Safari has a bug where 3D transforms render differently than
+					// other transforms. This means we should not minify a 3D transform
+					// into a 2D transform or it will cause a rendering difference in
+					// Safari.
+
+				case "matrix3d":
+					// specifies a 3D transformation as a 4x4 homogeneous matrix of 16
+					// values in column-major order.
+					if n == 31 {
+						// | m0 m4 m8  m12 |
+						// | m1 m5 m9  m13 |
+						// | m2 m6 m10 m14 |
+						// | m3 m7 m11 m15 |
+						mask := uint32(0)
+						for i := 0; i < 16; i++ {
+							if arg := args[i*2]; arg.IsZero() {
+								mask |= 1 << i
+							} else if arg.IsOne() {
+								mask |= (1 << 16) << i
+							}
+						}
+						const onlyScale = 0b1000_0000_0000_0000_0111_1011_1101_1110
+						if (mask & onlyScale) == onlyScale {
+							// | m0 0  0   0 |
+							// | 0  m5 0   0 |
+							// | 0  0  m10 0 |
+							// | 0  0  0   1 |
+							sx, sy := args[0], args[10]
+							if sx.IsOne() && sy.IsOne() {
+								token.Text = "scaleZ"
+								*token.Children = args[20:21]
+							} else {
+								token.Text = "scale3d"
+								*token.Children = append(append(args[0:2], args[10:12]...), args[20])
+							}
+						}
+
+						// Note: A "matrix3d" cannot be directly converted into a "translate3d"
+						// because "translate3d" requires units while "matrix3d" requires no
+						// units. I'm not sure exactly what the semantics are so I'm not
+						// sure if you can just add "px" or not. Even if that did work,
+						// you still couldn't substitute values containing "var()" since
+						// units would still not be substituted in that case.
+					}
+
+				case "translate3d":
+					// specifies a 3D translation by the vector [tx,ty,tz], with tx,
+					// ty and tz being the first, second and third translation-value
+					// parameters respectively.
+					if n == 5 {
+						tx, ty, tz := &args[0], &args[2], &args[4]
+						tx.TurnLengthOrPercentageIntoNumberIfZero()
+						ty.TurnLengthOrPercentageIntoNumberIfZero()
+						tz.TurnLengthIntoNumberIfZero()
+						if tx.IsZero() && ty.IsZero() {
+							// "translate3d(0, 0, tz)" => "translateZ(tz)"
+							token.Text = "translateZ"
+							*token.Children = args[4:]
+						}
+					}
+
+				case "translatez":
+					// specifies a 3D translation by the vector [0,0,tz] with the given
+					// amount in the Z direction.
+					if n == 1 {
+						args[0].TurnLengthIntoNumberIfZero()
+					}
+
+				case "scale3d":
+					// specifies a 3D scale operation by the [sx,sy,sz] scaling vector
+					// described by the 3 parameters.
+					if n == 5 {
+						sx, sy, sz := &args[0], &args[2], &args[4]
+						turnPercentIntoNumberIfShorter(sx)
+						turnPercentIntoNumberIfShorter(sy)
+						turnPercentIntoNumberIfShorter(sz)
+						if sx.IsOne() && sy.IsOne() {
+							// "scale3d(1, 1, sz)" => "scaleZ(sz)"
+							token.Text = "scaleZ"
+							*token.Children = args[4:]
+						}
+					}
+
+				case "scalez":
+					// specifies a 3D scale operation using the [1,1,sz] scaling vector,
+					// where sz is given as the parameter.
+					if n == 1 {
+						turnPercentIntoNumberIfShorter(&args[0])
+					}
+
+				case "rotate3d":
+					// specifies a 3D rotation by the angle specified in last parameter
+					// about the [x,y,z] direction vector described by the first three
+					// parameters. A direction vector that cannot be normalized, such as
+					// [0,0,0], will cause the rotation to not be applied.
+					if n == 7 {
+						x, y, z, angle := &args[0], &args[2], &args[4], &args[6]
+						angle.TurnLengthIntoNumberIfZero()
+						if x.IsOne() && y.IsZero() && z.IsZero() {
+							// "rotate3d(1, 0, 0, angle)" => "rotateX(angle)"
+							token.Text = "rotateX"
+							*token.Children = args[6:]
+						} else if x.IsZero() && y.IsOne() && z.IsZero() {
+							// "rotate3d(0, 1, 0, angle)" => "rotateY(angle)"
+							token.Text = "rotateY"
+							*token.Children = args[6:]
+						}
+					}
+
+				case "rotatex":
+					// same as rotate3d(1, 0, 0, <angle>).
+					if n == 1 {
+						args[0].TurnLengthIntoNumberIfZero()
+					}
+
+				case "rotatey":
+					// same as rotate3d(0, 1, 0, <angle>).
+					if n == 1 {
+						args[0].TurnLengthIntoNumberIfZero()
+					}
+
+				case "perspective":
+					// specifies a perspective projection matrix. This matrix scales
+					// points in X and Y based on their Z value, scaling points with
+					// positive Z values away from the origin, and those with negative Z
+					// values towards the origin. Points on the z=0 plane are unchanged.
+					// The parameter represents the distance of the z=0 plane from the
+					// viewer.
+					if n == 1 {
+						args[0].TurnLengthIntoNumberIfZero()
+					}
+				}
+
+				// Trim whitespace at the ends
+				if args := *token.Children; len(args) > 0 {
+					args[0].Whitespace &= ^css_ast.WhitespaceBefore
+					args[len(args)-1].Whitespace &= ^css_ast.WhitespaceAfter
+				}
+			}
+		}
+	}
+
+	return tokens
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_nesting.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_nesting.go
new file mode 100644
index 0000000..a95da13
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_nesting.go
@@ -0,0 +1,490 @@
+package css_parser
+
+import (
+	"fmt"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+func (p *parser) lowerNestingInRule(rule css_ast.Rule, results []css_ast.Rule) []css_ast.Rule {
+	switch r := rule.Data.(type) {
+	case *css_ast.RSelector:
+		scope := func(loc logger.Loc) css_ast.ComplexSelector {
+			return css_ast.ComplexSelector{
+				Selectors: []css_ast.CompoundSelector{{
+					SubclassSelectors: []css_ast.SubclassSelector{{
+						Range: logger.Range{Loc: loc},
+						Data:  &css_ast.SSPseudoClass{Name: "scope"},
+					}},
+				}},
+			}
+		}
+
+		parentSelectors := make([]css_ast.ComplexSelector, 0, len(r.Selectors))
+		for i, sel := range r.Selectors {
+			// Top-level "&" should be replaced with ":scope" to avoid recursion.
+			// From https://www.w3.org/TR/css-nesting-1/#nest-selector:
+			//
+			//   "When used in the selector of a nested style rule, the nesting
+			//   selector represents the elements matched by the parent rule. When
+			//   used in any other context, it represents the same elements as
+			//   :scope in that context (unless otherwise defined)."
+			//
+			substituted := make([]css_ast.CompoundSelector, 0, len(sel.Selectors))
+			for _, x := range sel.Selectors {
+				substituted = p.substituteAmpersandsInCompoundSelector(x, scope, substituted, keepLeadingCombinator)
+			}
+			r.Selectors[i] = css_ast.ComplexSelector{Selectors: substituted}
+
+			// Filter out pseudo elements because they are ignored by nested style
+			// rules. This is because pseudo-elements are not valid within :is():
+			// https://www.w3.org/TR/selectors-4/#matches-pseudo. This restriction
+			// may be relaxed in the future, but this restriction hash shipped so
+			// we're stuck with it: https://github.com/w3c/csswg-drafts/issues/7433.
+			//
+			// Note: This is only for the parent selector list that is used to
+			// substitute "&" within child rules. Do not filter out the pseudo
+			// element from the top-level selector list.
+			if !sel.UsesPseudoElement() {
+				parentSelectors = append(parentSelectors, css_ast.ComplexSelector{Selectors: substituted})
+			}
+		}
+
+		// Emit this selector before its nested children
+		start := len(results)
+		results = append(results, rule)
+
+		// Lower all children and filter out ones that become empty
+		context := lowerNestingContext{
+			parentSelectors: parentSelectors,
+			loweredRules:    results,
+		}
+		r.Rules = p.lowerNestingInRulesAndReturnRemaining(r.Rules, &context)
+
+		// Omit this selector entirely if it's now empty
+		if len(r.Rules) == 0 {
+			copy(context.loweredRules[start:], context.loweredRules[start+1:])
+			context.loweredRules = context.loweredRules[:len(context.loweredRules)-1]
+		}
+		return context.loweredRules
+
+	case *css_ast.RKnownAt:
+		var rules []css_ast.Rule
+		for _, child := range r.Rules {
+			rules = p.lowerNestingInRule(child, rules)
+		}
+		r.Rules = rules
+
+	case *css_ast.RAtLayer:
+		var rules []css_ast.Rule
+		for _, child := range r.Rules {
+			rules = p.lowerNestingInRule(child, rules)
+		}
+		r.Rules = rules
+	}
+
+	return append(results, rule)
+}
+
+// Lower all children and filter out ones that become empty
+func (p *parser) lowerNestingInRulesAndReturnRemaining(rules []css_ast.Rule, context *lowerNestingContext) []css_ast.Rule {
+	n := 0
+	for _, child := range rules {
+		child = p.lowerNestingInRuleWithContext(child, context)
+		if child.Data != nil {
+			rules[n] = child
+			n++
+		}
+	}
+	return rules[:n]
+}
+
+type lowerNestingContext struct {
+	parentSelectors []css_ast.ComplexSelector
+	loweredRules    []css_ast.Rule
+}
+
+func (p *parser) lowerNestingInRuleWithContext(rule css_ast.Rule, context *lowerNestingContext) css_ast.Rule {
+	switch r := rule.Data.(type) {
+	case *css_ast.RSelector:
+		// "a { & b {} }" => "a b {}"
+		// "a { &b {} }" => "a:is(b) {}"
+		// "a { &:hover {} }" => "a:hover {}"
+		// ".x { &b {} }" => "b.x {}"
+		// "a, b { .c, d {} }" => ":is(a, b) :is(.c, d) {}"
+		// "a, b { &.c, & d, e & {} }" => ":is(a, b).c, :is(a, b) d, e :is(a, b) {}"
+
+		// Pass 1: Canonicalize and analyze our selectors
+		canUseGroupDescendantCombinator := true // Can we do "parent «space» :is(...selectors)"?
+		canUseGroupSubSelector := true          // Can we do "parent«nospace»:is(...selectors)"?
+		var commonLeadingCombinator css_ast.Combinator
+		for i := range r.Selectors {
+			sel := &r.Selectors[i]
+
+			// Inject the implicit "&" now for simplicity later on
+			if sel.IsRelative() {
+				sel.Selectors = append([]css_ast.CompoundSelector{{NestingSelectorLoc: ast.MakeIndex32(uint32(rule.Loc.Start))}}, sel.Selectors...)
+			}
+
+			// Pseudo-elements aren't supported by ":is" (i.e. ":is(div, div::before)"
+			// is the same as ":is(div)") so we need to avoid generating ":is" if a
+			// pseudo-element is present.
+			if sel.UsesPseudoElement() {
+				canUseGroupDescendantCombinator = false
+				canUseGroupSubSelector = false
+			}
+
+			// Are all children of the form "& «something»"?
+			if len(sel.Selectors) < 2 || !sel.Selectors[0].IsSingleAmpersand() {
+				canUseGroupDescendantCombinator = false
+			} else {
+				// If all children are of the form "& «COMBINATOR» «something»", is «COMBINATOR» the same in all cases?
+				var combinator css_ast.Combinator
+				if len(sel.Selectors) >= 2 {
+					combinator = sel.Selectors[1].Combinator
+				}
+				if i == 0 {
+					commonLeadingCombinator = combinator
+				} else if commonLeadingCombinator.Byte != combinator.Byte {
+					canUseGroupDescendantCombinator = false
+				}
+			}
+
+			// Are all children of the form "&«something»"?
+			if first := sel.Selectors[0]; !first.HasNestingSelector() || first.IsSingleAmpersand() {
+				canUseGroupSubSelector = false
+			}
+		}
+
+		// Avoid generating ":is" if it's not supported
+		if p.options.unsupportedCSSFeatures.Has(compat.IsPseudoClass) && len(r.Selectors) > 1 {
+			canUseGroupDescendantCombinator = false
+			canUseGroupSubSelector = false
+		}
+
+		// Try to apply simplifications for shorter output
+		if canUseGroupDescendantCombinator {
+			// "& a, & b {}" => "& :is(a, b) {}"
+			// "& > a, & > b {}" => "& > :is(a, b) {}"
+			nestingSelectorLoc := r.Selectors[0].Selectors[0].NestingSelectorLoc
+			for i := range r.Selectors {
+				sel := &r.Selectors[i]
+				sel.Selectors = sel.Selectors[1:]
+			}
+			merged := p.multipleComplexSelectorsToSingleComplexSelector(r.Selectors)(rule.Loc)
+			merged.Selectors = append([]css_ast.CompoundSelector{{NestingSelectorLoc: nestingSelectorLoc}}, merged.Selectors...)
+			r.Selectors = []css_ast.ComplexSelector{merged}
+		} else if canUseGroupSubSelector {
+			// "&a, &b {}" => "&:is(a, b) {}"
+			// "> &a, > &b {}" => "> &:is(a, b) {}"
+			nestingSelectorLoc := r.Selectors[0].Selectors[0].NestingSelectorLoc
+			for i := range r.Selectors {
+				sel := &r.Selectors[i]
+				sel.Selectors[0].NestingSelectorLoc = ast.Index32{}
+			}
+			merged := p.multipleComplexSelectorsToSingleComplexSelector(r.Selectors)(rule.Loc)
+			merged.Selectors[0].NestingSelectorLoc = nestingSelectorLoc
+			r.Selectors = []css_ast.ComplexSelector{merged}
+		}
+
+		// Pass 2: Substitute "&" for the parent selector
+		if !p.options.unsupportedCSSFeatures.Has(compat.IsPseudoClass) || len(context.parentSelectors) <= 1 {
+			// If we can use ":is", or we don't have to because there's only one
+			// parent selector, or we are using ":is()" to match zero parent selectors
+			// (even if ":is" is unsupported), then substituting "&" for the parent
+			// selector is easy.
+			for i := range r.Selectors {
+				complex := &r.Selectors[i]
+				results := make([]css_ast.CompoundSelector, 0, len(complex.Selectors))
+				parent := p.multipleComplexSelectorsToSingleComplexSelector(context.parentSelectors)
+				for _, compound := range complex.Selectors {
+					results = p.substituteAmpersandsInCompoundSelector(compound, parent, results, keepLeadingCombinator)
+				}
+				complex.Selectors = results
+			}
+		} else {
+			// Otherwise if we can't use ":is", the transform is more complicated.
+			// Avoiding ":is" can lead to a combinatorial explosion of cases so we
+			// want to avoid this if possible. For example:
+			//
+			//   .first, .second, .third {
+			//     & > & {
+			//       color: red;
+			//     }
+			//   }
+			//
+			// If we can use ":is" (the easy case above) then we can do this:
+			//
+			//   :is(.first, .second, .third) > :is(.first, .second, .third) {
+			//     color: red;
+			//   }
+			//
+			// But if we can't use ":is" then we have to do this instead:
+			//
+			//   .first > .first,
+			//   .first > .second,
+			//   .first > .third,
+			//   .second > .first,
+			//   .second > .second,
+			//   .second > .third,
+			//   .third > .first,
+			//   .third > .second,
+			//   .third > .third {
+			//     color: red;
+			//   }
+			//
+			// That combinatorial explosion is what the loop below implements. Note
+			// that PostCSS's implementation of nesting gets this wrong. It generates
+			// this instead:
+			//
+			//   .first > .first,
+			//   .second > .second,
+			//   .third > .third {
+			//     color: red;
+			//   }
+			//
+			// That's not equivalent, so that's an incorrect transformation.
+			var selectors []css_ast.ComplexSelector
+			var indices []int
+			for {
+				// Every time we encounter another "&", add another dimension
+				offset := 0
+				parent := func(loc logger.Loc) css_ast.ComplexSelector {
+					if offset == len(indices) {
+						indices = append(indices, 0)
+					}
+					index := indices[offset]
+					offset++
+					return context.parentSelectors[index]
+				}
+
+				// Do the substitution for this particular combination
+				for i := range r.Selectors {
+					complex := r.Selectors[i]
+					results := make([]css_ast.CompoundSelector, 0, len(complex.Selectors))
+					for _, compound := range complex.Selectors {
+						results = p.substituteAmpersandsInCompoundSelector(compound, parent, results, keepLeadingCombinator)
+					}
+					complex.Selectors = results
+					selectors = append(selectors, complex)
+					offset = 0
+				}
+
+				// Do addition with carry on the indices across dimensions
+				carry := len(indices)
+				for carry > 0 {
+					index := &indices[carry-1]
+					if *index+1 < len(context.parentSelectors) {
+						*index++
+						break
+					}
+					*index = 0
+					carry--
+				}
+				if carry == 0 {
+					break
+				}
+			}
+			r.Selectors = selectors
+		}
+
+		// Lower all child rules using our newly substituted selector
+		context.loweredRules = p.lowerNestingInRule(rule, context.loweredRules)
+		return css_ast.Rule{}
+
+	case *css_ast.RKnownAt:
+		childContext := lowerNestingContext{parentSelectors: context.parentSelectors}
+		r.Rules = p.lowerNestingInRulesAndReturnRemaining(r.Rules, &childContext)
+
+		// "div { @media screen { color: red } }" "@media screen { div { color: red } }"
+		if len(r.Rules) > 0 {
+			childContext.loweredRules = append([]css_ast.Rule{{Loc: rule.Loc, Data: &css_ast.RSelector{
+				Selectors: context.parentSelectors,
+				Rules:     r.Rules,
+			}}}, childContext.loweredRules...)
+		}
+
+		// "div { @media screen { &:hover { color: red } } }" "@media screen { div:hover { color: red } }"
+		if len(childContext.loweredRules) > 0 {
+			r.Rules = childContext.loweredRules
+			context.loweredRules = append(context.loweredRules, rule)
+		}
+
+		return css_ast.Rule{}
+
+	case *css_ast.RAtLayer:
+		// Lower all children and filter out ones that become empty
+		childContext := lowerNestingContext{parentSelectors: context.parentSelectors}
+		r.Rules = p.lowerNestingInRulesAndReturnRemaining(r.Rules, &childContext)
+
+		// "div { @layer foo { color: red } }" "@layer foo { div { color: red } }"
+		if len(r.Rules) > 0 {
+			childContext.loweredRules = append([]css_ast.Rule{{Loc: rule.Loc, Data: &css_ast.RSelector{
+				Selectors: context.parentSelectors,
+				Rules:     r.Rules,
+			}}}, childContext.loweredRules...)
+		}
+
+		// "div { @layer foo { &:hover { color: red } } }" "@layer foo { div:hover { color: red } }"
+		// "div { @layer foo {} }" => "@layer foo {}" (layers have side effects, so don't remove empty ones)
+		r.Rules = childContext.loweredRules
+		context.loweredRules = append(context.loweredRules, rule)
+		return css_ast.Rule{}
+	}
+
+	return rule
+}
+
+type leadingCombinatorStrip uint8
+
+const (
+	keepLeadingCombinator leadingCombinatorStrip = iota
+	stripLeadingCombinator
+)
+
+func (p *parser) substituteAmpersandsInCompoundSelector(
+	sel css_ast.CompoundSelector,
+	replacementFn func(logger.Loc) css_ast.ComplexSelector,
+	results []css_ast.CompoundSelector,
+	strip leadingCombinatorStrip,
+) []css_ast.CompoundSelector {
+	if sel.HasNestingSelector() {
+		nestingSelectorLoc := logger.Loc{Start: int32(sel.NestingSelectorLoc.GetIndex())}
+		sel.NestingSelectorLoc = ast.Index32{}
+		replacement := replacementFn(nestingSelectorLoc)
+
+		// Convert the replacement to a single compound selector
+		var single css_ast.CompoundSelector
+		if sel.Combinator.Byte == 0 && (len(replacement.Selectors) == 1 || len(results) == 0) {
+			// ".foo { :hover & {} }" => ":hover .foo {}"
+			// ".foo .bar { &:hover {} }" => ".foo .bar:hover {}"
+			last := len(replacement.Selectors) - 1
+			results = append(results, replacement.Selectors[:last]...)
+			single = replacement.Selectors[last]
+			if strip == stripLeadingCombinator {
+				single.Combinator = css_ast.Combinator{}
+			}
+			sel.Combinator = single.Combinator
+		} else if len(replacement.Selectors) == 1 {
+			// ".foo { > &:hover {} }" => ".foo > .foo:hover {}"
+			single = replacement.Selectors[0]
+			if strip == stripLeadingCombinator {
+				single.Combinator = css_ast.Combinator{}
+			}
+		} else {
+			// ".foo .bar { :hover & {} }" => ":hover :is(.foo .bar) {}"
+			// ".foo .bar { > &:hover {} }" => ".foo .bar > :is(.foo .bar):hover {}"
+			p.reportNestingWithGeneratedPseudoClassIs(nestingSelectorLoc)
+			single = css_ast.CompoundSelector{
+				SubclassSelectors: []css_ast.SubclassSelector{{
+					Range: logger.Range{Loc: nestingSelectorLoc},
+					Data: &css_ast.SSPseudoClassWithSelectorList{
+						Kind:      css_ast.PseudoClassIs,
+						Selectors: []css_ast.ComplexSelector{replacement.CloneWithoutLeadingCombinator()},
+					},
+				}},
+			}
+		}
+
+		var subclassSelectorPrefix []css_ast.SubclassSelector
+
+		// Insert the type selector
+		if single.TypeSelector != nil {
+			if sel.TypeSelector != nil {
+				p.reportNestingWithGeneratedPseudoClassIs(nestingSelectorLoc)
+				subclassSelectorPrefix = append(subclassSelectorPrefix, css_ast.SubclassSelector{
+					Range: sel.TypeSelector.Range(),
+					Data: &css_ast.SSPseudoClassWithSelectorList{
+						Kind:      css_ast.PseudoClassIs,
+						Selectors: []css_ast.ComplexSelector{{Selectors: []css_ast.CompoundSelector{{TypeSelector: sel.TypeSelector}}}},
+					},
+				})
+			}
+			sel.TypeSelector = single.TypeSelector
+		}
+
+		// Insert the subclass selectors
+		subclassSelectorPrefix = append(subclassSelectorPrefix, single.SubclassSelectors...)
+
+		// Write the changes back
+		if len(subclassSelectorPrefix) > 0 {
+			sel.SubclassSelectors = append(subclassSelectorPrefix, sel.SubclassSelectors...)
+		}
+	}
+
+	// "div { :is(&.foo) {} }" => ":is(div.foo) {}"
+	for _, ss := range sel.SubclassSelectors {
+		if class, ok := ss.Data.(*css_ast.SSPseudoClassWithSelectorList); ok {
+			outer := make([]css_ast.ComplexSelector, 0, len(class.Selectors))
+			for _, complex := range class.Selectors {
+				inner := make([]css_ast.CompoundSelector, 0, len(complex.Selectors))
+				for _, sel := range complex.Selectors {
+					inner = p.substituteAmpersandsInCompoundSelector(sel, replacementFn, inner, stripLeadingCombinator)
+				}
+				outer = append(outer, css_ast.ComplexSelector{Selectors: inner})
+			}
+			class.Selectors = outer
+		}
+	}
+
+	return append(results, sel)
+}
+
+// Turn the list of selectors into a single selector by wrapping lists
+// without a single element with ":is(...)". Note that this may result
+// in an empty ":is()" selector (which matches nothing).
+func (p *parser) multipleComplexSelectorsToSingleComplexSelector(selectors []css_ast.ComplexSelector) func(logger.Loc) css_ast.ComplexSelector {
+	if len(selectors) == 1 {
+		return func(logger.Loc) css_ast.ComplexSelector {
+			return selectors[0]
+		}
+	}
+
+	var leadingCombinator css_ast.Combinator
+	clones := make([]css_ast.ComplexSelector, len(selectors))
+
+	for i, sel := range selectors {
+		// "> a, > b" => "> :is(a, b)" (the caller should have already checked that all leading combinators are the same)
+		leadingCombinator = sel.Selectors[0].Combinator
+		clones[i] = sel.CloneWithoutLeadingCombinator()
+	}
+
+	return func(loc logger.Loc) css_ast.ComplexSelector {
+		return css_ast.ComplexSelector{
+			Selectors: []css_ast.CompoundSelector{{
+				Combinator: leadingCombinator,
+				SubclassSelectors: []css_ast.SubclassSelector{{
+					Range: logger.Range{Loc: loc},
+					Data: &css_ast.SSPseudoClassWithSelectorList{
+						Kind:      css_ast.PseudoClassIs,
+						Selectors: clones,
+					},
+				}},
+			}},
+		}
+	}
+}
+
+func (p *parser) reportNestingWithGeneratedPseudoClassIs(nestingSelectorLoc logger.Loc) {
+	if p.options.unsupportedCSSFeatures.Has(compat.IsPseudoClass) {
+		_, didWarn := p.nestingWarnings[nestingSelectorLoc]
+		if didWarn {
+			// Only warn at each location once
+			return
+		}
+		if p.nestingWarnings == nil {
+			p.nestingWarnings = make(map[logger.Loc]struct{})
+		}
+		p.nestingWarnings[nestingSelectorLoc] = struct{}{}
+		text := "Transforming this CSS nesting syntax is not supported in the configured target environment"
+		if p.options.originalTargetEnv != "" {
+			text = fmt.Sprintf("%s (%s)", text, p.options.originalTargetEnv)
+		}
+		r := logger.Range{Loc: nestingSelectorLoc, Len: 1}
+		p.log.AddIDWithNotes(logger.MsgID_CSS_UnsupportedCSSNesting, logger.Warning, &p.tracker, r, text, []logger.MsgData{{
+			Text: "The nesting transform for this case must generate an \":is(...)\" but the configured target environment does not support the \":is\" pseudo-class."}})
+	}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_parser.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_parser.go
new file mode 100644
index 0000000..131ec5e
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_parser.go
@@ -0,0 +1,2374 @@
+package css_parser
+
+import (
+	"fmt"
+	"strings"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/config"
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+// This is mostly a normal CSS parser with one exception: the addition of
+// support for parsing https://drafts.csswg.org/css-nesting-1/.
+
+type parser struct {
+	log               logger.Log
+	source            logger.Source
+	tokens            []css_lexer.Token
+	allComments       []logger.Range
+	legalComments     []css_lexer.Comment
+	stack             []css_lexer.T
+	importRecords     []ast.ImportRecord
+	symbols           []ast.Symbol
+	composes          map[ast.Ref]*css_ast.Composes
+	localSymbols      []ast.LocRef
+	localScope        map[string]ast.LocRef
+	globalScope       map[string]ast.LocRef
+	nestingWarnings   map[logger.Loc]struct{}
+	tracker           logger.LineColumnTracker
+	enclosingAtMedia  [][]css_ast.Token
+	layersPreImport   [][]string
+	layersPostImport  [][]string
+	enclosingLayer    []string
+	anonLayerCount    int
+	index             int
+	legalCommentIndex int
+	inSelectorSubtree int
+	prevError         logger.Loc
+	options           Options
+	nestingIsPresent  bool
+	makeLocalSymbols  bool
+	hasSeenAtImport   bool
+}
+
+type Options struct {
+	cssPrefixData map[css_ast.D]compat.CSSPrefix
+
+	// This is an embedded struct. Always access these directly instead of off
+	// the name "optionsThatSupportStructuralEquality". This is only grouped like
+	// this to make the equality comparison easier and safer (and hopefully faster).
+	optionsThatSupportStructuralEquality
+}
+
+type symbolMode uint8
+
+const (
+	symbolModeDisabled symbolMode = iota
+	symbolModeGlobal
+	symbolModeLocal
+)
+
+type optionsThatSupportStructuralEquality struct {
+	originalTargetEnv      string
+	unsupportedCSSFeatures compat.CSSFeature
+	minifySyntax           bool
+	minifyWhitespace       bool
+	minifyIdentifiers      bool
+	symbolMode             symbolMode
+}
+
+func OptionsFromConfig(loader config.Loader, options *config.Options) Options {
+	var symbolMode symbolMode
+	switch loader {
+	case config.LoaderGlobalCSS:
+		symbolMode = symbolModeGlobal
+	case config.LoaderLocalCSS:
+		symbolMode = symbolModeLocal
+	}
+
+	return Options{
+		cssPrefixData: options.CSSPrefixData,
+
+		optionsThatSupportStructuralEquality: optionsThatSupportStructuralEquality{
+			minifySyntax:           options.MinifySyntax,
+			minifyWhitespace:       options.MinifyWhitespace,
+			minifyIdentifiers:      options.MinifyIdentifiers,
+			unsupportedCSSFeatures: options.UnsupportedCSSFeatures,
+			originalTargetEnv:      options.OriginalTargetEnv,
+			symbolMode:             symbolMode,
+		},
+	}
+}
+
+func (a *Options) Equal(b *Options) bool {
+	// Compare "optionsThatSupportStructuralEquality"
+	if a.optionsThatSupportStructuralEquality != b.optionsThatSupportStructuralEquality {
+		return false
+	}
+
+	// Compare "cssPrefixData"
+	if len(a.cssPrefixData) != len(b.cssPrefixData) {
+		return false
+	}
+	for k, va := range a.cssPrefixData {
+		vb, ok := b.cssPrefixData[k]
+		if !ok || va != vb {
+			return false
+		}
+	}
+	for k := range b.cssPrefixData {
+		if _, ok := b.cssPrefixData[k]; !ok {
+			return false
+		}
+	}
+
+	return true
+}
+
+func Parse(log logger.Log, source logger.Source, options Options) css_ast.AST {
+	result := css_lexer.Tokenize(log, source, css_lexer.Options{
+		RecordAllComments: options.minifyIdentifiers,
+	})
+	p := parser{
+		log:              log,
+		source:           source,
+		tracker:          logger.MakeLineColumnTracker(&source),
+		options:          options,
+		tokens:           result.Tokens,
+		allComments:      result.AllComments,
+		legalComments:    result.LegalComments,
+		prevError:        logger.Loc{Start: -1},
+		composes:         make(map[ast.Ref]*css_ast.Composes),
+		localScope:       make(map[string]ast.LocRef),
+		globalScope:      make(map[string]ast.LocRef),
+		makeLocalSymbols: options.symbolMode == symbolModeLocal,
+	}
+	rules := p.parseListOfRules(ruleContext{
+		isTopLevel:     true,
+		parseSelectors: true,
+	})
+	p.expect(css_lexer.TEndOfFile)
+	return css_ast.AST{
+		Rules:                rules,
+		CharFreq:             p.computeCharacterFrequency(),
+		Symbols:              p.symbols,
+		ImportRecords:        p.importRecords,
+		ApproximateLineCount: result.ApproximateLineCount,
+		SourceMapComment:     result.SourceMapComment,
+		LocalSymbols:         p.localSymbols,
+		LocalScope:           p.localScope,
+		GlobalScope:          p.globalScope,
+		Composes:             p.composes,
+		LayersPreImport:      p.layersPreImport,
+		LayersPostImport:     p.layersPostImport,
+	}
+}
+
+// Compute a character frequency histogram for everything that's not a bound
+// symbol. This is used to modify how minified names are generated for slightly
+// better gzip compression. Even though it's a very small win, we still do it
+// because it's simple to do and very cheap to compute.
+func (p *parser) computeCharacterFrequency() *ast.CharFreq {
+	if !p.options.minifyIdentifiers {
+		return nil
+	}
+
+	// Add everything in the file to the histogram
+	charFreq := &ast.CharFreq{}
+	charFreq.Scan(p.source.Contents, 1)
+
+	// Subtract out all comments
+	for _, commentRange := range p.allComments {
+		charFreq.Scan(p.source.TextForRange(commentRange), -1)
+	}
+
+	// Subtract out all import paths
+	for _, record := range p.importRecords {
+		if !record.SourceIndex.IsValid() {
+			charFreq.Scan(record.Path.Text, -1)
+		}
+	}
+
+	// Subtract out all symbols that will be minified
+	for _, symbol := range p.symbols {
+		if symbol.Kind == ast.SymbolLocalCSS {
+			charFreq.Scan(symbol.OriginalName, -int32(symbol.UseCountEstimate))
+		}
+	}
+
+	return charFreq
+}
+
+func (p *parser) advance() {
+	if p.index < len(p.tokens) {
+		p.index++
+	}
+}
+
+func (p *parser) at(index int) css_lexer.Token {
+	if index < len(p.tokens) {
+		return p.tokens[index]
+	}
+	return css_lexer.Token{
+		Kind:  css_lexer.TEndOfFile,
+		Range: logger.Range{Loc: logger.Loc{Start: int32(len(p.source.Contents))}},
+	}
+}
+
+func (p *parser) current() css_lexer.Token {
+	return p.at(p.index)
+}
+
+func (p *parser) next() css_lexer.Token {
+	return p.at(p.index + 1)
+}
+
+func (p *parser) raw() string {
+	t := p.current()
+	return p.source.Contents[t.Range.Loc.Start:t.Range.End()]
+}
+
+func (p *parser) decoded() string {
+	return p.current().DecodedText(p.source.Contents)
+}
+
+func (p *parser) peek(kind css_lexer.T) bool {
+	return kind == p.current().Kind
+}
+
+func (p *parser) eat(kind css_lexer.T) bool {
+	if p.peek(kind) {
+		p.advance()
+		return true
+	}
+	return false
+}
+
+func (p *parser) expect(kind css_lexer.T) bool {
+	return p.expectWithMatchingLoc(kind, logger.Loc{Start: -1})
+}
+
+func (p *parser) expectWithMatchingLoc(kind css_lexer.T, matchingLoc logger.Loc) bool {
+	if p.eat(kind) {
+		return true
+	}
+	t := p.current()
+	if (t.Flags & css_lexer.DidWarnAboutSingleLineComment) != 0 {
+		return false
+	}
+
+	var text string
+	var suggestion string
+	var notes []logger.MsgData
+
+	expected := kind.String()
+	if strings.HasPrefix(expected, "\"") && strings.HasSuffix(expected, "\"") {
+		suggestion = expected[1 : len(expected)-1]
+	}
+
+	if (kind == css_lexer.TSemicolon || kind == css_lexer.TColon) && p.index > 0 && p.at(p.index-1).Kind == css_lexer.TWhitespace {
+		// Have a nice error message for forgetting a trailing semicolon or colon
+		text = fmt.Sprintf("Expected %s", expected)
+		t = p.at(p.index - 1)
+	} else if (kind == css_lexer.TCloseBrace || kind == css_lexer.TCloseBracket || kind == css_lexer.TCloseParen) &&
+		matchingLoc.Start != -1 && int(matchingLoc.Start)+1 <= len(p.source.Contents) {
+		// Have a nice error message for forgetting a closing brace/bracket/parenthesis
+		c := p.source.Contents[matchingLoc.Start : matchingLoc.Start+1]
+		text = fmt.Sprintf("Expected %s to go with %q", expected, c)
+		notes = append(notes, p.tracker.MsgData(logger.Range{Loc: matchingLoc, Len: 1}, fmt.Sprintf("The unbalanced %q is here:", c)))
+	} else {
+		switch t.Kind {
+		case css_lexer.TEndOfFile, css_lexer.TWhitespace:
+			text = fmt.Sprintf("Expected %s but found %s", expected, t.Kind.String())
+			t.Range.Len = 0
+		case css_lexer.TBadURL, css_lexer.TUnterminatedString:
+			text = fmt.Sprintf("Expected %s but found %s", expected, t.Kind.String())
+		default:
+			text = fmt.Sprintf("Expected %s but found %q", expected, p.raw())
+		}
+	}
+
+	if t.Range.Loc.Start > p.prevError.Start {
+		data := p.tracker.MsgData(t.Range, text)
+		data.Location.Suggestion = suggestion
+		p.log.AddMsgID(logger.MsgID_CSS_CSSSyntaxError, logger.Msg{Kind: logger.Warning, Data: data, Notes: notes})
+		p.prevError = t.Range.Loc
+	}
+	return false
+}
+
+func (p *parser) unexpected() {
+	if t := p.current(); t.Range.Loc.Start > p.prevError.Start && (t.Flags&css_lexer.DidWarnAboutSingleLineComment) == 0 {
+		var text string
+		switch t.Kind {
+		case css_lexer.TEndOfFile, css_lexer.TWhitespace:
+			text = fmt.Sprintf("Unexpected %s", t.Kind.String())
+			t.Range.Len = 0
+		case css_lexer.TBadURL, css_lexer.TUnterminatedString:
+			text = fmt.Sprintf("Unexpected %s", t.Kind.String())
+		default:
+			text = fmt.Sprintf("Unexpected %q", p.raw())
+		}
+		p.log.AddID(logger.MsgID_CSS_CSSSyntaxError, logger.Warning, &p.tracker, t.Range, text)
+		p.prevError = t.Range.Loc
+	}
+}
+
+func (p *parser) symbolForName(loc logger.Loc, name string) ast.LocRef {
+	var kind ast.SymbolKind
+	var scope map[string]ast.LocRef
+
+	if p.makeLocalSymbols {
+		kind = ast.SymbolLocalCSS
+		scope = p.localScope
+	} else {
+		kind = ast.SymbolGlobalCSS
+		scope = p.globalScope
+	}
+
+	entry, ok := scope[name]
+	if !ok {
+		entry = ast.LocRef{
+			Loc: loc,
+			Ref: ast.Ref{
+				SourceIndex: p.source.Index,
+				InnerIndex:  uint32(len(p.symbols)),
+			},
+		}
+		p.symbols = append(p.symbols, ast.Symbol{
+			Kind:         kind,
+			OriginalName: name,
+			Link:         ast.InvalidRef,
+		})
+		scope[name] = entry
+		if kind == ast.SymbolLocalCSS {
+			p.localSymbols = append(p.localSymbols, entry)
+		}
+	}
+
+	p.symbols[entry.Ref.InnerIndex].UseCountEstimate++
+	return entry
+}
+
+func (p *parser) recordAtLayerRule(layers [][]string) {
+	if p.anonLayerCount > 0 {
+		return
+	}
+
+	for _, layer := range layers {
+		if len(p.enclosingLayer) > 0 {
+			clone := make([]string, 0, len(p.enclosingLayer)+len(layer))
+			layer = append(append(clone, p.enclosingLayer...), layer...)
+		}
+		p.layersPostImport = append(p.layersPostImport, layer)
+	}
+}
+
+type ruleContext struct {
+	isTopLevel     bool
+	parseSelectors bool
+}
+
+func (p *parser) parseListOfRules(context ruleContext) []css_ast.Rule {
+	atRuleContext := atRuleContext{}
+	if context.isTopLevel {
+		atRuleContext.charsetValidity = atRuleValid
+		atRuleContext.importValidity = atRuleValid
+		atRuleContext.isTopLevel = true
+	}
+	rules := []css_ast.Rule{}
+	didFindAtImport := false
+
+loop:
+	for {
+		if context.isTopLevel {
+			p.nestingIsPresent = false
+		}
+
+		// If there are any legal comments immediately before the current token,
+		// turn them all into comment rules and append them to the current rule list
+		for p.legalCommentIndex < len(p.legalComments) {
+			comment := p.legalComments[p.legalCommentIndex]
+			if comment.TokenIndexAfter > uint32(p.index) {
+				break
+			}
+			if comment.TokenIndexAfter == uint32(p.index) {
+				rules = append(rules, css_ast.Rule{Loc: comment.Loc, Data: &css_ast.RComment{Text: comment.Text}})
+			}
+			p.legalCommentIndex++
+		}
+
+		switch p.current().Kind {
+		case css_lexer.TEndOfFile:
+			break loop
+
+		case css_lexer.TCloseBrace:
+			if !context.isTopLevel {
+				break loop
+			}
+
+		case css_lexer.TWhitespace:
+			p.advance()
+			continue
+
+		case css_lexer.TAtKeyword:
+			rule := p.parseAtRule(atRuleContext)
+
+			// Disallow "@charset" and "@import" after other rules
+			if context.isTopLevel {
+				switch r := rule.Data.(type) {
+				case *css_ast.RAtCharset:
+					// This doesn't invalidate anything because it always comes first
+
+				case *css_ast.RAtImport:
+					didFindAtImport = true
+					if atRuleContext.charsetValidity == atRuleValid {
+						atRuleContext.afterLoc = rule.Loc
+						atRuleContext.charsetValidity = atRuleInvalidAfter
+					}
+
+				case *css_ast.RAtLayer:
+					if atRuleContext.charsetValidity == atRuleValid {
+						atRuleContext.afterLoc = rule.Loc
+						atRuleContext.charsetValidity = atRuleInvalidAfter
+					}
+
+					// From the specification: "Note: No @layer rules are allowed between
+					// @import and @namespace rules. Any @layer rule that comes after an
+					// @import or @namespace rule will cause any subsequent @import or
+					// @namespace rules to be ignored."
+					if atRuleContext.importValidity == atRuleValid && (r.Rules != nil || didFindAtImport) {
+						atRuleContext.afterLoc = rule.Loc
+						atRuleContext.charsetValidity = atRuleInvalidAfter
+						atRuleContext.importValidity = atRuleInvalidAfter
+					}
+
+				default:
+					if atRuleContext.importValidity == atRuleValid {
+						atRuleContext.afterLoc = rule.Loc
+						atRuleContext.charsetValidity = atRuleInvalidAfter
+						atRuleContext.importValidity = atRuleInvalidAfter
+					}
+				}
+			}
+
+			// Lower CSS nesting if it's not supported (but only at the top level)
+			if p.nestingIsPresent && p.options.unsupportedCSSFeatures.Has(compat.Nesting) && context.isTopLevel {
+				rules = p.lowerNestingInRule(rule, rules)
+			} else {
+				rules = append(rules, rule)
+			}
+			continue
+
+		case css_lexer.TCDO, css_lexer.TCDC:
+			if context.isTopLevel {
+				p.advance()
+				continue
+			}
+		}
+
+		if atRuleContext.importValidity == atRuleValid {
+			atRuleContext.afterLoc = p.current().Range.Loc
+			atRuleContext.charsetValidity = atRuleInvalidAfter
+			atRuleContext.importValidity = atRuleInvalidAfter
+		}
+
+		// Note: CSS recently changed to parse and discard declarations
+		// here instead of treating them as the start of a qualified rule.
+		// See also: https://github.com/w3c/csswg-drafts/issues/8834
+		if !context.isTopLevel {
+			if scan, index := p.scanForEndOfRule(); scan == endOfRuleSemicolon {
+				tokens := p.convertTokens(p.tokens[p.index:index])
+				rules = append(rules, css_ast.Rule{Loc: p.current().Range.Loc, Data: &css_ast.RBadDeclaration{Tokens: tokens}})
+				p.index = index + 1
+				continue
+			}
+		}
+
+		var rule css_ast.Rule
+		if context.parseSelectors {
+			rule = p.parseSelectorRule(context.isTopLevel, parseSelectorOpts{})
+		} else {
+			rule = p.parseQualifiedRule(parseQualifiedRuleOpts{isTopLevel: context.isTopLevel})
+		}
+
+		// Lower CSS nesting if it's not supported (but only at the top level)
+		if p.nestingIsPresent && p.options.unsupportedCSSFeatures.Has(compat.Nesting) && context.isTopLevel {
+			rules = p.lowerNestingInRule(rule, rules)
+		} else {
+			rules = append(rules, rule)
+		}
+	}
+
+	if p.options.minifySyntax {
+		rules = p.mangleRules(rules, context.isTopLevel)
+	}
+	return rules
+}
+
+type listOfDeclarationsOpts struct {
+	composesContext      *composesContext
+	canInlineNoOpNesting bool
+}
+
+func (p *parser) parseListOfDeclarations(opts listOfDeclarationsOpts) (list []css_ast.Rule) {
+	list = []css_ast.Rule{}
+	foundNesting := false
+
+	for {
+		switch p.current().Kind {
+		case css_lexer.TWhitespace, css_lexer.TSemicolon:
+			p.advance()
+
+		case css_lexer.TEndOfFile, css_lexer.TCloseBrace:
+			list = p.processDeclarations(list, opts.composesContext)
+			if p.options.minifySyntax {
+				list = p.mangleRules(list, false /* isTopLevel */)
+
+				// Pull out all unnecessarily-nested declarations and stick them at the end
+				if opts.canInlineNoOpNesting {
+					// "a { & { x: y } }" => "a { x: y }"
+					// "a { & { b: c } d: e }" => "a { d: e; b: c }"
+					if foundNesting {
+						var inlineDecls []css_ast.Rule
+						n := 0
+						for _, rule := range list {
+							if rule, ok := rule.Data.(*css_ast.RSelector); ok && len(rule.Selectors) == 1 {
+								if sel := rule.Selectors[0]; len(sel.Selectors) == 1 && sel.Selectors[0].IsSingleAmpersand() {
+									inlineDecls = append(inlineDecls, rule.Rules...)
+									continue
+								}
+							}
+							list[n] = rule
+							n++
+						}
+						list = append(list[:n], inlineDecls...)
+					}
+				} else {
+					// "a, b::before { & { x: y } }" => "a, b::before { & { x: y } }"
+				}
+			}
+			return
+
+		case css_lexer.TAtKeyword:
+			if p.inSelectorSubtree > 0 {
+				p.nestingIsPresent = true
+			}
+			list = append(list, p.parseAtRule(atRuleContext{
+				isDeclarationList:    true,
+				canInlineNoOpNesting: opts.canInlineNoOpNesting,
+			}))
+
+		// Reference: https://drafts.csswg.org/css-nesting-1/
+		default:
+			if scan, _ := p.scanForEndOfRule(); scan == endOfRuleOpenBrace {
+				p.nestingIsPresent = true
+				foundNesting = true
+				rule := p.parseSelectorRule(false, parseSelectorOpts{
+					isDeclarationContext: true,
+					composesContext:      opts.composesContext,
+				})
+
+				// If this rule was a single ":global" or ":local", inline it here. This
+				// is handled differently than a bare "&" with normal CSS nesting because
+				// that would be inlined at the end of the parent rule's body instead,
+				// which is probably unexpected (e.g. it would trip people up when trying
+				// to write rules in a specific order).
+				if sel, ok := rule.Data.(*css_ast.RSelector); ok && len(sel.Selectors) == 1 {
+					if first := sel.Selectors[0]; len(first.Selectors) == 1 {
+						if first := first.Selectors[0]; first.WasEmptyFromLocalOrGlobal && first.IsSingleAmpersand() {
+							list = append(list, sel.Rules...)
+							continue
+						}
+					}
+				}
+
+				list = append(list, rule)
+			} else {
+				list = append(list, p.parseDeclaration())
+			}
+		}
+	}
+}
+
+func (p *parser) mangleRules(rules []css_ast.Rule, isTopLevel bool) []css_ast.Rule {
+	// Remove empty rules
+	mangledRules := make([]css_ast.Rule, 0, len(rules))
+	var prevNonComment css_ast.R
+next:
+	for _, rule := range rules {
+		nextNonComment := rule.Data
+
+		switch r := rule.Data.(type) {
+		case *css_ast.RAtKeyframes:
+			// Do not remove empty "@keyframe foo {}" rules. Even empty rules still
+			// dispatch JavaScript animation events, so removing them changes
+			// behavior: https://bugzilla.mozilla.org/show_bug.cgi?id=1004377.
+
+		case *css_ast.RAtLayer:
+			if len(r.Rules) == 0 && len(r.Names) > 0 {
+				// Do not remove empty "@layer foo {}" rules. The specification says:
+				// "Cascade layers are sorted by the order in which they first are
+				// declared, with nested layers grouped within their parent layers
+				// before any unlayered rules." So removing empty rules could change
+				// the order in which they are first declared, and is therefore invalid.
+				//
+				// We can turn "@layer foo {}" into "@layer foo;" to be shorter. But
+				// don't collapse anonymous "@layer {}" into "@layer;" because that is
+				// a syntax error.
+				r.Rules = nil
+			} else if len(r.Rules) == 1 && len(r.Names) == 1 {
+				// Only collapse layers if each layer has exactly one name
+				if r2, ok := r.Rules[0].Data.(*css_ast.RAtLayer); ok && len(r2.Names) == 1 {
+					// "@layer a { @layer b {} }" => "@layer a.b;"
+					// "@layer a { @layer b { c {} } }" => "@layer a.b { c {} }"
+					r.Names[0] = append(r.Names[0], r2.Names[0]...)
+					r.Rules = r2.Rules
+				}
+			}
+
+		case *css_ast.RKnownAt:
+			if len(r.Rules) == 0 && atKnownRuleCanBeRemovedIfEmpty[r.AtToken] {
+				continue
+			}
+
+			// Unwrap "@media" rules that duplicate conditions from a parent "@media"
+			// rule. This is unlikely to be authored manually but can be automatically
+			// generated when using a CSS framework such as Tailwind.
+			//
+			//   @media (min-width: 1024px) {
+			//     .md\:class {
+			//       color: red;
+			//     }
+			//     @media (min-width: 1024px) {
+			//       .md\:class {
+			//         color: red;
+			//       }
+			//     }
+			//   }
+			//
+			// This converts that code into the following:
+			//
+			//   @media (min-width: 1024px) {
+			//     .md\:class {
+			//       color: red;
+			//     }
+			//     .md\:class {
+			//       color: red;
+			//     }
+			//   }
+			//
+			// Which can then be mangled further.
+			if strings.EqualFold(r.AtToken, "media") {
+				for _, prelude := range p.enclosingAtMedia {
+					if css_ast.TokensEqualIgnoringWhitespace(r.Prelude, prelude) {
+						mangledRules = append(mangledRules, r.Rules...)
+						continue next
+					}
+				}
+			}
+
+		case *css_ast.RSelector:
+			if len(r.Rules) == 0 {
+				continue
+			}
+
+			// Merge adjacent selectors with the same content
+			// "a { color: red; } b { color: red; }" => "a, b { color: red; }"
+			if prevNonComment != nil {
+				if r, ok := rule.Data.(*css_ast.RSelector); ok {
+					if prev, ok := prevNonComment.(*css_ast.RSelector); ok && css_ast.RulesEqual(r.Rules, prev.Rules, nil) &&
+						isSafeSelectors(r.Selectors) && isSafeSelectors(prev.Selectors) {
+					nextSelector:
+						for _, sel := range r.Selectors {
+							for _, prevSel := range prev.Selectors {
+								if sel.Equal(prevSel, nil) {
+									// Don't add duplicate selectors more than once
+									continue nextSelector
+								}
+							}
+							prev.Selectors = append(prev.Selectors, sel)
+						}
+						continue
+					}
+				}
+			}
+
+		case *css_ast.RComment:
+			nextNonComment = nil
+		}
+
+		if nextNonComment != nil {
+			prevNonComment = nextNonComment
+		}
+
+		mangledRules = append(mangledRules, rule)
+	}
+
+	// Mangle non-top-level rules using a back-to-front pass. Top-level rules
+	// will be mangled by the linker instead for cross-file rule mangling.
+	if !isTopLevel {
+		remover := MakeDuplicateRuleMangler(ast.SymbolMap{})
+		mangledRules = remover.RemoveDuplicateRulesInPlace(p.source.Index, mangledRules, p.importRecords)
+	}
+
+	return mangledRules
+}
+
+type ruleEntry struct {
+	data        css_ast.R
+	callCounter uint32
+}
+
+type hashEntry struct {
+	rules []ruleEntry
+}
+
+type callEntry struct {
+	importRecords []ast.ImportRecord
+	sourceIndex   uint32
+}
+
+type DuplicateRuleRemover struct {
+	entries map[uint32]hashEntry
+	calls   []callEntry
+	check   css_ast.CrossFileEqualityCheck
+}
+
+func MakeDuplicateRuleMangler(symbols ast.SymbolMap) DuplicateRuleRemover {
+	return DuplicateRuleRemover{
+		entries: make(map[uint32]hashEntry),
+		check:   css_ast.CrossFileEqualityCheck{Symbols: symbols},
+	}
+}
+
+func (remover *DuplicateRuleRemover) RemoveDuplicateRulesInPlace(sourceIndex uint32, rules []css_ast.Rule, importRecords []ast.ImportRecord) []css_ast.Rule {
+	// The caller may call this function multiple times, each with a different
+	// set of import records. Remember each set of import records for equality
+	// checks later.
+	callCounter := uint32(len(remover.calls))
+	remover.calls = append(remover.calls, callEntry{importRecords, sourceIndex})
+
+	// Remove duplicate rules, scanning from the back so we keep the last
+	// duplicate. Note that the linker calls this, so we do not want to do
+	// anything that modifies the rules themselves. One reason is that ASTs
+	// are immutable at the linking stage. Another reason is that merging
+	// CSS ASTs from separate files will mess up source maps because a single
+	// AST cannot simultaneously represent offsets from multiple files.
+	n := len(rules)
+	start := n
+skipRule:
+	for i := n - 1; i >= 0; i-- {
+		rule := rules[i]
+
+		// For duplicate rules, omit all but the last copy
+		if hash, ok := rule.Data.Hash(); ok {
+			entry := remover.entries[hash]
+			for _, current := range entry.rules {
+				var check *css_ast.CrossFileEqualityCheck
+
+				// If this rule was from another file, then pass along both arrays
+				// of import records so that the equality check for "url()" tokens
+				// can use them to check for equality.
+				if current.callCounter != callCounter {
+					// Reuse the same memory allocation
+					check = &remover.check
+					call := remover.calls[current.callCounter]
+					check.ImportRecordsA = importRecords
+					check.ImportRecordsB = call.importRecords
+					check.SourceIndexA = sourceIndex
+					check.SourceIndexB = call.sourceIndex
+				}
+
+				if rule.Data.Equal(current.data, check) {
+					continue skipRule
+				}
+			}
+			entry.rules = append(entry.rules, ruleEntry{
+				data:        rule.Data,
+				callCounter: callCounter,
+			})
+			remover.entries[hash] = entry
+		}
+
+		start--
+		rules[start] = rule
+	}
+
+	return rules[start:]
+}
+
+// Reference: https://developer.mozilla.org/en-US/docs/Web/HTML/Element
+var nonDeprecatedElementsSupportedByIE7 = map[string]bool{
+	"a":          true,
+	"abbr":       true,
+	"address":    true,
+	"area":       true,
+	"b":          true,
+	"base":       true,
+	"blockquote": true,
+	"body":       true,
+	"br":         true,
+	"button":     true,
+	"caption":    true,
+	"cite":       true,
+	"code":       true,
+	"col":        true,
+	"colgroup":   true,
+	"dd":         true,
+	"del":        true,
+	"dfn":        true,
+	"div":        true,
+	"dl":         true,
+	"dt":         true,
+	"em":         true,
+	"embed":      true,
+	"fieldset":   true,
+	"form":       true,
+	"h1":         true,
+	"h2":         true,
+	"h3":         true,
+	"h4":         true,
+	"h5":         true,
+	"h6":         true,
+	"head":       true,
+	"hr":         true,
+	"html":       true,
+	"i":          true,
+	"iframe":     true,
+	"img":        true,
+	"input":      true,
+	"ins":        true,
+	"kbd":        true,
+	"label":      true,
+	"legend":     true,
+	"li":         true,
+	"link":       true,
+	"map":        true,
+	"menu":       true,
+	"meta":       true,
+	"noscript":   true,
+	"object":     true,
+	"ol":         true,
+	"optgroup":   true,
+	"option":     true,
+	"p":          true,
+	"param":      true,
+	"pre":        true,
+	"q":          true,
+	"ruby":       true,
+	"s":          true,
+	"samp":       true,
+	"script":     true,
+	"select":     true,
+	"small":      true,
+	"span":       true,
+	"strong":     true,
+	"style":      true,
+	"sub":        true,
+	"sup":        true,
+	"table":      true,
+	"tbody":      true,
+	"td":         true,
+	"textarea":   true,
+	"tfoot":      true,
+	"th":         true,
+	"thead":      true,
+	"title":      true,
+	"tr":         true,
+	"u":          true,
+	"ul":         true,
+	"var":        true,
+}
+
+// This only returns true if all of these selectors are considered "safe" which
+// means that they are very likely to work in any browser a user might reasonably
+// be using. We do NOT want to merge adjacent qualified rules with the same body
+// if any of the selectors are unsafe, since then browsers which don't support
+// that particular feature would ignore the entire merged qualified rule:
+//
+//	Input:
+//	  a { color: red }
+//	  b { color: red }
+//	  input::-moz-placeholder { color: red }
+//
+//	Valid output:
+//	  a, b { color: red }
+//	  input::-moz-placeholder { color: red }
+//
+//	Invalid output:
+//	  a, b, input::-moz-placeholder { color: red }
+//
+// This considers IE 7 and above to be a browser that a user could possibly use.
+// Versions of IE less than 6 are not considered.
+func isSafeSelectors(complexSelectors []css_ast.ComplexSelector) bool {
+	for _, complex := range complexSelectors {
+		for _, compound := range complex.Selectors {
+			if compound.HasNestingSelector() {
+				// Bail because this is an extension: https://drafts.csswg.org/css-nesting-1/
+				return false
+			}
+
+			if compound.Combinator.Byte != 0 {
+				// "Before Internet Explorer 10, the combinator only works in standards mode"
+				// Reference: https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_Selectors
+				return false
+			}
+
+			if compound.TypeSelector != nil {
+				if compound.TypeSelector.NamespacePrefix != nil {
+					// Bail if we hit a namespace, which doesn't work in IE before version 9
+					// Reference: https://developer.mozilla.org/en-US/docs/Web/CSS/Type_selectors
+					return false
+				}
+
+				if compound.TypeSelector.Name.Kind == css_lexer.TIdent && !nonDeprecatedElementsSupportedByIE7[compound.TypeSelector.Name.Text] {
+					// Bail if this element is either deprecated or not supported in IE 7
+					return false
+				}
+			}
+
+			for _, ss := range compound.SubclassSelectors {
+				switch s := ss.Data.(type) {
+				case *css_ast.SSAttribute:
+					if s.MatcherModifier != 0 {
+						// Bail if we hit a case modifier, which doesn't work in IE at all
+						// Reference: https://developer.mozilla.org/en-US/docs/Web/CSS/Attribute_selectors
+						return false
+					}
+
+				case *css_ast.SSPseudoClass:
+					// Bail if this pseudo class doesn't match a hard-coded list that's
+					// known to work everywhere. For example, ":focus" doesn't work in IE 7.
+					// Reference: https://developer.mozilla.org/en-US/docs/Web/CSS/Pseudo-classes
+					if s.Args == nil && !s.IsElement {
+						switch s.Name {
+						case "active", "first-child", "hover", "link", "visited":
+							continue
+						}
+					}
+					return false
+
+				case *css_ast.SSPseudoClassWithSelectorList:
+					// These definitely don't work in IE 7
+					return false
+				}
+			}
+		}
+	}
+	return true
+}
+
+func (p *parser) parseURLOrString() (string, logger.Range, bool) {
+	t := p.current()
+	switch t.Kind {
+	case css_lexer.TString:
+		text := p.decoded()
+		p.advance()
+		return text, t.Range, true
+
+	case css_lexer.TURL:
+		text := p.decoded()
+		p.advance()
+		return text, t.Range, true
+
+	case css_lexer.TFunction:
+		if strings.EqualFold(p.decoded(), "url") {
+			matchingLoc := logger.Loc{Start: p.current().Range.End() - 1}
+			i := p.index + 1
+
+			// Skip over whitespace
+			for p.at(i).Kind == css_lexer.TWhitespace {
+				i++
+			}
+
+			// Consume a string
+			if p.at(i).Kind == css_lexer.TString {
+				stringIndex := i
+				i++
+
+				// Skip over whitespace
+				for p.at(i).Kind == css_lexer.TWhitespace {
+					i++
+				}
+
+				// Consume a closing parenthesis
+				if close := p.at(i).Kind; close == css_lexer.TCloseParen || close == css_lexer.TEndOfFile {
+					t := p.at(stringIndex)
+					text := t.DecodedText(p.source.Contents)
+					p.index = i
+					p.expectWithMatchingLoc(css_lexer.TCloseParen, matchingLoc)
+					return text, t.Range, true
+				}
+			}
+		}
+	}
+
+	return "", logger.Range{}, false
+}
+
+func (p *parser) expectURLOrString() (url string, r logger.Range, ok bool) {
+	url, r, ok = p.parseURLOrString()
+	if !ok {
+		p.expect(css_lexer.TURL)
+	}
+	return
+}
+
+type atRuleKind uint8
+
+const (
+	atRuleUnknown atRuleKind = iota
+	atRuleDeclarations
+	atRuleInheritContext
+	atRuleQualifiedOrEmpty
+	atRuleEmpty
+)
+
+var specialAtRules = map[string]atRuleKind{
+	"media":    atRuleInheritContext,
+	"supports": atRuleInheritContext,
+
+	"font-face": atRuleDeclarations,
+	"page":      atRuleDeclarations,
+
+	// These go inside "@page": https://www.w3.org/TR/css-page-3/#syntax-page-selector
+	"bottom-center":       atRuleDeclarations,
+	"bottom-left-corner":  atRuleDeclarations,
+	"bottom-left":         atRuleDeclarations,
+	"bottom-right-corner": atRuleDeclarations,
+	"bottom-right":        atRuleDeclarations,
+	"left-bottom":         atRuleDeclarations,
+	"left-middle":         atRuleDeclarations,
+	"left-top":            atRuleDeclarations,
+	"right-bottom":        atRuleDeclarations,
+	"right-middle":        atRuleDeclarations,
+	"right-top":           atRuleDeclarations,
+	"top-center":          atRuleDeclarations,
+	"top-left-corner":     atRuleDeclarations,
+	"top-left":            atRuleDeclarations,
+	"top-right-corner":    atRuleDeclarations,
+	"top-right":           atRuleDeclarations,
+
+	// These properties are very deprecated and appear to only be useful for
+	// mobile versions of internet explorer (which may no longer exist?), but
+	// they are used by the https://ant.design/ design system so we recognize
+	// them to avoid the warning.
+	//
+	//   Documentation: https://developer.mozilla.org/en-US/docs/Web/CSS/@viewport
+	//   Discussion: https://github.com/w3c/csswg-drafts/issues/4766
+	//
+	"viewport":     atRuleDeclarations,
+	"-ms-viewport": atRuleDeclarations,
+
+	// This feature has been removed from the web because it's actively harmful.
+	// However, there is one exception where "@-moz-document url-prefix() {" is
+	// accepted by Firefox to basically be an "if Firefox" conditional rule.
+	//
+	//   Documentation: https://developer.mozilla.org/en-US/docs/Web/CSS/@document
+	//   Discussion: https://bugzilla.mozilla.org/show_bug.cgi?id=1035091
+	//
+	"document":      atRuleInheritContext,
+	"-moz-document": atRuleInheritContext,
+
+	// This is a new feature that changes how the CSS rule cascade works. It can
+	// end in either a "{}" block or a ";" rule terminator so we need this special
+	// case to support both.
+	//
+	//   Documentation: https://developer.mozilla.org/en-US/docs/Web/CSS/@layer
+	//   Motivation: https://developer.chrome.com/blog/cascade-layers/
+	//
+	"layer": atRuleQualifiedOrEmpty,
+
+	// Reference: https://drafts.csswg.org/css-cascade-6/#scoped-styles
+	"scope": atRuleInheritContext,
+
+	// Reference: https://drafts.csswg.org/css-fonts-4/#font-palette-values
+	"font-palette-values": atRuleDeclarations,
+
+	// Documentation: https://developer.mozilla.org/en-US/docs/Web/CSS/@counter-style
+	// Reference: https://drafts.csswg.org/css-counter-styles/#the-counter-style-rule
+	"counter-style": atRuleDeclarations,
+
+	// Documentation: https://developer.mozilla.org/en-US/docs/Web/CSS/@font-feature-values
+	// Reference: https://drafts.csswg.org/css-fonts/#font-feature-values
+	"font-feature-values": atRuleDeclarations,
+	"annotation":          atRuleDeclarations,
+	"character-variant":   atRuleDeclarations,
+	"historical-forms":    atRuleDeclarations,
+	"ornaments":           atRuleDeclarations,
+	"styleset":            atRuleDeclarations,
+	"stylistic":           atRuleDeclarations,
+	"swash":               atRuleDeclarations,
+
+	// Container Queries
+	// Reference: https://drafts.csswg.org/css-contain-3/#container-rule
+	"container": atRuleInheritContext,
+
+	// Defining before-change style: the @starting-style rule
+	// Reference: https://drafts.csswg.org/css-transitions-2/#defining-before-change-style-the-starting-style-rule
+	"starting-style": atRuleInheritContext,
+
+	// Anchor Positioning
+	// Reference: https://drafts.csswg.org/css-anchor-position-1/#at-ruledef-position-try
+	"position-try": atRuleDeclarations,
+}
+
+var atKnownRuleCanBeRemovedIfEmpty = map[string]bool{
+	"media":     true,
+	"supports":  true,
+	"font-face": true,
+	"page":      true,
+
+	// https://www.w3.org/TR/css-page-3/#syntax-page-selector
+	"bottom-center":       true,
+	"bottom-left-corner":  true,
+	"bottom-left":         true,
+	"bottom-right-corner": true,
+	"bottom-right":        true,
+	"left-bottom":         true,
+	"left-middle":         true,
+	"left-top":            true,
+	"right-bottom":        true,
+	"right-middle":        true,
+	"right-top":           true,
+	"top-center":          true,
+	"top-left-corner":     true,
+	"top-left":            true,
+	"top-right-corner":    true,
+	"top-right":           true,
+
+	// https://drafts.csswg.org/css-cascade-6/#scoped-styles
+	"scope": true,
+
+	// https://drafts.csswg.org/css-fonts-4/#font-palette-values
+	"font-palette-values": true,
+
+	// https://drafts.csswg.org/css-contain-3/#container-rule
+	"container": true,
+}
+
+type atRuleValidity uint8
+
+const (
+	atRuleInvalid atRuleValidity = iota
+	atRuleValid
+	atRuleInvalidAfter
+)
+
+type atRuleContext struct {
+	afterLoc             logger.Loc
+	charsetValidity      atRuleValidity
+	importValidity       atRuleValidity
+	canInlineNoOpNesting bool
+	isDeclarationList    bool
+	isTopLevel           bool
+}
+
+func (p *parser) parseAtRule(context atRuleContext) css_ast.Rule {
+	// Parse the name
+	atToken := p.decoded()
+	atRange := p.current().Range
+	lowerAtToken := strings.ToLower(atToken)
+	kind := specialAtRules[lowerAtToken]
+	p.advance()
+
+	// Parse the prelude
+	preludeStart := p.index
+abortRuleParser:
+	switch lowerAtToken {
+	case "charset":
+		switch context.charsetValidity {
+		case atRuleInvalid:
+			p.log.AddID(logger.MsgID_CSS_InvalidAtCharset, logger.Warning, &p.tracker, atRange, "\"@charset\" must be the first rule in the file")
+
+		case atRuleInvalidAfter:
+			p.log.AddIDWithNotes(logger.MsgID_CSS_InvalidAtCharset, logger.Warning, &p.tracker, atRange,
+				"\"@charset\" must be the first rule in the file",
+				[]logger.MsgData{p.tracker.MsgData(logger.Range{Loc: context.afterLoc},
+					"This rule cannot come before a \"@charset\" rule")})
+
+		case atRuleValid:
+			kind = atRuleEmpty
+			p.expect(css_lexer.TWhitespace)
+			if p.peek(css_lexer.TString) {
+				encoding := p.decoded()
+				if !strings.EqualFold(encoding, "UTF-8") {
+					p.log.AddID(logger.MsgID_CSS_UnsupportedAtCharset, logger.Warning, &p.tracker, p.current().Range,
+						fmt.Sprintf("\"UTF-8\" will be used instead of unsupported charset %q", encoding))
+				}
+				p.advance()
+				p.expect(css_lexer.TSemicolon)
+				return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RAtCharset{Encoding: encoding}}
+			}
+			p.expect(css_lexer.TString)
+		}
+
+	case "import":
+		switch context.importValidity {
+		case atRuleInvalid:
+			p.log.AddID(logger.MsgID_CSS_InvalidAtImport, logger.Warning, &p.tracker, atRange, "\"@import\" is only valid at the top level")
+
+		case atRuleInvalidAfter:
+			p.log.AddIDWithNotes(logger.MsgID_CSS_InvalidAtImport, logger.Warning, &p.tracker, atRange,
+				"All \"@import\" rules must come first",
+				[]logger.MsgData{p.tracker.MsgData(logger.Range{Loc: context.afterLoc},
+					"This rule cannot come before an \"@import\" rule")})
+
+		case atRuleValid:
+			kind = atRuleEmpty
+			p.eat(css_lexer.TWhitespace)
+			if path, r, ok := p.expectURLOrString(); ok {
+				var conditions css_ast.ImportConditions
+				importConditionsStart := p.index
+				for {
+					if kind := p.current().Kind; kind == css_lexer.TSemicolon || kind == css_lexer.TOpenBrace ||
+						kind == css_lexer.TCloseBrace || kind == css_lexer.TEndOfFile {
+						break
+					}
+					p.parseComponentValue()
+				}
+				if p.current().Kind == css_lexer.TOpenBrace {
+					break // Avoid parsing an invalid "@import" rule
+				}
+				conditions.Media = p.convertTokens(p.tokens[importConditionsStart:p.index])
+
+				// Insert or remove whitespace before the first token
+				var importConditions *css_ast.ImportConditions
+				if len(conditions.Media) > 0 {
+					importConditions = &conditions
+
+					// Handle "layer()"
+					if t := conditions.Media[0]; (t.Kind == css_lexer.TIdent || t.Kind == css_lexer.TFunction) && strings.EqualFold(t.Text, "layer") {
+						conditions.Layers = conditions.Media[:1]
+						conditions.Media = conditions.Media[1:]
+					}
+
+					// Handle "supports()"
+					if len(conditions.Media) > 0 {
+						if t := conditions.Media[0]; t.Kind == css_lexer.TFunction && strings.EqualFold(t.Text, "supports") {
+							conditions.Supports = conditions.Media[:1]
+							conditions.Media = conditions.Media[1:]
+						}
+					}
+
+					// Remove leading and trailing whitespace
+					if len(conditions.Layers) > 0 {
+						conditions.Layers[0].Whitespace &= ^(css_ast.WhitespaceBefore | css_ast.WhitespaceAfter)
+					}
+					if len(conditions.Supports) > 0 {
+						conditions.Supports[0].Whitespace &= ^(css_ast.WhitespaceBefore | css_ast.WhitespaceAfter)
+					}
+					if n := len(conditions.Media); n > 0 {
+						conditions.Media[0].Whitespace &= ^css_ast.WhitespaceBefore
+						conditions.Media[n-1].Whitespace &= ^css_ast.WhitespaceAfter
+					}
+				}
+
+				p.expect(css_lexer.TSemicolon)
+				importRecordIndex := uint32(len(p.importRecords))
+				p.importRecords = append(p.importRecords, ast.ImportRecord{
+					Kind:  ast.ImportAt,
+					Path:  logger.Path{Text: path},
+					Range: r,
+				})
+
+				// Fill in the pre-import layers once we see the first "@import"
+				if !p.hasSeenAtImport {
+					p.hasSeenAtImport = true
+					p.layersPreImport = p.layersPostImport
+					p.layersPostImport = nil
+				}
+
+				return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RAtImport{
+					ImportRecordIndex: importRecordIndex,
+					ImportConditions:  importConditions,
+				}}
+			}
+		}
+
+	case "keyframes", "-webkit-keyframes", "-moz-keyframes", "-ms-keyframes", "-o-keyframes":
+		p.eat(css_lexer.TWhitespace)
+		nameLoc := p.current().Range.Loc
+		var name string
+
+		if p.peek(css_lexer.TIdent) {
+			name = p.decoded()
+			if isInvalidAnimationName(name) {
+				msg := logger.Msg{
+					ID:    logger.MsgID_CSS_CSSSyntaxError,
+					Kind:  logger.Warning,
+					Data:  p.tracker.MsgData(p.current().Range, fmt.Sprintf("Cannot use %q as a name for \"@keyframes\" without quotes", name)),
+					Notes: []logger.MsgData{{Text: fmt.Sprintf("You can put %q in quotes to prevent it from becoming a CSS keyword.", name)}},
+				}
+				msg.Data.Location.Suggestion = fmt.Sprintf("%q", name)
+				p.log.AddMsg(msg)
+				break
+			}
+			p.advance()
+		} else if p.peek(css_lexer.TString) {
+			// Note: Strings as names is allowed in the CSS specification and works in
+			// Firefox and Safari but Chrome has strangely decided to deliberately not
+			// support this. We always turn all string names into identifiers to avoid
+			// them silently breaking in Chrome.
+			name = p.decoded()
+			p.advance()
+			if !p.makeLocalSymbols && isInvalidAnimationName(name) {
+				break
+			}
+		} else if !p.expect(css_lexer.TIdent) {
+			break
+		}
+
+		p.eat(css_lexer.TWhitespace)
+		blockStart := p.index
+
+		matchingLoc := p.current().Range.Loc
+		if p.expect(css_lexer.TOpenBrace) {
+			var blocks []css_ast.KeyframeBlock
+
+		badSyntax:
+			for {
+				switch p.current().Kind {
+				case css_lexer.TWhitespace:
+					p.advance()
+					continue
+
+				case css_lexer.TCloseBrace:
+					closeBraceLoc := p.current().Range.Loc
+					p.advance()
+					return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RAtKeyframes{
+						AtToken:       atToken,
+						Name:          p.symbolForName(nameLoc, name),
+						Blocks:        blocks,
+						CloseBraceLoc: closeBraceLoc,
+					}}
+
+				case css_lexer.TEndOfFile:
+					break badSyntax
+
+				case css_lexer.TOpenBrace:
+					p.expect(css_lexer.TPercentage)
+					break badSyntax
+
+				default:
+					var selectors []string
+					var firstSelectorLoc logger.Loc
+
+				selectors:
+					for {
+						t := p.current()
+						switch t.Kind {
+						case css_lexer.TWhitespace:
+							p.advance()
+							continue
+
+						case css_lexer.TOpenBrace:
+							blockMatchingLoc := p.current().Range.Loc
+							p.advance()
+							rules := p.parseListOfDeclarations(listOfDeclarationsOpts{})
+							closeBraceLoc := p.current().Range.Loc
+							if !p.expectWithMatchingLoc(css_lexer.TCloseBrace, blockMatchingLoc) {
+								closeBraceLoc = logger.Loc{}
+							}
+
+							// "@keyframes { from {} to { color: red } }" => "@keyframes { to { color: red } }"
+							if !p.options.minifySyntax || len(rules) > 0 {
+								blocks = append(blocks, css_ast.KeyframeBlock{
+									Selectors:     selectors,
+									Rules:         rules,
+									Loc:           firstSelectorLoc,
+									CloseBraceLoc: closeBraceLoc,
+								})
+							}
+							break selectors
+
+						case css_lexer.TCloseBrace, css_lexer.TEndOfFile:
+							p.expect(css_lexer.TOpenBrace)
+							break badSyntax
+
+						case css_lexer.TIdent, css_lexer.TPercentage:
+							if firstSelectorLoc.Start == 0 {
+								firstSelectorLoc = p.current().Range.Loc
+							}
+							text := p.decoded()
+							if t.Kind == css_lexer.TIdent {
+								if strings.EqualFold(text, "from") {
+									if p.options.minifySyntax {
+										text = "0%" // "0%" is equivalent to but shorter than "from"
+									}
+								} else if !strings.EqualFold(text, "to") {
+									p.expect(css_lexer.TPercentage)
+								}
+							} else if p.options.minifySyntax && text == "100%" {
+								text = "to" // "to" is equivalent to but shorter than "100%"
+							}
+							selectors = append(selectors, text)
+							p.advance()
+
+							// Keyframe selectors are comma-separated
+							p.eat(css_lexer.TWhitespace)
+							if p.eat(css_lexer.TComma) {
+								p.eat(css_lexer.TWhitespace)
+								if k := p.current().Kind; k != css_lexer.TIdent && k != css_lexer.TPercentage {
+									p.expect(css_lexer.TPercentage)
+									break badSyntax
+								}
+							} else if k := p.current().Kind; k != css_lexer.TOpenBrace && k != css_lexer.TCloseBrace && k != css_lexer.TEndOfFile {
+								p.expect(css_lexer.TComma)
+								break badSyntax
+							}
+
+						default:
+							p.expect(css_lexer.TPercentage)
+							break badSyntax
+						}
+					}
+				}
+			}
+
+			// Otherwise, finish parsing the body and return an unknown rule
+			for !p.peek(css_lexer.TCloseBrace) && !p.peek(css_lexer.TEndOfFile) {
+				p.parseComponentValue()
+			}
+			p.expectWithMatchingLoc(css_lexer.TCloseBrace, matchingLoc)
+			prelude := p.convertTokens(p.tokens[preludeStart:blockStart])
+			block, _ := p.convertTokensHelper(p.tokens[blockStart:p.index], css_lexer.TEndOfFile, convertTokensOpts{allowImports: true})
+			return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RUnknownAt{AtToken: atToken, Prelude: prelude, Block: block}}
+		}
+
+	case "layer":
+		// Reference: https://developer.mozilla.org/en-US/docs/Web/CSS/@layer
+
+		// Read the layer name list
+		var names [][]string
+		p.eat(css_lexer.TWhitespace)
+		if p.peek(css_lexer.TIdent) {
+			for {
+				ident, ok := p.expectValidLayerNameIdent()
+				if !ok {
+					break abortRuleParser
+				}
+				name := []string{ident}
+				for {
+					p.eat(css_lexer.TWhitespace)
+					if !p.eat(css_lexer.TDelimDot) {
+						break
+					}
+					p.eat(css_lexer.TWhitespace)
+					ident, ok := p.expectValidLayerNameIdent()
+					if !ok {
+						break abortRuleParser
+					}
+					name = append(name, ident)
+				}
+				names = append(names, name)
+				p.eat(css_lexer.TWhitespace)
+				if !p.eat(css_lexer.TComma) {
+					break
+				}
+				p.eat(css_lexer.TWhitespace)
+			}
+		}
+
+		// Read the optional block
+		matchingLoc := p.current().Range.Loc
+		if len(names) <= 1 && p.eat(css_lexer.TOpenBrace) {
+			p.recordAtLayerRule(names)
+			oldEnclosingLayer := p.enclosingLayer
+			if len(names) == 1 {
+				p.enclosingLayer = append(p.enclosingLayer, names[0]...)
+			} else {
+				p.anonLayerCount++
+			}
+			var rules []css_ast.Rule
+			if context.isDeclarationList {
+				rules = p.parseListOfDeclarations(listOfDeclarationsOpts{
+					canInlineNoOpNesting: context.canInlineNoOpNesting,
+				})
+			} else {
+				rules = p.parseListOfRules(ruleContext{
+					parseSelectors: true,
+				})
+			}
+			if len(names) != 1 {
+				p.anonLayerCount--
+			}
+			p.enclosingLayer = oldEnclosingLayer
+			closeBraceLoc := p.current().Range.Loc
+			if !p.expectWithMatchingLoc(css_lexer.TCloseBrace, matchingLoc) {
+				closeBraceLoc = logger.Loc{}
+			}
+			return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RAtLayer{Names: names, Rules: rules, CloseBraceLoc: closeBraceLoc}}
+		}
+
+		// Handle lack of a block
+		if len(names) >= 1 && p.eat(css_lexer.TSemicolon) {
+			p.recordAtLayerRule(names)
+			return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RAtLayer{Names: names}}
+		}
+
+		// Otherwise there's some kind of syntax error
+		switch p.current().Kind {
+		case css_lexer.TEndOfFile:
+			p.expect(css_lexer.TSemicolon)
+			p.recordAtLayerRule(names)
+			return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RAtLayer{Names: names}}
+
+		case css_lexer.TCloseBrace:
+			p.expect(css_lexer.TSemicolon)
+			if !context.isTopLevel {
+				p.recordAtLayerRule(names)
+				return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RAtLayer{Names: names}}
+			}
+
+		case css_lexer.TOpenBrace:
+			p.expect(css_lexer.TSemicolon)
+
+		default:
+			p.unexpected()
+		}
+
+	default:
+		if kind == atRuleUnknown && lowerAtToken == "namespace" {
+			// CSS namespaces are a weird feature that appears to only really be
+			// useful for styling XML. And the world has moved on from XHTML to
+			// HTML5 so pretty much no one uses CSS namespaces anymore. They are
+			// also complicated to support in a bundler because CSS namespaces are
+			// file-scoped, which means:
+			//
+			// * Default namespaces can be different in different files, in which
+			//   case some default namespaces would have to be converted to prefixed
+			//   namespaces to avoid collisions.
+			//
+			// * Prefixed namespaces from different files can use the same name, in
+			//   which case some prefixed namespaces would need to be renamed to
+			//   avoid collisions.
+			//
+			// Instead of implementing all of that for an extremely obscure feature,
+			// CSS namespaces are just explicitly not supported.
+			p.log.AddID(logger.MsgID_CSS_UnsupportedAtNamespace, logger.Warning, &p.tracker, atRange, "\"@namespace\" rules are not supported")
+		}
+	}
+
+	// Parse an unknown prelude
+prelude:
+	for {
+		switch p.current().Kind {
+		case css_lexer.TOpenBrace, css_lexer.TEndOfFile:
+			break prelude
+
+		case css_lexer.TSemicolon, css_lexer.TCloseBrace:
+			prelude := p.convertTokens(p.tokens[preludeStart:p.index])
+
+			switch kind {
+			case atRuleQualifiedOrEmpty:
+				// Parse a known at rule below
+				break prelude
+
+			case atRuleEmpty, atRuleUnknown:
+				// Parse an unknown at rule
+				p.expect(css_lexer.TSemicolon)
+				return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RUnknownAt{AtToken: atToken, Prelude: prelude}}
+
+			default:
+				// Report an error for rules that should have blocks
+				p.expect(css_lexer.TOpenBrace)
+				p.eat(css_lexer.TSemicolon)
+				return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RUnknownAt{AtToken: atToken, Prelude: prelude}}
+			}
+
+		default:
+			p.parseComponentValue()
+		}
+	}
+	prelude := p.convertTokens(p.tokens[preludeStart:p.index])
+	blockStart := p.index
+
+	switch kind {
+	case atRuleEmpty:
+		// Report an error for rules that shouldn't have blocks
+		p.expect(css_lexer.TSemicolon)
+		p.parseBlock(css_lexer.TOpenBrace, css_lexer.TCloseBrace)
+		block := p.convertTokens(p.tokens[blockStart:p.index])
+		return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RUnknownAt{AtToken: atToken, Prelude: prelude, Block: block}}
+
+	case atRuleDeclarations:
+		// Parse known rules whose blocks always consist of declarations
+		matchingLoc := p.current().Range.Loc
+		p.expect(css_lexer.TOpenBrace)
+		rules := p.parseListOfDeclarations(listOfDeclarationsOpts{})
+		closeBraceLoc := p.current().Range.Loc
+		if !p.expectWithMatchingLoc(css_lexer.TCloseBrace, matchingLoc) {
+			closeBraceLoc = logger.Loc{}
+		}
+
+		// Handle local names for "@counter-style"
+		if len(prelude) == 1 && lowerAtToken == "counter-style" {
+			if t := &prelude[0]; t.Kind == css_lexer.TIdent {
+				t.Kind = css_lexer.TSymbol
+				t.PayloadIndex = p.symbolForName(t.Loc, t.Text).Ref.InnerIndex
+			}
+		}
+
+		return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RKnownAt{AtToken: atToken, Prelude: prelude, Rules: rules, CloseBraceLoc: closeBraceLoc}}
+
+	case atRuleInheritContext:
+		// Parse known rules whose blocks consist of whatever the current context is
+		matchingLoc := p.current().Range.Loc
+		p.expect(css_lexer.TOpenBrace)
+		var rules []css_ast.Rule
+
+		// Push the "@media" conditions
+		isAtMedia := lowerAtToken == "media"
+		if isAtMedia {
+			p.enclosingAtMedia = append(p.enclosingAtMedia, prelude)
+		}
+
+		// Parse the block for this rule
+		if context.isDeclarationList {
+			rules = p.parseListOfDeclarations(listOfDeclarationsOpts{
+				canInlineNoOpNesting: context.canInlineNoOpNesting,
+			})
+		} else {
+			rules = p.parseListOfRules(ruleContext{
+				parseSelectors: true,
+			})
+		}
+
+		// Pop the "@media" conditions
+		if isAtMedia {
+			p.enclosingAtMedia = p.enclosingAtMedia[:len(p.enclosingAtMedia)-1]
+		}
+
+		closeBraceLoc := p.current().Range.Loc
+		if !p.expectWithMatchingLoc(css_lexer.TCloseBrace, matchingLoc) {
+			closeBraceLoc = logger.Loc{}
+		}
+
+		// Handle local names for "@container"
+		if len(prelude) >= 1 && lowerAtToken == "container" {
+			if t := &prelude[0]; t.Kind == css_lexer.TIdent && strings.ToLower(t.Text) != "not" {
+				t.Kind = css_lexer.TSymbol
+				t.PayloadIndex = p.symbolForName(t.Loc, t.Text).Ref.InnerIndex
+			}
+		}
+
+		return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RKnownAt{AtToken: atToken, Prelude: prelude, Rules: rules, CloseBraceLoc: closeBraceLoc}}
+
+	case atRuleQualifiedOrEmpty:
+		matchingLoc := p.current().Range.Loc
+		if p.eat(css_lexer.TOpenBrace) {
+			rules := p.parseListOfRules(ruleContext{
+				parseSelectors: true,
+			})
+			closeBraceLoc := p.current().Range.Loc
+			if !p.expectWithMatchingLoc(css_lexer.TCloseBrace, matchingLoc) {
+				closeBraceLoc = logger.Loc{}
+			}
+			return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RKnownAt{AtToken: atToken, Prelude: prelude, Rules: rules, CloseBraceLoc: closeBraceLoc}}
+		}
+		p.expect(css_lexer.TSemicolon)
+		return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RKnownAt{AtToken: atToken, Prelude: prelude}}
+
+	default:
+		// Otherwise, parse an unknown rule
+		p.parseBlock(css_lexer.TOpenBrace, css_lexer.TCloseBrace)
+		block, _ := p.convertTokensHelper(p.tokens[blockStart:p.index], css_lexer.TEndOfFile, convertTokensOpts{allowImports: true})
+		return css_ast.Rule{Loc: atRange.Loc, Data: &css_ast.RUnknownAt{AtToken: atToken, Prelude: prelude, Block: block}}
+	}
+}
+
+func (p *parser) expectValidLayerNameIdent() (string, bool) {
+	r := p.current().Range
+	text := p.decoded()
+	if !p.expect(css_lexer.TIdent) {
+		return "", false
+	}
+	switch text {
+	case "initial", "inherit", "unset":
+		p.log.AddID(logger.MsgID_CSS_InvalidAtLayer, logger.Warning, &p.tracker, r, fmt.Sprintf("%q cannot be used as a layer name", text))
+		p.prevError = r.Loc
+		return "", false
+	}
+	return text, true
+}
+
+func (p *parser) convertTokens(tokens []css_lexer.Token) []css_ast.Token {
+	result, _ := p.convertTokensHelper(tokens, css_lexer.TEndOfFile, convertTokensOpts{})
+	return result
+}
+
+type convertTokensOpts struct {
+	allowImports         bool
+	verbatimWhitespace   bool
+	isInsideCalcFunction bool
+}
+
+func (p *parser) convertTokensHelper(tokens []css_lexer.Token, close css_lexer.T, opts convertTokensOpts) ([]css_ast.Token, []css_lexer.Token) {
+	result := []css_ast.Token{}
+	var nextWhitespace css_ast.WhitespaceFlags
+
+	// Enable verbatim whitespace mode when the first two non-whitespace tokens
+	// are a CSS variable name followed by a colon. This is because it could be
+	// a form of CSS variable usage, and removing whitespace could potentially
+	// break this usage. For example, the following CSS is ignored by Chrome if
+	// the whitespace isn't preserved:
+	//
+	//   @supports (--foo: ) {
+	//     html { background: green; }
+	//   }
+	//
+	// Strangely whitespace removal doesn't cause the declaration to be ignored
+	// in Firefox or Safari, so there's definitely a browser bug somewhere.
+	if !opts.verbatimWhitespace {
+		for i, t := range tokens {
+			if t.Kind == css_lexer.TWhitespace {
+				continue
+			}
+			if t.Kind == css_lexer.TIdent && strings.HasPrefix(t.DecodedText(p.source.Contents), "--") {
+				for _, t := range tokens[i+1:] {
+					if t.Kind == css_lexer.TWhitespace {
+						continue
+					}
+					if t.Kind == css_lexer.TColon {
+						opts.verbatimWhitespace = true
+					}
+					break
+				}
+			}
+			break
+		}
+	}
+
+loop:
+	for len(tokens) > 0 {
+		t := tokens[0]
+		tokens = tokens[1:]
+		if t.Kind == close {
+			break loop
+		}
+		token := css_ast.Token{
+			Loc:        t.Range.Loc,
+			Kind:       t.Kind,
+			Text:       t.DecodedText(p.source.Contents),
+			Whitespace: nextWhitespace,
+		}
+		nextWhitespace = 0
+
+		// Warn about invalid "+" and "-" operators that break the containing "calc()"
+		if opts.isInsideCalcFunction && t.Kind.IsNumeric() && len(result) > 0 && result[len(result)-1].Kind.IsNumeric() &&
+			(strings.HasPrefix(token.Text, "+") || strings.HasPrefix(token.Text, "-")) {
+			// "calc(1+2)" and "calc(1-2)" are invalid
+			p.log.AddID(logger.MsgID_CSS_InvalidCalc, logger.Warning, &p.tracker, logger.Range{Loc: t.Range.Loc, Len: 1},
+				fmt.Sprintf("The %q operator only works if there is whitespace on both sides", token.Text[:1]))
+		}
+
+		switch t.Kind {
+		case css_lexer.TWhitespace:
+			if last := len(result) - 1; last >= 0 {
+				result[last].Whitespace |= css_ast.WhitespaceAfter
+			}
+			nextWhitespace = css_ast.WhitespaceBefore
+			continue
+
+		case css_lexer.TDelimPlus, css_lexer.TDelimMinus:
+			// Warn about invalid "+" and "-" operators that break the containing "calc()"
+			if opts.isInsideCalcFunction && len(tokens) > 0 {
+				if len(result) == 0 || result[len(result)-1].Kind == css_lexer.TComma {
+					// "calc(-(1 + 2))" is invalid
+					p.log.AddID(logger.MsgID_CSS_InvalidCalc, logger.Warning, &p.tracker, t.Range,
+						fmt.Sprintf("%q can only be used as an infix operator, not a prefix operator", token.Text))
+				} else if token.Whitespace != css_ast.WhitespaceBefore || tokens[0].Kind != css_lexer.TWhitespace {
+					// "calc(1- 2)" and "calc(1 -(2))" are invalid
+					p.log.AddID(logger.MsgID_CSS_InvalidCalc, logger.Warning, &p.tracker, t.Range,
+						fmt.Sprintf("The %q operator only works if there is whitespace on both sides", token.Text))
+				}
+			}
+
+		case css_lexer.TNumber:
+			if p.options.minifySyntax {
+				if text, ok := mangleNumber(token.Text); ok {
+					token.Text = text
+				}
+			}
+
+		case css_lexer.TPercentage:
+			if p.options.minifySyntax {
+				if text, ok := mangleNumber(token.PercentageValue()); ok {
+					token.Text = text + "%"
+				}
+			}
+
+		case css_lexer.TDimension:
+			token.UnitOffset = t.UnitOffset
+
+			if p.options.minifySyntax {
+				if text, ok := mangleNumber(token.DimensionValue()); ok {
+					token.Text = text + token.DimensionUnit()
+					token.UnitOffset = uint16(len(text))
+				}
+
+				if value, unit, ok := mangleDimension(token.DimensionValue(), token.DimensionUnit()); ok {
+					token.Text = value + unit
+					token.UnitOffset = uint16(len(value))
+				}
+			}
+
+		case css_lexer.TURL:
+			token.PayloadIndex = uint32(len(p.importRecords))
+			var flags ast.ImportRecordFlags
+			if !opts.allowImports {
+				flags |= ast.IsUnused
+			}
+			p.importRecords = append(p.importRecords, ast.ImportRecord{
+				Kind:  ast.ImportURL,
+				Path:  logger.Path{Text: token.Text},
+				Range: t.Range,
+				Flags: flags,
+			})
+			token.Text = ""
+
+		case css_lexer.TFunction:
+			var nested []css_ast.Token
+			original := tokens
+			nestedOpts := opts
+			if strings.EqualFold(token.Text, "var") {
+				// CSS variables require verbatim whitespace for correctness
+				nestedOpts.verbatimWhitespace = true
+			}
+			if strings.EqualFold(token.Text, "calc") {
+				nestedOpts.isInsideCalcFunction = true
+			}
+			nested, tokens = p.convertTokensHelper(tokens, css_lexer.TCloseParen, nestedOpts)
+			token.Children = &nested
+
+			// Apply "calc" simplification rules when minifying
+			if p.options.minifySyntax && strings.EqualFold(token.Text, "calc") {
+				token = p.tryToReduceCalcExpression(token)
+			}
+
+			// Treat a URL function call with a string just like a URL token
+			if strings.EqualFold(token.Text, "url") && len(nested) == 1 && nested[0].Kind == css_lexer.TString {
+				token.Kind = css_lexer.TURL
+				token.Text = ""
+				token.Children = nil
+				token.PayloadIndex = uint32(len(p.importRecords))
+				var flags ast.ImportRecordFlags
+				if !opts.allowImports {
+					flags |= ast.IsUnused
+				}
+				p.importRecords = append(p.importRecords, ast.ImportRecord{
+					Kind:  ast.ImportURL,
+					Path:  logger.Path{Text: nested[0].Text},
+					Range: original[0].Range,
+					Flags: flags,
+				})
+			}
+
+		case css_lexer.TOpenParen:
+			var nested []css_ast.Token
+			nested, tokens = p.convertTokensHelper(tokens, css_lexer.TCloseParen, opts)
+			token.Children = &nested
+
+		case css_lexer.TOpenBrace:
+			var nested []css_ast.Token
+			nested, tokens = p.convertTokensHelper(tokens, css_lexer.TCloseBrace, opts)
+
+			// Pretty-printing: insert leading and trailing whitespace when not minifying
+			if !opts.verbatimWhitespace && !p.options.minifyWhitespace && len(nested) > 0 {
+				nested[0].Whitespace |= css_ast.WhitespaceBefore
+				nested[len(nested)-1].Whitespace |= css_ast.WhitespaceAfter
+			}
+
+			token.Children = &nested
+
+		case css_lexer.TOpenBracket:
+			var nested []css_ast.Token
+			nested, tokens = p.convertTokensHelper(tokens, css_lexer.TCloseBracket, opts)
+			token.Children = &nested
+		}
+
+		result = append(result, token)
+	}
+
+	if !opts.verbatimWhitespace {
+		for i := range result {
+			token := &result[i]
+
+			// Always remove leading and trailing whitespace
+			if i == 0 {
+				token.Whitespace &= ^css_ast.WhitespaceBefore
+			}
+			if i+1 == len(result) {
+				token.Whitespace &= ^css_ast.WhitespaceAfter
+			}
+
+			switch token.Kind {
+			case css_lexer.TComma:
+				// Assume that whitespace can always be removed before a comma
+				token.Whitespace &= ^css_ast.WhitespaceBefore
+				if i > 0 {
+					result[i-1].Whitespace &= ^css_ast.WhitespaceAfter
+				}
+
+				// Assume whitespace can always be added after a comma
+				if p.options.minifyWhitespace {
+					token.Whitespace &= ^css_ast.WhitespaceAfter
+					if i+1 < len(result) {
+						result[i+1].Whitespace &= ^css_ast.WhitespaceBefore
+					}
+				} else {
+					token.Whitespace |= css_ast.WhitespaceAfter
+					if i+1 < len(result) {
+						result[i+1].Whitespace |= css_ast.WhitespaceBefore
+					}
+				}
+			}
+		}
+	}
+
+	// Insert an explicit whitespace token if we're in verbatim mode and all
+	// tokens were whitespace. In this case there is no token to attach the
+	// whitespace before/after flags so this is the only way to represent this.
+	// This is the only case where this function generates an explicit whitespace
+	// token. It represents whitespace as flags in all other cases.
+	if opts.verbatimWhitespace && len(result) == 0 && nextWhitespace == css_ast.WhitespaceBefore {
+		result = append(result, css_ast.Token{
+			Kind: css_lexer.TWhitespace,
+		})
+	}
+
+	return result, tokens
+}
+
+func shiftDot(text string, dotOffset int) (string, bool) {
+	// This doesn't handle numbers with exponents
+	if strings.ContainsAny(text, "eE") {
+		return "", false
+	}
+
+	// Handle a leading sign
+	sign := ""
+	if len(text) > 0 && (text[0] == '-' || text[0] == '+') {
+		sign = text[:1]
+		text = text[1:]
+	}
+
+	// Remove the dot
+	dot := strings.IndexByte(text, '.')
+	if dot == -1 {
+		dot = len(text)
+	} else {
+		text = text[:dot] + text[dot+1:]
+	}
+
+	// Move the dot
+	dot += dotOffset
+
+	// Remove any leading zeros before the dot
+	for len(text) > 0 && dot > 0 && text[0] == '0' {
+		text = text[1:]
+		dot--
+	}
+
+	// Remove any trailing zeros after the dot
+	for len(text) > 0 && len(text) > dot && text[len(text)-1] == '0' {
+		text = text[:len(text)-1]
+	}
+
+	// Does this number have no fractional component?
+	if dot >= len(text) {
+		trailingZeros := strings.Repeat("0", dot-len(text))
+		return fmt.Sprintf("%s%s%s", sign, text, trailingZeros), true
+	}
+
+	// Potentially add leading zeros
+	if dot < 0 {
+		text = strings.Repeat("0", -dot) + text
+		dot = 0
+	}
+
+	// Insert the dot again
+	return fmt.Sprintf("%s%s.%s", sign, text[:dot], text[dot:]), true
+}
+
+func mangleDimension(value string, unit string) (string, string, bool) {
+	const msLen = 2
+	const sLen = 1
+
+	// Mangle times: https://developer.mozilla.org/en-US/docs/Web/CSS/time
+	if strings.EqualFold(unit, "ms") {
+		if shifted, ok := shiftDot(value, -3); ok && len(shifted)+sLen < len(value)+msLen {
+			// Convert "ms" to "s" if shorter
+			return shifted, "s", true
+		}
+	}
+	if strings.EqualFold(unit, "s") {
+		if shifted, ok := shiftDot(value, 3); ok && len(shifted)+msLen < len(value)+sLen {
+			// Convert "s" to "ms" if shorter
+			return shifted, "ms", true
+		}
+	}
+
+	return "", "", false
+}
+
+func mangleNumber(t string) (string, bool) {
+	original := t
+
+	if dot := strings.IndexByte(t, '.'); dot != -1 {
+		// Remove trailing zeros
+		for len(t) > 0 && t[len(t)-1] == '0' {
+			t = t[:len(t)-1]
+		}
+
+		// Remove the decimal point if it's unnecessary
+		if dot+1 == len(t) {
+			t = t[:dot]
+			if t == "" || t == "+" || t == "-" {
+				t += "0"
+			}
+		} else {
+			// Remove a leading zero
+			if len(t) >= 3 && t[0] == '0' && t[1] == '.' && t[2] >= '0' && t[2] <= '9' {
+				t = t[1:]
+			} else if len(t) >= 4 && (t[0] == '+' || t[0] == '-') && t[1] == '0' && t[2] == '.' && t[3] >= '0' && t[3] <= '9' {
+				t = t[0:1] + t[2:]
+			}
+		}
+	}
+
+	return t, t != original
+}
+
+func (p *parser) parseSelectorRule(isTopLevel bool, opts parseSelectorOpts) css_ast.Rule {
+	// Save and restore the local symbol state in case there are any bare
+	// ":global" or ":local" annotations. The effect of these should be scoped
+	// to within the selector rule.
+	local := p.makeLocalSymbols
+	preludeStart := p.index
+
+	// Try parsing the prelude as a selector list
+	if list, ok := p.parseSelectorList(opts); ok {
+		canInlineNoOpNesting := true
+		for _, sel := range list {
+			// We cannot transform the CSS "a, b::before { & { color: red } }" into
+			// "a, b::before { color: red }" because it's basically equivalent to
+			// ":is(a, b::before) { color: red }" which only applies to "a", not to
+			// "b::before" because pseudo-elements are not valid within :is():
+			// https://www.w3.org/TR/selectors-4/#matches-pseudo. This restriction
+			// may be relaxed in the future, but this restriction hash shipped so
+			// we're stuck with it: https://github.com/w3c/csswg-drafts/issues/7433.
+			if sel.UsesPseudoElement() {
+				canInlineNoOpNesting = false
+				break
+			}
+		}
+		selector := css_ast.RSelector{Selectors: list}
+		matchingLoc := p.current().Range.Loc
+		if p.expect(css_lexer.TOpenBrace) {
+			p.inSelectorSubtree++
+			declOpts := listOfDeclarationsOpts{
+				canInlineNoOpNesting: canInlineNoOpNesting,
+			}
+
+			// Prepare for "composes" declarations
+			if opts.composesContext != nil && len(list) == 1 && len(list[0].Selectors) == 1 && list[0].Selectors[0].IsSingleAmpersand() {
+				// Support code like this:
+				//
+				//   .foo {
+				//     :local { composes: bar }
+				//     :global { composes: baz }
+				//   }
+				//
+				declOpts.composesContext = opts.composesContext
+			} else {
+				composesContext := composesContext{parentRange: list[0].Selectors[0].Range()}
+				if opts.composesContext != nil {
+					composesContext.problemRange = opts.composesContext.parentRange
+				}
+				for _, sel := range list {
+					first := sel.Selectors[0]
+					if first.Combinator.Byte != 0 {
+						composesContext.problemRange = logger.Range{Loc: first.Combinator.Loc, Len: 1}
+					} else if first.TypeSelector != nil {
+						composesContext.problemRange = first.TypeSelector.Range()
+					} else if first.NestingSelectorLoc.IsValid() {
+						composesContext.problemRange = logger.Range{Loc: logger.Loc{Start: int32(first.NestingSelectorLoc.GetIndex())}, Len: 1}
+					} else {
+						for i, ss := range first.SubclassSelectors {
+							class, ok := ss.Data.(*css_ast.SSClass)
+							if i > 0 || !ok {
+								composesContext.problemRange = ss.Range
+							} else {
+								composesContext.parentRefs = append(composesContext.parentRefs, class.Name.Ref)
+							}
+						}
+					}
+					if composesContext.problemRange.Len > 0 {
+						break
+					}
+					if len(sel.Selectors) > 1 {
+						composesContext.problemRange = sel.Selectors[1].Range()
+						break
+					}
+				}
+				declOpts.composesContext = &composesContext
+			}
+
+			selector.Rules = p.parseListOfDeclarations(declOpts)
+			p.inSelectorSubtree--
+			closeBraceLoc := p.current().Range.Loc
+			if p.expectWithMatchingLoc(css_lexer.TCloseBrace, matchingLoc) {
+				selector.CloseBraceLoc = closeBraceLoc
+			}
+			p.makeLocalSymbols = local
+			return css_ast.Rule{Loc: p.tokens[preludeStart].Range.Loc, Data: &selector}
+		}
+	}
+
+	p.makeLocalSymbols = local
+	p.index = preludeStart
+
+	// Otherwise, parse a generic qualified rule
+	return p.parseQualifiedRule(parseQualifiedRuleOpts{
+		isAlreadyInvalid:     true,
+		isTopLevel:           isTopLevel,
+		isDeclarationContext: opts.isDeclarationContext,
+	})
+}
+
+type parseQualifiedRuleOpts struct {
+	isAlreadyInvalid     bool
+	isTopLevel           bool
+	isDeclarationContext bool
+}
+
+func (p *parser) parseQualifiedRule(opts parseQualifiedRuleOpts) css_ast.Rule {
+	preludeStart := p.index
+	preludeLoc := p.current().Range.Loc
+
+loop:
+	for {
+		switch p.current().Kind {
+		case css_lexer.TOpenBrace, css_lexer.TEndOfFile:
+			break loop
+
+		case css_lexer.TCloseBrace:
+			if !opts.isTopLevel {
+				break loop
+			}
+
+		case css_lexer.TSemicolon:
+			if opts.isDeclarationContext {
+				return css_ast.Rule{Loc: preludeLoc, Data: &css_ast.RBadDeclaration{
+					Tokens: p.convertTokens(p.tokens[preludeStart:p.index]),
+				}}
+			}
+		}
+
+		p.parseComponentValue()
+	}
+
+	qualified := css_ast.RQualified{
+		Prelude: p.convertTokens(p.tokens[preludeStart:p.index]),
+	}
+
+	matchingLoc := p.current().Range.Loc
+	if p.eat(css_lexer.TOpenBrace) {
+		qualified.Rules = p.parseListOfDeclarations(listOfDeclarationsOpts{})
+		closeBraceLoc := p.current().Range.Loc
+		if p.expectWithMatchingLoc(css_lexer.TCloseBrace, matchingLoc) {
+			qualified.CloseBraceLoc = closeBraceLoc
+		}
+	} else if !opts.isAlreadyInvalid {
+		p.expect(css_lexer.TOpenBrace)
+	}
+
+	return css_ast.Rule{Loc: preludeLoc, Data: &qualified}
+}
+
+type endOfRuleScan uint8
+
+const (
+	endOfRuleUnknown endOfRuleScan = iota
+	endOfRuleSemicolon
+	endOfRuleOpenBrace
+)
+
+// Note: This was a late change to the CSS nesting syntax.
+// See also: https://github.com/w3c/csswg-drafts/issues/7961
+func (p *parser) scanForEndOfRule() (endOfRuleScan, int) {
+	var initialStack [4]css_lexer.T
+	stack := initialStack[:0]
+
+	for i, t := range p.tokens[p.index:] {
+		switch t.Kind {
+		case css_lexer.TSemicolon:
+			if len(stack) == 0 {
+				return endOfRuleSemicolon, p.index + i
+			}
+
+		case css_lexer.TFunction, css_lexer.TOpenParen:
+			stack = append(stack, css_lexer.TCloseParen)
+
+		case css_lexer.TOpenBracket:
+			stack = append(stack, css_lexer.TCloseBracket)
+
+		case css_lexer.TOpenBrace:
+			if len(stack) == 0 {
+				return endOfRuleOpenBrace, p.index + i
+			}
+			stack = append(stack, css_lexer.TCloseBrace)
+
+		case css_lexer.TCloseParen, css_lexer.TCloseBracket:
+			if n := len(stack); n > 0 && t.Kind == stack[n-1] {
+				stack = stack[:n-1]
+			}
+
+		case css_lexer.TCloseBrace:
+			if n := len(stack); n > 0 && t.Kind == stack[n-1] {
+				stack = stack[:n-1]
+			} else {
+				return endOfRuleUnknown, -1
+			}
+		}
+	}
+
+	return endOfRuleUnknown, -1
+}
+
+func (p *parser) parseDeclaration() css_ast.Rule {
+	// Parse the key
+	keyStart := p.index
+	keyRange := p.tokens[keyStart].Range
+	keyIsIdent := p.expect(css_lexer.TIdent)
+	ok := false
+	if keyIsIdent {
+		p.eat(css_lexer.TWhitespace)
+		ok = p.eat(css_lexer.TColon)
+	}
+
+	// Parse the value
+	valueStart := p.index
+stop:
+	for {
+		switch p.current().Kind {
+		case css_lexer.TEndOfFile, css_lexer.TSemicolon, css_lexer.TCloseBrace:
+			break stop
+
+		default:
+			p.parseComponentValue()
+		}
+	}
+
+	// Stop now if this is not a valid declaration
+	if !ok {
+		if keyIsIdent {
+			if end := keyRange.End(); end > p.prevError.Start {
+				p.prevError.Start = end
+				data := p.tracker.MsgData(logger.Range{Loc: logger.Loc{Start: end}}, "Expected \":\"")
+				data.Location.Suggestion = ":"
+				p.log.AddMsgID(logger.MsgID_CSS_CSSSyntaxError, logger.Msg{
+					Kind: logger.Warning,
+					Data: data,
+				})
+			}
+		}
+
+		return css_ast.Rule{Loc: keyRange.Loc, Data: &css_ast.RBadDeclaration{
+			Tokens: p.convertTokens(p.tokens[keyStart:p.index]),
+		}}
+	}
+
+	keyToken := p.tokens[keyStart]
+	keyText := keyToken.DecodedText(p.source.Contents)
+	value := p.tokens[valueStart:p.index]
+	verbatimWhitespace := strings.HasPrefix(keyText, "--")
+
+	// Remove trailing "!important"
+	important := false
+	i := len(value) - 1
+	if i >= 0 && value[i].Kind == css_lexer.TWhitespace {
+		i--
+	}
+	if i >= 0 && value[i].Kind == css_lexer.TIdent && strings.EqualFold(value[i].DecodedText(p.source.Contents), "important") {
+		i--
+		if i >= 0 && value[i].Kind == css_lexer.TWhitespace {
+			i--
+		}
+		if i >= 0 && value[i].Kind == css_lexer.TDelimExclamation {
+			value = value[:i]
+			important = true
+		}
+	}
+
+	result, _ := p.convertTokensHelper(value, css_lexer.TEndOfFile, convertTokensOpts{
+		allowImports: true,
+
+		// CSS variables require verbatim whitespace for correctness
+		verbatimWhitespace: verbatimWhitespace,
+	})
+
+	// Insert or remove whitespace before the first token
+	if !verbatimWhitespace && len(result) > 0 {
+		if p.options.minifyWhitespace {
+			result[0].Whitespace &= ^css_ast.WhitespaceBefore
+		} else {
+			result[0].Whitespace |= css_ast.WhitespaceBefore
+		}
+	}
+
+	lowerKeyText := strings.ToLower(keyText)
+	key := css_ast.KnownDeclarations[lowerKeyText]
+
+	// Attempt to point out trivial typos
+	if key == css_ast.DUnknown {
+		if corrected, ok := css_ast.MaybeCorrectDeclarationTypo(lowerKeyText); ok {
+			data := p.tracker.MsgData(keyToken.Range, fmt.Sprintf("%q is not a known CSS property", keyText))
+			data.Location.Suggestion = corrected
+			p.log.AddMsgID(logger.MsgID_CSS_UnsupportedCSSProperty, logger.Msg{Kind: logger.Warning, Data: data,
+				Notes: []logger.MsgData{{Text: fmt.Sprintf("Did you mean %q instead?", corrected)}}})
+		}
+	}
+
+	return css_ast.Rule{Loc: keyRange.Loc, Data: &css_ast.RDeclaration{
+		Key:       key,
+		KeyText:   keyText,
+		KeyRange:  keyToken.Range,
+		Value:     result,
+		Important: important,
+	}}
+}
+
+func (p *parser) parseComponentValue() {
+	switch p.current().Kind {
+	case css_lexer.TFunction:
+		p.parseBlock(css_lexer.TFunction, css_lexer.TCloseParen)
+
+	case css_lexer.TOpenParen:
+		p.parseBlock(css_lexer.TOpenParen, css_lexer.TCloseParen)
+
+	case css_lexer.TOpenBrace:
+		p.parseBlock(css_lexer.TOpenBrace, css_lexer.TCloseBrace)
+
+	case css_lexer.TOpenBracket:
+		p.parseBlock(css_lexer.TOpenBracket, css_lexer.TCloseBracket)
+
+	case css_lexer.TEndOfFile:
+		p.unexpected()
+
+	default:
+		p.advance()
+	}
+}
+
+func (p *parser) parseBlock(open css_lexer.T, close css_lexer.T) {
+	current := p.current()
+	matchingStart := current.Range.End() - 1
+	if p.expect(open) {
+		for !p.eat(close) {
+			if p.peek(css_lexer.TEndOfFile) {
+				p.expectWithMatchingLoc(close, logger.Loc{Start: matchingStart})
+				return
+			}
+
+			p.parseComponentValue()
+		}
+	}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_parser_selector.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_parser_selector.go
new file mode 100644
index 0000000..d766e8e
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_parser_selector.go
@@ -0,0 +1,979 @@
+package css_parser
+
+import (
+	"fmt"
+	"strings"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type parseSelectorOpts struct {
+	composesContext        *composesContext
+	pseudoClassKind        css_ast.PseudoClassKind
+	isDeclarationContext   bool
+	stopOnCloseParen       bool
+	onlyOneComplexSelector bool
+	noLeadingCombinator    bool
+}
+
+func (p *parser) parseSelectorList(opts parseSelectorOpts) (list []css_ast.ComplexSelector, ok bool) {
+	// Parse the first selector
+	sel, good := p.parseComplexSelector(parseComplexSelectorOpts{
+		parseSelectorOpts: opts,
+		isFirst:           true,
+	})
+	if !good {
+		return
+	}
+	list = p.flattenLocalAndGlobalSelectors(list, sel)
+
+	// Parse the remaining selectors
+	if opts.onlyOneComplexSelector {
+		if t := p.current(); t.Kind == css_lexer.TComma {
+			p.prevError = t.Range.Loc
+			kind := fmt.Sprintf(":%s(...)", opts.pseudoClassKind.String())
+			p.log.AddIDWithNotes(logger.MsgID_CSS_CSSSyntaxError, logger.Warning, &p.tracker, t.Range,
+				fmt.Sprintf("Unexpected \",\" inside %q", kind),
+				[]logger.MsgData{{Text: fmt.Sprintf("Different CSS tools behave differently in this case, so esbuild doesn't allow it. "+
+					"Either remove this comma or split this selector up into multiple comma-separated %q selectors instead.", kind)}})
+			return
+		}
+	} else {
+	skip:
+		for {
+			p.eat(css_lexer.TWhitespace)
+			if !p.eat(css_lexer.TComma) {
+				break
+			}
+			p.eat(css_lexer.TWhitespace)
+			sel, good := p.parseComplexSelector(parseComplexSelectorOpts{
+				parseSelectorOpts: opts,
+			})
+			if !good {
+				return
+			}
+
+			// Omit duplicate selectors
+			if p.options.minifySyntax {
+				for _, existing := range list {
+					if sel.Equal(existing, nil) {
+						continue skip
+					}
+				}
+			}
+
+			list = p.flattenLocalAndGlobalSelectors(list, sel)
+		}
+	}
+
+	if p.options.minifySyntax {
+		for i := 1; i < len(list); i++ {
+			if analyzeLeadingAmpersand(list[i], opts.isDeclarationContext) != cannotRemoveLeadingAmpersand {
+				list[i].Selectors = list[i].Selectors[1:]
+			}
+		}
+
+		switch analyzeLeadingAmpersand(list[0], opts.isDeclarationContext) {
+		case canAlwaysRemoveLeadingAmpersand:
+			list[0].Selectors = list[0].Selectors[1:]
+
+		case canRemoveLeadingAmpersandIfNotFirst:
+			for i := 1; i < len(list); i++ {
+				if sel := list[i].Selectors[0]; !sel.HasNestingSelector() && (sel.Combinator.Byte != 0 || sel.TypeSelector == nil) {
+					list[0].Selectors = list[0].Selectors[1:]
+					list[0], list[i] = list[i], list[0]
+					break
+				}
+			}
+		}
+	}
+
+	ok = true
+	return
+}
+
+func mergeCompoundSelectors(target *css_ast.CompoundSelector, source css_ast.CompoundSelector) {
+	// ".foo:local(&)" => "&.foo"
+	if source.HasNestingSelector() && !target.HasNestingSelector() {
+		target.NestingSelectorLoc = source.NestingSelectorLoc
+	}
+
+	if source.TypeSelector != nil {
+		if target.TypeSelector == nil {
+			// ".foo:local(div)" => "div.foo"
+			target.TypeSelector = source.TypeSelector
+		} else {
+			// "div:local(span)" => "div:is(span)"
+			//
+			// Note: All other implementations of this (Lightning CSS, PostCSS, and
+			// Webpack) do something really weird here. They do this instead:
+			//
+			// "div:local(span)" => "divspan"
+			//
+			// But that just seems so obviously wrong that I'm not going to do that.
+			target.SubclassSelectors = append(target.SubclassSelectors, css_ast.SubclassSelector{
+				Range: source.TypeSelector.Range(),
+				Data: &css_ast.SSPseudoClassWithSelectorList{
+					Kind:      css_ast.PseudoClassIs,
+					Selectors: []css_ast.ComplexSelector{{Selectors: []css_ast.CompoundSelector{{TypeSelector: source.TypeSelector}}}},
+				},
+			})
+		}
+	}
+
+	// ".foo:local(.bar)" => ".foo.bar"
+	target.SubclassSelectors = append(target.SubclassSelectors, source.SubclassSelectors...)
+}
+
+func containsLocalOrGlobalSelector(sel css_ast.ComplexSelector) bool {
+	for _, s := range sel.Selectors {
+		for _, ss := range s.SubclassSelectors {
+			switch pseudo := ss.Data.(type) {
+			case *css_ast.SSPseudoClass:
+				if pseudo.Name == "global" || pseudo.Name == "local" {
+					return true
+				}
+
+			case *css_ast.SSPseudoClassWithSelectorList:
+				if pseudo.Kind == css_ast.PseudoClassGlobal || pseudo.Kind == css_ast.PseudoClassLocal {
+					return true
+				}
+			}
+		}
+	}
+	return false
+}
+
+// This handles the ":local()" and ":global()" annotations from CSS modules
+func (p *parser) flattenLocalAndGlobalSelectors(list []css_ast.ComplexSelector, sel css_ast.ComplexSelector) []css_ast.ComplexSelector {
+	// Only do the work to flatten the whole list if there's a ":local" or a ":global"
+	if p.options.symbolMode != symbolModeDisabled && containsLocalOrGlobalSelector(sel) {
+		var selectors []css_ast.CompoundSelector
+
+		for _, s := range sel.Selectors {
+			oldSubclassSelectors := s.SubclassSelectors
+			s.SubclassSelectors = make([]css_ast.SubclassSelector, 0, len(oldSubclassSelectors))
+
+			for _, ss := range oldSubclassSelectors {
+				switch pseudo := ss.Data.(type) {
+				case *css_ast.SSPseudoClass:
+					if pseudo.Name == "global" || pseudo.Name == "local" {
+						// Remove bare ":global" and ":local" pseudo-classes
+						continue
+					}
+
+				case *css_ast.SSPseudoClassWithSelectorList:
+					if pseudo.Kind == css_ast.PseudoClassGlobal || pseudo.Kind == css_ast.PseudoClassLocal {
+						inner := pseudo.Selectors[0].Selectors
+
+						// Replace this pseudo-class with all inner compound selectors.
+						// The first inner compound selector is merged with the compound
+						// selector before it and the last inner compound selector is
+						// merged with the compound selector after it:
+						//
+						// "div:local(.a .b):hover" => "div.a b:hover"
+						//
+						// This behavior is really strange since this is not how anything
+						// involving pseudo-classes in real CSS works at all. However, all
+						// other implementations (Lightning CSS, PostCSS, and Webpack) are
+						// consistent with this strange behavior, so we do it too.
+						if inner[0].Combinator.Byte == 0 {
+							mergeCompoundSelectors(&s, inner[0])
+							inner = inner[1:]
+						} else {
+							// "div:local(+ .foo):hover" => "div + .foo:hover"
+						}
+						if n := len(inner); n > 0 {
+							if !s.IsInvalidBecauseEmpty() {
+								// Don't add this selector if it consisted only of a bare ":global" or ":local"
+								selectors = append(selectors, s)
+							}
+							selectors = append(selectors, inner[:n-1]...)
+							s = inner[n-1]
+						}
+						continue
+					}
+				}
+
+				s.SubclassSelectors = append(s.SubclassSelectors, ss)
+			}
+
+			if !s.IsInvalidBecauseEmpty() {
+				// Don't add this selector if it consisted only of a bare ":global" or ":local"
+				selectors = append(selectors, s)
+			}
+		}
+
+		if len(selectors) == 0 {
+			// Treat a bare ":global" or ":local" as a bare "&" nesting selector
+			selectors = append(selectors, css_ast.CompoundSelector{
+				NestingSelectorLoc:        ast.MakeIndex32(uint32(sel.Selectors[0].Range().Loc.Start)),
+				WasEmptyFromLocalOrGlobal: true,
+			})
+
+			// Make sure we report that nesting is present so that it can be lowered
+			p.nestingIsPresent = true
+		}
+
+		sel.Selectors = selectors
+	}
+
+	return append(list, sel)
+}
+
+type leadingAmpersand uint8
+
+const (
+	cannotRemoveLeadingAmpersand leadingAmpersand = iota
+	canAlwaysRemoveLeadingAmpersand
+	canRemoveLeadingAmpersandIfNotFirst
+)
+
+func analyzeLeadingAmpersand(sel css_ast.ComplexSelector, isDeclarationContext bool) leadingAmpersand {
+	if len(sel.Selectors) > 1 {
+		if first := sel.Selectors[0]; first.IsSingleAmpersand() {
+			if second := sel.Selectors[1]; second.Combinator.Byte == 0 && second.HasNestingSelector() {
+				// ".foo { & &.bar {} }" => ".foo { & &.bar {} }"
+			} else if second.Combinator.Byte != 0 || second.TypeSelector == nil || !isDeclarationContext {
+				// "& + div {}" => "+ div {}"
+				// "& div {}" => "div {}"
+				// ".foo { & + div {} }" => ".foo { + div {} }"
+				// ".foo { & + &.bar {} }" => ".foo { + &.bar {} }"
+				// ".foo { & :hover {} }" => ".foo { :hover {} }"
+				return canAlwaysRemoveLeadingAmpersand
+			} else {
+				// ".foo { & div {} }"
+				// ".foo { .bar, & div {} }" => ".foo { .bar, div {} }"
+				return canRemoveLeadingAmpersandIfNotFirst
+			}
+		}
+	} else {
+		// "& {}" => "& {}"
+	}
+	return cannotRemoveLeadingAmpersand
+}
+
+type parseComplexSelectorOpts struct {
+	parseSelectorOpts
+	isFirst bool
+}
+
+func (p *parser) parseComplexSelector(opts parseComplexSelectorOpts) (result css_ast.ComplexSelector, ok bool) {
+	// This is an extension: https://drafts.csswg.org/css-nesting-1/
+	var combinator css_ast.Combinator
+	if !opts.noLeadingCombinator {
+		combinator = p.parseCombinator()
+		if combinator.Byte != 0 {
+			p.nestingIsPresent = true
+			p.eat(css_lexer.TWhitespace)
+		}
+	}
+
+	// Parent
+	sel, good := p.parseCompoundSelector(parseComplexSelectorOpts{
+		parseSelectorOpts: opts.parseSelectorOpts,
+		isFirst:           opts.isFirst,
+	})
+	if !good {
+		return
+	}
+	sel.Combinator = combinator
+	result.Selectors = append(result.Selectors, sel)
+
+	stop := css_lexer.TOpenBrace
+	if opts.stopOnCloseParen {
+		stop = css_lexer.TCloseParen
+	}
+	for {
+		p.eat(css_lexer.TWhitespace)
+		if p.peek(css_lexer.TEndOfFile) || p.peek(css_lexer.TComma) || p.peek(stop) {
+			break
+		}
+
+		// Optional combinator
+		combinator := p.parseCombinator()
+		if combinator.Byte != 0 {
+			p.eat(css_lexer.TWhitespace)
+		}
+
+		// Child
+		sel, good := p.parseCompoundSelector(parseComplexSelectorOpts{
+			parseSelectorOpts: opts.parseSelectorOpts,
+		})
+		if !good {
+			return
+		}
+		sel.Combinator = combinator
+		result.Selectors = append(result.Selectors, sel)
+	}
+
+	ok = true
+	return
+}
+
+func (p *parser) nameToken() css_ast.NameToken {
+	t := p.current()
+	return css_ast.NameToken{
+		Kind:  t.Kind,
+		Range: t.Range,
+		Text:  p.decoded(),
+	}
+}
+
+func (p *parser) parseCompoundSelector(opts parseComplexSelectorOpts) (sel css_ast.CompoundSelector, ok bool) {
+	startLoc := p.current().Range.Loc
+
+	// This is an extension: https://drafts.csswg.org/css-nesting-1/
+	hasLeadingNestingSelector := p.peek(css_lexer.TDelimAmpersand)
+	if hasLeadingNestingSelector {
+		p.nestingIsPresent = true
+		sel.NestingSelectorLoc = ast.MakeIndex32(uint32(startLoc.Start))
+		p.advance()
+	}
+
+	// Parse the type selector
+	typeSelectorLoc := p.current().Range.Loc
+	switch p.current().Kind {
+	case css_lexer.TDelimBar, css_lexer.TIdent, css_lexer.TDelimAsterisk:
+		nsName := css_ast.NamespacedName{}
+		if !p.peek(css_lexer.TDelimBar) {
+			nsName.Name = p.nameToken()
+			p.advance()
+		} else {
+			// Hack: Create an empty "identifier" to represent this
+			nsName.Name.Kind = css_lexer.TIdent
+		}
+		if p.eat(css_lexer.TDelimBar) {
+			if !p.peek(css_lexer.TIdent) && !p.peek(css_lexer.TDelimAsterisk) {
+				p.expect(css_lexer.TIdent)
+				return
+			}
+			prefix := nsName.Name
+			nsName.NamespacePrefix = &prefix
+			nsName.Name = p.nameToken()
+			p.advance()
+		}
+		sel.TypeSelector = &nsName
+	}
+
+	// Parse the subclass selectors
+subclassSelectors:
+	for {
+		subclassToken := p.current()
+
+		switch subclassToken.Kind {
+		case css_lexer.THash:
+			if (subclassToken.Flags & css_lexer.IsID) == 0 {
+				break subclassSelectors
+			}
+			nameLoc := logger.Loc{Start: subclassToken.Range.Loc.Start + 1}
+			name := p.decoded()
+			sel.SubclassSelectors = append(sel.SubclassSelectors, css_ast.SubclassSelector{
+				Range: subclassToken.Range,
+				Data: &css_ast.SSHash{
+					Name: p.symbolForName(nameLoc, name),
+				},
+			})
+			p.advance()
+
+		case css_lexer.TDelimDot:
+			p.advance()
+			nameRange := p.current().Range
+			name := p.decoded()
+			sel.SubclassSelectors = append(sel.SubclassSelectors, css_ast.SubclassSelector{
+				Range: logger.Range{Loc: subclassToken.Range.Loc, Len: nameRange.End() - subclassToken.Range.Loc.Start},
+				Data: &css_ast.SSClass{
+					Name: p.symbolForName(nameRange.Loc, name),
+				},
+			})
+			if !p.expect(css_lexer.TIdent) {
+				return
+			}
+
+		case css_lexer.TOpenBracket:
+			attr, r := p.parseAttributeSelector()
+			if r.Len == 0 {
+				return
+			}
+			sel.SubclassSelectors = append(sel.SubclassSelectors, css_ast.SubclassSelector{
+				Range: r,
+				Data:  &attr,
+			})
+
+		case css_lexer.TColon:
+			if p.next().Kind == css_lexer.TColon {
+				// Special-case the start of the pseudo-element selector section
+				for p.current().Kind == css_lexer.TColon {
+					firstColonLoc := p.current().Range.Loc
+					isElement := p.next().Kind == css_lexer.TColon
+					if isElement {
+						p.advance()
+					}
+					pseudo, r := p.parsePseudoClassSelector(firstColonLoc, isElement)
+
+					// https://www.w3.org/TR/selectors-4/#single-colon-pseudos
+					// The four Level 2 pseudo-elements (::before, ::after, ::first-line,
+					// and ::first-letter) may, for legacy reasons, be represented using
+					// the <pseudo-class-selector> grammar, with only a single ":"
+					// character at their start.
+					if p.options.minifySyntax && isElement {
+						if pseudo, ok := pseudo.(*css_ast.SSPseudoClass); ok && len(pseudo.Args) == 0 {
+							switch pseudo.Name {
+							case "before", "after", "first-line", "first-letter":
+								pseudo.IsElement = false
+							}
+						}
+					}
+
+					sel.SubclassSelectors = append(sel.SubclassSelectors, css_ast.SubclassSelector{
+						Range: r,
+						Data:  pseudo,
+					})
+				}
+				break subclassSelectors
+			}
+
+			pseudo, r := p.parsePseudoClassSelector(subclassToken.Range.Loc, false)
+			sel.SubclassSelectors = append(sel.SubclassSelectors, css_ast.SubclassSelector{
+				Range: r,
+				Data:  pseudo,
+			})
+
+		case css_lexer.TDelimAmpersand:
+			// This is an extension: https://drafts.csswg.org/css-nesting-1/
+			p.nestingIsPresent = true
+			sel.NestingSelectorLoc = ast.MakeIndex32(uint32(subclassToken.Range.Loc.Start))
+			p.advance()
+
+		default:
+			break subclassSelectors
+		}
+	}
+
+	// The compound selector must be non-empty
+	if sel.IsInvalidBecauseEmpty() {
+		p.unexpected()
+		return
+	}
+
+	// Note: "&div {}" was originally valid, but is now an invalid selector:
+	// https://github.com/w3c/csswg-drafts/issues/8662#issuecomment-1514977935.
+	// This is because SASS already uses that syntax to mean something very
+	// different, so that syntax has been removed to avoid mistakes.
+	if hasLeadingNestingSelector && sel.TypeSelector != nil {
+		r := logger.Range{Loc: typeSelectorLoc, Len: p.at(p.index-1).Range.End() - typeSelectorLoc.Start}
+		text := sel.TypeSelector.Name.Text
+		if sel.TypeSelector.NamespacePrefix != nil {
+			text = fmt.Sprintf("%s|%s", sel.TypeSelector.NamespacePrefix.Text, text)
+		}
+		var howToFix string
+		suggestion := p.source.TextForRange(r)
+		if opts.isFirst {
+			suggestion = fmt.Sprintf(":is(%s)", suggestion)
+			howToFix = "You can wrap this selector in \":is(...)\" as a workaround. "
+		} else {
+			r = logger.Range{Loc: startLoc, Len: r.End() - startLoc.Start}
+			suggestion += "&"
+			howToFix = "You can move the \"&\" to the end of this selector as a workaround. "
+		}
+		msg := logger.Msg{
+			Kind: logger.Warning,
+			Data: p.tracker.MsgData(r, fmt.Sprintf("Cannot use type selector %q directly after nesting selector \"&\"", text)),
+			Notes: []logger.MsgData{{Text: "CSS nesting syntax does not allow the \"&\" selector to come before a type selector. " +
+				howToFix +
+				"This restriction exists to avoid problems with SASS nesting, where the same syntax means something very different " +
+				"that has no equivalent in real CSS (appending a suffix to the parent selector)."}},
+		}
+		msg.Data.Location.Suggestion = suggestion
+		p.log.AddMsgID(logger.MsgID_CSS_CSSSyntaxError, msg)
+		return
+	}
+
+	// The type selector must always come first
+	switch p.current().Kind {
+	case css_lexer.TDelimBar, css_lexer.TIdent, css_lexer.TDelimAsterisk:
+		p.unexpected()
+		return
+	}
+
+	ok = true
+	return
+}
+
+func (p *parser) parseAttributeSelector() (attr css_ast.SSAttribute, r logger.Range) {
+	matchingLoc := p.current().Range.Loc
+	p.advance()
+
+	// Parse the namespaced name
+	switch p.current().Kind {
+	case css_lexer.TDelimBar, css_lexer.TDelimAsterisk:
+		// "[|x]"
+		// "[*|x]"
+		if p.peek(css_lexer.TDelimAsterisk) {
+			prefix := p.nameToken()
+			p.advance()
+			attr.NamespacedName.NamespacePrefix = &prefix
+		} else {
+			// "[|attr]" is equivalent to "[attr]". From the specification:
+			// "In keeping with the Namespaces in the XML recommendation, default
+			// namespaces do not apply to attributes, therefore attribute selectors
+			// without a namespace component apply only to attributes that have no
+			// namespace (equivalent to |attr)."
+		}
+		if !p.expect(css_lexer.TDelimBar) {
+			return
+		}
+		attr.NamespacedName.Name = p.nameToken()
+		if !p.expect(css_lexer.TIdent) {
+			return
+		}
+
+	default:
+		// "[x]"
+		// "[x|y]"
+		attr.NamespacedName.Name = p.nameToken()
+		if !p.expect(css_lexer.TIdent) {
+			return
+		}
+		if p.next().Kind != css_lexer.TDelimEquals && p.eat(css_lexer.TDelimBar) {
+			prefix := attr.NamespacedName.Name
+			attr.NamespacedName.NamespacePrefix = &prefix
+			attr.NamespacedName.Name = p.nameToken()
+			if !p.expect(css_lexer.TIdent) {
+				return
+			}
+		}
+	}
+
+	// Parse the optional matcher operator
+	p.eat(css_lexer.TWhitespace)
+	if p.eat(css_lexer.TDelimEquals) {
+		attr.MatcherOp = "="
+	} else {
+		switch p.current().Kind {
+		case css_lexer.TDelimTilde:
+			attr.MatcherOp = "~="
+		case css_lexer.TDelimBar:
+			attr.MatcherOp = "|="
+		case css_lexer.TDelimCaret:
+			attr.MatcherOp = "^="
+		case css_lexer.TDelimDollar:
+			attr.MatcherOp = "$="
+		case css_lexer.TDelimAsterisk:
+			attr.MatcherOp = "*="
+		}
+		if attr.MatcherOp != "" {
+			p.advance()
+			if !p.expect(css_lexer.TDelimEquals) {
+				return
+			}
+		}
+	}
+
+	// Parse the optional matcher value
+	if attr.MatcherOp != "" {
+		p.eat(css_lexer.TWhitespace)
+		if !p.peek(css_lexer.TString) && !p.peek(css_lexer.TIdent) {
+			p.unexpected()
+		}
+		attr.MatcherValue = p.decoded()
+		p.advance()
+		p.eat(css_lexer.TWhitespace)
+		if p.peek(css_lexer.TIdent) {
+			if modifier := p.decoded(); len(modifier) == 1 {
+				if c := modifier[0]; c == 'i' || c == 'I' || c == 's' || c == 'S' {
+					attr.MatcherModifier = c
+					p.advance()
+				}
+			}
+		}
+	}
+
+	closeRange := p.current().Range
+	if !p.expectWithMatchingLoc(css_lexer.TCloseBracket, matchingLoc) {
+		closeRange.Len = 0
+	}
+	r = logger.Range{Loc: matchingLoc, Len: closeRange.End() - matchingLoc.Start}
+	return
+}
+
+func (p *parser) parsePseudoClassSelector(loc logger.Loc, isElement bool) (css_ast.SS, logger.Range) {
+	p.advance()
+
+	if p.peek(css_lexer.TFunction) {
+		text := p.decoded()
+		matchingLoc := logger.Loc{Start: p.current().Range.End() - 1}
+		p.advance()
+
+		// Potentially parse a pseudo-class with a selector list
+		if !isElement {
+			var kind css_ast.PseudoClassKind
+			local := p.makeLocalSymbols
+			ok := true
+			switch text {
+			case "global":
+				kind = css_ast.PseudoClassGlobal
+				if p.options.symbolMode != symbolModeDisabled {
+					local = false
+				}
+			case "has":
+				kind = css_ast.PseudoClassHas
+			case "is":
+				kind = css_ast.PseudoClassIs
+			case "local":
+				kind = css_ast.PseudoClassLocal
+				if p.options.symbolMode != symbolModeDisabled {
+					local = true
+				}
+			case "not":
+				kind = css_ast.PseudoClassNot
+			case "nth-child":
+				kind = css_ast.PseudoClassNthChild
+			case "nth-last-child":
+				kind = css_ast.PseudoClassNthLastChild
+			case "nth-of-type":
+				kind = css_ast.PseudoClassNthOfType
+			case "nth-last-of-type":
+				kind = css_ast.PseudoClassNthLastOfType
+			case "where":
+				kind = css_ast.PseudoClassWhere
+			default:
+				ok = false
+			}
+			if ok {
+				old := p.index
+				if kind.HasNthIndex() {
+					p.eat(css_lexer.TWhitespace)
+
+					// Parse the "An+B" syntax
+					if index, ok := p.parseNthIndex(); ok {
+						var selectors []css_ast.ComplexSelector
+
+						// Parse the optional "of" clause
+						if (kind == css_ast.PseudoClassNthChild || kind == css_ast.PseudoClassNthLastChild) &&
+							p.peek(css_lexer.TIdent) && strings.EqualFold(p.decoded(), "of") {
+							p.advance()
+							p.eat(css_lexer.TWhitespace)
+
+							// Contain the effects of ":local" and ":global"
+							oldLocal := p.makeLocalSymbols
+							selectors, ok = p.parseSelectorList(parseSelectorOpts{
+								stopOnCloseParen:    true,
+								noLeadingCombinator: true,
+							})
+							p.makeLocalSymbols = oldLocal
+						}
+
+						// "2n+0" => "2n"
+						if p.options.minifySyntax {
+							index.Minify()
+						}
+
+						// Match the closing ")"
+						if ok {
+							closeRange := p.current().Range
+							if !p.expectWithMatchingLoc(css_lexer.TCloseParen, matchingLoc) {
+								closeRange.Len = 0
+							}
+							return &css_ast.SSPseudoClassWithSelectorList{Kind: kind, Selectors: selectors, Index: index},
+								logger.Range{Loc: loc, Len: closeRange.End() - loc.Start}
+						}
+					}
+				} else {
+					p.eat(css_lexer.TWhitespace)
+
+					// ":local" forces local names and ":global" forces global names
+					oldLocal := p.makeLocalSymbols
+					p.makeLocalSymbols = local
+					selectors, ok := p.parseSelectorList(parseSelectorOpts{
+						pseudoClassKind:        kind,
+						stopOnCloseParen:       true,
+						onlyOneComplexSelector: kind == css_ast.PseudoClassGlobal || kind == css_ast.PseudoClassLocal,
+					})
+					p.makeLocalSymbols = oldLocal
+
+					// Match the closing ")"
+					if ok {
+						closeRange := p.current().Range
+						if !p.expectWithMatchingLoc(css_lexer.TCloseParen, matchingLoc) {
+							closeRange.Len = 0
+						}
+						return &css_ast.SSPseudoClassWithSelectorList{Kind: kind, Selectors: selectors},
+							logger.Range{Loc: loc, Len: closeRange.End() - loc.Start}
+					}
+				}
+				p.index = old
+			}
+		}
+
+		args := p.convertTokens(p.parseAnyValue())
+		closeRange := p.current().Range
+		if !p.expectWithMatchingLoc(css_lexer.TCloseParen, matchingLoc) {
+			closeRange.Len = 0
+		}
+		return &css_ast.SSPseudoClass{IsElement: isElement, Name: text, Args: args},
+			logger.Range{Loc: loc, Len: closeRange.End() - loc.Start}
+	}
+
+	nameRange := p.current().Range
+	name := p.decoded()
+	sel := css_ast.SSPseudoClass{IsElement: isElement}
+	if p.expect(css_lexer.TIdent) {
+		sel.Name = name
+
+		// ":local .local_name :global .global_name {}"
+		// ":local { .local_name { :global { .global_name {} } }"
+		if p.options.symbolMode != symbolModeDisabled {
+			switch name {
+			case "local":
+				p.makeLocalSymbols = true
+			case "global":
+				p.makeLocalSymbols = false
+			}
+		}
+	} else {
+		nameRange.Len = 0
+	}
+	return &sel, logger.Range{Loc: loc, Len: nameRange.End() - loc.Start}
+}
+
+func (p *parser) parseAnyValue() []css_lexer.Token {
+	// Reference: https://drafts.csswg.org/css-syntax-3/#typedef-declaration-value
+
+	p.stack = p.stack[:0] // Reuse allocated memory
+	start := p.index
+
+loop:
+	for {
+		switch p.current().Kind {
+		case css_lexer.TCloseParen, css_lexer.TCloseBracket, css_lexer.TCloseBrace:
+			last := len(p.stack) - 1
+			if last < 0 || !p.peek(p.stack[last]) {
+				break loop
+			}
+			p.stack = p.stack[:last]
+
+		case css_lexer.TSemicolon, css_lexer.TDelimExclamation:
+			if len(p.stack) == 0 {
+				break loop
+			}
+
+		case css_lexer.TOpenParen, css_lexer.TFunction:
+			p.stack = append(p.stack, css_lexer.TCloseParen)
+
+		case css_lexer.TOpenBracket:
+			p.stack = append(p.stack, css_lexer.TCloseBracket)
+
+		case css_lexer.TOpenBrace:
+			p.stack = append(p.stack, css_lexer.TCloseBrace)
+
+		case css_lexer.TEndOfFile:
+			break loop
+		}
+
+		p.advance()
+	}
+
+	tokens := p.tokens[start:p.index]
+	if len(tokens) == 0 {
+		p.unexpected()
+	}
+	return tokens
+}
+
+func (p *parser) parseCombinator() css_ast.Combinator {
+	t := p.current()
+
+	switch t.Kind {
+	case css_lexer.TDelimGreaterThan:
+		p.advance()
+		return css_ast.Combinator{Loc: t.Range.Loc, Byte: '>'}
+
+	case css_lexer.TDelimPlus:
+		p.advance()
+		return css_ast.Combinator{Loc: t.Range.Loc, Byte: '+'}
+
+	case css_lexer.TDelimTilde:
+		p.advance()
+		return css_ast.Combinator{Loc: t.Range.Loc, Byte: '~'}
+
+	default:
+		return css_ast.Combinator{}
+	}
+}
+
+func parseInteger(text string) (string, bool) {
+	n := len(text)
+	if n == 0 {
+		return "", false
+	}
+
+	// Trim leading zeros
+	start := 0
+	for start < n && text[start] == '0' {
+		start++
+	}
+
+	// Make sure remaining characters are digits
+	if start == n {
+		return "0", true
+	}
+	for i := start; i < n; i++ {
+		if c := text[i]; c < '0' || c > '9' {
+			return "", false
+		}
+	}
+	return text[start:], true
+}
+
+func (p *parser) parseNthIndex() (css_ast.NthIndex, bool) {
+	type sign uint8
+	const (
+		none sign = iota
+		negative
+		positive
+	)
+
+	// Reference: https://drafts.csswg.org/css-syntax-3/#anb-microsyntax
+	t0 := p.current()
+	text0 := p.decoded()
+
+	// Handle "even" and "odd"
+	if t0.Kind == css_lexer.TIdent && (text0 == "even" || text0 == "odd") {
+		p.advance()
+		p.eat(css_lexer.TWhitespace)
+		return css_ast.NthIndex{B: text0}, true
+	}
+
+	// Handle a single number
+	if t0.Kind == css_lexer.TNumber {
+		bNeg := false
+		if strings.HasPrefix(text0, "-") {
+			bNeg = true
+			text0 = text0[1:]
+		} else {
+			text0 = strings.TrimPrefix(text0, "+")
+		}
+		if b, ok := parseInteger(text0); ok {
+			if bNeg {
+				b = "-" + b
+			}
+			p.advance()
+			p.eat(css_lexer.TWhitespace)
+			return css_ast.NthIndex{B: b}, true
+		}
+		p.unexpected()
+		return css_ast.NthIndex{}, false
+	}
+
+	aSign := none
+	if p.eat(css_lexer.TDelimPlus) {
+		aSign = positive
+		t0 = p.current()
+		text0 = p.decoded()
+	}
+
+	// Everything from here must be able to contain an "n"
+	if t0.Kind != css_lexer.TIdent && t0.Kind != css_lexer.TDimension {
+		p.unexpected()
+		return css_ast.NthIndex{}, false
+	}
+
+	// Check for a leading sign
+	if aSign == none {
+		if strings.HasPrefix(text0, "-") {
+			aSign = negative
+			text0 = text0[1:]
+		} else {
+			text0 = strings.TrimPrefix(text0, "+")
+		}
+	}
+
+	// The string must contain an "n"
+	n := strings.IndexByte(text0, 'n')
+	if n < 0 {
+		p.unexpected()
+		return css_ast.NthIndex{}, false
+	}
+
+	// Parse the number before the "n"
+	var a string
+	if n == 0 {
+		if aSign == negative {
+			a = "-1"
+		} else {
+			a = "1"
+		}
+	} else if aInt, ok := parseInteger(text0[:n]); ok {
+		if aSign == negative {
+			aInt = "-" + aInt
+		}
+		a = aInt
+	} else {
+		p.unexpected()
+		return css_ast.NthIndex{}, false
+	}
+	text0 = text0[n+1:]
+
+	// Parse the stuff after the "n"
+	bSign := none
+	if strings.HasPrefix(text0, "-") {
+		text0 = text0[1:]
+		if b, ok := parseInteger(text0); ok {
+			p.advance()
+			p.eat(css_lexer.TWhitespace)
+			return css_ast.NthIndex{A: a, B: "-" + b}, true
+		}
+		bSign = negative
+	}
+	if text0 != "" {
+		p.unexpected()
+		return css_ast.NthIndex{}, false
+	}
+	p.advance()
+	p.eat(css_lexer.TWhitespace)
+
+	// Parse an optional sign delimiter
+	if bSign == none {
+		if p.eat(css_lexer.TDelimMinus) {
+			bSign = negative
+			p.eat(css_lexer.TWhitespace)
+		} else if p.eat(css_lexer.TDelimPlus) {
+			bSign = positive
+			p.eat(css_lexer.TWhitespace)
+		}
+	}
+
+	// Parse an optional trailing number
+	t1 := p.current()
+	text1 := p.decoded()
+	if t1.Kind == css_lexer.TNumber {
+		if bSign == none {
+			if strings.HasPrefix(text1, "-") {
+				bSign = negative
+				text1 = text1[1:]
+			} else if strings.HasPrefix(text1, "+") {
+				text1 = text1[1:]
+			}
+		}
+		if b, ok := parseInteger(text1); ok {
+			if bSign == negative {
+				b = "-" + b
+			}
+			p.advance()
+			p.eat(css_lexer.TWhitespace)
+			return css_ast.NthIndex{A: a, B: b}, true
+		}
+	}
+
+	// If there is a trailing sign, then there must also be a trailing number
+	if bSign != none {
+		p.expect(css_lexer.TNumber)
+		return css_ast.NthIndex{}, false
+	}
+
+	return css_ast.NthIndex{A: a}, true
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_reduce_calc.go b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_reduce_calc.go
new file mode 100644
index 0000000..19b4582
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_parser/css_reduce_calc.go
@@ -0,0 +1,605 @@
+package css_parser
+
+import (
+	"fmt"
+	"math"
+	"strconv"
+	"strings"
+
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+func (p *parser) tryToReduceCalcExpression(token css_ast.Token) css_ast.Token {
+	if term := tryToParseCalcTerm(*token.Children); term != nil {
+		whitespace := css_ast.WhitespaceBefore | css_ast.WhitespaceAfter
+		if p.options.minifyWhitespace {
+			whitespace = 0
+		}
+		term = term.partiallySimplify()
+		if result, ok := term.convertToToken(whitespace); ok {
+			if result.Kind == css_lexer.TOpenParen {
+				result.Kind = css_lexer.TFunction
+				result.Text = "calc"
+			}
+			result.Loc = token.Loc
+			result.Whitespace = css_ast.WhitespaceBefore | css_ast.WhitespaceAfter
+			return result
+		}
+	}
+	return token
+}
+
+type calcTermWithOp struct {
+	data  calcTerm
+	opLoc logger.Loc
+}
+
+// See: https://www.w3.org/TR/css-values-4/#calc-internal
+type calcTerm interface {
+	convertToToken(whitespace css_ast.WhitespaceFlags) (css_ast.Token, bool)
+	partiallySimplify() calcTerm
+}
+
+type calcSum struct {
+	terms []calcTermWithOp
+}
+
+type calcProduct struct {
+	terms []calcTermWithOp
+}
+
+type calcNegate struct {
+	term calcTermWithOp
+}
+
+type calcInvert struct {
+	term calcTermWithOp
+}
+
+type calcNumeric struct {
+	unit   string
+	number float64
+	loc    logger.Loc
+}
+
+type calcValue struct {
+	token                css_ast.Token
+	isInvalidPlusOrMinus bool
+}
+
+func floatToStringForCalc(a float64) (string, bool) {
+	// Handle non-finite cases
+	if math.IsNaN(a) || math.IsInf(a, 0) {
+		return "", false
+	}
+
+	// Print the number as a string
+	text := fmt.Sprintf("%.05f", a)
+	for text[len(text)-1] == '0' {
+		text = text[:len(text)-1]
+	}
+	if text[len(text)-1] == '.' {
+		text = text[:len(text)-1]
+	}
+	if strings.HasPrefix(text, "0.") {
+		text = text[1:]
+	} else if strings.HasPrefix(text, "-0.") {
+		text = "-" + text[2:]
+	}
+
+	// Bail if the number is not exactly represented
+	if number, err := strconv.ParseFloat(text, 64); err != nil || number != a {
+		return "", false
+	}
+
+	return text, true
+}
+
+func (c *calcSum) convertToToken(whitespace css_ast.WhitespaceFlags) (css_ast.Token, bool) {
+	// Specification: https://www.w3.org/TR/css-values-4/#calc-serialize
+	tokens := make([]css_ast.Token, 0, len(c.terms)*2)
+
+	// ALGORITHM DEVIATION: Avoid parenthesizing product nodes inside sum nodes
+	if product, ok := c.terms[0].data.(*calcProduct); ok {
+		token, ok := product.convertToToken(whitespace)
+		if !ok {
+			return css_ast.Token{}, false
+		}
+		tokens = append(tokens, *token.Children...)
+	} else {
+		token, ok := c.terms[0].data.convertToToken(whitespace)
+		if !ok {
+			return css_ast.Token{}, false
+		}
+		tokens = append(tokens, token)
+	}
+
+	for _, term := range c.terms[1:] {
+		// If child is a Negate node, append " - " to s, then serialize the Negate’s child and append the result to s.
+		if negate, ok := term.data.(*calcNegate); ok {
+			token, ok := negate.term.data.convertToToken(whitespace)
+			if !ok {
+				return css_ast.Token{}, false
+			}
+			tokens = append(tokens, css_ast.Token{
+				Loc:        term.opLoc,
+				Kind:       css_lexer.TDelimMinus,
+				Text:       "-",
+				Whitespace: css_ast.WhitespaceBefore | css_ast.WhitespaceAfter,
+			}, token)
+			continue
+		}
+
+		// If child is a negative numeric value, append " - " to s, then serialize the negation of child as normal and append the result to s.
+		if numeric, ok := term.data.(*calcNumeric); ok && numeric.number < 0 {
+			clone := *numeric
+			clone.number = -clone.number
+			token, ok := clone.convertToToken(whitespace)
+			if !ok {
+				return css_ast.Token{}, false
+			}
+			tokens = append(tokens, css_ast.Token{
+				Loc:        term.opLoc,
+				Kind:       css_lexer.TDelimMinus,
+				Text:       "-",
+				Whitespace: css_ast.WhitespaceBefore | css_ast.WhitespaceAfter,
+			}, token)
+			continue
+		}
+
+		// Otherwise, append " + " to s, then serialize child and append the result to s.
+		tokens = append(tokens, css_ast.Token{
+			Loc:        term.opLoc,
+			Kind:       css_lexer.TDelimPlus,
+			Text:       "+",
+			Whitespace: css_ast.WhitespaceBefore | css_ast.WhitespaceAfter,
+		})
+
+		// ALGORITHM DEVIATION: Avoid parenthesizing product nodes inside sum nodes
+		if product, ok := term.data.(*calcProduct); ok {
+			token, ok := product.convertToToken(whitespace)
+			if !ok {
+				return css_ast.Token{}, false
+			}
+			tokens = append(tokens, *token.Children...)
+		} else {
+			token, ok := term.data.convertToToken(whitespace)
+			if !ok {
+				return css_ast.Token{}, false
+			}
+			tokens = append(tokens, token)
+		}
+	}
+
+	return css_ast.Token{
+		Loc:      tokens[0].Loc,
+		Kind:     css_lexer.TOpenParen,
+		Text:     "(",
+		Children: &tokens,
+	}, true
+}
+
+func (c *calcProduct) convertToToken(whitespace css_ast.WhitespaceFlags) (css_ast.Token, bool) {
+	// Specification: https://www.w3.org/TR/css-values-4/#calc-serialize
+	tokens := make([]css_ast.Token, 0, len(c.terms)*2)
+	token, ok := c.terms[0].data.convertToToken(whitespace)
+	if !ok {
+		return css_ast.Token{}, false
+	}
+	tokens = append(tokens, token)
+
+	for _, term := range c.terms[1:] {
+		// If child is an Invert node, append " / " to s, then serialize the Invert’s child and append the result to s.
+		if invert, ok := term.data.(*calcInvert); ok {
+			token, ok := invert.term.data.convertToToken(whitespace)
+			if !ok {
+				return css_ast.Token{}, false
+			}
+			tokens = append(tokens, css_ast.Token{
+				Loc:        term.opLoc,
+				Kind:       css_lexer.TDelimSlash,
+				Text:       "/",
+				Whitespace: whitespace,
+			}, token)
+			continue
+		}
+
+		// Otherwise, append " * " to s, then serialize child and append the result to s.
+		token, ok := term.data.convertToToken(whitespace)
+		if !ok {
+			return css_ast.Token{}, false
+		}
+		tokens = append(tokens, css_ast.Token{
+			Loc:        term.opLoc,
+			Kind:       css_lexer.TDelimAsterisk,
+			Text:       "*",
+			Whitespace: whitespace,
+		}, token)
+	}
+
+	return css_ast.Token{
+		Loc:      tokens[0].Loc,
+		Kind:     css_lexer.TOpenParen,
+		Text:     "(",
+		Children: &tokens,
+	}, true
+}
+
+func (c *calcNegate) convertToToken(whitespace css_ast.WhitespaceFlags) (css_ast.Token, bool) {
+	// Specification: https://www.w3.org/TR/css-values-4/#calc-serialize
+	token, ok := c.term.data.convertToToken(whitespace)
+	if !ok {
+		return css_ast.Token{}, false
+	}
+	return css_ast.Token{
+		Kind: css_lexer.TOpenParen,
+		Text: "(",
+		Children: &[]css_ast.Token{
+			{Loc: c.term.opLoc, Kind: css_lexer.TNumber, Text: "-1"},
+			{Loc: c.term.opLoc, Kind: css_lexer.TDelimSlash, Text: "*", Whitespace: css_ast.WhitespaceBefore | css_ast.WhitespaceAfter},
+			token,
+		},
+	}, true
+}
+
+func (c *calcInvert) convertToToken(whitespace css_ast.WhitespaceFlags) (css_ast.Token, bool) {
+	// Specification: https://www.w3.org/TR/css-values-4/#calc-serialize
+	token, ok := c.term.data.convertToToken(whitespace)
+	if !ok {
+		return css_ast.Token{}, false
+	}
+	return css_ast.Token{
+		Kind: css_lexer.TOpenParen,
+		Text: "(",
+		Children: &[]css_ast.Token{
+			{Loc: c.term.opLoc, Kind: css_lexer.TNumber, Text: "1"},
+			{Loc: c.term.opLoc, Kind: css_lexer.TDelimSlash, Text: "/", Whitespace: css_ast.WhitespaceBefore | css_ast.WhitespaceAfter},
+			token,
+		},
+	}, true
+}
+
+func (c *calcNumeric) convertToToken(whitespace css_ast.WhitespaceFlags) (css_ast.Token, bool) {
+	text, ok := floatToStringForCalc(c.number)
+	if !ok {
+		return css_ast.Token{}, false
+	}
+	if c.unit == "" {
+		return css_ast.Token{
+			Loc:  c.loc,
+			Kind: css_lexer.TNumber,
+			Text: text,
+		}, true
+	}
+	if c.unit == "%" {
+		return css_ast.Token{
+			Loc:  c.loc,
+			Kind: css_lexer.TPercentage,
+			Text: text + "%",
+		}, true
+	}
+	return css_ast.Token{
+		Loc:        c.loc,
+		Kind:       css_lexer.TDimension,
+		Text:       text + c.unit,
+		UnitOffset: uint16(len(text)),
+	}, true
+}
+
+func (c *calcValue) convertToToken(whitespace css_ast.WhitespaceFlags) (css_ast.Token, bool) {
+	t := c.token
+	t.Whitespace = 0
+	return t, true
+}
+
+func (c *calcSum) partiallySimplify() calcTerm {
+	// Specification: https://www.w3.org/TR/css-values-4/#calc-simplification
+
+	// For each of root’s children that are Sum nodes, replace them with their children.
+	terms := make([]calcTermWithOp, 0, len(c.terms))
+	for _, term := range c.terms {
+		term.data = term.data.partiallySimplify()
+		if sum, ok := term.data.(*calcSum); ok {
+			terms = append(terms, sum.terms...)
+		} else {
+			terms = append(terms, term)
+		}
+	}
+
+	// For each set of root’s children that are numeric values with identical units, remove
+	// those children and replace them with a single numeric value containing the sum of the
+	// removed nodes, and with the same unit. (E.g. combine numbers, combine percentages,
+	// combine px values, etc.)
+	for i := 0; i < len(terms); i++ {
+		term := terms[i]
+		if numeric, ok := term.data.(*calcNumeric); ok {
+			end := i + 1
+			for j := end; j < len(terms); j++ {
+				term2 := terms[j]
+				if numeric2, ok := term2.data.(*calcNumeric); ok && strings.EqualFold(numeric2.unit, numeric.unit) {
+					numeric.number += numeric2.number
+				} else {
+					terms[end] = term2
+					end++
+				}
+			}
+			terms = terms[:end]
+		}
+	}
+
+	// If root has only a single child at this point, return the child.
+	if len(terms) == 1 {
+		return terms[0].data
+	}
+
+	// Otherwise, return root.
+	c.terms = terms
+	return c
+}
+
+func (c *calcProduct) partiallySimplify() calcTerm {
+	// Specification: https://www.w3.org/TR/css-values-4/#calc-simplification
+
+	// For each of root’s children that are Product nodes, replace them with their children.
+	terms := make([]calcTermWithOp, 0, len(c.terms))
+	for _, term := range c.terms {
+		term.data = term.data.partiallySimplify()
+		if product, ok := term.data.(*calcProduct); ok {
+			terms = append(terms, product.terms...)
+		} else {
+			terms = append(terms, term)
+		}
+	}
+
+	// If root has multiple children that are numbers (not percentages or dimensions), remove
+	// them and replace them with a single number containing the product of the removed nodes.
+	for i, term := range terms {
+		if numeric, ok := term.data.(*calcNumeric); ok && numeric.unit == "" {
+			end := i + 1
+			for j := end; j < len(terms); j++ {
+				term2 := terms[j]
+				if numeric2, ok := term2.data.(*calcNumeric); ok && numeric2.unit == "" {
+					numeric.number *= numeric2.number
+				} else {
+					terms[end] = term2
+					end++
+				}
+			}
+			terms = terms[:end]
+			break
+		}
+	}
+
+	// If root contains only numeric values and/or Invert nodes containing numeric values,
+	// and multiplying the types of all the children (noting that the type of an Invert
+	// node is the inverse of its child’s type) results in a type that matches any of the
+	// types that a math function can resolve to, return the result of multiplying all the
+	// values of the children (noting that the value of an Invert node is the reciprocal
+	// of its child’s value), expressed in the result’s canonical unit.
+	if len(terms) == 2 {
+		// Right now, only handle the case of two numbers, one of which has no unit
+		if first, ok := terms[0].data.(*calcNumeric); ok {
+			if second, ok := terms[1].data.(*calcNumeric); ok {
+				if first.unit == "" {
+					second.number *= first.number
+					return second
+				}
+				if second.unit == "" {
+					first.number *= second.number
+					return first
+				}
+			}
+		}
+	}
+
+	// ALGORITHM DEVIATION: Divide instead of multiply if the reciprocal is shorter
+	for i := 1; i < len(terms); i++ {
+		if numeric, ok := terms[i].data.(*calcNumeric); ok {
+			reciprocal := 1 / numeric.number
+			if multiply, ok := floatToStringForCalc(numeric.number); ok {
+				if divide, ok := floatToStringForCalc(reciprocal); ok && len(divide) < len(multiply) {
+					numeric.number = reciprocal
+					terms[i].data = &calcInvert{term: calcTermWithOp{
+						data:  numeric,
+						opLoc: terms[i].opLoc,
+					}}
+				}
+			}
+		}
+	}
+
+	// If root has only a single child at this point, return the child.
+	if len(terms) == 1 {
+		return terms[0].data
+	}
+
+	// Otherwise, return root.
+	c.terms = terms
+	return c
+}
+
+func (c *calcNegate) partiallySimplify() calcTerm {
+	// Specification: https://www.w3.org/TR/css-values-4/#calc-simplification
+
+	c.term.data = c.term.data.partiallySimplify()
+
+	// If root’s child is a numeric value, return an equivalent numeric value, but with the value negated (0 - value).
+	if numeric, ok := c.term.data.(*calcNumeric); ok {
+		numeric.number = -numeric.number
+		return numeric
+	}
+
+	// If root’s child is a Negate node, return the child’s child.
+	if negate, ok := c.term.data.(*calcNegate); ok {
+		return negate.term.data
+	}
+
+	return c
+}
+
+func (c *calcInvert) partiallySimplify() calcTerm {
+	// Specification: https://www.w3.org/TR/css-values-4/#calc-simplification
+
+	c.term.data = c.term.data.partiallySimplify()
+
+	// If root’s child is a number (not a percentage or dimension) return the reciprocal of the child’s value.
+	if numeric, ok := c.term.data.(*calcNumeric); ok && numeric.unit == "" {
+		numeric.number = 1 / numeric.number
+		return numeric
+	}
+
+	// If root’s child is an Invert node, return the child’s child.
+	if invert, ok := c.term.data.(*calcInvert); ok {
+		return invert.term.data
+	}
+
+	return c
+}
+
+func (c *calcNumeric) partiallySimplify() calcTerm {
+	return c
+}
+
+func (c *calcValue) partiallySimplify() calcTerm {
+	return c
+}
+
+func tryToParseCalcTerm(tokens []css_ast.Token) calcTerm {
+	// Specification: https://www.w3.org/TR/css-values-4/#calc-internal
+	terms := make([]calcTermWithOp, len(tokens))
+
+	for i, token := range tokens {
+		var term calcTerm
+		if token.Kind == css_lexer.TFunction && strings.EqualFold(token.Text, "var") {
+			// Using "var()" should bail because it can expand to any number of tokens
+			return nil
+		} else if token.Kind == css_lexer.TOpenParen || (token.Kind == css_lexer.TFunction && strings.EqualFold(token.Text, "calc")) {
+			term = tryToParseCalcTerm(*token.Children)
+			if term == nil {
+				return nil
+			}
+		} else if token.Kind == css_lexer.TNumber {
+			if number, err := strconv.ParseFloat(token.Text, 64); err == nil {
+				term = &calcNumeric{loc: token.Loc, number: number}
+			} else {
+				term = &calcValue{token: token}
+			}
+		} else if token.Kind == css_lexer.TPercentage {
+			if number, err := strconv.ParseFloat(token.PercentageValue(), 64); err == nil {
+				term = &calcNumeric{loc: token.Loc, number: number, unit: "%"}
+			} else {
+				term = &calcValue{token: token}
+			}
+		} else if token.Kind == css_lexer.TDimension {
+			if number, err := strconv.ParseFloat(token.DimensionValue(), 64); err == nil {
+				term = &calcNumeric{loc: token.Loc, number: number, unit: token.DimensionUnit()}
+			} else {
+				term = &calcValue{token: token}
+			}
+		} else if token.Kind == css_lexer.TIdent && strings.EqualFold(token.Text, "Infinity") {
+			term = &calcNumeric{loc: token.Loc, number: math.Inf(1)}
+		} else if token.Kind == css_lexer.TIdent && strings.EqualFold(token.Text, "-Infinity") {
+			term = &calcNumeric{loc: token.Loc, number: math.Inf(-1)}
+		} else if token.Kind == css_lexer.TIdent && strings.EqualFold(token.Text, "NaN") {
+			term = &calcNumeric{loc: token.Loc, number: math.NaN()}
+		} else {
+			term = &calcValue{
+				token: token,
+
+				// From the specification: "In addition, whitespace is required on both sides of the
+				// + and - operators. (The * and / operators can be used without white space around them.)"
+				isInvalidPlusOrMinus: i > 0 && i+1 < len(tokens) &&
+					(token.Kind == css_lexer.TDelimPlus || token.Kind == css_lexer.TDelimMinus) &&
+					(((token.Whitespace&css_ast.WhitespaceBefore) == 0 && (tokens[i-1].Whitespace&css_ast.WhitespaceAfter) == 0) ||
+						(token.Whitespace&css_ast.WhitespaceAfter) == 0 && (tokens[i+1].Whitespace&css_ast.WhitespaceBefore) == 0),
+			}
+		}
+		terms[i].data = term
+	}
+
+	// Collect children into Product and Invert nodes
+	first := 1
+	for first+1 < len(terms) {
+		// If this is a "*" or "/" operator
+		if value, ok := terms[first].data.(*calcValue); ok && (value.token.Kind == css_lexer.TDelimAsterisk || value.token.Kind == css_lexer.TDelimSlash) {
+			// Scan over the run
+			last := first
+			for last+3 < len(terms) {
+				if value, ok := terms[last+2].data.(*calcValue); ok && (value.token.Kind == css_lexer.TDelimAsterisk || value.token.Kind == css_lexer.TDelimSlash) {
+					last += 2
+				} else {
+					break
+				}
+			}
+
+			// Generate a node for the run
+			product := calcProduct{terms: make([]calcTermWithOp, (last-first)/2+2)}
+			for i := range product.terms {
+				term := terms[first+i*2-1]
+				if i > 0 {
+					op := terms[first+i*2-2].data.(*calcValue).token
+					term.opLoc = op.Loc
+					if op.Kind == css_lexer.TDelimSlash {
+						term.data = &calcInvert{term: term}
+					}
+				}
+				product.terms[i] = term
+			}
+
+			// Replace the run with a single node
+			terms[first-1].data = &product
+			terms = append(terms[:first], terms[last+2:]...)
+			continue
+		}
+
+		first++
+	}
+
+	// Collect children into Sum and Negate nodes
+	first = 1
+	for first+1 < len(terms) {
+		// If this is a "+" or "-" operator
+		if value, ok := terms[first].data.(*calcValue); ok && !value.isInvalidPlusOrMinus &&
+			(value.token.Kind == css_lexer.TDelimPlus || value.token.Kind == css_lexer.TDelimMinus) {
+			// Scan over the run
+			last := first
+			for last+3 < len(terms) {
+				if value, ok := terms[last+2].data.(*calcValue); ok && !value.isInvalidPlusOrMinus &&
+					(value.token.Kind == css_lexer.TDelimPlus || value.token.Kind == css_lexer.TDelimMinus) {
+					last += 2
+				} else {
+					break
+				}
+			}
+
+			// Generate a node for the run
+			sum := calcSum{terms: make([]calcTermWithOp, (last-first)/2+2)}
+			for i := range sum.terms {
+				term := terms[first+i*2-1]
+				if i > 0 {
+					op := terms[first+i*2-2].data.(*calcValue).token
+					term.opLoc = op.Loc
+					if op.Kind == css_lexer.TDelimMinus {
+						term.data = &calcNegate{term: term}
+					}
+				}
+				sum.terms[i] = term
+			}
+
+			// Replace the run with a single node
+			terms[first-1].data = &sum
+			terms = append(terms[:first], terms[last+2:]...)
+			continue
+		}
+
+		first++
+	}
+
+	// This only succeeds if everything reduces to a single term
+	if len(terms) == 1 {
+		return terms[0].data
+	}
+	return nil
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/css_printer/css_printer.go b/source/vendor/github.com/evanw/esbuild/internal/css_printer/css_printer.go
new file mode 100644
index 0000000..c010074
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/css_printer/css_printer.go
@@ -0,0 +1,1141 @@
+package css_printer
+
+import (
+	"fmt"
+	"strings"
+	"unicode/utf8"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/config"
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/logger"
+	"github.com/evanw/esbuild/internal/sourcemap"
+)
+
+const quoteForURL byte = 0
+
+type printer struct {
+	options                Options
+	symbols                ast.SymbolMap
+	importRecords          []ast.ImportRecord
+	css                    []byte
+	hasLegalComment        map[string]struct{}
+	extractedLegalComments []string
+	jsonMetadataImports    []string
+	builder                sourcemap.ChunkBuilder
+	oldLineStart           int
+	oldLineEnd             int
+}
+
+type Options struct {
+	// This will be present if the input file had a source map. In that case we
+	// want to map all the way back to the original input file(s).
+	InputSourceMap *sourcemap.SourceMap
+
+	// If we're writing out a source map, this table of line start indices lets
+	// us do binary search on to figure out what line a given AST node came from
+	LineOffsetTables []sourcemap.LineOffsetTable
+
+	// Local symbol renaming results go here
+	LocalNames map[ast.Ref]string
+
+	LineLimit           int
+	InputSourceIndex    uint32
+	UnsupportedFeatures compat.CSSFeature
+	MinifyWhitespace    bool
+	ASCIIOnly           bool
+	SourceMap           config.SourceMap
+	AddSourceMappings   bool
+	LegalComments       config.LegalComments
+	NeedsMetafile       bool
+}
+
+type PrintResult struct {
+	CSS                    []byte
+	ExtractedLegalComments []string
+	JSONMetadataImports    []string
+
+	// This source map chunk just contains the VLQ-encoded offsets for the "CSS"
+	// field above. It's not a full source map. The bundler will be joining many
+	// source map chunks together to form the final source map.
+	SourceMapChunk sourcemap.Chunk
+}
+
+func Print(tree css_ast.AST, symbols ast.SymbolMap, options Options) PrintResult {
+	p := printer{
+		options:       options,
+		symbols:       symbols,
+		importRecords: tree.ImportRecords,
+		builder:       sourcemap.MakeChunkBuilder(options.InputSourceMap, options.LineOffsetTables, options.ASCIIOnly),
+	}
+	for _, rule := range tree.Rules {
+		p.printRule(rule, 0, false)
+	}
+	result := PrintResult{
+		CSS:                    p.css,
+		ExtractedLegalComments: p.extractedLegalComments,
+		JSONMetadataImports:    p.jsonMetadataImports,
+	}
+	if options.SourceMap != config.SourceMapNone {
+		// This is expensive. Only do this if it's necessary. For example, skipping
+		// this if it's not needed sped up end-to-end parsing and printing of a
+		// large CSS file from 66ms to 52ms (around 25% faster).
+		result.SourceMapChunk = p.builder.GenerateChunk(p.css)
+	}
+	return result
+}
+
+func (p *printer) recordImportPathForMetafile(importRecordIndex uint32) {
+	if p.options.NeedsMetafile {
+		record := p.importRecords[importRecordIndex]
+		external := ""
+		if (record.Flags & ast.ShouldNotBeExternalInMetafile) == 0 {
+			external = ",\n          \"external\": true"
+		}
+		p.jsonMetadataImports = append(p.jsonMetadataImports, fmt.Sprintf("\n        {\n          \"path\": %s,\n          \"kind\": %s%s\n        }",
+			helpers.QuoteForJSON(record.Path.Text, p.options.ASCIIOnly),
+			helpers.QuoteForJSON(record.Kind.StringForMetafile(), p.options.ASCIIOnly),
+			external))
+	}
+}
+
+func (p *printer) printRule(rule css_ast.Rule, indent int32, omitTrailingSemicolon bool) {
+	if r, ok := rule.Data.(*css_ast.RComment); ok {
+		switch p.options.LegalComments {
+		case config.LegalCommentsNone:
+			return
+
+		case config.LegalCommentsEndOfFile,
+			config.LegalCommentsLinkedWithComment,
+			config.LegalCommentsExternalWithoutComment:
+
+			// Don't record the same legal comment more than once per file
+			if p.hasLegalComment == nil {
+				p.hasLegalComment = make(map[string]struct{})
+			} else if _, ok := p.hasLegalComment[r.Text]; ok {
+				return
+			}
+			p.hasLegalComment[r.Text] = struct{}{}
+			p.extractedLegalComments = append(p.extractedLegalComments, r.Text)
+			return
+		}
+	}
+
+	if p.options.LineLimit > 0 {
+		p.printNewlinePastLineLimit(indent)
+	}
+
+	if p.options.AddSourceMappings {
+		shouldPrintMapping := true
+		if indent == 0 || p.options.MinifyWhitespace {
+			switch rule.Data.(type) {
+			case *css_ast.RSelector, *css_ast.RQualified, *css_ast.RBadDeclaration:
+				// These rules will begin with a potentially more accurate mapping. We
+				// shouldn't print a mapping here if there's no indent in between this
+				// mapping and the rule.
+				shouldPrintMapping = false
+			}
+		}
+		if shouldPrintMapping {
+			p.builder.AddSourceMapping(rule.Loc, "", p.css)
+		}
+	}
+
+	if !p.options.MinifyWhitespace {
+		p.printIndent(indent)
+	}
+
+	switch r := rule.Data.(type) {
+	case *css_ast.RAtCharset:
+		// It's not valid to remove the space in between these two tokens
+		p.print("@charset ")
+
+		// It's not valid to print the string with single quotes
+		p.printQuotedWithQuote(r.Encoding, '"', 0)
+		p.print(";")
+
+	case *css_ast.RAtImport:
+		if p.options.MinifyWhitespace {
+			p.print("@import")
+		} else {
+			p.print("@import ")
+		}
+		record := p.importRecords[r.ImportRecordIndex]
+		var flags printQuotedFlags
+		if record.Flags.Has(ast.ContainsUniqueKey) {
+			flags |= printQuotedNoWrap
+		}
+		p.printQuoted(record.Path.Text, flags)
+		p.recordImportPathForMetafile(r.ImportRecordIndex)
+		if conditions := r.ImportConditions; conditions != nil {
+			space := !p.options.MinifyWhitespace
+			if len(conditions.Layers) > 0 {
+				if space {
+					p.print(" ")
+				}
+				p.printTokens(conditions.Layers, printTokensOpts{})
+				space = true
+			}
+			if len(conditions.Supports) > 0 {
+				if space {
+					p.print(" ")
+				}
+				p.printTokens(conditions.Supports, printTokensOpts{})
+				space = true
+			}
+			if len(conditions.Media) > 0 {
+				if space {
+					p.print(" ")
+				}
+				p.printTokens(conditions.Media, printTokensOpts{})
+			}
+		}
+		p.print(";")
+
+	case *css_ast.RAtKeyframes:
+		p.print("@")
+		p.printIdent(r.AtToken, identNormal, mayNeedWhitespaceAfter)
+		p.print(" ")
+		p.printSymbol(r.Name.Loc, r.Name.Ref, identNormal, canDiscardWhitespaceAfter)
+		if !p.options.MinifyWhitespace {
+			p.print(" ")
+		}
+		if p.options.MinifyWhitespace {
+			p.print("{")
+		} else {
+			p.print("{\n")
+		}
+		indent++
+		for _, block := range r.Blocks {
+			if p.options.AddSourceMappings {
+				p.builder.AddSourceMapping(block.Loc, "", p.css)
+			}
+			if !p.options.MinifyWhitespace {
+				p.printIndent(indent)
+			}
+			for i, sel := range block.Selectors {
+				if i > 0 {
+					if p.options.MinifyWhitespace {
+						p.print(",")
+					} else {
+						p.print(", ")
+					}
+				}
+				p.print(sel)
+			}
+			if !p.options.MinifyWhitespace {
+				p.print(" ")
+			}
+			p.printRuleBlock(block.Rules, indent, block.CloseBraceLoc)
+			if !p.options.MinifyWhitespace {
+				p.print("\n")
+			}
+		}
+		indent--
+		if p.options.AddSourceMappings && r.CloseBraceLoc.Start != 0 {
+			p.builder.AddSourceMapping(r.CloseBraceLoc, "", p.css)
+		}
+		if !p.options.MinifyWhitespace {
+			p.printIndent(indent)
+		}
+		p.print("}")
+
+	case *css_ast.RKnownAt:
+		p.print("@")
+		whitespace := mayNeedWhitespaceAfter
+		if len(r.Prelude) == 0 {
+			whitespace = canDiscardWhitespaceAfter
+		}
+		p.printIdent(r.AtToken, identNormal, whitespace)
+		if (!p.options.MinifyWhitespace && r.Rules != nil) || len(r.Prelude) > 0 {
+			p.print(" ")
+		}
+		p.printTokens(r.Prelude, printTokensOpts{})
+		if r.Rules == nil {
+			p.print(";")
+		} else {
+			if !p.options.MinifyWhitespace && len(r.Prelude) > 0 {
+				p.print(" ")
+			}
+			p.printRuleBlock(r.Rules, indent, r.CloseBraceLoc)
+		}
+
+	case *css_ast.RUnknownAt:
+		p.print("@")
+		whitespace := mayNeedWhitespaceAfter
+		if len(r.Prelude) == 0 {
+			whitespace = canDiscardWhitespaceAfter
+		}
+		p.printIdent(r.AtToken, identNormal, whitespace)
+		if (!p.options.MinifyWhitespace && len(r.Block) != 0) || len(r.Prelude) > 0 {
+			p.print(" ")
+		}
+		p.printTokens(r.Prelude, printTokensOpts{})
+		if !p.options.MinifyWhitespace && len(r.Block) != 0 && len(r.Prelude) > 0 {
+			p.print(" ")
+		}
+		if len(r.Block) == 0 {
+			p.print(";")
+		} else {
+			p.printTokens(r.Block, printTokensOpts{})
+		}
+
+	case *css_ast.RSelector:
+		p.printComplexSelectors(r.Selectors, indent, layoutMultiLine)
+		if !p.options.MinifyWhitespace {
+			p.print(" ")
+		}
+		p.printRuleBlock(r.Rules, indent, r.CloseBraceLoc)
+
+	case *css_ast.RQualified:
+		hasWhitespaceAfter := p.printTokens(r.Prelude, printTokensOpts{})
+		if !hasWhitespaceAfter && !p.options.MinifyWhitespace {
+			p.print(" ")
+		}
+		p.printRuleBlock(r.Rules, indent, r.CloseBraceLoc)
+
+	case *css_ast.RDeclaration:
+		p.printIdent(r.KeyText, identNormal, canDiscardWhitespaceAfter)
+		p.print(":")
+		hasWhitespaceAfter := p.printTokens(r.Value, printTokensOpts{
+			indent:        indent,
+			isDeclaration: true,
+		})
+		if r.Important {
+			if !hasWhitespaceAfter && !p.options.MinifyWhitespace && len(r.Value) > 0 {
+				p.print(" ")
+			}
+			p.print("!important")
+		}
+		if !omitTrailingSemicolon {
+			p.print(";")
+		}
+
+	case *css_ast.RBadDeclaration:
+		p.printTokens(r.Tokens, printTokensOpts{})
+		if !omitTrailingSemicolon {
+			p.print(";")
+		}
+
+	case *css_ast.RComment:
+		p.printIndentedComment(indent, r.Text)
+
+	case *css_ast.RAtLayer:
+		p.print("@layer")
+		for i, parts := range r.Names {
+			if i == 0 {
+				p.print(" ")
+			} else if !p.options.MinifyWhitespace {
+				p.print(", ")
+			} else {
+				p.print(",")
+			}
+			p.print(strings.Join(parts, "."))
+		}
+		if r.Rules == nil {
+			p.print(";")
+		} else {
+			if !p.options.MinifyWhitespace {
+				p.print(" ")
+			}
+			p.printRuleBlock(r.Rules, indent, r.CloseBraceLoc)
+		}
+
+	default:
+		panic("Internal error")
+	}
+
+	if !p.options.MinifyWhitespace {
+		p.print("\n")
+	}
+}
+
+func (p *printer) printIndentedComment(indent int32, text string) {
+	// Avoid generating a comment containing the character sequence "</style"
+	if !p.options.UnsupportedFeatures.Has(compat.InlineStyle) {
+		text = helpers.EscapeClosingTag(text, "/style")
+	}
+
+	// Re-indent multi-line comments
+	for {
+		newline := strings.IndexByte(text, '\n')
+		if newline == -1 {
+			break
+		}
+		p.print(text[:newline+1])
+		if !p.options.MinifyWhitespace {
+			p.printIndent(indent)
+		}
+		text = text[newline+1:]
+	}
+	p.print(text)
+}
+
+func (p *printer) printRuleBlock(rules []css_ast.Rule, indent int32, closeBraceLoc logger.Loc) {
+	if p.options.MinifyWhitespace {
+		p.print("{")
+	} else {
+		p.print("{\n")
+	}
+
+	for i, decl := range rules {
+		omitTrailingSemicolon := p.options.MinifyWhitespace && i+1 == len(rules)
+		p.printRule(decl, indent+1, omitTrailingSemicolon)
+	}
+
+	if p.options.AddSourceMappings && closeBraceLoc.Start != 0 {
+		p.builder.AddSourceMapping(closeBraceLoc, "", p.css)
+	}
+	if !p.options.MinifyWhitespace {
+		p.printIndent(indent)
+	}
+	p.print("}")
+}
+
+type selectorLayout uint8
+
+const (
+	layoutMultiLine selectorLayout = iota
+	layoutSingleLine
+)
+
+func (p *printer) printComplexSelectors(selectors []css_ast.ComplexSelector, indent int32, layout selectorLayout) {
+	for i, complex := range selectors {
+		if i > 0 {
+			if p.options.MinifyWhitespace {
+				p.print(",")
+				if p.options.LineLimit > 0 {
+					p.printNewlinePastLineLimit(indent)
+				}
+			} else if layout == layoutMultiLine {
+				p.print(",\n")
+				p.printIndent(indent)
+			} else {
+				p.print(", ")
+			}
+		}
+
+		for j, compound := range complex.Selectors {
+			p.printCompoundSelector(compound, j == 0, j+1 == len(complex.Selectors), indent)
+		}
+	}
+}
+
+func (p *printer) printCompoundSelector(sel css_ast.CompoundSelector, isFirst bool, isLast bool, indent int32) {
+	if !isFirst && sel.Combinator.Byte == 0 {
+		// A space is required in between compound selectors if there is no
+		// combinator in the middle. It's fine to convert "a + b" into "a+b"
+		// but not to convert "a b" into "ab".
+		if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit(indent) {
+			p.print(" ")
+		}
+	}
+
+	if sel.Combinator.Byte != 0 {
+		if !isFirst && !p.options.MinifyWhitespace {
+			p.print(" ")
+		}
+
+		if p.options.AddSourceMappings {
+			p.builder.AddSourceMapping(sel.Combinator.Loc, "", p.css)
+		}
+		p.css = append(p.css, sel.Combinator.Byte)
+
+		if (p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit(indent)) && !p.options.MinifyWhitespace {
+			p.print(" ")
+		}
+	}
+
+	if sel.TypeSelector != nil {
+		whitespace := mayNeedWhitespaceAfter
+		if len(sel.SubclassSelectors) > 0 {
+			// There is no chance of whitespace before a subclass selector or pseudo
+			// class selector
+			whitespace = canDiscardWhitespaceAfter
+		}
+		p.printNamespacedName(*sel.TypeSelector, whitespace)
+	}
+
+	if sel.HasNestingSelector() {
+		if p.options.AddSourceMappings {
+			p.builder.AddSourceMapping(logger.Loc{Start: int32(sel.NestingSelectorLoc.GetIndex())}, "", p.css)
+		}
+
+		p.print("&")
+	}
+
+	for i, ss := range sel.SubclassSelectors {
+		whitespace := mayNeedWhitespaceAfter
+
+		// There is no chance of whitespace between subclass selectors
+		if i+1 < len(sel.SubclassSelectors) {
+			whitespace = canDiscardWhitespaceAfter
+		}
+
+		if p.options.AddSourceMappings {
+			p.builder.AddSourceMapping(ss.Range.Loc, "", p.css)
+		}
+
+		switch s := ss.Data.(type) {
+		case *css_ast.SSHash:
+			p.print("#")
+
+			// This deliberately does not use identHash. From the specification:
+			// "In <id-selector>, the <hash-token>'s value must be an identifier."
+			p.printSymbol(s.Name.Loc, s.Name.Ref, identNormal, whitespace)
+
+		case *css_ast.SSClass:
+			p.print(".")
+			p.printSymbol(s.Name.Loc, s.Name.Ref, identNormal, whitespace)
+
+		case *css_ast.SSAttribute:
+			p.print("[")
+			p.printNamespacedName(s.NamespacedName, canDiscardWhitespaceAfter)
+			if s.MatcherOp != "" {
+				p.print(s.MatcherOp)
+				printAsIdent := false
+
+				// Print the value as an identifier if it's possible
+				if css_lexer.WouldStartIdentifierWithoutEscapes(s.MatcherValue) {
+					printAsIdent = true
+					for _, c := range s.MatcherValue {
+						if !css_lexer.IsNameContinue(c) {
+							printAsIdent = false
+							break
+						}
+					}
+				}
+
+				if printAsIdent {
+					p.printIdent(s.MatcherValue, identNormal, canDiscardWhitespaceAfter)
+				} else {
+					p.printQuoted(s.MatcherValue, 0)
+				}
+			}
+			if s.MatcherModifier != 0 {
+				p.print(" ")
+				p.print(string(rune(s.MatcherModifier)))
+			}
+			p.print("]")
+
+		case *css_ast.SSPseudoClass:
+			p.printPseudoClassSelector(*s, whitespace)
+
+		case *css_ast.SSPseudoClassWithSelectorList:
+			p.print(":")
+			p.print(s.Kind.String())
+			p.print("(")
+			if s.Index.A != "" || s.Index.B != "" {
+				p.printNthIndex(s.Index)
+				if len(s.Selectors) > 0 {
+					if p.options.MinifyWhitespace && s.Selectors[0].Selectors[0].TypeSelector == nil {
+						p.print(" of")
+					} else {
+						p.print(" of ")
+					}
+				}
+			}
+			p.printComplexSelectors(s.Selectors, indent, layoutSingleLine)
+			p.print(")")
+
+		default:
+			panic("Internal error")
+		}
+	}
+}
+
+func (p *printer) printNthIndex(index css_ast.NthIndex) {
+	if index.A != "" {
+		if index.A == "-1" {
+			p.print("-")
+		} else if index.A != "1" {
+			p.print(index.A)
+		}
+		p.print("n")
+		if index.B != "" {
+			if !strings.HasPrefix(index.B, "-") {
+				p.print("+")
+			}
+			p.print(index.B)
+		}
+	} else if index.B != "" {
+		p.print(index.B)
+	}
+}
+
+func (p *printer) printNamespacedName(nsName css_ast.NamespacedName, whitespace trailingWhitespace) {
+	if prefix := nsName.NamespacePrefix; prefix != nil {
+		if p.options.AddSourceMappings {
+			p.builder.AddSourceMapping(prefix.Range.Loc, "", p.css)
+		}
+
+		switch prefix.Kind {
+		case css_lexer.TIdent:
+			p.printIdent(prefix.Text, identNormal, canDiscardWhitespaceAfter)
+		case css_lexer.TDelimAsterisk:
+			p.print("*")
+		default:
+			panic("Internal error")
+		}
+
+		p.print("|")
+	}
+
+	if p.options.AddSourceMappings {
+		p.builder.AddSourceMapping(nsName.Name.Range.Loc, "", p.css)
+	}
+
+	switch nsName.Name.Kind {
+	case css_lexer.TIdent:
+		p.printIdent(nsName.Name.Text, identNormal, whitespace)
+	case css_lexer.TDelimAsterisk:
+		p.print("*")
+	case css_lexer.TDelimAmpersand:
+		p.print("&")
+	default:
+		panic("Internal error")
+	}
+}
+
+func (p *printer) printPseudoClassSelector(pseudo css_ast.SSPseudoClass, whitespace trailingWhitespace) {
+	if pseudo.IsElement {
+		p.print("::")
+	} else {
+		p.print(":")
+	}
+
+	// This checks for "nil" so we can distinguish ":is()" from ":is"
+	if pseudo.Args != nil {
+		p.printIdent(pseudo.Name, identNormal, canDiscardWhitespaceAfter)
+		p.print("(")
+		p.printTokens(pseudo.Args, printTokensOpts{})
+		p.print(")")
+	} else {
+		p.printIdent(pseudo.Name, identNormal, whitespace)
+	}
+}
+
+func (p *printer) print(text string) {
+	p.css = append(p.css, text...)
+}
+
+func bestQuoteCharForString(text string, forURL bool) byte {
+	forURLCost := 0
+	singleCost := 2
+	doubleCost := 2
+
+	for _, c := range text {
+		switch c {
+		case '\'':
+			forURLCost++
+			singleCost++
+
+		case '"':
+			forURLCost++
+			doubleCost++
+
+		case '(', ')', ' ', '\t':
+			forURLCost++
+
+		case '\\', '\n', '\r', '\f':
+			forURLCost++
+			singleCost++
+			doubleCost++
+		}
+	}
+
+	// Quotes can sometimes be omitted for URL tokens
+	if forURL && forURLCost < singleCost && forURLCost < doubleCost {
+		return quoteForURL
+	}
+
+	// Prefer double quotes to single quotes if there is no cost difference
+	if singleCost < doubleCost {
+		return '\''
+	}
+
+	return '"'
+}
+
+type printQuotedFlags uint8
+
+const (
+	printQuotedNoWrap printQuotedFlags = 1 << iota
+)
+
+func (p *printer) printQuoted(text string, flags printQuotedFlags) {
+	p.printQuotedWithQuote(text, bestQuoteCharForString(text, false), flags)
+}
+
+type escapeKind uint8
+
+const (
+	escapeNone escapeKind = iota
+	escapeBackslash
+	escapeHex
+)
+
+func (p *printer) printWithEscape(c rune, escape escapeKind, remainingText string, mayNeedWhitespaceAfter bool) {
+	var temp [utf8.UTFMax]byte
+
+	if escape == escapeBackslash && ((c >= '0' && c <= '9') || (c >= 'a' && c <= 'f') || (c >= 'A' && c <= 'F')) {
+		// Hexadecimal characters cannot use a plain backslash escape
+		escape = escapeHex
+	}
+
+	switch escape {
+	case escapeNone:
+		width := utf8.EncodeRune(temp[:], c)
+		p.css = append(p.css, temp[:width]...)
+
+	case escapeBackslash:
+		p.css = append(p.css, '\\')
+		width := utf8.EncodeRune(temp[:], c)
+		p.css = append(p.css, temp[:width]...)
+
+	case escapeHex:
+		text := fmt.Sprintf("\\%x", c)
+		p.css = append(p.css, text...)
+
+		// Make sure the next character is not interpreted as part of the escape sequence
+		if len(text) < 1+6 {
+			if next := utf8.RuneLen(c); next < len(remainingText) {
+				c = rune(remainingText[next])
+				if c == ' ' || c == '\t' || (c >= '0' && c <= '9') || (c >= 'a' && c <= 'f') || (c >= 'A' && c <= 'F') {
+					p.css = append(p.css, ' ')
+				}
+			} else if mayNeedWhitespaceAfter {
+				// If the last character is a hexadecimal escape, print a space afterwards
+				// for the escape sequence to consume. That way we're sure it won't
+				// accidentally consume a semantically significant space afterward.
+				p.css = append(p.css, ' ')
+			}
+		}
+	}
+}
+
+// Note: This function is hot in profiles
+func (p *printer) printQuotedWithQuote(text string, quote byte, flags printQuotedFlags) {
+	if quote != quoteForURL {
+		p.css = append(p.css, quote)
+	}
+
+	n := len(text)
+	i := 0
+	runStart := 0
+
+	// Only compute the line length if necessary
+	var startLineLength int
+	wrapLongLines := false
+	if p.options.LineLimit > 0 && quote != quoteForURL && (flags&printQuotedNoWrap) == 0 {
+		startLineLength = p.currentLineLength()
+		if startLineLength > p.options.LineLimit {
+			startLineLength = p.options.LineLimit
+		}
+		wrapLongLines = true
+	}
+
+	for i < n {
+		// Wrap long lines that are over the limit using escaped newlines
+		if wrapLongLines && startLineLength+i >= p.options.LineLimit {
+			if runStart < i {
+				p.css = append(p.css, text[runStart:i]...)
+				runStart = i
+			}
+			p.css = append(p.css, "\\\n"...)
+			startLineLength -= p.options.LineLimit
+		}
+
+		c, width := utf8.DecodeRuneInString(text[i:])
+		escape := escapeNone
+
+		switch c {
+		case '\x00', '\r', '\n', '\f':
+			// Use a hexadecimal escape for characters that would be invalid escapes
+			escape = escapeHex
+
+		case '\\', rune(quote):
+			escape = escapeBackslash
+
+		case '(', ')', ' ', '\t', '"', '\'':
+			// These characters must be escaped in URL tokens
+			if quote == quoteForURL {
+				escape = escapeBackslash
+			}
+
+		case '/':
+			// Avoid generating the sequence "</style" in CSS code
+			if !p.options.UnsupportedFeatures.Has(compat.InlineStyle) && i >= 1 && text[i-1] == '<' && i+6 <= len(text) && strings.EqualFold(text[i+1:i+6], "style") {
+				escape = escapeBackslash
+			}
+
+		default:
+			if (p.options.ASCIIOnly && c >= 0x80) || c == '\uFEFF' {
+				escape = escapeHex
+			}
+		}
+
+		if escape != escapeNone {
+			if runStart < i {
+				p.css = append(p.css, text[runStart:i]...)
+			}
+			p.printWithEscape(c, escape, text[i:], false)
+			runStart = i + width
+		}
+		i += width
+	}
+
+	if runStart < n {
+		p.css = append(p.css, text[runStart:]...)
+	}
+
+	if quote != quoteForURL {
+		p.css = append(p.css, quote)
+	}
+}
+
+func (p *printer) currentLineLength() int {
+	css := p.css
+	n := len(css)
+	stop := p.oldLineEnd
+
+	// Update "oldLineStart" to the start of the current line
+	for i := n; i > stop; i-- {
+		if c := css[i-1]; c == '\r' || c == '\n' {
+			p.oldLineStart = i
+			break
+		}
+	}
+
+	p.oldLineEnd = n
+	return n - p.oldLineStart
+}
+
+func (p *printer) printNewlinePastLineLimit(indent int32) bool {
+	if p.currentLineLength() < p.options.LineLimit {
+		return false
+	}
+	p.print("\n")
+	if !p.options.MinifyWhitespace {
+		p.printIndent(indent)
+	}
+	return true
+}
+
+type identMode uint8
+
+const (
+	identNormal identMode = iota
+	identHash
+	identDimensionUnit
+	identDimensionUnitAfterExponent
+)
+
+type trailingWhitespace uint8
+
+const (
+	mayNeedWhitespaceAfter trailingWhitespace = iota
+	canDiscardWhitespaceAfter
+)
+
+// Note: This function is hot in profiles
+func (p *printer) printIdent(text string, mode identMode, whitespace trailingWhitespace) {
+	n := len(text)
+
+	// Special escape behavior for the first character
+	initialEscape := escapeNone
+	switch mode {
+	case identNormal:
+		if !css_lexer.WouldStartIdentifierWithoutEscapes(text) {
+			initialEscape = escapeBackslash
+		}
+	case identDimensionUnit, identDimensionUnitAfterExponent:
+		if !css_lexer.WouldStartIdentifierWithoutEscapes(text) {
+			initialEscape = escapeBackslash
+		} else if n > 0 {
+			if c := text[0]; c >= '0' && c <= '9' {
+				// Unit: "2x"
+				initialEscape = escapeHex
+			} else if (c == 'e' || c == 'E') && mode != identDimensionUnitAfterExponent {
+				if n >= 2 && text[1] >= '0' && text[1] <= '9' {
+					// Unit: "e2x"
+					initialEscape = escapeHex
+				} else if n >= 3 && text[1] == '-' && text[2] >= '0' && text[2] <= '9' {
+					// Unit: "e-2x"
+					initialEscape = escapeHex
+				}
+			}
+		}
+	}
+
+	// Fast path: the identifier does not need to be escaped. This fast path is
+	// important for performance. For example, doing this sped up end-to-end
+	// parsing and printing of a large CSS file from 84ms to 66ms (around 25%
+	// faster).
+	if initialEscape == escapeNone {
+		for i := 0; i < n; i++ {
+			if c := text[i]; c >= 0x80 || !css_lexer.IsNameContinue(rune(c)) {
+				goto slowPath
+			}
+		}
+		p.css = append(p.css, text...)
+		return
+	slowPath:
+	}
+
+	// Slow path: the identifier needs to be escaped
+	for i, c := range text {
+		escape := escapeNone
+
+		if p.options.ASCIIOnly && c >= 0x80 {
+			escape = escapeHex
+		} else if c == '\r' || c == '\n' || c == '\f' || c == '\uFEFF' {
+			// Use a hexadecimal escape for characters that would be invalid escapes
+			escape = escapeHex
+		} else {
+			// Escape non-identifier characters
+			if !css_lexer.IsNameContinue(c) {
+				escape = escapeBackslash
+			}
+
+			// Special escape behavior for the first character
+			if i == 0 && initialEscape != escapeNone {
+				escape = initialEscape
+			}
+		}
+
+		// If the last character is a hexadecimal escape, print a space afterwards
+		// for the escape sequence to consume. That way we're sure it won't
+		// accidentally consume a semantically significant space afterward.
+		mayNeedWhitespaceAfter := whitespace == mayNeedWhitespaceAfter && escape != escapeNone && i+utf8.RuneLen(c) == n
+		p.printWithEscape(c, escape, text[i:], mayNeedWhitespaceAfter)
+	}
+}
+
+func (p *printer) printSymbol(loc logger.Loc, ref ast.Ref, mode identMode, whitespace trailingWhitespace) {
+	ref = ast.FollowSymbols(p.symbols, ref)
+	originalName := p.symbols.Get(ref).OriginalName
+	name, ok := p.options.LocalNames[ref]
+	if !ok {
+		name = originalName
+	}
+	if p.options.AddSourceMappings {
+		if originalName == name {
+			originalName = ""
+		}
+		p.builder.AddSourceMapping(loc, originalName, p.css)
+	}
+	p.printIdent(name, mode, whitespace)
+}
+
+func (p *printer) printIndent(indent int32) {
+	n := int(indent)
+	if p.options.LineLimit > 0 && n*2 >= p.options.LineLimit {
+		n = p.options.LineLimit / 2
+	}
+	for i := 0; i < n; i++ {
+		p.css = append(p.css, "  "...)
+	}
+}
+
+type printTokensOpts struct {
+	indent               int32
+	multiLineCommaPeriod uint8
+	isDeclaration        bool
+}
+
+func functionMultiLineCommaPeriod(token css_ast.Token) uint8 {
+	if token.Kind == css_lexer.TFunction {
+		commaCount := 0
+		for _, t := range *token.Children {
+			if t.Kind == css_lexer.TComma {
+				commaCount++
+			}
+		}
+
+		switch strings.ToLower(token.Text) {
+		case "linear-gradient", "radial-gradient", "conic-gradient",
+			"repeating-linear-gradient", "repeating-radial-gradient", "repeating-conic-gradient":
+			if commaCount >= 2 {
+				return 1
+			}
+
+		case "matrix":
+			if commaCount == 5 {
+				return 2
+			}
+
+		case "matrix3d":
+			if commaCount == 15 {
+				return 4
+			}
+		}
+	}
+	return 0
+}
+
+func (p *printer) printTokens(tokens []css_ast.Token, opts printTokensOpts) bool {
+	hasWhitespaceAfter := len(tokens) > 0 && (tokens[0].Whitespace&css_ast.WhitespaceBefore) != 0
+
+	// Pretty-print long comma-separated declarations of 3 or more items
+	commaPeriod := int(opts.multiLineCommaPeriod)
+	if !p.options.MinifyWhitespace && opts.isDeclaration {
+		commaCount := 0
+		for _, t := range tokens {
+			if t.Kind == css_lexer.TComma {
+				commaCount++
+				if commaCount >= 2 {
+					commaPeriod = 1
+					break
+				}
+			}
+			if t.Kind == css_lexer.TFunction && functionMultiLineCommaPeriod(t) > 0 {
+				commaPeriod = 1
+				break
+			}
+		}
+	}
+
+	commaCount := 0
+	for i, t := range tokens {
+		if t.Kind == css_lexer.TComma {
+			commaCount++
+		}
+		if t.Kind == css_lexer.TWhitespace {
+			hasWhitespaceAfter = true
+			continue
+		}
+		if hasWhitespaceAfter {
+			if commaPeriod > 0 && (i == 0 || (tokens[i-1].Kind == css_lexer.TComma && commaCount%commaPeriod == 0)) {
+				p.print("\n")
+				p.printIndent(opts.indent + 1)
+			} else if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit(opts.indent+1) {
+				p.print(" ")
+			}
+		}
+		hasWhitespaceAfter = (t.Whitespace&css_ast.WhitespaceAfter) != 0 ||
+			(i+1 < len(tokens) && (tokens[i+1].Whitespace&css_ast.WhitespaceBefore) != 0)
+
+		whitespace := mayNeedWhitespaceAfter
+		if !hasWhitespaceAfter {
+			whitespace = canDiscardWhitespaceAfter
+		}
+
+		if p.options.AddSourceMappings {
+			p.builder.AddSourceMapping(t.Loc, "", p.css)
+		}
+
+		switch t.Kind {
+		case css_lexer.TIdent:
+			p.printIdent(t.Text, identNormal, whitespace)
+
+		case css_lexer.TSymbol:
+			ref := ast.Ref{SourceIndex: p.options.InputSourceIndex, InnerIndex: t.PayloadIndex}
+			p.printSymbol(t.Loc, ref, identNormal, whitespace)
+
+		case css_lexer.TFunction:
+			p.printIdent(t.Text, identNormal, whitespace)
+			p.print("(")
+
+		case css_lexer.TDimension:
+			value := t.DimensionValue()
+			p.print(value)
+			mode := identDimensionUnit
+			if strings.ContainsAny(value, "eE") {
+				mode = identDimensionUnitAfterExponent
+			}
+			p.printIdent(t.DimensionUnit(), mode, whitespace)
+
+		case css_lexer.TAtKeyword:
+			p.print("@")
+			p.printIdent(t.Text, identNormal, whitespace)
+
+		case css_lexer.THash:
+			p.print("#")
+			p.printIdent(t.Text, identHash, whitespace)
+
+		case css_lexer.TString:
+			p.printQuoted(t.Text, 0)
+
+		case css_lexer.TURL:
+			record := p.importRecords[t.PayloadIndex]
+			text := record.Path.Text
+			tryToAvoidQuote := true
+			var flags printQuotedFlags
+			if record.Flags.Has(ast.ContainsUniqueKey) {
+				flags |= printQuotedNoWrap
+
+				// If the caller will be substituting a path here later using string
+				// substitution, then we can't be sure that it will form a valid URL
+				// token when unquoted (e.g. it may contain spaces). So we need to
+				// quote the unique key here just in case. For more info see this
+				// issue: https://github.com/evanw/esbuild/issues/3410
+				tryToAvoidQuote = false
+			} else if p.options.LineLimit > 0 && p.currentLineLength()+len(text) >= p.options.LineLimit {
+				tryToAvoidQuote = false
+			}
+			p.print("url(")
+			p.printQuotedWithQuote(text, bestQuoteCharForString(text, tryToAvoidQuote), flags)
+			p.print(")")
+			p.recordImportPathForMetafile(t.PayloadIndex)
+
+		case css_lexer.TUnterminatedString:
+			// We must end this with a newline so that this string stays unterminated
+			p.print(t.Text)
+			p.print("\n")
+			if !p.options.MinifyWhitespace {
+				p.printIndent(opts.indent)
+			}
+			hasWhitespaceAfter = false
+
+		default:
+			p.print(t.Text)
+		}
+
+		if t.Children != nil {
+			childCommaPeriod := uint8(0)
+
+			if commaPeriod > 0 && opts.isDeclaration {
+				childCommaPeriod = functionMultiLineCommaPeriod(t)
+			}
+
+			if childCommaPeriod > 0 {
+				opts.indent++
+				if !p.options.MinifyWhitespace {
+					p.print("\n")
+					p.printIndent(opts.indent + 1)
+				}
+			}
+
+			p.printTokens(*t.Children, printTokensOpts{
+				indent:               opts.indent,
+				multiLineCommaPeriod: childCommaPeriod,
+			})
+
+			if childCommaPeriod > 0 {
+				opts.indent--
+			}
+
+			switch t.Kind {
+			case css_lexer.TFunction:
+				p.print(")")
+
+			case css_lexer.TOpenParen:
+				p.print(")")
+
+			case css_lexer.TOpenBrace:
+				p.print("}")
+
+			case css_lexer.TOpenBracket:
+				p.print("]")
+			}
+		}
+	}
+	if hasWhitespaceAfter {
+		p.print(" ")
+	}
+	return hasWhitespaceAfter
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/fs/error_other.go b/source/vendor/github.com/evanw/esbuild/internal/fs/error_other.go
new file mode 100644
index 0000000..bd8647d
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/fs/error_other.go
@@ -0,0 +1,9 @@
+//go:build (!js || !wasm) && !windows
+// +build !js !wasm
+// +build !windows
+
+package fs
+
+func is_ERROR_INVALID_NAME(err error) bool {
+	return false
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/fs/error_wasm+windows.go b/source/vendor/github.com/evanw/esbuild/internal/fs/error_wasm+windows.go
new file mode 100644
index 0000000..1ddecff
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/fs/error_wasm+windows.go
@@ -0,0 +1,17 @@
+//go:build (js && wasm) || windows
+// +build js,wasm windows
+
+package fs
+
+import "syscall"
+
+// This check is here in a conditionally-compiled file because Go's standard
+// library for Plan 9 doesn't define a type called "syscall.Errno". Plan 9 is
+// not a supported operating system but someone wanted to be able to compile
+// esbuild for Plan 9 anyway.
+func is_ERROR_INVALID_NAME(err error) bool {
+	// This has been copied from golang.org/x/sys/windows
+	const ERROR_INVALID_NAME syscall.Errno = 123
+
+	return err == ERROR_INVALID_NAME
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/fs/filepath.go b/source/vendor/github.com/evanw/esbuild/internal/fs/filepath.go
new file mode 100644
index 0000000..edab22c
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/fs/filepath.go
@@ -0,0 +1,649 @@
+// Code in this file has been forked from the "filepath" module in the Go
+// source code to work around bugs with the WebAssembly build target. More
+// information about why here: https://github.com/golang/go/issues/43768.
+
+////////////////////////////////////////////////////////////////////////////////
+
+// Copyright (c) 2009 The Go Authors. All rights reserved.
+//
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+//    * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+//    * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following disclaimer
+// in the documentation and/or other materials provided with the
+// distribution.
+//    * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived from
+// this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+package fs
+
+import (
+	"errors"
+	"os"
+	"strings"
+	"syscall"
+)
+
+type goFilepath struct {
+	cwd           string
+	isWindows     bool
+	pathSeparator byte
+}
+
+func isSlash(c uint8) bool {
+	return c == '\\' || c == '/'
+}
+
+// reservedNames lists reserved Windows names. Search for PRN in
+// https://docs.microsoft.com/en-us/windows/desktop/fileio/naming-a-file
+// for details.
+var reservedNames = []string{
+	"CON", "PRN", "AUX", "NUL",
+	"COM1", "COM2", "COM3", "COM4", "COM5", "COM6", "COM7", "COM8", "COM9",
+	"LPT1", "LPT2", "LPT3", "LPT4", "LPT5", "LPT6", "LPT7", "LPT8", "LPT9",
+}
+
+// isReservedName returns true, if path is Windows reserved name.
+// See reservedNames for the full list.
+func isReservedName(path string) bool {
+	if len(path) == 0 {
+		return false
+	}
+	for _, reserved := range reservedNames {
+		if strings.EqualFold(path, reserved) {
+			return true
+		}
+	}
+	return false
+}
+
+// IsAbs reports whether the path is absolute.
+func (fp goFilepath) isAbs(path string) bool {
+	if !fp.isWindows {
+		return strings.HasPrefix(path, "/")
+	}
+	if isReservedName(path) {
+		return true
+	}
+	l := fp.volumeNameLen(path)
+	if l == 0 {
+		return false
+	}
+	path = path[l:]
+	if path == "" {
+		return false
+	}
+	return isSlash(path[0])
+}
+
+// Abs returns an absolute representation of path.
+// If the path is not absolute it will be joined with the current
+// working directory to turn it into an absolute path. The absolute
+// path name for a given file is not guaranteed to be unique.
+// Abs calls Clean on the result.
+func (fp goFilepath) abs(path string) (string, error) {
+	if fp.isAbs(path) {
+		return fp.clean(path), nil
+	}
+	return fp.join([]string{fp.cwd, path}), nil
+}
+
+// IsPathSeparator reports whether c is a directory separator character.
+func (fp goFilepath) isPathSeparator(c uint8) bool {
+	return c == '/' || (fp.isWindows && c == '\\')
+}
+
+// volumeNameLen returns length of the leading volume name on Windows.
+// It returns 0 elsewhere.
+func (fp goFilepath) volumeNameLen(path string) int {
+	if !fp.isWindows {
+		return 0
+	}
+	if len(path) < 2 {
+		return 0
+	}
+	// with drive letter
+	c := path[0]
+	if path[1] == ':' && ('a' <= c && c <= 'z' || 'A' <= c && c <= 'Z') {
+		return 2
+	}
+	// is it UNC? https://msdn.microsoft.com/en-us/library/windows/desktop/aa365247(v=vs.85).aspx
+	if l := len(path); l >= 5 && isSlash(path[0]) && isSlash(path[1]) &&
+		!isSlash(path[2]) && path[2] != '.' {
+		// first, leading `\\` and next shouldn't be `\`. its server name.
+		for n := 3; n < l-1; n++ {
+			// second, next '\' shouldn't be repeated.
+			if isSlash(path[n]) {
+				n++
+				// third, following something characters. its share name.
+				if !isSlash(path[n]) {
+					if path[n] == '.' {
+						break
+					}
+					for ; n < l; n++ {
+						if isSlash(path[n]) {
+							break
+						}
+					}
+					return n
+				}
+				break
+			}
+		}
+	}
+	return 0
+}
+
+// EvalSymlinks returns the path name after the evaluation of any symbolic
+// links.
+// If path is relative the result will be relative to the current directory,
+// unless one of the components is an absolute symbolic link.
+// EvalSymlinks calls Clean on the result.
+func (fp goFilepath) evalSymlinks(path string) (string, error) {
+	volLen := fp.volumeNameLen(path)
+	pathSeparator := string(fp.pathSeparator)
+
+	if volLen < len(path) && fp.isPathSeparator(path[volLen]) {
+		volLen++
+	}
+	vol := path[:volLen]
+	dest := vol
+	linksWalked := 0
+	for start, end := volLen, volLen; start < len(path); start = end {
+		for start < len(path) && fp.isPathSeparator(path[start]) {
+			start++
+		}
+		end = start
+		for end < len(path) && !fp.isPathSeparator(path[end]) {
+			end++
+		}
+
+		// On Windows, "." can be a symlink.
+		// We look it up, and use the value if it is absolute.
+		// If not, we just return ".".
+		isWindowsDot := fp.isWindows && path[fp.volumeNameLen(path):] == "."
+
+		// The next path component is in path[start:end].
+		if end == start {
+			// No more path components.
+			break
+		} else if path[start:end] == "." && !isWindowsDot {
+			// Ignore path component ".".
+			continue
+		} else if path[start:end] == ".." {
+			// Back up to previous component if possible.
+			// Note that volLen includes any leading slash.
+
+			// Set r to the index of the last slash in dest,
+			// after the volume.
+			var r int
+			for r = len(dest) - 1; r >= volLen; r-- {
+				if fp.isPathSeparator(dest[r]) {
+					break
+				}
+			}
+			if r < volLen || dest[r+1:] == ".." {
+				// Either path has no slashes
+				// (it's empty or just "C:")
+				// or it ends in a ".." we had to keep.
+				// Either way, keep this "..".
+				if len(dest) > volLen {
+					dest += pathSeparator
+				}
+				dest += ".."
+			} else {
+				// Discard everything since the last slash.
+				dest = dest[:r]
+			}
+			continue
+		}
+
+		// Ordinary path component. Add it to result.
+
+		if len(dest) > fp.volumeNameLen(dest) && !fp.isPathSeparator(dest[len(dest)-1]) {
+			dest += pathSeparator
+		}
+
+		dest += path[start:end]
+
+		// Resolve symlink.
+
+		fi, err := os.Lstat(dest)
+		if err != nil {
+			return "", err
+		}
+
+		if fi.Mode()&os.ModeSymlink == 0 {
+			if !fi.Mode().IsDir() && end < len(path) {
+				return "", syscall.ENOTDIR
+			}
+			continue
+		}
+
+		// Found symlink.
+
+		linksWalked++
+		if linksWalked > 255 {
+			return "", errors.New("EvalSymlinks: too many links")
+		}
+
+		link, err := os.Readlink(dest)
+		if err != nil {
+			return "", err
+		}
+
+		if isWindowsDot && !fp.isAbs(link) {
+			// On Windows, if "." is a relative symlink,
+			// just return ".".
+			break
+		}
+
+		path = link + path[end:]
+
+		v := fp.volumeNameLen(link)
+		if v > 0 {
+			// Symlink to drive name is an absolute path.
+			if v < len(link) && fp.isPathSeparator(link[v]) {
+				v++
+			}
+			vol = link[:v]
+			dest = vol
+			end = len(vol)
+		} else if len(link) > 0 && fp.isPathSeparator(link[0]) {
+			// Symlink to absolute path.
+			dest = link[:1]
+			end = 1
+		} else {
+			// Symlink to relative path; replace last
+			// path component in dest.
+			var r int
+			for r = len(dest) - 1; r >= volLen; r-- {
+				if fp.isPathSeparator(dest[r]) {
+					break
+				}
+			}
+			if r < volLen {
+				dest = vol
+			} else {
+				dest = dest[:r]
+			}
+			end = 0
+		}
+	}
+	return fp.clean(dest), nil
+}
+
+// A lazybuf is a lazily constructed path buffer.
+// It supports append, reading previously appended bytes,
+// and retrieving the final string. It does not allocate a buffer
+// to hold the output until that output diverges from s.
+type lazybuf struct {
+	path       string
+	volAndPath string
+	buf        []byte
+	w          int
+	volLen     int
+}
+
+func (b *lazybuf) index(i int) byte {
+	if b.buf != nil {
+		return b.buf[i]
+	}
+	return b.path[i]
+}
+
+func (b *lazybuf) append(c byte) {
+	if b.buf == nil {
+		if b.w < len(b.path) && b.path[b.w] == c {
+			b.w++
+			return
+		}
+		b.buf = make([]byte, len(b.path))
+		copy(b.buf, b.path[:b.w])
+	}
+	b.buf[b.w] = c
+	b.w++
+}
+
+func (b *lazybuf) string() string {
+	if b.buf == nil {
+		return b.volAndPath[:b.volLen+b.w]
+	}
+	return b.volAndPath[:b.volLen] + string(b.buf[:b.w])
+}
+
+// FromSlash returns the result of replacing each slash ('/') character
+// in path with a separator character. Multiple slashes are replaced
+// by multiple separators.
+func (fp goFilepath) fromSlash(path string) string {
+	if !fp.isWindows {
+		return path
+	}
+	return strings.ReplaceAll(path, "/", "\\")
+}
+
+// Clean returns the shortest path name equivalent to path
+// by purely lexical processing. It applies the following rules
+// iteratively until no further processing can be done:
+//
+//  1. Replace multiple Separator elements with a single one.
+//  2. Eliminate each . path name element (the current directory).
+//  3. Eliminate each inner .. path name element (the parent directory)
+//     along with the non-.. element that precedes it.
+//  4. Eliminate .. elements that begin a rooted path:
+//     that is, replace "/.." by "/" at the beginning of a path,
+//     assuming Separator is '/'.
+//
+// The returned path ends in a slash only if it represents a root directory,
+// such as "/" on Unix or `C:\` on Windows.
+//
+// Finally, any occurrences of slash are replaced by Separator.
+//
+// If the result of this process is an empty string, Clean
+// returns the string ".".
+//
+// See also Rob Pike, "Lexical File Names in Plan 9 or
+// Getting Dot-Dot Right,"
+// https://9p.io/sys/doc/lexnames.html
+func (fp goFilepath) clean(path string) string {
+	originalPath := path
+	volLen := fp.volumeNameLen(path)
+	path = path[volLen:]
+	if path == "" {
+		if volLen > 1 && originalPath[1] != ':' {
+			// should be UNC
+			return fp.fromSlash(originalPath)
+		}
+		return originalPath + "."
+	}
+	rooted := fp.isPathSeparator(path[0])
+
+	// Invariants:
+	//	reading from path; r is index of next byte to process.
+	//	writing to buf; w is index of next byte to write.
+	//	dotdot is index in buf where .. must stop, either because
+	//		it is the leading slash or it is a leading ../../.. prefix.
+	n := len(path)
+	out := lazybuf{path: path, volAndPath: originalPath, volLen: volLen}
+	r, dotdot := 0, 0
+	if rooted {
+		out.append(fp.pathSeparator)
+		r, dotdot = 1, 1
+	}
+
+	for r < n {
+		switch {
+		case fp.isPathSeparator(path[r]):
+			// empty path element
+			r++
+		case path[r] == '.' && (r+1 == n || fp.isPathSeparator(path[r+1])):
+			// . element
+			r++
+		case path[r] == '.' && path[r+1] == '.' && (r+2 == n || fp.isPathSeparator(path[r+2])):
+			// .. element: remove to last separator
+			r += 2
+			switch {
+			case out.w > dotdot:
+				// can backtrack
+				out.w--
+				for out.w > dotdot && !fp.isPathSeparator(out.index(out.w)) {
+					out.w--
+				}
+			case !rooted:
+				// cannot backtrack, but not rooted, so append .. element.
+				if out.w > 0 {
+					out.append(fp.pathSeparator)
+				}
+				out.append('.')
+				out.append('.')
+				dotdot = out.w
+			}
+		default:
+			// real path element.
+			// add slash if needed
+			if rooted && out.w != 1 || !rooted && out.w != 0 {
+				out.append(fp.pathSeparator)
+			}
+			// copy element
+			for ; r < n && !fp.isPathSeparator(path[r]); r++ {
+				out.append(path[r])
+			}
+		}
+	}
+
+	// Turn empty string into "."
+	if out.w == 0 {
+		out.append('.')
+	}
+
+	return fp.fromSlash(out.string())
+}
+
+// VolumeName returns leading volume name.
+// Given "C:\foo\bar" it returns "C:" on Windows.
+// Given "\\host\share\foo" it returns "\\host\share".
+// On other platforms it returns "".
+func (fp goFilepath) volumeName(path string) string {
+	return path[:fp.volumeNameLen(path)]
+}
+
+// Base returns the last element of path.
+// Trailing path separators are removed before extracting the last element.
+// If the path is empty, Base returns ".".
+// If the path consists entirely of separators, Base returns a single separator.
+func (fp goFilepath) base(path string) string {
+	if path == "" {
+		return "."
+	}
+	// Strip trailing slashes.
+	for len(path) > 0 && fp.isPathSeparator(path[len(path)-1]) {
+		path = path[0 : len(path)-1]
+	}
+	// Throw away volume name
+	path = path[len(fp.volumeName(path)):]
+	// Find the last element
+	i := len(path) - 1
+	for i >= 0 && !fp.isPathSeparator(path[i]) {
+		i--
+	}
+	if i >= 0 {
+		path = path[i+1:]
+	}
+	// If empty now, it had only slashes.
+	if path == "" {
+		return string(fp.pathSeparator)
+	}
+	return path
+}
+
+// Dir returns all but the last element of path, typically the path's directory.
+// After dropping the final element, Dir calls Clean on the path and trailing
+// slashes are removed.
+// If the path is empty, Dir returns ".".
+// If the path consists entirely of separators, Dir returns a single separator.
+// The returned path does not end in a separator unless it is the root directory.
+func (fp goFilepath) dir(path string) string {
+	vol := fp.volumeName(path)
+	i := len(path) - 1
+	for i >= len(vol) && !fp.isPathSeparator(path[i]) {
+		i--
+	}
+	dir := fp.clean(path[len(vol) : i+1])
+	if dir == "." && len(vol) > 2 {
+		// must be UNC
+		return vol
+	}
+	return vol + dir
+}
+
+// Ext returns the file name extension used by path.
+// The extension is the suffix beginning at the final dot
+// in the final element of path; it is empty if there is
+// no dot.
+func (fp goFilepath) ext(path string) string {
+	for i := len(path) - 1; i >= 0 && !fp.isPathSeparator(path[i]); i-- {
+		if path[i] == '.' {
+			return path[i:]
+		}
+	}
+	return ""
+}
+
+// Join joins any number of path elements into a single path,
+// separating them with an OS specific Separator. Empty elements
+// are ignored. The result is Cleaned. However, if the argument
+// list is empty or all its elements are empty, Join returns
+// an empty string.
+// On Windows, the result will only be a UNC path if the first
+// non-empty element is a UNC path.
+func (fp goFilepath) join(elem []string) string {
+	for i, e := range elem {
+		if e != "" {
+			if fp.isWindows {
+				return fp.joinNonEmpty(elem[i:])
+			}
+			return fp.clean(strings.Join(elem[i:], string(fp.pathSeparator)))
+		}
+	}
+	return ""
+}
+
+// joinNonEmpty is like join, but it assumes that the first element is non-empty.
+func (fp goFilepath) joinNonEmpty(elem []string) string {
+	if len(elem[0]) == 2 && elem[0][1] == ':' {
+		// First element is drive letter without terminating slash.
+		// Keep path relative to current directory on that drive.
+		// Skip empty elements.
+		i := 1
+		for ; i < len(elem); i++ {
+			if elem[i] != "" {
+				break
+			}
+		}
+		return fp.clean(elem[0] + strings.Join(elem[i:], string(fp.pathSeparator)))
+	}
+	// The following logic prevents Join from inadvertently creating a
+	// UNC path on Windows. Unless the first element is a UNC path, Join
+	// shouldn't create a UNC path. See golang.org/issue/9167.
+	p := fp.clean(strings.Join(elem, string(fp.pathSeparator)))
+	if !fp.isUNC(p) {
+		return p
+	}
+	// p == UNC only allowed when the first element is a UNC path.
+	head := fp.clean(elem[0])
+	if fp.isUNC(head) {
+		return p
+	}
+	// head + tail == UNC, but joining two non-UNC paths should not result
+	// in a UNC path. Undo creation of UNC path.
+	tail := fp.clean(strings.Join(elem[1:], string(fp.pathSeparator)))
+	if head[len(head)-1] == fp.pathSeparator {
+		return head + tail
+	}
+	return head + string(fp.pathSeparator) + tail
+}
+
+// isUNC reports whether path is a UNC path.
+func (fp goFilepath) isUNC(path string) bool {
+	return fp.volumeNameLen(path) > 2
+}
+
+// Rel returns a relative path that is lexically equivalent to targpath when
+// joined to basepath with an intervening separator. That is,
+// Join(basepath, Rel(basepath, targpath)) is equivalent to targpath itself.
+// On success, the returned path will always be relative to basepath,
+// even if basepath and targpath share no elements.
+// An error is returned if targpath can't be made relative to basepath or if
+// knowing the current working directory would be necessary to compute it.
+// Rel calls Clean on the result.
+func (fp goFilepath) rel(basepath, targpath string) (string, error) {
+	baseVol := fp.volumeName(basepath)
+	targVol := fp.volumeName(targpath)
+	base := fp.clean(basepath)
+	targ := fp.clean(targpath)
+	if fp.sameWord(targ, base) {
+		return ".", nil
+	}
+	base = base[len(baseVol):]
+	targ = targ[len(targVol):]
+	if base == "." {
+		base = ""
+	}
+	// Can't use IsAbs - `\a` and `a` are both relative in Windows.
+	baseSlashed := len(base) > 0 && base[0] == fp.pathSeparator
+	targSlashed := len(targ) > 0 && targ[0] == fp.pathSeparator
+	if baseSlashed != targSlashed || !fp.sameWord(baseVol, targVol) {
+		return "", errors.New("Rel: can't make " + targpath + " relative to " + basepath)
+	}
+	// Position base[b0:bi] and targ[t0:ti] at the first differing elements.
+	bl := len(base)
+	tl := len(targ)
+	var b0, bi, t0, ti int
+	for {
+		for bi < bl && base[bi] != fp.pathSeparator {
+			bi++
+		}
+		for ti < tl && targ[ti] != fp.pathSeparator {
+			ti++
+		}
+		if !fp.sameWord(targ[t0:ti], base[b0:bi]) {
+			break
+		}
+		if bi < bl {
+			bi++
+		}
+		if ti < tl {
+			ti++
+		}
+		b0 = bi
+		t0 = ti
+	}
+	if base[b0:bi] == ".." {
+		return "", errors.New("Rel: can't make " + targpath + " relative to " + basepath)
+	}
+	if b0 != bl {
+		// Base elements left. Must go up before going down.
+		seps := strings.Count(base[b0:bl], string(fp.pathSeparator))
+		size := 2 + seps*3
+		if tl != t0 {
+			size += 1 + tl - t0
+		}
+		buf := make([]byte, size)
+		n := copy(buf, "..")
+		for i := 0; i < seps; i++ {
+			buf[n] = fp.pathSeparator
+			copy(buf[n+1:], "..")
+			n += 3
+		}
+		if t0 != tl {
+			buf[n] = fp.pathSeparator
+			copy(buf[n+1:], targ[t0:])
+		}
+		return string(buf), nil
+	}
+	return targ[t0:], nil
+}
+
+func (fp goFilepath) sameWord(a, b string) bool {
+	if !fp.isWindows {
+		return a == b
+	}
+	return strings.EqualFold(a, b)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/fs/fs.go b/source/vendor/github.com/evanw/esbuild/internal/fs/fs.go
new file mode 100644
index 0000000..ccfcc6c
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/fs/fs.go
@@ -0,0 +1,287 @@
+package fs
+
+// Most of esbuild's internals use this file system abstraction instead of
+// using native file system APIs. This lets us easily mock the file system
+// for tests and also implement Yarn's virtual ".zip" file system overlay.
+
+import (
+	"errors"
+	"os"
+	"sort"
+	"strings"
+	"sync"
+	"syscall"
+)
+
+type EntryKind uint8
+
+const (
+	DirEntry  EntryKind = 1
+	FileEntry EntryKind = 2
+)
+
+type Entry struct {
+	symlink  string
+	dir      string
+	base     string
+	mutex    sync.Mutex
+	kind     EntryKind
+	needStat bool
+}
+
+func (e *Entry) Kind(fs FS) EntryKind {
+	e.mutex.Lock()
+	defer e.mutex.Unlock()
+	if e.needStat {
+		e.needStat = false
+		e.symlink, e.kind = fs.kind(e.dir, e.base)
+	}
+	return e.kind
+}
+
+func (e *Entry) Symlink(fs FS) string {
+	e.mutex.Lock()
+	defer e.mutex.Unlock()
+	if e.needStat {
+		e.needStat = false
+		e.symlink, e.kind = fs.kind(e.dir, e.base)
+	}
+	return e.symlink
+}
+
+type accessedEntries struct {
+	wasPresent map[string]bool
+
+	// If this is nil, "SortedKeys()" was not accessed. This means we should
+	// check for whether this directory has changed or not by seeing if any of
+	// the entries in the "wasPresent" map have changed in "present or not"
+	// status, since the only access was to individual entries via "Get()".
+	//
+	// If this is non-nil, "SortedKeys()" was accessed. This means we should
+	// check for whether this directory has changed or not by checking the
+	// "allEntries" array for equality with the existing entries list, since the
+	// code asked for all entries and may have used the presence or absence of
+	// entries in that list.
+	//
+	// The goal of having these two checks is to be as narrow as possible to
+	// avoid unnecessary rebuilds. If only "Get()" is called on a few entries,
+	// then we won't invalidate the build if random unrelated entries are added
+	// or removed. But if "SortedKeys()" is called, we need to invalidate the
+	// build if anything about the set of entries in this directory is changed.
+	allEntries []string
+
+	mutex sync.Mutex
+}
+
+type DirEntries struct {
+	data            map[string]*Entry
+	accessedEntries *accessedEntries
+	dir             string
+}
+
+func MakeEmptyDirEntries(dir string) DirEntries {
+	return DirEntries{dir: dir, data: make(map[string]*Entry)}
+}
+
+type DifferentCase struct {
+	Dir    string
+	Query  string
+	Actual string
+}
+
+func (entries DirEntries) Get(query string) (*Entry, *DifferentCase) {
+	if entries.data != nil {
+		key := strings.ToLower(query)
+		entry := entries.data[key]
+
+		// Track whether this specific entry was present or absent for watch mode
+		if accessed := entries.accessedEntries; accessed != nil {
+			accessed.mutex.Lock()
+			accessed.wasPresent[key] = entry != nil
+			accessed.mutex.Unlock()
+		}
+
+		if entry != nil {
+			if entry.base != query {
+				return entry, &DifferentCase{
+					Dir:    entries.dir,
+					Query:  query,
+					Actual: entry.base,
+				}
+			}
+			return entry, nil
+		}
+	}
+
+	return nil, nil
+}
+
+// This function lets you "peek" at the number of entries without watch mode
+// considering the number of entries as having been observed. This is used when
+// generating debug log messages to log the number of entries without causing
+// watch mode to rebuild when the number of entries has been changed.
+func (entries DirEntries) PeekEntryCount() int {
+	if entries.data != nil {
+		return len(entries.data)
+	}
+	return 0
+}
+
+func (entries DirEntries) SortedKeys() (keys []string) {
+	if entries.data != nil {
+		keys = make([]string, 0, len(entries.data))
+		for _, entry := range entries.data {
+			keys = append(keys, entry.base)
+		}
+		sort.Strings(keys)
+
+		// Track the exact set of all entries for watch mode
+		if entries.accessedEntries != nil {
+			entries.accessedEntries.mutex.Lock()
+			entries.accessedEntries.allEntries = keys
+			entries.accessedEntries.mutex.Unlock()
+		}
+
+		return keys
+	}
+
+	return
+}
+
+type OpenedFile interface {
+	Len() int
+	Read(start int, end int) ([]byte, error)
+	Close() error
+}
+
+type InMemoryOpenedFile struct {
+	Contents []byte
+}
+
+func (f *InMemoryOpenedFile) Len() int {
+	return len(f.Contents)
+}
+
+func (f *InMemoryOpenedFile) Read(start int, end int) ([]byte, error) {
+	return []byte(f.Contents[start:end]), nil
+}
+
+func (f *InMemoryOpenedFile) Close() error {
+	return nil
+}
+
+type FS interface {
+	// The returned map is immutable and is cached across invocations. Do not
+	// mutate it.
+	ReadDirectory(path string) (entries DirEntries, canonicalError error, originalError error)
+	ReadFile(path string) (contents string, canonicalError error, originalError error)
+	OpenFile(path string) (result OpenedFile, canonicalError error, originalError error)
+
+	// This is a key made from the information returned by "stat". It is intended
+	// to be different if the file has been edited, and to otherwise be equal if
+	// the file has not been edited. It should usually work, but no guarantees.
+	//
+	// See https://apenwarr.ca/log/20181113 for more information about why this
+	// can be broken. For example, writing to a file with mmap on WSL on Windows
+	// won't change this key. Hopefully this isn't too much of an issue.
+	//
+	// Additional reading:
+	// - https://github.com/npm/npm/pull/20027
+	// - https://github.com/golang/go/commit/7dea509703eb5ad66a35628b12a678110fbb1f72
+	ModKey(path string) (ModKey, error)
+
+	// This is part of the interface because the mock interface used for tests
+	// should not depend on file system behavior (i.e. different slashes for
+	// Windows) while the real interface should.
+	IsAbs(path string) bool
+	Abs(path string) (string, bool)
+	Dir(path string) string
+	Base(path string) string
+	Ext(path string) string
+	Join(parts ...string) string
+	Cwd() string
+	Rel(base string, target string) (string, bool)
+	EvalSymlinks(path string) (string, bool)
+
+	// This is used in the implementation of "Entry"
+	kind(dir string, base string) (symlink string, kind EntryKind)
+
+	// This is a set of all files used and all directories checked. The build
+	// must be invalidated if any of these watched files change.
+	WatchData() WatchData
+}
+
+type WatchData struct {
+	// These functions return a non-empty path as a string if the file system
+	// entry has been modified. For files, the returned path is the same as the
+	// file path. For directories, the returned path is either the directory
+	// itself or a file in the directory that was changed.
+	Paths map[string]func() string
+}
+
+type ModKey struct {
+	// What gets filled in here is OS-dependent
+	inode      uint64
+	size       int64
+	mtime_sec  int64
+	mtime_nsec int64
+	mode       uint32
+	uid        uint32
+}
+
+// Some file systems have a time resolution of only a few seconds. If a mtime
+// value is too new, we won't be able to tell if it has been recently modified
+// or not. So we only use mtimes for comparison if they are sufficiently old.
+// Apparently the FAT file system has a resolution of two seconds according to
+// this article: https://en.wikipedia.org/wiki/Stat_(system_call).
+const modKeySafetyGap = 3 // In seconds
+var modKeyUnusable = errors.New("The modification key is unusable")
+
+// Limit the number of files open simultaneously to avoid ulimit issues
+var fileOpenLimit = make(chan bool, 32)
+
+func BeforeFileOpen() {
+	// This will block if the number of open files is already at the limit
+	fileOpenLimit <- false
+}
+
+func AfterFileClose() {
+	<-fileOpenLimit
+}
+
+// This is a fork of "os.MkdirAll" to work around bugs with the WebAssembly
+// build target. More information here: https://github.com/golang/go/issues/43768.
+func MkdirAll(fs FS, path string, perm os.FileMode) error {
+	// Run "Join" once to run "Clean" on the path, which removes trailing slashes
+	return mkdirAll(fs, fs.Join(path), perm)
+}
+
+func mkdirAll(fs FS, path string, perm os.FileMode) error {
+	// Fast path: if we can tell whether path is a directory or file, stop with success or error.
+	if dir, err := os.Stat(path); err == nil {
+		if dir.IsDir() {
+			return nil
+		}
+		return &os.PathError{Op: "mkdir", Path: path, Err: syscall.ENOTDIR}
+	}
+
+	// Slow path: make sure parent exists and then call Mkdir for path.
+	if parent := fs.Dir(path); parent != path {
+		// Create parent.
+		if err := mkdirAll(fs, parent, perm); err != nil {
+			return err
+		}
+	}
+
+	// Parent now exists; invoke Mkdir and use its result.
+	if err := os.Mkdir(path, perm); err != nil {
+		// Handle arguments like "foo/." by
+		// double-checking that directory doesn't exist.
+		dir, err1 := os.Lstat(path)
+		if err1 == nil && dir.IsDir() {
+			return nil
+		}
+		return err
+	}
+	return nil
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/fs/fs_mock.go b/source/vendor/github.com/evanw/esbuild/internal/fs/fs_mock.go
new file mode 100644
index 0000000..8626b59
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/fs/fs_mock.go
@@ -0,0 +1,294 @@
+package fs
+
+// This is a mock implementation of the "fs" module for use with tests. It does
+// not actually read from the file system. Instead, it reads from a pre-specified
+// map of file paths to files.
+
+import (
+	"errors"
+	"path"
+	"strings"
+	"syscall"
+)
+
+type MockKind uint8
+
+const (
+	MockUnix MockKind = iota
+	MockWindows
+)
+
+type mockFS struct {
+	dirs          map[string]DirEntries
+	files         map[string]string
+	absWorkingDir string
+	Kind          MockKind
+}
+
+func MockFS(input map[string]string, kind MockKind, absWorkingDir string) FS {
+	dirs := make(map[string]DirEntries)
+	files := make(map[string]string)
+
+	for k, v := range input {
+		key := k
+		if kind == MockWindows {
+			key = "C:" + strings.ReplaceAll(key, "/", "\\")
+		}
+		files[key] = v
+		original := k
+
+		// Build the directory map
+		for {
+			kDir := path.Dir(k)
+			key := kDir
+			if kind == MockWindows {
+				key = "C:" + strings.ReplaceAll(key, "/", "\\")
+			}
+			dir, ok := dirs[key]
+			if !ok {
+				dir = DirEntries{dir: key, data: make(map[string]*Entry)}
+				dirs[key] = dir
+			}
+			if kDir == k {
+				break
+			}
+			base := path.Base(k)
+			if k == original {
+				dir.data[strings.ToLower(base)] = &Entry{kind: FileEntry, base: base}
+			} else {
+				dir.data[strings.ToLower(base)] = &Entry{kind: DirEntry, base: base}
+			}
+			k = kDir
+		}
+	}
+
+	return &mockFS{dirs, files, absWorkingDir, kind}
+}
+
+func (fs *mockFS) ReadDirectory(path string) (DirEntries, error, error) {
+	if fs.Kind == MockWindows {
+		path = strings.ReplaceAll(path, "/", "\\")
+	}
+
+	var slash byte = '/'
+	if fs.Kind == MockWindows {
+		slash = '\\'
+	}
+
+	// Trim trailing slashes before lookup
+	firstSlash := strings.IndexByte(path, slash)
+	for {
+		i := strings.LastIndexByte(path, slash)
+		if i != len(path)-1 || i <= firstSlash {
+			break
+		}
+		path = path[:i]
+	}
+
+	if dir, ok := fs.dirs[path]; ok {
+		return dir, nil, nil
+	}
+	return DirEntries{}, syscall.ENOENT, syscall.ENOENT
+}
+
+func (fs *mockFS) ReadFile(path string) (string, error, error) {
+	if fs.Kind == MockWindows {
+		path = strings.ReplaceAll(path, "/", "\\")
+	}
+	if contents, ok := fs.files[path]; ok {
+		return contents, nil, nil
+	}
+	return "", syscall.ENOENT, syscall.ENOENT
+}
+
+func (fs *mockFS) OpenFile(path string) (OpenedFile, error, error) {
+	if fs.Kind == MockWindows {
+		path = strings.ReplaceAll(path, "/", "\\")
+	}
+	if contents, ok := fs.files[path]; ok {
+		return &InMemoryOpenedFile{Contents: []byte(contents)}, nil, nil
+	}
+	return nil, syscall.ENOENT, syscall.ENOENT
+}
+
+func (fs *mockFS) ModKey(path string) (ModKey, error) {
+	return ModKey{}, errors.New("This is not available during tests")
+}
+
+func win2unix(p string) string {
+	if strings.HasPrefix(p, "C:\\") {
+		p = p[2:]
+	}
+	p = strings.ReplaceAll(p, "\\", "/")
+	return p
+}
+
+func unix2win(p string) string {
+	p = strings.ReplaceAll(p, "/", "\\")
+	if strings.HasPrefix(p, "\\") {
+		p = "C:" + p
+	}
+	return p
+}
+
+func (fs *mockFS) IsAbs(p string) bool {
+	if fs.Kind == MockWindows {
+		p = win2unix(p)
+	}
+	return path.IsAbs(p)
+}
+
+func (fs *mockFS) Abs(p string) (string, bool) {
+	if fs.Kind == MockWindows {
+		p = win2unix(p)
+	}
+
+	p = path.Clean(path.Join("/", p))
+
+	if fs.Kind == MockWindows {
+		p = unix2win(p)
+	}
+
+	return p, true
+}
+
+func (fs *mockFS) Dir(p string) string {
+	if fs.Kind == MockWindows {
+		p = win2unix(p)
+	}
+
+	p = path.Dir(p)
+
+	if fs.Kind == MockWindows {
+		p = unix2win(p)
+	}
+
+	return p
+}
+
+func (fs *mockFS) Base(p string) string {
+	if fs.Kind == MockWindows {
+		p = win2unix(p)
+	}
+
+	p = path.Base(p)
+
+	if fs.Kind == MockWindows && p == "/" {
+		p = "\\"
+	}
+
+	return p
+}
+
+func (fs *mockFS) Ext(p string) string {
+	if fs.Kind == MockWindows {
+		p = win2unix(p)
+	}
+
+	return path.Ext(p)
+}
+
+func (fs *mockFS) Join(parts ...string) string {
+	if fs.Kind == MockWindows {
+		converted := make([]string, len(parts))
+		for i, part := range parts {
+			converted[i] = win2unix(part)
+		}
+		parts = converted
+	}
+
+	p := path.Clean(path.Join(parts...))
+
+	if fs.Kind == MockWindows {
+		p = unix2win(p)
+	}
+
+	return p
+}
+
+func (fs *mockFS) Cwd() string {
+	return fs.absWorkingDir
+}
+
+func splitOnSlash(path string) (string, string) {
+	if slash := strings.IndexByte(path, '/'); slash != -1 {
+		return path[:slash], path[slash+1:]
+	}
+	return path, ""
+}
+
+func (fs *mockFS) Rel(base string, target string) (string, bool) {
+	if fs.Kind == MockWindows {
+		base = win2unix(base)
+		target = win2unix(target)
+	}
+
+	base = path.Clean(base)
+	target = path.Clean(target)
+
+	// Go's implementation does these checks
+	if base == target {
+		return ".", true
+	}
+	if base == "." {
+		base = ""
+	}
+
+	// Go's implementation fails when this condition is false. I believe this is
+	// because of this part of the contract, from Go's documentation: "An error
+	// is returned if targpath can't be made relative to basepath or if knowing
+	// the current working directory would be necessary to compute it."
+	if (len(base) > 0 && base[0] == '/') != (len(target) > 0 && target[0] == '/') {
+		return "", false
+	}
+
+	// Find the common parent directory
+	for {
+		bHead, bTail := splitOnSlash(base)
+		tHead, tTail := splitOnSlash(target)
+		if bHead != tHead {
+			break
+		}
+		base = bTail
+		target = tTail
+	}
+
+	// Stop now if base is a subpath of target
+	if base == "" {
+		if fs.Kind == MockWindows {
+			target = unix2win(target)
+		}
+		return target, true
+	}
+
+	// Traverse up to the common parent
+	commonParent := strings.Repeat("../", strings.Count(base, "/")+1)
+
+	// Stop now if target is a subpath of base
+	if target == "" {
+		target = commonParent[:len(commonParent)-1]
+		if fs.Kind == MockWindows {
+			target = unix2win(target)
+		}
+		return target, true
+	}
+
+	// Otherwise, down to the parent
+	target = commonParent + target
+	if fs.Kind == MockWindows {
+		target = unix2win(target)
+	}
+	return target, true
+}
+
+func (fs *mockFS) EvalSymlinks(path string) (string, bool) {
+	return "", false
+}
+
+func (fs *mockFS) kind(dir string, base string) (symlink string, kind EntryKind) {
+	panic("This should never be called")
+}
+
+func (fs *mockFS) WatchData() WatchData {
+	panic("This should never be called")
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/fs/fs_real.go b/source/vendor/github.com/evanw/esbuild/internal/fs/fs_real.go
new file mode 100644
index 0000000..412eb68
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/fs/fs_real.go
@@ -0,0 +1,543 @@
+package fs
+
+import (
+	"fmt"
+	"io"
+	"io/ioutil"
+	"os"
+	"sort"
+	"strings"
+	"sync"
+	"syscall"
+)
+
+type realFS struct {
+	// Stores the file entries for directories we've listed before
+	entries map[string]entriesOrErr
+
+	// This stores data that will end up being returned by "WatchData()"
+	watchData map[string]privateWatchData
+
+	// When building with WebAssembly, the Go compiler doesn't correctly handle
+	// platform-specific path behavior. Hack around these bugs by compiling
+	// support for both Unix and Windows paths into all executables and switch
+	// between them at run-time instead.
+	fp goFilepath
+
+	entriesMutex sync.Mutex
+	watchMutex   sync.Mutex
+
+	// If true, do not use the "entries" cache
+	doNotCacheEntries bool
+}
+
+type entriesOrErr struct {
+	canonicalError error
+	originalError  error
+	entries        DirEntries
+}
+
+type watchState uint8
+
+const (
+	stateNone                  watchState = iota
+	stateDirHasAccessedEntries            // Compare "accessedEntries"
+	stateDirUnreadable                    // Compare directory readability
+	stateFileHasModKey                    // Compare "modKey"
+	stateFileNeedModKey                   // Need to transition to "stateFileHasModKey" or "stateFileUnusableModKey" before "WatchData()" returns
+	stateFileMissing                      // Compare file presence
+	stateFileUnusableModKey               // Compare "fileContents"
+)
+
+type privateWatchData struct {
+	accessedEntries *accessedEntries
+	fileContents    string
+	modKey          ModKey
+	state           watchState
+}
+
+type RealFSOptions struct {
+	AbsWorkingDir string
+	WantWatchData bool
+	DoNotCache    bool
+}
+
+func RealFS(options RealFSOptions) (FS, error) {
+	var fp goFilepath
+	if CheckIfWindows() {
+		fp.isWindows = true
+		fp.pathSeparator = '\\'
+	} else {
+		fp.isWindows = false
+		fp.pathSeparator = '/'
+	}
+
+	// Come up with a default working directory if one was not specified
+	fp.cwd = options.AbsWorkingDir
+	if fp.cwd == "" {
+		if cwd, err := os.Getwd(); err == nil {
+			fp.cwd = cwd
+		} else if fp.isWindows {
+			fp.cwd = "C:\\"
+		} else {
+			fp.cwd = "/"
+		}
+	} else if !fp.isAbs(fp.cwd) {
+		return nil, fmt.Errorf("The working directory %q is not an absolute path", fp.cwd)
+	}
+
+	// Resolve symlinks in the current working directory. Symlinks are resolved
+	// when input file paths are converted to absolute paths because we need to
+	// recognize an input file as unique even if it has multiple symlinks
+	// pointing to it. The build will generate relative paths from the current
+	// working directory to the absolute input file paths for error messages,
+	// so the current working directory should be processed the same way. Not
+	// doing this causes test failures with esbuild when run from inside a
+	// symlinked directory.
+	//
+	// This deliberately ignores errors due to e.g. infinite loops. If there is
+	// an error, we will just use the original working directory and likely
+	// encounter an error later anyway. And if we don't encounter an error
+	// later, then the current working directory didn't even matter and the
+	// error is unimportant.
+	if path, err := fp.evalSymlinks(fp.cwd); err == nil {
+		fp.cwd = path
+	}
+
+	// Only allocate memory for watch data if necessary
+	var watchData map[string]privateWatchData
+	if options.WantWatchData {
+		watchData = make(map[string]privateWatchData)
+	}
+
+	var result FS = &realFS{
+		entries:           make(map[string]entriesOrErr),
+		fp:                fp,
+		watchData:         watchData,
+		doNotCacheEntries: options.DoNotCache,
+	}
+
+	// Add a wrapper that lets us traverse into ".zip" files. This is what yarn
+	// uses as a package format when in yarn is in its "PnP" mode.
+	result = &zipFS{
+		inner:    result,
+		zipFiles: make(map[string]*zipFile),
+	}
+
+	return result, nil
+}
+
+func (fs *realFS) ReadDirectory(dir string) (entries DirEntries, canonicalError error, originalError error) {
+	if !fs.doNotCacheEntries {
+		// First, check the cache
+		cached, ok := func() (cached entriesOrErr, ok bool) {
+			fs.entriesMutex.Lock()
+			defer fs.entriesMutex.Unlock()
+			cached, ok = fs.entries[dir]
+			return
+		}()
+		if ok {
+			// Cache hit: stop now
+			return cached.entries, cached.canonicalError, cached.originalError
+		}
+	}
+
+	// Cache miss: read the directory entries
+	names, canonicalError, originalError := fs.readdir(dir)
+	entries = DirEntries{dir: dir, data: make(map[string]*Entry)}
+
+	// Unwrap to get the underlying error
+	if pathErr, ok := canonicalError.(*os.PathError); ok {
+		canonicalError = pathErr.Unwrap()
+	}
+
+	if canonicalError == nil {
+		for _, name := range names {
+			// Call "stat" lazily for performance. The "@material-ui/icons" package
+			// contains a directory with over 11,000 entries in it and running "stat"
+			// for each entry was a big performance issue for that package.
+			entries.data[strings.ToLower(name)] = &Entry{
+				dir:      dir,
+				base:     name,
+				needStat: true,
+			}
+		}
+	}
+
+	// Store data for watch mode
+	if fs.watchData != nil {
+		defer fs.watchMutex.Unlock()
+		fs.watchMutex.Lock()
+		state := stateDirHasAccessedEntries
+		if canonicalError != nil {
+			state = stateDirUnreadable
+		}
+		entries.accessedEntries = &accessedEntries{wasPresent: make(map[string]bool)}
+		fs.watchData[dir] = privateWatchData{
+			accessedEntries: entries.accessedEntries,
+			state:           state,
+		}
+	}
+
+	// Update the cache unconditionally. Even if the read failed, we don't want to
+	// retry again later. The directory is inaccessible so trying again is wasted.
+	if canonicalError != nil {
+		entries.data = nil
+	}
+	if !fs.doNotCacheEntries {
+		fs.entriesMutex.Lock()
+		defer fs.entriesMutex.Unlock()
+		fs.entries[dir] = entriesOrErr{
+			entries:        entries,
+			canonicalError: canonicalError,
+			originalError:  originalError,
+		}
+	}
+	return entries, canonicalError, originalError
+}
+
+func (fs *realFS) ReadFile(path string) (contents string, canonicalError error, originalError error) {
+	BeforeFileOpen()
+	defer AfterFileClose()
+	buffer, originalError := ioutil.ReadFile(path)
+	canonicalError = fs.canonicalizeError(originalError)
+
+	// Allocate the string once
+	fileContents := string(buffer)
+
+	// Store data for watch mode
+	if fs.watchData != nil {
+		defer fs.watchMutex.Unlock()
+		fs.watchMutex.Lock()
+		data, ok := fs.watchData[path]
+		if canonicalError != nil {
+			data.state = stateFileMissing
+		} else if !ok || data.state == stateDirUnreadable {
+			// Note: If "ReadDirectory" is called before "ReadFile" with this same
+			// path, then "data.state" will be "stateDirUnreadable". In that case
+			// we want to transition to "stateFileNeedModKey" because it's a file.
+			data.state = stateFileNeedModKey
+		}
+		data.fileContents = fileContents
+		fs.watchData[path] = data
+	}
+
+	return fileContents, canonicalError, originalError
+}
+
+type realOpenedFile struct {
+	handle *os.File
+	len    int
+}
+
+func (f *realOpenedFile) Len() int {
+	return f.len
+}
+
+func (f *realOpenedFile) Read(start int, end int) ([]byte, error) {
+	bytes := make([]byte, end-start)
+	remaining := bytes
+
+	_, err := f.handle.Seek(int64(start), io.SeekStart)
+	if err != nil {
+		return nil, err
+	}
+
+	for len(remaining) > 0 {
+		n, err := f.handle.Read(remaining)
+		if err != nil && n <= 0 {
+			return nil, err
+		}
+		remaining = remaining[n:]
+	}
+
+	return bytes, nil
+}
+
+func (f *realOpenedFile) Close() error {
+	return f.handle.Close()
+}
+
+func (fs *realFS) OpenFile(path string) (OpenedFile, error, error) {
+	BeforeFileOpen()
+	defer AfterFileClose()
+
+	f, err := os.Open(path)
+	if err != nil {
+		return nil, fs.canonicalizeError(err), err
+	}
+
+	info, err := f.Stat()
+	if err != nil {
+		f.Close()
+		return nil, fs.canonicalizeError(err), err
+	}
+
+	return &realOpenedFile{f, int(info.Size())}, nil, nil
+}
+
+func (fs *realFS) ModKey(path string) (ModKey, error) {
+	BeforeFileOpen()
+	defer AfterFileClose()
+	key, err := modKey(path)
+
+	// Store data for watch mode
+	if fs.watchData != nil {
+		defer fs.watchMutex.Unlock()
+		fs.watchMutex.Lock()
+		data, ok := fs.watchData[path]
+		if !ok {
+			if err == modKeyUnusable {
+				data.state = stateFileUnusableModKey
+			} else if err != nil {
+				data.state = stateFileMissing
+			} else {
+				data.state = stateFileHasModKey
+			}
+		} else if data.state == stateFileNeedModKey {
+			data.state = stateFileHasModKey
+		}
+		data.modKey = key
+		fs.watchData[path] = data
+	}
+
+	return key, err
+}
+
+func (fs *realFS) IsAbs(p string) bool {
+	return fs.fp.isAbs(p)
+}
+
+func (fs *realFS) Abs(p string) (string, bool) {
+	abs, err := fs.fp.abs(p)
+	return abs, err == nil
+}
+
+func (fs *realFS) Dir(p string) string {
+	return fs.fp.dir(p)
+}
+
+func (fs *realFS) Base(p string) string {
+	return fs.fp.base(p)
+}
+
+func (fs *realFS) Ext(p string) string {
+	return fs.fp.ext(p)
+}
+
+func (fs *realFS) Join(parts ...string) string {
+	return fs.fp.clean(fs.fp.join(parts))
+}
+
+func (fs *realFS) Cwd() string {
+	return fs.fp.cwd
+}
+
+func (fs *realFS) Rel(base string, target string) (string, bool) {
+	if rel, err := fs.fp.rel(base, target); err == nil {
+		return rel, true
+	}
+	return "", false
+}
+
+func (fs *realFS) EvalSymlinks(path string) (string, bool) {
+	if path, err := fs.fp.evalSymlinks(path); err == nil {
+		return path, true
+	}
+	return "", false
+}
+
+func (fs *realFS) readdir(dirname string) (entries []string, canonicalError error, originalError error) {
+	BeforeFileOpen()
+	defer AfterFileClose()
+	f, originalError := os.Open(dirname)
+	canonicalError = fs.canonicalizeError(originalError)
+
+	// Stop now if there was an error
+	if canonicalError != nil {
+		return nil, canonicalError, originalError
+	}
+
+	defer f.Close()
+	entries, originalError = f.Readdirnames(-1)
+	canonicalError = originalError
+
+	// Unwrap to get the underlying error
+	if syscallErr, ok := canonicalError.(*os.SyscallError); ok {
+		canonicalError = syscallErr.Unwrap()
+	}
+
+	// Don't convert ENOTDIR to ENOENT here. ENOTDIR is a legitimate error
+	// condition for Readdirnames() on non-Windows platforms.
+
+	// Go's WebAssembly implementation returns EINVAL instead of ENOTDIR if we
+	// call "readdir" on a file. Canonicalize this to ENOTDIR so esbuild's path
+	// resolution code continues traversing instead of failing with an error.
+	// https://github.com/golang/go/blob/2449bbb5e614954ce9e99c8a481ea2ee73d72d61/src/syscall/fs_js.go#L144
+	if pathErr, ok := canonicalError.(*os.PathError); ok && pathErr.Unwrap() == syscall.EINVAL {
+		canonicalError = syscall.ENOTDIR
+	}
+
+	return entries, canonicalError, originalError
+}
+
+func (fs *realFS) canonicalizeError(err error) error {
+	// Unwrap to get the underlying error
+	if pathErr, ok := err.(*os.PathError); ok {
+		err = pathErr.Unwrap()
+	}
+
+	// Windows is much more restrictive than Unix about file names. If a file name
+	// is invalid, it will return ERROR_INVALID_NAME. Treat this as ENOENT (i.e.
+	// "the file does not exist") so that the resolver continues trying to resolve
+	// the path on this failure instead of aborting with an error.
+	if fs.fp.isWindows && is_ERROR_INVALID_NAME(err) {
+		err = syscall.ENOENT
+	}
+
+	// Windows returns ENOTDIR here even though nothing we've done yet has asked
+	// for a directory. This really means ENOENT on Windows. Return ENOENT here
+	// so callers that check for ENOENT will successfully detect this file as
+	// missing.
+	if err == syscall.ENOTDIR {
+		err = syscall.ENOENT
+	}
+
+	return err
+}
+
+func (fs *realFS) kind(dir string, base string) (symlink string, kind EntryKind) {
+	entryPath := fs.fp.join([]string{dir, base})
+
+	// Use "lstat" since we want information about symbolic links
+	BeforeFileOpen()
+	defer AfterFileClose()
+	stat, err := os.Lstat(entryPath)
+	if err != nil {
+		return
+	}
+	mode := stat.Mode()
+
+	// Follow symlinks now so the cache contains the translation
+	if (mode & os.ModeSymlink) != 0 {
+		link, err := fs.fp.evalSymlinks(entryPath)
+		if err != nil {
+			return // Skip over this entry
+		}
+
+		// Re-run "lstat" on the symlink target to see if it's a file or not
+		stat2, err2 := os.Lstat(link)
+		if err2 != nil {
+			return // Skip over this entry
+		}
+		mode = stat2.Mode()
+		if (mode & os.ModeSymlink) != 0 {
+			return // This should no longer be a symlink, so this is unexpected
+		}
+		symlink = link
+	}
+
+	// We consider the entry either a directory or a file
+	if (mode & os.ModeDir) != 0 {
+		kind = DirEntry
+	} else {
+		kind = FileEntry
+	}
+	return
+}
+
+func (fs *realFS) WatchData() WatchData {
+	paths := make(map[string]func() string)
+
+	for path, data := range fs.watchData {
+		// Each closure below needs its own copy of these loop variables
+		path := path
+		data := data
+
+		// Each function should return true if the state has been changed
+		if data.state == stateFileNeedModKey {
+			key, err := modKey(path)
+			if err == modKeyUnusable {
+				data.state = stateFileUnusableModKey
+			} else if err != nil {
+				data.state = stateFileMissing
+			} else {
+				data.state = stateFileHasModKey
+				data.modKey = key
+			}
+		}
+
+		switch data.state {
+		case stateDirUnreadable:
+			paths[path] = func() string {
+				_, err, _ := fs.readdir(path)
+				if err == nil {
+					return path
+				}
+				return ""
+			}
+
+		case stateDirHasAccessedEntries:
+			paths[path] = func() string {
+				names, err, _ := fs.readdir(path)
+				if err != nil {
+					return path
+				}
+				data.accessedEntries.mutex.Lock()
+				defer data.accessedEntries.mutex.Unlock()
+				if allEntries := data.accessedEntries.allEntries; allEntries != nil {
+					// Check all entries
+					if len(names) != len(allEntries) {
+						return path
+					}
+					sort.Strings(names)
+					for i, s := range names {
+						if s != allEntries[i] {
+							return path
+						}
+					}
+				} else {
+					// Check individual entries
+					lookup := make(map[string]string, len(names))
+					for _, name := range names {
+						lookup[strings.ToLower(name)] = name
+					}
+					for name, wasPresent := range data.accessedEntries.wasPresent {
+						if originalName, isPresent := lookup[name]; wasPresent != isPresent {
+							return fs.Join(path, originalName)
+						}
+					}
+				}
+				return ""
+			}
+
+		case stateFileMissing:
+			paths[path] = func() string {
+				if info, err := os.Stat(path); err == nil && !info.IsDir() {
+					return path
+				}
+				return ""
+			}
+
+		case stateFileHasModKey:
+			paths[path] = func() string {
+				if key, err := modKey(path); err != nil || key != data.modKey {
+					return path
+				}
+				return ""
+			}
+
+		case stateFileUnusableModKey:
+			paths[path] = func() string {
+				if buffer, err := ioutil.ReadFile(path); err != nil || string(buffer) != data.fileContents {
+					return path
+				}
+				return ""
+			}
+		}
+	}
+
+	return WatchData{
+		Paths: paths,
+	}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/fs/fs_zip.go b/source/vendor/github.com/evanw/esbuild/internal/fs/fs_zip.go
new file mode 100644
index 0000000..58a7b8b
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/fs/fs_zip.go
@@ -0,0 +1,405 @@
+package fs
+
+// The Yarn package manager (https://yarnpkg.com/) has a custom installation
+// strategy called "Plug'n'Play" where they install packages as zip files
+// instead of directory trees, and then modify node to treat zip files like
+// directories. This reduces package installation time because Yarn now only
+// has to copy a single file per package instead of a whole directory tree.
+// However, it introduces overhead at run-time because the virtual file system
+// is written in JavaScript.
+//
+// This file contains esbuild's implementation of the behavior that treats zip
+// files like directories. It implements the "FS" interface and wraps an inner
+// "FS" interface that treats zip files like files. That way it can run both on
+// a real file system and a mock file system.
+//
+// This file also implements another Yarn-specific behavior where certain paths
+// containing the special path segments "__virtual__" or "$$virtual" have some
+// unusual behavior. See the code below for details.
+
+import (
+	"archive/zip"
+	"io/ioutil"
+	"strconv"
+	"strings"
+	"sync"
+	"syscall"
+)
+
+type zipFS struct {
+	inner FS
+
+	zipFilesMutex sync.Mutex
+	zipFiles      map[string]*zipFile
+}
+
+type zipFile struct {
+	reader *zip.ReadCloser
+	err    error
+
+	dirs  map[string]*compressedDir
+	files map[string]*compressedFile
+	wait  sync.WaitGroup
+}
+
+type compressedDir struct {
+	entries map[string]EntryKind
+	path    string
+
+	// Compatible entries are decoded lazily
+	mutex      sync.Mutex
+	dirEntries DirEntries
+}
+
+type compressedFile struct {
+	compressed *zip.File
+
+	// The file is decompressed lazily
+	mutex    sync.Mutex
+	contents string
+	err      error
+	wasRead  bool
+}
+
+func (fs *zipFS) checkForZip(path string, kind EntryKind) (*zipFile, string) {
+	var zipPath string
+	var pathTail string
+
+	// Do a quick check for a ".zip" in the path at all
+	path = strings.ReplaceAll(path, "\\", "/")
+	if i := strings.Index(path, ".zip/"); i != -1 {
+		zipPath = path[:i+len(".zip")]
+		pathTail = path[i+len(".zip/"):]
+	} else if kind == DirEntry && strings.HasSuffix(path, ".zip") {
+		zipPath = path
+	} else {
+		return nil, ""
+	}
+
+	// If there is one, then check whether it's a file on the file system or not
+	fs.zipFilesMutex.Lock()
+	archive := fs.zipFiles[zipPath]
+	if archive != nil {
+		fs.zipFilesMutex.Unlock()
+		archive.wait.Wait()
+	} else {
+		archive = &zipFile{}
+		archive.wait.Add(1)
+		fs.zipFiles[zipPath] = archive
+		fs.zipFilesMutex.Unlock()
+		defer archive.wait.Done()
+
+		// Try reading the zip archive if it's not in the cache
+		tryToReadZipArchive(zipPath, archive)
+	}
+
+	if archive.err != nil {
+		return nil, ""
+	}
+	return archive, pathTail
+}
+
+func tryToReadZipArchive(zipPath string, archive *zipFile) {
+	reader, err := zip.OpenReader(zipPath)
+	if err != nil {
+		archive.err = err
+		return
+	}
+
+	dirs := make(map[string]*compressedDir)
+	files := make(map[string]*compressedFile)
+	seeds := []string{}
+
+	// Build an index of all files in the archive
+	for _, file := range reader.File {
+		baseName := strings.TrimSuffix(file.Name, "/")
+		dirPath := ""
+		if slash := strings.LastIndexByte(baseName, '/'); slash != -1 {
+			dirPath = baseName[:slash]
+			baseName = baseName[slash+1:]
+		}
+		if file.FileInfo().IsDir() {
+			// Handle a directory
+			lowerDir := strings.ToLower(dirPath)
+			if _, ok := dirs[lowerDir]; !ok {
+				dir := &compressedDir{
+					path:    dirPath,
+					entries: make(map[string]EntryKind),
+				}
+
+				// List the same directory both with and without the slash
+				dirs[lowerDir] = dir
+				dirs[lowerDir+"/"] = dir
+				seeds = append(seeds, lowerDir)
+			}
+		} else {
+			// Handle a file
+			files[strings.ToLower(file.Name)] = &compressedFile{compressed: file}
+			lowerDir := strings.ToLower(dirPath)
+			dir, ok := dirs[lowerDir]
+			if !ok {
+				dir = &compressedDir{
+					path:    dirPath,
+					entries: make(map[string]EntryKind),
+				}
+
+				// List the same directory both with and without the slash
+				dirs[lowerDir] = dir
+				dirs[lowerDir+"/"] = dir
+				seeds = append(seeds, lowerDir)
+			}
+			dir.entries[baseName] = FileEntry
+		}
+	}
+
+	// Populate child directories
+	for _, baseName := range seeds {
+		for baseName != "" {
+			dirPath := ""
+			if slash := strings.LastIndexByte(baseName, '/'); slash != -1 {
+				dirPath = baseName[:slash]
+				baseName = baseName[slash+1:]
+			}
+			lowerDir := strings.ToLower(dirPath)
+			dir, ok := dirs[lowerDir]
+			if !ok {
+				dir = &compressedDir{
+					path:    dirPath,
+					entries: make(map[string]EntryKind),
+				}
+
+				// List the same directory both with and without the slash
+				dirs[lowerDir] = dir
+				dirs[lowerDir+"/"] = dir
+			}
+			dir.entries[baseName] = DirEntry
+			baseName = dirPath
+		}
+	}
+
+	archive.dirs = dirs
+	archive.files = files
+	archive.reader = reader
+}
+
+func (fs *zipFS) ReadDirectory(path string) (entries DirEntries, canonicalError error, originalError error) {
+	path = mangleYarnPnPVirtualPath(path)
+
+	entries, canonicalError, originalError = fs.inner.ReadDirectory(path)
+
+	// Only continue if reading this path as a directory caused an error that's
+	// consistent with trying to read a zip file as a directory. Note that EINVAL
+	// is produced by the file system in Go's WebAssembly implementation.
+	if canonicalError != syscall.ENOENT && canonicalError != syscall.ENOTDIR && canonicalError != syscall.EINVAL {
+		return
+	}
+
+	// If the directory doesn't exist, try reading from an enclosing zip archive
+	zip, pathTail := fs.checkForZip(path, DirEntry)
+	if zip == nil {
+		return
+	}
+
+	// Does the zip archive have this directory?
+	dir, ok := zip.dirs[strings.ToLower(pathTail)]
+	if !ok {
+		return DirEntries{}, syscall.ENOENT, syscall.ENOENT
+	}
+
+	// Check whether it has already been converted
+	dir.mutex.Lock()
+	defer dir.mutex.Unlock()
+	if dir.dirEntries.data != nil {
+		return dir.dirEntries, nil, nil
+	}
+
+	// Otherwise, fill in the entries
+	dir.dirEntries = DirEntries{dir: path, data: make(map[string]*Entry, len(dir.entries))}
+	for name, kind := range dir.entries {
+		dir.dirEntries.data[strings.ToLower(name)] = &Entry{
+			dir:  path,
+			base: name,
+			kind: kind,
+		}
+	}
+
+	return dir.dirEntries, nil, nil
+}
+
+func (fs *zipFS) ReadFile(path string) (contents string, canonicalError error, originalError error) {
+	path = mangleYarnPnPVirtualPath(path)
+
+	contents, canonicalError, originalError = fs.inner.ReadFile(path)
+	if canonicalError != syscall.ENOENT {
+		return
+	}
+
+	// If the file doesn't exist, try reading from an enclosing zip archive
+	zip, pathTail := fs.checkForZip(path, FileEntry)
+	if zip == nil {
+		return
+	}
+
+	// Does the zip archive have this file?
+	file, ok := zip.files[strings.ToLower(pathTail)]
+	if !ok {
+		return "", syscall.ENOENT, syscall.ENOENT
+	}
+
+	// Check whether it has already been read
+	file.mutex.Lock()
+	defer file.mutex.Unlock()
+	if file.wasRead {
+		return file.contents, file.err, file.err
+	}
+	file.wasRead = true
+
+	// If not, try to open it
+	reader, err := file.compressed.Open()
+	if err != nil {
+		file.err = err
+		return "", err, err
+	}
+	defer reader.Close()
+
+	// Then try to read it
+	bytes, err := ioutil.ReadAll(reader)
+	if err != nil {
+		file.err = err
+		return "", err, err
+	}
+
+	file.contents = string(bytes)
+	return file.contents, nil, nil
+}
+
+func (fs *zipFS) OpenFile(path string) (result OpenedFile, canonicalError error, originalError error) {
+	path = mangleYarnPnPVirtualPath(path)
+
+	result, canonicalError, originalError = fs.inner.OpenFile(path)
+	return
+}
+
+func (fs *zipFS) ModKey(path string) (modKey ModKey, err error) {
+	path = mangleYarnPnPVirtualPath(path)
+
+	modKey, err = fs.inner.ModKey(path)
+	return
+}
+
+func (fs *zipFS) IsAbs(path string) bool {
+	return fs.inner.IsAbs(path)
+}
+
+func (fs *zipFS) Abs(path string) (string, bool) {
+	return fs.inner.Abs(path)
+}
+
+func (fs *zipFS) Dir(path string) string {
+	if prefix, suffix, ok := ParseYarnPnPVirtualPath(path); ok && suffix == "" {
+		return prefix
+	}
+	return fs.inner.Dir(path)
+}
+
+func (fs *zipFS) Base(path string) string {
+	return fs.inner.Base(path)
+}
+
+func (fs *zipFS) Ext(path string) string {
+	return fs.inner.Ext(path)
+}
+
+func (fs *zipFS) Join(parts ...string) string {
+	return fs.inner.Join(parts...)
+}
+
+func (fs *zipFS) Cwd() string {
+	return fs.inner.Cwd()
+}
+
+func (fs *zipFS) Rel(base string, target string) (string, bool) {
+	return fs.inner.Rel(base, target)
+}
+
+func (fs *zipFS) EvalSymlinks(path string) (string, bool) {
+	return fs.inner.EvalSymlinks(path)
+}
+
+func (fs *zipFS) kind(dir string, base string) (symlink string, kind EntryKind) {
+	return fs.inner.kind(dir, base)
+}
+
+func (fs *zipFS) WatchData() WatchData {
+	return fs.inner.WatchData()
+}
+
+func ParseYarnPnPVirtualPath(path string) (string, string, bool) {
+	i := 0
+
+	for {
+		start := i
+		slash := strings.IndexAny(path[i:], "/\\")
+		if slash == -1 {
+			break
+		}
+		i += slash + 1
+
+		// Replace the segments "__virtual__/<segment>/<n>" with N times the ".."
+		// operation. Note: The "__virtual__" folder name appeared with Yarn 3.0.
+		// Earlier releases used "$$virtual", but it was changed after discovering
+		// that this pattern triggered bugs in software where paths were used as
+		// either regexps or replacement. For example, "$$" found in the second
+		// parameter of "String.prototype.replace" silently turned into "$".
+		if segment := path[start : i-1]; segment == "__virtual__" || segment == "$$virtual" {
+			if slash := strings.IndexAny(path[i:], "/\\"); slash != -1 {
+				var count string
+				var suffix string
+				j := i + slash + 1
+
+				// Find the range of the count
+				if slash := strings.IndexAny(path[j:], "/\\"); slash != -1 {
+					count = path[j : j+slash]
+					suffix = path[j+slash:]
+				} else {
+					count = path[j:]
+				}
+
+				// Parse the count
+				if n, err := strconv.ParseInt(count, 10, 64); err == nil {
+					prefix := path[:start]
+
+					// Apply N times the ".." operator
+					for n > 0 && (strings.HasSuffix(prefix, "/") || strings.HasSuffix(prefix, "\\")) {
+						slash := strings.LastIndexAny(prefix[:len(prefix)-1], "/\\")
+						if slash == -1 {
+							break
+						}
+						prefix = prefix[:slash+1]
+						n--
+					}
+
+					// Make sure the prefix and suffix work well when joined together
+					if suffix == "" && strings.IndexAny(prefix, "/\\") != strings.LastIndexAny(prefix, "/\\") {
+						prefix = prefix[:len(prefix)-1]
+					} else if prefix == "" {
+						prefix = "."
+					} else if strings.HasPrefix(suffix, "/") || strings.HasPrefix(suffix, "\\") {
+						suffix = suffix[1:]
+					}
+
+					return prefix, suffix, true
+				}
+			}
+		}
+	}
+
+	return "", "", false
+}
+
+func mangleYarnPnPVirtualPath(path string) string {
+	if prefix, suffix, ok := ParseYarnPnPVirtualPath(path); ok {
+		return prefix + suffix
+	}
+	return path
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/fs/iswin_other.go b/source/vendor/github.com/evanw/esbuild/internal/fs/iswin_other.go
new file mode 100644
index 0000000..1aa2037
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/fs/iswin_other.go
@@ -0,0 +1,9 @@
+//go:build (!js || !wasm) && !windows
+// +build !js !wasm
+// +build !windows
+
+package fs
+
+func CheckIfWindows() bool {
+	return false
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/fs/iswin_wasm.go b/source/vendor/github.com/evanw/esbuild/internal/fs/iswin_wasm.go
new file mode 100644
index 0000000..b44a60e
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/fs/iswin_wasm.go
@@ -0,0 +1,25 @@
+//go:build js && wasm
+// +build js,wasm
+
+package fs
+
+import (
+	"os"
+)
+
+var checkedIfWindows bool
+var cachedIfWindows bool
+
+func CheckIfWindows() bool {
+	if !checkedIfWindows {
+		checkedIfWindows = true
+
+		// Hack: Assume that we're on Windows if we're running WebAssembly and
+		// the "C:\\" directory exists. This is a workaround for a bug in Go's
+		// WebAssembly support: https://github.com/golang/go/issues/43768.
+		_, err := os.Stat("C:\\")
+		cachedIfWindows = err == nil
+	}
+
+	return cachedIfWindows
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/fs/iswin_windows.go b/source/vendor/github.com/evanw/esbuild/internal/fs/iswin_windows.go
new file mode 100644
index 0000000..6f0128d
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/fs/iswin_windows.go
@@ -0,0 +1,8 @@
+//go:build windows
+// +build windows
+
+package fs
+
+func CheckIfWindows() bool {
+	return true
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/fs/modkey_other.go b/source/vendor/github.com/evanw/esbuild/internal/fs/modkey_other.go
new file mode 100644
index 0000000..4999e80
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/fs/modkey_other.go
@@ -0,0 +1,35 @@
+//go:build !darwin && !freebsd && !linux
+// +build !darwin,!freebsd,!linux
+
+package fs
+
+import (
+	"os"
+	"time"
+)
+
+var zeroTime time.Time
+
+func modKey(path string) (ModKey, error) {
+	info, err := os.Stat(path)
+	if err != nil {
+		return ModKey{}, err
+	}
+
+	// We can't detect changes if the file system zeros out the modification time
+	mtime := info.ModTime()
+	if mtime == zeroTime || mtime.Unix() == 0 {
+		return ModKey{}, modKeyUnusable
+	}
+
+	// Don't generate a modification key if the file is too new
+	if mtime.Add(modKeySafetyGap * time.Second).After(time.Now()) {
+		return ModKey{}, modKeyUnusable
+	}
+
+	return ModKey{
+		size:      info.Size(),
+		mtime_sec: mtime.Unix(),
+		mode:      uint32(info.Mode()),
+	}, nil
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/fs/modkey_unix.go b/source/vendor/github.com/evanw/esbuild/internal/fs/modkey_unix.go
new file mode 100644
index 0000000..c7c7f9c
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/fs/modkey_unix.go
@@ -0,0 +1,41 @@
+//go:build darwin || freebsd || linux
+// +build darwin freebsd linux
+
+package fs
+
+import (
+	"time"
+
+	"golang.org/x/sys/unix"
+)
+
+func modKey(path string) (ModKey, error) {
+	stat := unix.Stat_t{}
+	if err := unix.Stat(path, &stat); err != nil {
+		return ModKey{}, err
+	}
+
+	// We can't detect changes if the file system zeros out the modification time
+	if stat.Mtim.Sec == 0 && stat.Mtim.Nsec == 0 {
+		return ModKey{}, modKeyUnusable
+	}
+
+	// Don't generate a modification key if the file is too new
+	now, err := unix.TimeToTimespec(time.Now())
+	if err != nil {
+		return ModKey{}, err
+	}
+	mtimeSec := stat.Mtim.Sec + modKeySafetyGap
+	if mtimeSec > now.Sec || (mtimeSec == now.Sec && stat.Mtim.Nsec > now.Nsec) {
+		return ModKey{}, modKeyUnusable
+	}
+
+	return ModKey{
+		inode:      stat.Ino,
+		size:       stat.Size,
+		mtime_sec:  int64(stat.Mtim.Sec),
+		mtime_nsec: int64(stat.Mtim.Nsec),
+		mode:       uint32(stat.Mode),
+		uid:        stat.Uid,
+	}, nil
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/graph/graph.go b/source/vendor/github.com/evanw/esbuild/internal/graph/graph.go
new file mode 100644
index 0000000..a030797
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/graph/graph.go
@@ -0,0 +1,431 @@
+package graph
+
+// This graph represents the set of files that the linker operates on. Each
+// linker has a separate one of these graphs (there is one linker when code
+// splitting is on, but one linker per entry point when code splitting is off).
+//
+// The input data to the linker constructor must be considered immutable because
+// it's shared between linker invocations and is also stored in the cache for
+// incremental builds.
+//
+// The linker constructor makes a shallow clone of the input data and is careful
+// to pre-clone ahead of time the AST fields that it may modify. The Go language
+// doesn't have any type system features for immutability so this has to be
+// manually enforced. Please be careful.
+
+import (
+	"sort"
+	"sync"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/logger"
+	"github.com/evanw/esbuild/internal/runtime"
+)
+
+type entryPointKind uint8
+
+const (
+	entryPointNone entryPointKind = iota
+	entryPointUserSpecified
+	entryPointDynamicImport
+)
+
+type LinkerFile struct {
+	// This holds all entry points that can reach this file. It will be used to
+	// assign the parts in this file to a chunk.
+	EntryBits helpers.BitSet
+
+	// This is lazily-allocated because it's only needed if there are warnings
+	// logged, which should be relatively rare.
+	lazyLineColumnTracker *logger.LineColumnTracker
+
+	InputFile InputFile
+
+	// The minimum number of links in the module graph to get from an entry point
+	// to this file
+	DistanceFromEntryPoint uint32
+
+	// If "entryPointKind" is not "entryPointNone", this is the index of the
+	// corresponding entry point chunk.
+	EntryPointChunkIndex uint32
+
+	// This file is an entry point if and only if this is not "entryPointNone".
+	// Note that dynamically-imported files are allowed to also be specified by
+	// the user as top-level entry points, so some dynamically-imported files
+	// may be "entryPointUserSpecified" instead of "entryPointDynamicImport".
+	entryPointKind entryPointKind
+
+	// This is true if this file has been marked as live by the tree shaking
+	// algorithm.
+	IsLive bool
+}
+
+func (f *LinkerFile) IsEntryPoint() bool {
+	return f.entryPointKind != entryPointNone
+}
+
+func (f *LinkerFile) IsUserSpecifiedEntryPoint() bool {
+	return f.entryPointKind == entryPointUserSpecified
+}
+
+// Note: This is not guarded by a mutex. Make sure this isn't called from a
+// parallel part of the code.
+func (f *LinkerFile) LineColumnTracker() *logger.LineColumnTracker {
+	if f.lazyLineColumnTracker == nil {
+		tracker := logger.MakeLineColumnTracker(&f.InputFile.Source)
+		f.lazyLineColumnTracker = &tracker
+	}
+	return f.lazyLineColumnTracker
+}
+
+type EntryPoint struct {
+	// This may be an absolute path or a relative path. If absolute, it will
+	// eventually be turned into a relative path by computing the path relative
+	// to the "outbase" directory. Then this relative path will be joined onto
+	// the "outdir" directory to form the final output path for this entry point.
+	OutputPath string
+
+	// This is the source index of the entry point. This file must have a valid
+	// entry point kind (i.e. not "none").
+	SourceIndex uint32
+
+	// Manually specified output paths are ignored when computing the default
+	// "outbase" directory, which is computed as the lowest common ancestor of
+	// all automatically generated output paths.
+	OutputPathWasAutoGenerated bool
+}
+
+type LinkerGraph struct {
+	Files       []LinkerFile
+	entryPoints []EntryPoint
+	Symbols     ast.SymbolMap
+
+	// This is for cross-module inlining of TypeScript enum constants
+	TSEnums map[ast.Ref]map[string]js_ast.TSEnumValue
+
+	// This is for cross-module inlining of detected inlinable constants
+	ConstValues map[ast.Ref]js_ast.ConstValue
+
+	// We should avoid traversing all files in the bundle, because the linker
+	// should be able to run a linking operation on a large bundle where only
+	// a few files are needed (e.g. an incremental compilation scenario). This
+	// holds all files that could possibly be reached through the entry points.
+	// If you need to iterate over all files in the linking operation, iterate
+	// over this array. This array is also sorted in a deterministic ordering
+	// to help ensure deterministic builds (source indices are random).
+	ReachableFiles []uint32
+
+	// This maps from unstable source index to stable reachable file index. This
+	// is useful as a deterministic key for sorting if you need to sort something
+	// containing a source index (such as "ast.Ref" symbol references).
+	StableSourceIndices []uint32
+}
+
+func CloneLinkerGraph(
+	inputFiles []InputFile,
+	reachableFiles []uint32,
+	originalEntryPoints []EntryPoint,
+	codeSplitting bool,
+) LinkerGraph {
+	entryPoints := append([]EntryPoint{}, originalEntryPoints...)
+	symbols := ast.NewSymbolMap(len(inputFiles))
+	files := make([]LinkerFile, len(inputFiles))
+
+	// Mark all entry points so we don't add them again for import() expressions
+	for _, entryPoint := range entryPoints {
+		files[entryPoint.SourceIndex].entryPointKind = entryPointUserSpecified
+	}
+
+	// Clone various things since we may mutate them later. Do this in parallel
+	// for a speedup (around ~2x faster for this function in the three.js
+	// benchmark on a 6-core laptop).
+	var dynamicImportEntryPoints []uint32
+	var dynamicImportEntryPointsMutex sync.Mutex
+	waitGroup := sync.WaitGroup{}
+	waitGroup.Add(len(reachableFiles))
+	stableSourceIndices := make([]uint32, len(inputFiles))
+	for stableIndex, sourceIndex := range reachableFiles {
+		// Create a way to convert source indices to a stable ordering
+		stableSourceIndices[sourceIndex] = uint32(stableIndex)
+
+		go func(sourceIndex uint32) {
+			file := &files[sourceIndex]
+			file.InputFile = inputFiles[sourceIndex]
+
+			switch repr := file.InputFile.Repr.(type) {
+			case *JSRepr:
+				// Clone the representation
+				{
+					clone := *repr
+					repr = &clone
+					file.InputFile.Repr = repr
+				}
+
+				// Clone the symbol map
+				fileSymbols := append([]ast.Symbol{}, repr.AST.Symbols...)
+				symbols.SymbolsForSource[sourceIndex] = fileSymbols
+				repr.AST.Symbols = nil
+
+				// Clone the parts
+				repr.AST.Parts = append([]js_ast.Part{}, repr.AST.Parts...)
+				for i := range repr.AST.Parts {
+					part := &repr.AST.Parts[i]
+					clone := make(map[ast.Ref]js_ast.SymbolUse, len(part.SymbolUses))
+					for ref, uses := range part.SymbolUses {
+						clone[ref] = uses
+					}
+					part.SymbolUses = clone
+				}
+
+				// Clone the import records
+				repr.AST.ImportRecords = append([]ast.ImportRecord{}, repr.AST.ImportRecords...)
+
+				// Add dynamic imports as additional entry points if code splitting is active
+				if codeSplitting {
+					for importRecordIndex := range repr.AST.ImportRecords {
+						if record := &repr.AST.ImportRecords[importRecordIndex]; record.SourceIndex.IsValid() && record.Kind == ast.ImportDynamic {
+							dynamicImportEntryPointsMutex.Lock()
+							dynamicImportEntryPoints = append(dynamicImportEntryPoints, record.SourceIndex.GetIndex())
+							dynamicImportEntryPointsMutex.Unlock()
+
+							// Remove import assertions for dynamic imports of additional
+							// entry points so that they don't mess with the run-time behavior.
+							// For example, "import('./foo.json', { assert: { type: 'json' } })"
+							// will likely be converted into an import of a JavaScript file and
+							// leaving the import assertion there will prevent it from working.
+							record.AssertOrWith = nil
+						}
+					}
+				}
+
+				// Clone the import map
+				namedImports := make(map[ast.Ref]js_ast.NamedImport, len(repr.AST.NamedImports))
+				for k, v := range repr.AST.NamedImports {
+					namedImports[k] = v
+				}
+				repr.AST.NamedImports = namedImports
+
+				// Clone the export map
+				resolvedExports := make(map[string]ExportData)
+				for alias, name := range repr.AST.NamedExports {
+					resolvedExports[alias] = ExportData{
+						Ref:         name.Ref,
+						SourceIndex: sourceIndex,
+						NameLoc:     name.AliasLoc,
+					}
+				}
+
+				// Clone the top-level scope so we can generate more variables
+				{
+					new := &js_ast.Scope{}
+					*new = *repr.AST.ModuleScope
+					new.Generated = append([]ast.Ref{}, new.Generated...)
+					repr.AST.ModuleScope = new
+				}
+
+				// Also associate some default metadata with the file
+				repr.Meta.ResolvedExports = resolvedExports
+				repr.Meta.IsProbablyTypeScriptType = make(map[ast.Ref]bool)
+				repr.Meta.ImportsToBind = make(map[ast.Ref]ImportData)
+
+			case *CSSRepr:
+				// Clone the representation
+				{
+					clone := *repr
+					repr = &clone
+					file.InputFile.Repr = repr
+				}
+
+				// Clone the symbol map
+				fileSymbols := append([]ast.Symbol{}, repr.AST.Symbols...)
+				symbols.SymbolsForSource[sourceIndex] = fileSymbols
+				repr.AST.Symbols = nil
+
+				// Clone the import records
+				repr.AST.ImportRecords = append([]ast.ImportRecord{}, repr.AST.ImportRecords...)
+			}
+
+			// All files start off as far as possible from an entry point
+			file.DistanceFromEntryPoint = ^uint32(0)
+			waitGroup.Done()
+		}(sourceIndex)
+	}
+	waitGroup.Wait()
+
+	// Process dynamic entry points after merging control flow again
+	stableEntryPoints := make([]int, 0, len(dynamicImportEntryPoints))
+	for _, sourceIndex := range dynamicImportEntryPoints {
+		if otherFile := &files[sourceIndex]; otherFile.entryPointKind == entryPointNone {
+			stableEntryPoints = append(stableEntryPoints, int(stableSourceIndices[sourceIndex]))
+			otherFile.entryPointKind = entryPointDynamicImport
+		}
+	}
+
+	// Make sure to add dynamic entry points in a deterministic order
+	sort.Ints(stableEntryPoints)
+	for _, stableIndex := range stableEntryPoints {
+		entryPoints = append(entryPoints, EntryPoint{SourceIndex: reachableFiles[stableIndex]})
+	}
+
+	// Do a final quick pass over all files
+	var tsEnums map[ast.Ref]map[string]js_ast.TSEnumValue
+	var constValues map[ast.Ref]js_ast.ConstValue
+	bitCount := uint(len(entryPoints))
+	for _, sourceIndex := range reachableFiles {
+		file := &files[sourceIndex]
+
+		// Allocate the entry bit set now that the number of entry points is known
+		file.EntryBits = helpers.NewBitSet(bitCount)
+
+		// Merge TypeScript enums together into one big map. There likely aren't
+		// too many enum definitions relative to the overall size of the code so
+		// it should be fine to just merge them together in serial.
+		if repr, ok := file.InputFile.Repr.(*JSRepr); ok && repr.AST.TSEnums != nil {
+			if tsEnums == nil {
+				tsEnums = make(map[ast.Ref]map[string]js_ast.TSEnumValue)
+			}
+			for ref, enum := range repr.AST.TSEnums {
+				tsEnums[ref] = enum
+			}
+		}
+
+		// Also merge const values into one big map as well
+		if repr, ok := file.InputFile.Repr.(*JSRepr); ok && repr.AST.ConstValues != nil {
+			if constValues == nil {
+				constValues = make(map[ast.Ref]js_ast.ConstValue)
+			}
+			for ref, value := range repr.AST.ConstValues {
+				constValues[ref] = value
+			}
+		}
+	}
+
+	return LinkerGraph{
+		Symbols:             symbols,
+		TSEnums:             tsEnums,
+		ConstValues:         constValues,
+		entryPoints:         entryPoints,
+		Files:               files,
+		ReachableFiles:      reachableFiles,
+		StableSourceIndices: stableSourceIndices,
+	}
+}
+
+// Prevent packages that depend on us from adding or removing entry points
+func (g *LinkerGraph) EntryPoints() []EntryPoint {
+	return g.entryPoints
+}
+
+func (g *LinkerGraph) AddPartToFile(sourceIndex uint32, part js_ast.Part) uint32 {
+	// Invariant: this map is never null
+	if part.SymbolUses == nil {
+		part.SymbolUses = make(map[ast.Ref]js_ast.SymbolUse)
+	}
+
+	repr := g.Files[sourceIndex].InputFile.Repr.(*JSRepr)
+	partIndex := uint32(len(repr.AST.Parts))
+	repr.AST.Parts = append(repr.AST.Parts, part)
+
+	// Invariant: the parts for all top-level symbols can be found in the file-level map
+	for _, declaredSymbol := range part.DeclaredSymbols {
+		if declaredSymbol.IsTopLevel {
+			// Check for an existing overlay
+			partIndices, ok := repr.Meta.TopLevelSymbolToPartsOverlay[declaredSymbol.Ref]
+
+			// If missing, initialize using the original values from the parser
+			if !ok {
+				partIndices = append(partIndices, repr.AST.TopLevelSymbolToPartsFromParser[declaredSymbol.Ref]...)
+			}
+
+			// Add this part to the overlay
+			partIndices = append(partIndices, partIndex)
+			if repr.Meta.TopLevelSymbolToPartsOverlay == nil {
+				repr.Meta.TopLevelSymbolToPartsOverlay = make(map[ast.Ref][]uint32)
+			}
+			repr.Meta.TopLevelSymbolToPartsOverlay[declaredSymbol.Ref] = partIndices
+		}
+	}
+
+	return partIndex
+}
+
+func (g *LinkerGraph) GenerateNewSymbol(sourceIndex uint32, kind ast.SymbolKind, originalName string) ast.Ref {
+	sourceSymbols := &g.Symbols.SymbolsForSource[sourceIndex]
+
+	ref := ast.Ref{
+		SourceIndex: sourceIndex,
+		InnerIndex:  uint32(len(*sourceSymbols)),
+	}
+
+	*sourceSymbols = append(*sourceSymbols, ast.Symbol{
+		Kind:         kind,
+		OriginalName: originalName,
+		Link:         ast.InvalidRef,
+	})
+
+	generated := &g.Files[sourceIndex].InputFile.Repr.(*JSRepr).AST.ModuleScope.Generated
+	*generated = append(*generated, ref)
+	return ref
+}
+
+func (g *LinkerGraph) GenerateSymbolImportAndUse(
+	sourceIndex uint32,
+	partIndex uint32,
+	ref ast.Ref,
+	useCount uint32,
+	sourceIndexToImportFrom uint32,
+) {
+	if useCount == 0 {
+		return
+	}
+
+	repr := g.Files[sourceIndex].InputFile.Repr.(*JSRepr)
+	part := &repr.AST.Parts[partIndex]
+
+	// Mark this symbol as used by this part
+	use := part.SymbolUses[ref]
+	use.CountEstimate += useCount
+	part.SymbolUses[ref] = use
+
+	// Uphold invariants about the CommonJS "exports" and "module" symbols
+	if ref == repr.AST.ExportsRef {
+		repr.AST.UsesExportsRef = true
+	}
+	if ref == repr.AST.ModuleRef {
+		repr.AST.UsesModuleRef = true
+	}
+
+	// Track that this specific symbol was imported
+	if sourceIndexToImportFrom != sourceIndex {
+		repr.Meta.ImportsToBind[ref] = ImportData{
+			SourceIndex: sourceIndexToImportFrom,
+			Ref:         ref,
+		}
+	}
+
+	// Pull in all parts that declare this symbol
+	targetRepr := g.Files[sourceIndexToImportFrom].InputFile.Repr.(*JSRepr)
+	for _, partIndex := range targetRepr.TopLevelSymbolToParts(ref) {
+		part.Dependencies = append(part.Dependencies, js_ast.Dependency{
+			SourceIndex: sourceIndexToImportFrom,
+			PartIndex:   partIndex,
+		})
+	}
+}
+
+func (g *LinkerGraph) GenerateRuntimeSymbolImportAndUse(
+	sourceIndex uint32,
+	partIndex uint32,
+	name string,
+	useCount uint32,
+) {
+	if useCount == 0 {
+		return
+	}
+
+	runtimeRepr := g.Files[runtime.SourceIndex].InputFile.Repr.(*JSRepr)
+	ref := runtimeRepr.AST.NamedExports[name].Ref
+	g.GenerateSymbolImportAndUse(sourceIndex, partIndex, ref, useCount, runtime.SourceIndex)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/graph/input.go b/source/vendor/github.com/evanw/esbuild/internal/graph/input.go
new file mode 100644
index 0000000..7faa763
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/graph/input.go
@@ -0,0 +1,127 @@
+package graph
+
+// The code in this file mainly represents data that passes from the scan phase
+// to the compile phase of the bundler. There is currently one exception: the
+// "meta" member of the JavaScript file representation. That could have been
+// stored separately but is stored together for convenience and to avoid an
+// extra level of indirection. Instead it's kept in a separate type to keep
+// things organized.
+
+import (
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/config"
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/logger"
+	"github.com/evanw/esbuild/internal/resolver"
+	"github.com/evanw/esbuild/internal/sourcemap"
+)
+
+type InputFile struct {
+	Repr           InputFileRepr
+	InputSourceMap *sourcemap.SourceMap
+
+	// If this file ends up being used in the bundle, these are additional files
+	// that must be written to the output directory. It's used by the "file" and
+	// "copy" loaders.
+	AdditionalFiles            []OutputFile
+	UniqueKeyForAdditionalFile string
+
+	SideEffects SideEffects
+	Source      logger.Source
+	Loader      config.Loader
+
+	OmitFromSourceMapsAndMetafile bool
+}
+
+type OutputFile struct {
+	// If "AbsMetadataFile" is present, this will be filled out with information
+	// about this file in JSON format. This is a partial JSON file that will be
+	// fully assembled later.
+	JSONMetadataChunk string
+
+	AbsPath      string
+	Contents     []byte
+	IsExecutable bool
+}
+
+type SideEffects struct {
+	// This is optional additional information for use in error messages
+	Data *resolver.SideEffectsData
+
+	Kind SideEffectsKind
+}
+
+type SideEffectsKind uint8
+
+const (
+	// The default value conservatively considers all files to have side effects.
+	HasSideEffects SideEffectsKind = iota
+
+	// This file was listed as not having side effects by a "package.json"
+	// file in one of our containing directories with a "sideEffects" field.
+	NoSideEffects_PackageJSON
+
+	// This file is considered to have no side effects because the AST was empty
+	// after parsing finished. This should be the case for ".d.ts" files.
+	NoSideEffects_EmptyAST
+
+	// This file was loaded using a data-oriented loader (e.g. "text") that is
+	// known to not have side effects.
+	NoSideEffects_PureData
+
+	// Same as above but it came from a plugin. We don't want to warn about
+	// unused imports to these files since running the plugin is a side effect.
+	// Removing the import would not call the plugin which is observable.
+	NoSideEffects_PureData_FromPlugin
+)
+
+type InputFileRepr interface {
+	ImportRecords() *[]ast.ImportRecord
+}
+
+type JSRepr struct {
+	Meta JSReprMeta
+	AST  js_ast.AST
+
+	// If present, this is the CSS file that this JavaScript stub corresponds to.
+	// A JavaScript stub is automatically generated for a CSS file when it's
+	// imported from a JavaScript file.
+	CSSSourceIndex ast.Index32
+}
+
+func (repr *JSRepr) ImportRecords() *[]ast.ImportRecord {
+	return &repr.AST.ImportRecords
+}
+
+func (repr *JSRepr) TopLevelSymbolToParts(ref ast.Ref) []uint32 {
+	// Overlay the mutable map from the linker
+	if parts, ok := repr.Meta.TopLevelSymbolToPartsOverlay[ref]; ok {
+		return parts
+	}
+
+	// Fall back to the immutable map from the parser
+	return repr.AST.TopLevelSymbolToPartsFromParser[ref]
+}
+
+type CSSRepr struct {
+	AST css_ast.AST
+
+	// If present, this is the JavaScript stub corresponding to this CSS file.
+	// A JavaScript stub is automatically generated for a CSS file when it's
+	// imported from a JavaScript file.
+	JSSourceIndex ast.Index32
+}
+
+func (repr *CSSRepr) ImportRecords() *[]ast.ImportRecord {
+	return &repr.AST.ImportRecords
+}
+
+type CopyRepr struct {
+	// The URL that replaces the contents of any import record paths for this file
+	URLForCode string
+}
+
+func (repr *CopyRepr) ImportRecords() *[]ast.ImportRecord {
+	return nil
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/graph/meta.go b/source/vendor/github.com/evanw/esbuild/internal/graph/meta.go
new file mode 100644
index 0000000..208056c
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/graph/meta.go
@@ -0,0 +1,205 @@
+package graph
+
+// The code in this file represents data that is required by the compile phase
+// of the bundler but that is not required by the scan phase.
+
+import (
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type WrapKind uint8
+
+const (
+	WrapNone WrapKind = iota
+
+	// The module will be bundled CommonJS-style like this:
+	//
+	//   // foo.ts
+	//   let require_foo = __commonJS((exports, module) => {
+	//     exports.foo = 123;
+	//   });
+	//
+	//   // bar.ts
+	//   let foo = flag ? require_foo() : null;
+	//
+	WrapCJS
+
+	// The module will be bundled ESM-style like this:
+	//
+	//   // foo.ts
+	//   var foo, foo_exports = {};
+	//   __export(foo_exports, {
+	//     foo: () => foo
+	//   });
+	//   let init_foo = __esm(() => {
+	//     foo = 123;
+	//   });
+	//
+	//   // bar.ts
+	//   let foo = flag ? (init_foo(), __toCommonJS(foo_exports)) : null;
+	//
+	WrapESM
+)
+
+// This contains linker-specific metadata corresponding to a "file" struct
+// from the initial scan phase of the bundler. It's separated out because it's
+// conceptually only used for a single linking operation and because multiple
+// linking operations may be happening in parallel with different metadata for
+// the same file.
+type JSReprMeta struct {
+	// This is only for TypeScript files. If an import symbol is in this map, it
+	// means the import couldn't be found and doesn't actually exist. This is not
+	// an error in TypeScript because the import is probably just a type.
+	//
+	// Normally we remove all unused imports for TypeScript files during parsing,
+	// which automatically removes type-only imports. But there are certain re-
+	// export situations where it's impossible to tell if an import is a type or
+	// not:
+	//
+	//   import {typeOrNotTypeWhoKnows} from 'path';
+	//   export {typeOrNotTypeWhoKnows};
+	//
+	// Really people should be using the TypeScript "isolatedModules" flag with
+	// bundlers like this one that compile TypeScript files independently without
+	// type checking. That causes the TypeScript type checker to emit the error
+	// "Re-exporting a type when the '--isolatedModules' flag is provided requires
+	// using 'export type'." But we try to be robust to such code anyway.
+	IsProbablyTypeScriptType map[ast.Ref]bool
+
+	// Imports are matched with exports in a separate pass from when the matched
+	// exports are actually bound to the imports. Here "binding" means adding non-
+	// local dependencies on the parts in the exporting file that declare the
+	// exported symbol to all parts in the importing file that use the imported
+	// symbol.
+	//
+	// This must be a separate pass because of the "probably TypeScript type"
+	// check above. We can't generate the part for the export namespace until
+	// we've matched imports with exports because the generated code must omit
+	// type-only imports in the export namespace code. And we can't bind exports
+	// to imports until the part for the export namespace is generated since that
+	// part needs to participate in the binding.
+	//
+	// This array holds the deferred imports to bind so the pass can be split
+	// into two separate passes.
+	ImportsToBind map[ast.Ref]ImportData
+
+	// This includes both named exports and re-exports.
+	//
+	// Named exports come from explicit export statements in the original file,
+	// and are copied from the "NamedExports" field in the AST.
+	//
+	// Re-exports come from other files and are the result of resolving export
+	// star statements (i.e. "export * from 'foo'").
+	ResolvedExports     map[string]ExportData
+	ResolvedExportStar  *ExportData
+	ResolvedExportTypos *helpers.TypoDetector
+
+	// Never iterate over "resolvedExports" directly. Instead, iterate over this
+	// array. Some exports in that map aren't meant to end up in generated code.
+	// This array excludes these exports and is also sorted, which avoids non-
+	// determinism due to random map iteration order.
+	SortedAndFilteredExportAliases []string
+
+	// This is merged on top of the corresponding map from the parser in the AST.
+	// You should call "TopLevelSymbolToParts" to access this instead of accessing
+	// it directly.
+	TopLevelSymbolToPartsOverlay map[ast.Ref][]uint32
+
+	// If this is an entry point, this array holds a reference to one free
+	// temporary symbol for each entry in "sortedAndFilteredExportAliases".
+	// These may be needed to store copies of CommonJS re-exports in ESM.
+	CJSExportCopies []ast.Ref
+
+	// The index of the automatically-generated part used to represent the
+	// CommonJS or ESM wrapper. This part is empty and is only useful for tree
+	// shaking and code splitting. The wrapper can't be inserted into the part
+	// because the wrapper contains other parts, which can't be represented by
+	// the current part system. Only wrapped files have one of these.
+	WrapperPartIndex ast.Index32
+
+	// The index of the automatically-generated part used to handle entry point
+	// specific stuff. If a certain part is needed by the entry point, it's added
+	// as a dependency of this part. This is important for parts that are marked
+	// as removable when unused and that are not used by anything else. Only
+	// entry point files have one of these.
+	EntryPointPartIndex ast.Index32
+
+	// This is true if this file is affected by top-level await, either by having
+	// a top-level await inside this file or by having an import/export statement
+	// that transitively imports such a file. It is forbidden to call "require()"
+	// on these files since they are evaluated asynchronously.
+	IsAsyncOrHasAsyncDependency bool
+
+	Wrap WrapKind
+
+	// If true, we need to insert "var exports = {};". This is the case for ESM
+	// files when the import namespace is captured via "import * as" and also
+	// when they are the target of a "require()" call.
+	NeedsExportsVariable bool
+
+	// If true, the "__export(exports, { ... })" call will be force-included even
+	// if there are no parts that reference "exports". Otherwise this call will
+	// be removed due to the tree shaking pass. This is used when for entry point
+	// files when code related to the current output format needs to reference
+	// the "exports" variable.
+	ForceIncludeExportsForEntryPoint bool
+
+	// This is set when we need to pull in the "__export" symbol in to the part
+	// at "nsExportPartIndex". This can't be done in "createExportsForFile"
+	// because of concurrent map hazards. Instead, it must be done later.
+	NeedsExportSymbolFromRuntime bool
+
+	// Wrapped files must also ensure that their dependencies are wrapped. This
+	// flag is used during the traversal that enforces this invariant, and is used
+	// to detect when the fixed point has been reached.
+	DidWrapDependencies bool
+}
+
+type ImportData struct {
+	// This is an array of intermediate statements that re-exported this symbol
+	// in a chain before getting to the final symbol. This can be done either with
+	// "export * from" or "export {} from". If this is done with "export * from"
+	// then this may not be the result of a single chain but may instead form
+	// a diamond shape if this same symbol was re-exported multiple times from
+	// different files.
+	ReExports []js_ast.Dependency
+
+	NameLoc     logger.Loc // Optional, goes with sourceIndex, ignore if zero
+	Ref         ast.Ref
+	SourceIndex uint32
+}
+
+type ExportData struct {
+	// Export star resolution happens first before import resolution. That means
+	// it cannot yet determine if duplicate names from export star resolution are
+	// ambiguous (point to different symbols) or not (point to the same symbol).
+	// This issue can happen in the following scenario:
+	//
+	//   // entry.js
+	//   export * from './a'
+	//   export * from './b'
+	//
+	//   // a.js
+	//   export * from './c'
+	//
+	//   // b.js
+	//   export {x} from './c'
+	//
+	//   // c.js
+	//   export let x = 1, y = 2
+	//
+	// In this case "entry.js" should have two exports "x" and "y", neither of
+	// which are ambiguous. To handle this case, ambiguity resolution must be
+	// deferred until import resolution time. That is done using this array.
+	PotentiallyAmbiguousExportStarRefs []ImportData
+
+	Ref ast.Ref
+
+	// This is the file that the named export above came from. This will be
+	// different from the file that contains this object if this is a re-export.
+	NameLoc     logger.Loc // Optional, goes with sourceIndex, ignore if zero
+	SourceIndex uint32
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/bitset.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/bitset.go
new file mode 100644
index 0000000..47b0c1c
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/bitset.go
@@ -0,0 +1,27 @@
+package helpers
+
+import "bytes"
+
+type BitSet struct {
+	entries []byte
+}
+
+func NewBitSet(bitCount uint) BitSet {
+	return BitSet{make([]byte, (bitCount+7)/8)}
+}
+
+func (bs BitSet) HasBit(bit uint) bool {
+	return (bs.entries[bit/8] & (1 << (bit & 7))) != 0
+}
+
+func (bs BitSet) SetBit(bit uint) {
+	bs.entries[bit/8] |= 1 << (bit & 7)
+}
+
+func (bs BitSet) Equals(other BitSet) bool {
+	return bytes.Equal(bs.entries, other.entries)
+}
+
+func (bs BitSet) String() string {
+	return string(bs.entries)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/comment.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/comment.go
new file mode 100644
index 0000000..95b8684
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/comment.go
@@ -0,0 +1,29 @@
+package helpers
+
+import (
+	"strings"
+)
+
+func EscapeClosingTag(text string, slashTag string) string {
+	if slashTag == "" {
+		return text
+	}
+	i := strings.Index(text, "</")
+	if i < 0 {
+		return text
+	}
+	var b strings.Builder
+	for {
+		b.WriteString(text[:i+1])
+		text = text[i+1:]
+		if len(text) >= len(slashTag) && strings.EqualFold(text[:len(slashTag)], slashTag) {
+			b.WriteByte('\\')
+		}
+		i = strings.Index(text, "</")
+		if i < 0 {
+			break
+		}
+	}
+	b.WriteString(text)
+	return b.String()
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/dataurl.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/dataurl.go
new file mode 100644
index 0000000..2b5004c
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/dataurl.go
@@ -0,0 +1,72 @@
+package helpers
+
+import (
+	"encoding/base64"
+	"fmt"
+	"strings"
+	"unicode/utf8"
+)
+
+// Returns the shorter of either a base64-encoded or percent-escaped data URL
+func EncodeStringAsShortestDataURL(mimeType string, text string) string {
+	encoded := base64.StdEncoding.EncodeToString([]byte(text))
+	url := fmt.Sprintf("data:%s;base64,%s", mimeType, encoded)
+	if percentURL, ok := EncodeStringAsPercentEscapedDataURL(mimeType, text); ok && len(percentURL) < len(url) {
+		return percentURL
+	}
+	return url
+}
+
+// See "scripts/dataurl-escapes.html" for how this was derived
+func EncodeStringAsPercentEscapedDataURL(mimeType string, text string) (string, bool) {
+	hex := "0123456789ABCDEF"
+	sb := strings.Builder{}
+	n := len(text)
+	i := 0
+	runStart := 0
+	sb.WriteString("data:")
+	sb.WriteString(mimeType)
+	sb.WriteByte(',')
+
+	// Scan for trailing characters that need to be escaped
+	trailingStart := n
+	for trailingStart > 0 {
+		if c := text[trailingStart-1]; c > 0x20 || c == '\t' || c == '\n' || c == '\r' {
+			break
+		}
+		trailingStart--
+	}
+
+	for i < n {
+		c, width := utf8.DecodeRuneInString(text[i:])
+
+		// We can't encode invalid UTF-8 data
+		if c == utf8.RuneError && width == 1 {
+			return "", false
+		}
+
+		// Escape this character if needed
+		if c == '\t' || c == '\n' || c == '\r' || c == '#' || i >= trailingStart ||
+			(c == '%' && i+2 < n && isHex(text[i+1]) && isHex(text[i+2])) {
+			if runStart < i {
+				sb.WriteString(text[runStart:i])
+			}
+			sb.WriteByte('%')
+			sb.WriteByte(hex[c>>4])
+			sb.WriteByte(hex[c&15])
+			runStart = i + width
+		}
+
+		i += width
+	}
+
+	if runStart < n {
+		sb.WriteString(text[runStart:])
+	}
+
+	return sb.String(), true
+}
+
+func isHex(c byte) bool {
+	return c >= '0' && c <= '9' || c >= 'a' && c <= 'f' || c >= 'A' && c <= 'F'
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/float.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/float.go
new file mode 100644
index 0000000..02b3ac9
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/float.go
@@ -0,0 +1,158 @@
+package helpers
+
+import "math"
+
+// This wraps float64 math operations. Why does this exist? The Go compiler
+// contains some optimizations to take advantage of "fused multiply and add"
+// (FMA) instructions on certain processors. These instructions lead to
+// different output on those processors, which means esbuild's output is no
+// longer deterministic across all platforms. From the Go specification itself
+// (https://go.dev/ref/spec#Floating_point_operators):
+//
+//	An implementation may combine multiple floating-point operations into a
+//	single fused operation, possibly across statements, and produce a result
+//	that differs from the value obtained by executing and rounding the
+//	instructions individually. An explicit floating-point type conversion
+//	rounds to the precision of the target type, preventing fusion that would
+//	discard that rounding.
+//
+//	For instance, some architectures provide a "fused multiply and add" (FMA)
+//	instruction that computes x*y + z without rounding the intermediate result
+//	x*y.
+//
+// Therefore we need to add explicit type conversions such as "float64(x)" to
+// prevent optimizations that break correctness. Rather than adding them on a
+// case-by-case basis as real correctness issues are discovered, we instead
+// preemptively force them to be added everywhere by using this wrapper type
+// for all floating-point math.
+type F64 struct {
+	value float64
+}
+
+func NewF64(a float64) F64 {
+	return F64{value: float64(a)}
+}
+
+func (a F64) Value() float64 {
+	return a.value
+}
+
+func (a F64) IsNaN() bool {
+	return math.IsNaN(a.value)
+}
+
+func (a F64) Neg() F64 {
+	return NewF64(-a.value)
+}
+
+func (a F64) Abs() F64 {
+	return NewF64(math.Abs(a.value))
+}
+
+func (a F64) Sin() F64 {
+	return NewF64(math.Sin(a.value))
+}
+
+func (a F64) Cos() F64 {
+	return NewF64(math.Cos(a.value))
+}
+
+func (a F64) Log2() F64 {
+	return NewF64(math.Log2(a.value))
+}
+
+func (a F64) Round() F64 {
+	return NewF64(math.Round(a.value))
+}
+
+func (a F64) Floor() F64 {
+	return NewF64(math.Floor(a.value))
+}
+
+func (a F64) Ceil() F64 {
+	return NewF64(math.Ceil(a.value))
+}
+
+func (a F64) Squared() F64 {
+	return a.Mul(a)
+}
+
+func (a F64) Cubed() F64 {
+	return a.Mul(a).Mul(a)
+}
+
+func (a F64) Sqrt() F64 {
+	return NewF64(math.Sqrt(a.value))
+}
+
+func (a F64) Cbrt() F64 {
+	return NewF64(math.Cbrt(a.value))
+}
+
+func (a F64) Add(b F64) F64 {
+	return NewF64(a.value + b.value)
+}
+
+func (a F64) AddConst(b float64) F64 {
+	return NewF64(a.value + b)
+}
+
+func (a F64) Sub(b F64) F64 {
+	return NewF64(a.value - b.value)
+}
+
+func (a F64) SubConst(b float64) F64 {
+	return NewF64(a.value - b)
+}
+
+func (a F64) Mul(b F64) F64 {
+	return NewF64(a.value * b.value)
+}
+
+func (a F64) MulConst(b float64) F64 {
+	return NewF64(a.value * b)
+}
+
+func (a F64) Div(b F64) F64 {
+	return NewF64(a.value / b.value)
+}
+
+func (a F64) DivConst(b float64) F64 {
+	return NewF64(a.value / b)
+}
+
+func (a F64) Pow(b F64) F64 {
+	return NewF64(math.Pow(a.value, b.value))
+}
+
+func (a F64) PowConst(b float64) F64 {
+	return NewF64(math.Pow(a.value, b))
+}
+
+func (a F64) Atan2(b F64) F64 {
+	return NewF64(math.Atan2(a.value, b.value))
+}
+
+func (a F64) WithSignFrom(b F64) F64 {
+	return NewF64(math.Copysign(a.value, b.value))
+}
+
+func Min2(a F64, b F64) F64 {
+	return NewF64(math.Min(a.value, b.value))
+}
+
+func Max2(a F64, b F64) F64 {
+	return NewF64(math.Max(a.value, b.value))
+}
+
+func Min3(a F64, b F64, c F64) F64 {
+	return NewF64(math.Min(math.Min(a.value, b.value), c.value))
+}
+
+func Max3(a F64, b F64, c F64) F64 {
+	return NewF64(math.Max(math.Max(a.value, b.value), c.value))
+}
+
+func Lerp(a F64, b F64, t F64) F64 {
+	return b.Sub(a).Mul(t).Add(a)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/glob.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/glob.go
new file mode 100644
index 0000000..c8ffa8b
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/glob.go
@@ -0,0 +1,58 @@
+package helpers
+
+import "strings"
+
+type GlobWildcard uint8
+
+const (
+	GlobNone GlobWildcard = iota
+	GlobAllExceptSlash
+	GlobAllIncludingSlash
+)
+
+type GlobPart struct {
+	Prefix   string
+	Wildcard GlobWildcard
+}
+
+// The returned array will always be at least one element. If there are no
+// wildcards then it will be exactly one element, and if there are wildcards
+// then it will be more than one element.
+func ParseGlobPattern(text string) (pattern []GlobPart) {
+	for {
+		star := strings.IndexByte(text, '*')
+		if star < 0 {
+			pattern = append(pattern, GlobPart{Prefix: text})
+			break
+		}
+		count := 1
+		for star+count < len(text) && text[star+count] == '*' {
+			count++
+		}
+		wildcard := GlobAllExceptSlash
+
+		// Allow both "/" and "\" as slashes
+		if count > 1 && (star == 0 || text[star-1] == '/' || text[star-1] == '\\') &&
+			(star+count == len(text) || text[star+count] == '/' || text[star+count] == '\\') {
+			wildcard = GlobAllIncludingSlash // A "globstar" path segment
+		}
+
+		pattern = append(pattern, GlobPart{Prefix: text[:star], Wildcard: wildcard})
+		text = text[star+count:]
+	}
+	return
+}
+
+func GlobPatternToString(pattern []GlobPart) string {
+	sb := strings.Builder{}
+	for _, part := range pattern {
+		sb.WriteString(part.Prefix)
+		switch part.Wildcard {
+		case GlobAllExceptSlash:
+			sb.WriteByte('*')
+		case GlobAllIncludingSlash:
+			sb.WriteString("**")
+		}
+	}
+	return sb.String()
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/hash.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/hash.go
new file mode 100644
index 0000000..d702886
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/hash.go
@@ -0,0 +1,14 @@
+package helpers
+
+// From: http://boost.sourceforge.net/doc/html/boost/hash_combine.html
+func HashCombine(seed uint32, hash uint32) uint32 {
+	return seed ^ (hash + 0x9e3779b9 + (seed << 6) + (seed >> 2))
+}
+
+func HashCombineString(seed uint32, text string) uint32 {
+	seed = HashCombine(seed, uint32(len(text)))
+	for _, c := range text {
+		seed = HashCombine(seed, uint32(c))
+	}
+	return seed
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/joiner.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/joiner.go
new file mode 100644
index 0000000..649f80a
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/joiner.go
@@ -0,0 +1,86 @@
+package helpers
+
+import (
+	"bytes"
+	"strings"
+)
+
+// This provides an efficient way to join lots of big string and byte slices
+// together. It avoids the cost of repeatedly reallocating as the buffer grows
+// by measuring exactly how big the buffer should be and then allocating once.
+// This is a measurable speedup.
+type Joiner struct {
+	strings  []joinerString
+	bytes    []joinerBytes
+	length   uint32
+	lastByte byte
+}
+
+type joinerString struct {
+	data   string
+	offset uint32
+}
+
+type joinerBytes struct {
+	data   []byte
+	offset uint32
+}
+
+func (j *Joiner) AddString(data string) {
+	if len(data) > 0 {
+		j.lastByte = data[len(data)-1]
+	}
+	j.strings = append(j.strings, joinerString{data, j.length})
+	j.length += uint32(len(data))
+}
+
+func (j *Joiner) AddBytes(data []byte) {
+	if len(data) > 0 {
+		j.lastByte = data[len(data)-1]
+	}
+	j.bytes = append(j.bytes, joinerBytes{data, j.length})
+	j.length += uint32(len(data))
+}
+
+func (j *Joiner) LastByte() byte {
+	return j.lastByte
+}
+
+func (j *Joiner) Length() uint32 {
+	return j.length
+}
+
+func (j *Joiner) EnsureNewlineAtEnd() {
+	if j.length > 0 && j.lastByte != '\n' {
+		j.AddString("\n")
+	}
+}
+
+func (j *Joiner) Done() []byte {
+	if len(j.strings) == 0 && len(j.bytes) == 1 && j.bytes[0].offset == 0 {
+		// No need to allocate if there was only a single byte array written
+		return j.bytes[0].data
+	}
+	buffer := make([]byte, j.length)
+	for _, item := range j.strings {
+		copy(buffer[item.offset:], item.data)
+	}
+	for _, item := range j.bytes {
+		copy(buffer[item.offset:], item.data)
+	}
+	return buffer
+}
+
+func (j *Joiner) Contains(s string, b []byte) bool {
+	for _, item := range j.strings {
+		if strings.Contains(item.data, s) {
+			return true
+		}
+	}
+	for _, item := range j.bytes {
+		if bytes.Contains(item.data, b) {
+			return true
+		}
+	}
+	return false
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/mime.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/mime.go
new file mode 100644
index 0000000..f928a84
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/mime.go
@@ -0,0 +1,49 @@
+package helpers
+
+import "strings"
+
+var builtinTypesLower = map[string]string{
+	// Text
+	".css":      "text/css; charset=utf-8",
+	".htm":      "text/html; charset=utf-8",
+	".html":     "text/html; charset=utf-8",
+	".js":       "text/javascript; charset=utf-8",
+	".json":     "application/json; charset=utf-8",
+	".markdown": "text/markdown; charset=utf-8",
+	".md":       "text/markdown; charset=utf-8",
+	".mjs":      "text/javascript; charset=utf-8",
+	".xhtml":    "application/xhtml+xml; charset=utf-8",
+	".xml":      "text/xml; charset=utf-8",
+
+	// Images
+	".avif": "image/avif",
+	".gif":  "image/gif",
+	".jpeg": "image/jpeg",
+	".jpg":  "image/jpeg",
+	".png":  "image/png",
+	".svg":  "image/svg+xml",
+	".webp": "image/webp",
+
+	// Fonts
+	".eot":   "application/vnd.ms-fontobject",
+	".otf":   "font/otf",
+	".sfnt":  "font/sfnt",
+	".ttf":   "font/ttf",
+	".woff":  "font/woff",
+	".woff2": "font/woff2",
+
+	// Other
+	".pdf":         "application/pdf",
+	".wasm":        "application/wasm",
+	".webmanifest": "application/manifest+json",
+}
+
+// This is used instead of Go's built-in "mime.TypeByExtension" function because
+// that function is broken on Windows: https://github.com/golang/go/issues/32350.
+func MimeTypeByExtension(ext string) string {
+	contentType := builtinTypesLower[ext]
+	if contentType == "" {
+		contentType = builtinTypesLower[strings.ToLower(ext)]
+	}
+	return contentType
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/path.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/path.go
new file mode 100644
index 0000000..87e90b8
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/path.go
@@ -0,0 +1,22 @@
+package helpers
+
+import "strings"
+
+func IsInsideNodeModules(path string) bool {
+	for {
+		// This is written in a platform-independent manner because it's run on
+		// user-specified paths which can be arbitrary non-file-system things. So
+		// for example Windows paths may end up being used on Unix or URLs may end
+		// up being used on Windows. Be consistently agnostic to which kind of
+		// slash is used on all platforms.
+		slash := strings.LastIndexAny(path, "/\\")
+		if slash == -1 {
+			return false
+		}
+		dir, base := path[:slash], path[slash+1:]
+		if base == "node_modules" {
+			return true
+		}
+		path = dir
+	}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/quote.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/quote.go
new file mode 100644
index 0000000..a505ad4
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/quote.go
@@ -0,0 +1,142 @@
+package helpers
+
+import "unicode/utf8"
+
+const hexChars = "0123456789ABCDEF"
+const firstASCII = 0x20
+const lastASCII = 0x7E
+const firstHighSurrogate = 0xD800
+const firstLowSurrogate = 0xDC00
+const lastLowSurrogate = 0xDFFF
+
+func canPrintWithoutEscape(c rune, asciiOnly bool) bool {
+	if c <= lastASCII {
+		return c >= firstASCII && c != '\\' && c != '"'
+	} else {
+		return !asciiOnly && c != '\uFEFF' && (c < firstHighSurrogate || c > lastLowSurrogate)
+	}
+}
+
+func QuoteSingle(text string, asciiOnly bool) []byte {
+	return internalQuote(text, asciiOnly, '\'')
+}
+
+func QuoteForJSON(text string, asciiOnly bool) []byte {
+	return internalQuote(text, asciiOnly, '"')
+}
+
+func internalQuote(text string, asciiOnly bool, quoteChar byte) []byte {
+	// Estimate the required length
+	lenEstimate := 2
+	for _, c := range text {
+		if canPrintWithoutEscape(c, asciiOnly) {
+			lenEstimate += utf8.RuneLen(c)
+		} else {
+			switch c {
+			case '\b', '\f', '\n', '\r', '\t', '\\':
+				lenEstimate += 2
+			case '"':
+				if quoteChar == '"' {
+					lenEstimate += 2
+				}
+			case '\'':
+				if quoteChar == '\'' {
+					lenEstimate += 2
+				}
+			default:
+				if c <= 0xFFFF {
+					lenEstimate += 6
+				} else {
+					lenEstimate += 12
+				}
+			}
+		}
+	}
+
+	// Preallocate the array
+	bytes := make([]byte, 0, lenEstimate)
+	i := 0
+	n := len(text)
+	bytes = append(bytes, quoteChar)
+
+	for i < n {
+		c, width := DecodeWTF8Rune(text[i:])
+
+		// Fast path: a run of characters that don't need escaping
+		if canPrintWithoutEscape(c, asciiOnly) {
+			start := i
+			i += width
+			for i < n {
+				c, width = DecodeWTF8Rune(text[i:])
+				if !canPrintWithoutEscape(c, asciiOnly) {
+					break
+				}
+				i += width
+			}
+			bytes = append(bytes, text[start:i]...)
+			continue
+		}
+
+		switch c {
+		case '\b':
+			bytes = append(bytes, "\\b"...)
+			i++
+
+		case '\f':
+			bytes = append(bytes, "\\f"...)
+			i++
+
+		case '\n':
+			bytes = append(bytes, "\\n"...)
+			i++
+
+		case '\r':
+			bytes = append(bytes, "\\r"...)
+			i++
+
+		case '\t':
+			bytes = append(bytes, "\\t"...)
+			i++
+
+		case '\\':
+			bytes = append(bytes, "\\\\"...)
+			i++
+
+		case '"':
+			if quoteChar == '"' {
+				bytes = append(bytes, "\\\""...)
+			} else {
+				bytes = append(bytes, '"')
+			}
+			i++
+
+		case '\'':
+			if quoteChar == '\'' {
+				bytes = append(bytes, "\\'"...)
+			} else {
+				bytes = append(bytes, '\'')
+			}
+			i++
+
+		default:
+			i += width
+			if c <= 0xFFFF {
+				bytes = append(
+					bytes,
+					'\\', 'u', hexChars[c>>12], hexChars[(c>>8)&15], hexChars[(c>>4)&15], hexChars[c&15],
+				)
+			} else {
+				c -= 0x10000
+				lo := firstHighSurrogate + ((c >> 10) & 0x3FF)
+				hi := firstLowSurrogate + (c & 0x3FF)
+				bytes = append(
+					bytes,
+					'\\', 'u', hexChars[lo>>12], hexChars[(lo>>8)&15], hexChars[(lo>>4)&15], hexChars[lo&15],
+					'\\', 'u', hexChars[hi>>12], hexChars[(hi>>8)&15], hexChars[(hi>>4)&15], hexChars[hi&15],
+				)
+			}
+		}
+	}
+
+	return append(bytes, quoteChar)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/serializer.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/serializer.go
new file mode 100644
index 0000000..9b3ae0b
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/serializer.go
@@ -0,0 +1,26 @@
+package helpers
+
+import "sync"
+
+// Each call to "Enter(i)" doesn't start until "Leave(i-1)" is called
+type Serializer struct {
+	flags []sync.WaitGroup
+}
+
+func MakeSerializer(count int) Serializer {
+	flags := make([]sync.WaitGroup, count)
+	for i := 0; i < count; i++ {
+		flags[i].Add(1)
+	}
+	return Serializer{flags: flags}
+}
+
+func (s *Serializer) Enter(i int) {
+	if i > 0 {
+		s.flags[i-1].Wait()
+	}
+}
+
+func (s *Serializer) Leave(i int) {
+	s.flags[i].Done()
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/stack.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/stack.go
new file mode 100644
index 0000000..0c2e91c
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/stack.go
@@ -0,0 +1,50 @@
+package helpers
+
+import (
+	"runtime/debug"
+	"strings"
+)
+
+func PrettyPrintedStack() string {
+	lines := strings.Split(strings.TrimSpace(string(debug.Stack())), "\n")
+
+	// Strip the first "goroutine" line
+	if len(lines) > 0 {
+		if first := lines[0]; strings.HasPrefix(first, "goroutine ") && strings.HasSuffix(first, ":") {
+			lines = lines[1:]
+		}
+	}
+
+	sb := strings.Builder{}
+
+	for _, line := range lines {
+		// Indented lines are source locations
+		if strings.HasPrefix(line, "\t") {
+			line = line[1:]
+			line = strings.TrimPrefix(line, "github.com/evanw/esbuild/")
+			if offset := strings.LastIndex(line, " +0x"); offset != -1 {
+				line = line[:offset]
+			}
+			sb.WriteString(" (")
+			sb.WriteString(line)
+			sb.WriteString(")")
+			continue
+		}
+
+		// Other lines are function calls
+		if sb.Len() > 0 {
+			sb.WriteByte('\n')
+		}
+		if strings.HasSuffix(line, ")") {
+			if paren := strings.LastIndexByte(line, '('); paren != -1 {
+				line = line[:paren]
+			}
+		}
+		if slash := strings.LastIndexByte(line, '/'); slash != -1 {
+			line = line[slash+1:]
+		}
+		sb.WriteString(line)
+	}
+
+	return sb.String()
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/strings.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/strings.go
new file mode 100644
index 0000000..e077892
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/strings.go
@@ -0,0 +1,41 @@
+package helpers
+
+import (
+	"fmt"
+	"strings"
+)
+
+func StringArraysEqual(a []string, b []string) bool {
+	if len(a) != len(b) {
+		return false
+	}
+	for i, x := range a {
+		if x != b[i] {
+			return false
+		}
+	}
+	return true
+}
+
+func StringArrayArraysEqual(a [][]string, b [][]string) bool {
+	if len(a) != len(b) {
+		return false
+	}
+	for i, x := range a {
+		if !StringArraysEqual(x, b[i]) {
+			return false
+		}
+	}
+	return true
+}
+
+func StringArrayToQuotedCommaSeparatedString(a []string) string {
+	sb := strings.Builder{}
+	for i, str := range a {
+		if i > 0 {
+			sb.WriteString(", ")
+		}
+		sb.WriteString(fmt.Sprintf("%q", str))
+	}
+	return sb.String()
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/timer.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/timer.go
new file mode 100644
index 0000000..4502b23
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/timer.go
@@ -0,0 +1,94 @@
+package helpers
+
+import (
+	"fmt"
+	"strings"
+	"sync"
+	"time"
+
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type Timer struct {
+	data  []timerData
+	mutex sync.Mutex
+}
+
+type timerData struct {
+	time  time.Time
+	name  string
+	isEnd bool
+}
+
+func (t *Timer) Begin(name string) {
+	if t != nil {
+		t.data = append(t.data, timerData{
+			name: name,
+			time: time.Now(),
+		})
+	}
+}
+
+func (t *Timer) End(name string) {
+	if t != nil {
+		t.data = append(t.data, timerData{
+			name:  name,
+			time:  time.Now(),
+			isEnd: true,
+		})
+	}
+}
+
+func (t *Timer) Fork() *Timer {
+	if t != nil {
+		return &Timer{}
+	}
+	return nil
+}
+
+func (t *Timer) Join(other *Timer) {
+	if t != nil && other != nil {
+		t.mutex.Lock()
+		defer t.mutex.Unlock()
+		t.data = append(t.data, other.data...)
+	}
+}
+
+func (t *Timer) Log(log logger.Log) {
+	if t == nil {
+		return
+	}
+
+	type pair struct {
+		timerData
+		index uint32
+	}
+
+	var notes []logger.MsgData
+	var stack []pair
+	indent := 0
+
+	for _, item := range t.data {
+		if !item.isEnd {
+			top := pair{timerData: item, index: uint32(len(notes))}
+			notes = append(notes, logger.MsgData{DisableMaximumWidth: true})
+			stack = append(stack, top)
+			indent++
+		} else {
+			indent--
+			last := len(stack) - 1
+			top := stack[last]
+			stack = stack[:last]
+			if item.name != top.name {
+				panic("Internal error")
+			}
+			notes[top.index].Text = fmt.Sprintf("%s%s: %dms",
+				strings.Repeat("  ", indent),
+				top.name,
+				item.time.Sub(top.time).Milliseconds())
+		}
+	}
+
+	log.AddIDWithNotes(logger.MsgID_None, logger.Info, nil, logger.Range{},
+		"Timing information (times may not nest hierarchically due to parallelism)", notes)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/typos.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/typos.go
new file mode 100644
index 0000000..deef87e
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/typos.go
@@ -0,0 +1,38 @@
+package helpers
+
+import "unicode/utf8"
+
+type TypoDetector struct {
+	oneCharTypos map[string]string
+}
+
+func MakeTypoDetector(valid []string) TypoDetector {
+	detector := TypoDetector{oneCharTypos: make(map[string]string)}
+
+	// Add all combinations of each valid word with one character missing
+	for _, correct := range valid {
+		if len(correct) > 3 {
+			for i, ch := range correct {
+				detector.oneCharTypos[correct[:i]+correct[i+utf8.RuneLen(ch):]] = correct
+			}
+		}
+	}
+
+	return detector
+}
+
+func (detector TypoDetector) MaybeCorrectTypo(typo string) (string, bool) {
+	// Check for a single deleted character
+	if corrected, ok := detector.oneCharTypos[typo]; ok {
+		return corrected, true
+	}
+
+	// Check for a single misplaced character
+	for i, ch := range typo {
+		if corrected, ok := detector.oneCharTypos[typo[:i]+typo[i+utf8.RuneLen(ch):]]; ok {
+			return corrected, true
+		}
+	}
+
+	return "", false
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/utf.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/utf.go
new file mode 100644
index 0000000..59b24bb
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/utf.go
@@ -0,0 +1,230 @@
+package helpers
+
+import (
+	"strings"
+	"unicode/utf8"
+)
+
+func ContainsNonBMPCodePoint(text string) bool {
+	for _, c := range text {
+		if c > 0xFFFF {
+			return true
+		}
+	}
+	return false
+}
+
+// This does "ContainsNonBMPCodePoint(UTF16ToString(text))" without any allocations
+func ContainsNonBMPCodePointUTF16(text []uint16) bool {
+	if n := len(text); n > 0 {
+		for i, c := range text[:n-1] {
+			// Check for a high surrogate
+			if c >= 0xD800 && c <= 0xDBFF {
+				// Check for a low surrogate
+				if c2 := text[i+1]; c2 >= 0xDC00 && c2 <= 0xDFFF {
+					return true
+				}
+			}
+		}
+	}
+	return false
+}
+
+func StringToUTF16(text string) []uint16 {
+	decoded := make([]uint16, 0, len(text))
+	for _, c := range text {
+		if c <= 0xFFFF {
+			decoded = append(decoded, uint16(c))
+		} else {
+			c -= 0x10000
+			decoded = append(decoded, uint16(0xD800+((c>>10)&0x3FF)), uint16(0xDC00+(c&0x3FF)))
+		}
+	}
+	return decoded
+}
+
+func UTF16ToString(text []uint16) string {
+	var temp [utf8.UTFMax]byte
+	b := strings.Builder{}
+	n := len(text)
+	for i := 0; i < n; i++ {
+		r1 := rune(text[i])
+		if r1 >= 0xD800 && r1 <= 0xDBFF && i+1 < n {
+			if r2 := rune(text[i+1]); r2 >= 0xDC00 && r2 <= 0xDFFF {
+				r1 = (r1-0xD800)<<10 | (r2 - 0xDC00) + 0x10000
+				i++
+			}
+		}
+		width := encodeWTF8Rune(temp[:], r1)
+		b.Write(temp[:width])
+	}
+	return b.String()
+}
+
+func UTF16ToStringWithValidation(text []uint16) (string, uint16, bool) {
+	var temp [utf8.UTFMax]byte
+	b := strings.Builder{}
+	n := len(text)
+	for i := 0; i < n; i++ {
+		r1 := rune(text[i])
+		if r1 >= 0xD800 && r1 <= 0xDBFF {
+			if i+1 < n {
+				if r2 := rune(text[i+1]); r2 >= 0xDC00 && r2 <= 0xDFFF {
+					r1 = (r1-0xD800)<<10 | (r2 - 0xDC00) + 0x10000
+					i++
+				} else {
+					return "", uint16(r1), false
+				}
+			} else {
+				return "", uint16(r1), false
+			}
+		} else if r1 >= 0xDC00 && r1 <= 0xDFFF {
+			return "", uint16(r1), false
+		}
+		width := encodeWTF8Rune(temp[:], r1)
+		b.Write(temp[:width])
+	}
+	return b.String(), 0, true
+}
+
+// Does "UTF16ToString(text) == str" without a temporary allocation
+func UTF16EqualsString(text []uint16, str string) bool {
+	if len(text) > len(str) {
+		// Strings can't be equal if UTF-16 encoding is longer than UTF-8 encoding
+		return false
+	}
+	var temp [utf8.UTFMax]byte
+	n := len(text)
+	j := 0
+	for i := 0; i < n; i++ {
+		r1 := rune(text[i])
+		if r1 >= 0xD800 && r1 <= 0xDBFF && i+1 < n {
+			if r2 := rune(text[i+1]); r2 >= 0xDC00 && r2 <= 0xDFFF {
+				r1 = (r1-0xD800)<<10 | (r2 - 0xDC00) + 0x10000
+				i++
+			}
+		}
+		width := encodeWTF8Rune(temp[:], r1)
+		if j+width > len(str) {
+			return false
+		}
+		for k := 0; k < width; k++ {
+			if temp[k] != str[j] {
+				return false
+			}
+			j++
+		}
+	}
+	return j == len(str)
+}
+
+func UTF16EqualsUTF16(a []uint16, b []uint16) bool {
+	if len(a) == len(b) {
+		for i, c := range a {
+			if c != b[i] {
+				return false
+			}
+		}
+		return true
+	}
+	return false
+}
+
+// This is a clone of "utf8.EncodeRune" that has been modified to encode using
+// WTF-8 instead. See https://simonsapin.github.io/wtf-8/ for more info.
+func encodeWTF8Rune(p []byte, r rune) int {
+	// Negative values are erroneous. Making it unsigned addresses the problem.
+	switch i := uint32(r); {
+	case i <= 0x7F:
+		p[0] = byte(r)
+		return 1
+	case i <= 0x7FF:
+		_ = p[1] // eliminate bounds checks
+		p[0] = 0xC0 | byte(r>>6)
+		p[1] = 0x80 | byte(r)&0x3F
+		return 2
+	case i > utf8.MaxRune:
+		r = utf8.RuneError
+		fallthrough
+	case i <= 0xFFFF:
+		_ = p[2] // eliminate bounds checks
+		p[0] = 0xE0 | byte(r>>12)
+		p[1] = 0x80 | byte(r>>6)&0x3F
+		p[2] = 0x80 | byte(r)&0x3F
+		return 3
+	default:
+		_ = p[3] // eliminate bounds checks
+		p[0] = 0xF0 | byte(r>>18)
+		p[1] = 0x80 | byte(r>>12)&0x3F
+		p[2] = 0x80 | byte(r>>6)&0x3F
+		p[3] = 0x80 | byte(r)&0x3F
+		return 4
+	}
+}
+
+// This is a clone of "utf8.DecodeRuneInString" that has been modified to
+// decode using WTF-8 instead. See https://simonsapin.github.io/wtf-8/ for
+// more info.
+func DecodeWTF8Rune(s string) (rune, int) {
+	n := len(s)
+	if n < 1 {
+		return utf8.RuneError, 0
+	}
+
+	s0 := s[0]
+	if s0 < 0x80 {
+		return rune(s0), 1
+	}
+
+	var sz int
+	if (s0 & 0xE0) == 0xC0 {
+		sz = 2
+	} else if (s0 & 0xF0) == 0xE0 {
+		sz = 3
+	} else if (s0 & 0xF8) == 0xF0 {
+		sz = 4
+	} else {
+		return utf8.RuneError, 1
+	}
+
+	if n < sz {
+		return utf8.RuneError, 0
+	}
+
+	s1 := s[1]
+	if (s1 & 0xC0) != 0x80 {
+		return utf8.RuneError, 1
+	}
+
+	if sz == 2 {
+		cp := rune(s0&0x1F)<<6 | rune(s1&0x3F)
+		if cp < 0x80 {
+			return utf8.RuneError, 1
+		}
+		return cp, 2
+	}
+	s2 := s[2]
+
+	if (s2 & 0xC0) != 0x80 {
+		return utf8.RuneError, 1
+	}
+
+	if sz == 3 {
+		cp := rune(s0&0x0F)<<12 | rune(s1&0x3F)<<6 | rune(s2&0x3F)
+		if cp < 0x0800 {
+			return utf8.RuneError, 1
+		}
+		return cp, 3
+	}
+	s3 := s[3]
+
+	if (s3 & 0xC0) != 0x80 {
+		return utf8.RuneError, 1
+	}
+
+	cp := rune(s0&0x07)<<18 | rune(s1&0x3F)<<12 | rune(s2&0x3F)<<6 | rune(s3&0x3F)
+	if cp < 0x010000 || cp > 0x10FFFF {
+		return utf8.RuneError, 1
+	}
+	return cp, 4
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/helpers/waitgroup.go b/source/vendor/github.com/evanw/esbuild/internal/helpers/waitgroup.go
new file mode 100644
index 0000000..850e088
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/helpers/waitgroup.go
@@ -0,0 +1,37 @@
+package helpers
+
+import "sync/atomic"
+
+// Go's "sync.WaitGroup" is not thread-safe. Specifically it's not safe to call
+// "Add" concurrently with "Wait", which is problematic because we have a case
+// where we would like to do that.
+//
+// This is a simple alternative implementation of "sync.WaitGroup" that is
+// thread-safe and that works for our purposes. We don't need to worry about
+// multiple waiters so the implementation can be very simple.
+type ThreadSafeWaitGroup struct {
+	counter int32
+	channel chan struct{}
+}
+
+func MakeThreadSafeWaitGroup() *ThreadSafeWaitGroup {
+	return &ThreadSafeWaitGroup{
+		channel: make(chan struct{}, 1),
+	}
+}
+
+func (wg *ThreadSafeWaitGroup) Add(delta int32) {
+	if counter := atomic.AddInt32(&wg.counter, delta); counter == 0 {
+		wg.channel <- struct{}{}
+	} else if counter < 0 {
+		panic("sync: negative WaitGroup counter")
+	}
+}
+
+func (wg *ThreadSafeWaitGroup) Done() {
+	wg.Add(-1)
+}
+
+func (wg *ThreadSafeWaitGroup) Wait() {
+	<-wg.channel
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/js_ast/js_ast.go b/source/vendor/github.com/evanw/esbuild/internal/js_ast/js_ast.go
new file mode 100644
index 0000000..f8d3fe3
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/js_ast/js_ast.go
@@ -0,0 +1,1841 @@
+package js_ast
+
+import (
+	"strconv"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+// Every module (i.e. file) is parsed into a separate AST data structure. For
+// efficiency, the parser also resolves all scopes and binds all symbols in the
+// tree.
+//
+// Identifiers in the tree are referenced by a Ref, which is a pointer into the
+// symbol table for the file. The symbol table is stored as a top-level field
+// in the AST so it can be accessed without traversing the tree. For example,
+// a renaming pass can iterate over the symbol table without touching the tree.
+//
+// Parse trees are intended to be immutable. That makes it easy to build an
+// incremental compiler with a "watch" mode that can avoid re-parsing files
+// that have already been parsed. Any passes that operate on an AST after it
+// has been parsed should create a copy of the mutated parts of the tree
+// instead of mutating the original tree.
+
+type L uint8
+
+// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Operator_Precedence
+const (
+	LLowest L = iota
+	LComma
+	LSpread
+	LYield
+	LAssign
+	LConditional
+	LNullishCoalescing
+	LLogicalOr
+	LLogicalAnd
+	LBitwiseOr
+	LBitwiseXor
+	LBitwiseAnd
+	LEquals
+	LCompare
+	LShift
+	LAdd
+	LMultiply
+	LExponentiation
+	LPrefix
+	LPostfix
+	LNew
+	LCall
+	LMember
+)
+
+type OpCode uint8
+
+func (op OpCode) IsPrefix() bool {
+	return op < UnOpPostDec
+}
+
+func (op OpCode) UnaryAssignTarget() AssignTarget {
+	if op >= UnOpPreDec && op <= UnOpPostInc {
+		return AssignTargetUpdate
+	}
+	return AssignTargetNone
+}
+
+func (op OpCode) IsLeftAssociative() bool {
+	return op >= BinOpAdd && op < BinOpComma && op != BinOpPow
+}
+
+func (op OpCode) IsRightAssociative() bool {
+	return op >= BinOpAssign || op == BinOpPow
+}
+
+func (op OpCode) BinaryAssignTarget() AssignTarget {
+	if op == BinOpAssign {
+		return AssignTargetReplace
+	}
+	if op > BinOpAssign {
+		return AssignTargetUpdate
+	}
+	return AssignTargetNone
+}
+
+func (op OpCode) IsShortCircuit() bool {
+	switch op {
+	case BinOpLogicalOr, BinOpLogicalOrAssign,
+		BinOpLogicalAnd, BinOpLogicalAndAssign,
+		BinOpNullishCoalescing, BinOpNullishCoalescingAssign:
+		return true
+	}
+	return false
+}
+
+type AssignTarget uint8
+
+const (
+	AssignTargetNone    AssignTarget = iota
+	AssignTargetReplace              // "a = b"
+	AssignTargetUpdate               // "a += b"
+)
+
+// If you add a new token, remember to add it to "OpTable" too
+const (
+	// Prefix
+	UnOpPos OpCode = iota
+	UnOpNeg
+	UnOpCpl
+	UnOpNot
+	UnOpVoid
+	UnOpTypeof
+	UnOpDelete
+
+	// Prefix update
+	UnOpPreDec
+	UnOpPreInc
+
+	// Postfix update
+	UnOpPostDec
+	UnOpPostInc
+
+	// Left-associative
+	BinOpAdd
+	BinOpSub
+	BinOpMul
+	BinOpDiv
+	BinOpRem
+	BinOpPow
+	BinOpLt
+	BinOpLe
+	BinOpGt
+	BinOpGe
+	BinOpIn
+	BinOpInstanceof
+	BinOpShl
+	BinOpShr
+	BinOpUShr
+	BinOpLooseEq
+	BinOpLooseNe
+	BinOpStrictEq
+	BinOpStrictNe
+	BinOpNullishCoalescing
+	BinOpLogicalOr
+	BinOpLogicalAnd
+	BinOpBitwiseOr
+	BinOpBitwiseAnd
+	BinOpBitwiseXor
+
+	// Non-associative
+	BinOpComma
+
+	// Right-associative
+	BinOpAssign
+	BinOpAddAssign
+	BinOpSubAssign
+	BinOpMulAssign
+	BinOpDivAssign
+	BinOpRemAssign
+	BinOpPowAssign
+	BinOpShlAssign
+	BinOpShrAssign
+	BinOpUShrAssign
+	BinOpBitwiseOrAssign
+	BinOpBitwiseAndAssign
+	BinOpBitwiseXorAssign
+	BinOpNullishCoalescingAssign
+	BinOpLogicalOrAssign
+	BinOpLogicalAndAssign
+)
+
+type OpTableEntry struct {
+	Text      string
+	Level     L
+	IsKeyword bool
+}
+
+var OpTable = []OpTableEntry{
+	// Prefix
+	{"+", LPrefix, false},
+	{"-", LPrefix, false},
+	{"~", LPrefix, false},
+	{"!", LPrefix, false},
+	{"void", LPrefix, true},
+	{"typeof", LPrefix, true},
+	{"delete", LPrefix, true},
+
+	// Prefix update
+	{"--", LPrefix, false},
+	{"++", LPrefix, false},
+
+	// Postfix update
+	{"--", LPostfix, false},
+	{"++", LPostfix, false},
+
+	// Left-associative
+	{"+", LAdd, false},
+	{"-", LAdd, false},
+	{"*", LMultiply, false},
+	{"/", LMultiply, false},
+	{"%", LMultiply, false},
+	{"**", LExponentiation, false}, // Right-associative
+	{"<", LCompare, false},
+	{"<=", LCompare, false},
+	{">", LCompare, false},
+	{">=", LCompare, false},
+	{"in", LCompare, true},
+	{"instanceof", LCompare, true},
+	{"<<", LShift, false},
+	{">>", LShift, false},
+	{">>>", LShift, false},
+	{"==", LEquals, false},
+	{"!=", LEquals, false},
+	{"===", LEquals, false},
+	{"!==", LEquals, false},
+	{"??", LNullishCoalescing, false},
+	{"||", LLogicalOr, false},
+	{"&&", LLogicalAnd, false},
+	{"|", LBitwiseOr, false},
+	{"&", LBitwiseAnd, false},
+	{"^", LBitwiseXor, false},
+
+	// Non-associative
+	{",", LComma, false},
+
+	// Right-associative
+	{"=", LAssign, false},
+	{"+=", LAssign, false},
+	{"-=", LAssign, false},
+	{"*=", LAssign, false},
+	{"/=", LAssign, false},
+	{"%=", LAssign, false},
+	{"**=", LAssign, false},
+	{"<<=", LAssign, false},
+	{">>=", LAssign, false},
+	{">>>=", LAssign, false},
+	{"|=", LAssign, false},
+	{"&=", LAssign, false},
+	{"^=", LAssign, false},
+	{"??=", LAssign, false},
+	{"||=", LAssign, false},
+	{"&&=", LAssign, false},
+}
+
+type Decorator struct {
+	Value            Expr
+	AtLoc            logger.Loc
+	OmitNewlineAfter bool
+}
+
+type PropertyKind uint8
+
+const (
+	PropertyField PropertyKind = iota
+	PropertyMethod
+	PropertyGetter
+	PropertySetter
+	PropertyAutoAccessor
+	PropertySpread
+	PropertyDeclareOrAbstract
+	PropertyClassStaticBlock
+)
+
+// This returns true if and only if this property matches the "MethodDefinition"
+// grammar from the specification. That means it's one of the following forms:
+//
+//	foo() {}
+//	*foo() {}
+//	async foo() {}
+//	async *foo() {}
+//	get foo() {}
+//	set foo(_) {}
+//
+// If this returns true, the "ValueOrNil" field of the property is always an
+// "EFunction" expression and it is always printed as a method.
+func (kind PropertyKind) IsMethodDefinition() bool {
+	return kind == PropertyMethod || kind == PropertyGetter || kind == PropertySetter
+}
+
+type ClassStaticBlock struct {
+	Block SBlock
+	Loc   logger.Loc
+}
+
+type PropertyFlags uint8
+
+const (
+	PropertyIsComputed PropertyFlags = 1 << iota
+	PropertyIsStatic
+	PropertyWasShorthand
+	PropertyPreferQuotedKey
+)
+
+func (flags PropertyFlags) Has(flag PropertyFlags) bool {
+	return (flags & flag) != 0
+}
+
+type Property struct {
+	ClassStaticBlock *ClassStaticBlock
+
+	Key Expr
+
+	// This is omitted for class fields
+	ValueOrNil Expr
+
+	// This is used when parsing a pattern that uses default values:
+	//
+	//   [a = 1] = [];
+	//   ({a = 1} = {});
+	//
+	// It's also used for class fields:
+	//
+	//   class Foo { a = 1 }
+	//
+	InitializerOrNil Expr
+
+	Decorators []Decorator
+
+	Loc             logger.Loc
+	CloseBracketLoc logger.Loc
+	Kind            PropertyKind
+	Flags           PropertyFlags
+}
+
+type PropertyBinding struct {
+	Key               Expr
+	Value             Binding
+	DefaultValueOrNil Expr
+	Loc               logger.Loc
+	CloseBracketLoc   logger.Loc
+	IsComputed        bool
+	IsSpread          bool
+	PreferQuotedKey   bool
+}
+
+type Arg struct {
+	Binding      Binding
+	DefaultOrNil Expr
+	Decorators   []Decorator
+
+	// "constructor(public x: boolean) {}"
+	IsTypeScriptCtorField bool
+}
+
+type Fn struct {
+	Name         *ast.LocRef
+	Args         []Arg
+	Body         FnBody
+	ArgumentsRef ast.Ref
+	OpenParenLoc logger.Loc
+
+	IsAsync     bool
+	IsGenerator bool
+	HasRestArg  bool
+	HasIfScope  bool
+
+	// See: https://github.com/rollup/rollup/pull/5024
+	HasNoSideEffectsComment bool
+
+	// This is true if the function is a method
+	IsUniqueFormalParameters bool
+}
+
+type FnBody struct {
+	Block SBlock
+	Loc   logger.Loc
+}
+
+type Class struct {
+	Decorators    []Decorator
+	Name          *ast.LocRef
+	ExtendsOrNil  Expr
+	Properties    []Property
+	ClassKeyword  logger.Range
+	BodyLoc       logger.Loc
+	CloseBraceLoc logger.Loc
+
+	// If true, JavaScript decorators (i.e. not TypeScript experimental
+	// decorators) should be lowered. This is the case either if JavaScript
+	// decorators are not supported in the configured target environment, or
+	// if "useDefineForClassFields" is set to false and this class has
+	// decorators on it. Note that this flag is not necessarily set to true if
+	// "useDefineForClassFields" is false and a class has an "accessor" even
+	// though the accessor feature comes from the decorator specification.
+	ShouldLowerStandardDecorators bool
+
+	// If true, property field initializers cannot be assumed to have no side
+	// effects. For example:
+	//
+	//   class Foo {
+	//     static set foo(x) { importantSideEffect(x) }
+	//   }
+	//   class Bar extends Foo {
+	//     foo = 1
+	//   }
+	//
+	// This happens in TypeScript when "useDefineForClassFields" is disabled
+	// because TypeScript (and esbuild) transforms the above class into this:
+	//
+	//   class Foo {
+	//     static set foo(x) { importantSideEffect(x); }
+	//   }
+	//   class Bar extends Foo {
+	//   }
+	//   Bar.foo = 1;
+	//
+	UseDefineForClassFields bool
+}
+
+type ArrayBinding struct {
+	Binding           Binding
+	DefaultValueOrNil Expr
+	Loc               logger.Loc
+}
+
+type Binding struct {
+	Data B
+	Loc  logger.Loc
+}
+
+// This interface is never called. Its purpose is to encode a variant type in
+// Go's type system.
+type B interface{ isBinding() }
+
+func (*BMissing) isBinding()    {}
+func (*BIdentifier) isBinding() {}
+func (*BArray) isBinding()      {}
+func (*BObject) isBinding()     {}
+
+type BMissing struct{}
+
+type BIdentifier struct{ Ref ast.Ref }
+
+type BArray struct {
+	Items           []ArrayBinding
+	CloseBracketLoc logger.Loc
+	HasSpread       bool
+	IsSingleLine    bool
+}
+
+type BObject struct {
+	Properties    []PropertyBinding
+	CloseBraceLoc logger.Loc
+	IsSingleLine  bool
+}
+
+type Expr struct {
+	Data E
+	Loc  logger.Loc
+}
+
+// This interface is never called. Its purpose is to encode a variant type in
+// Go's type system.
+type E interface{ isExpr() }
+
+func (*EArray) isExpr()                {}
+func (*EUnary) isExpr()                {}
+func (*EBinary) isExpr()               {}
+func (*EBoolean) isExpr()              {}
+func (*ESuper) isExpr()                {}
+func (*ENull) isExpr()                 {}
+func (*EUndefined) isExpr()            {}
+func (*EThis) isExpr()                 {}
+func (*ENew) isExpr()                  {}
+func (*ENewTarget) isExpr()            {}
+func (*EImportMeta) isExpr()           {}
+func (*ECall) isExpr()                 {}
+func (*EDot) isExpr()                  {}
+func (*EIndex) isExpr()                {}
+func (*EArrow) isExpr()                {}
+func (*EFunction) isExpr()             {}
+func (*EClass) isExpr()                {}
+func (*EIdentifier) isExpr()           {}
+func (*EImportIdentifier) isExpr()     {}
+func (*EPrivateIdentifier) isExpr()    {}
+func (*ENameOfSymbol) isExpr()         {}
+func (*EJSXElement) isExpr()           {}
+func (*EJSXText) isExpr()              {}
+func (*EMissing) isExpr()              {}
+func (*ENumber) isExpr()               {}
+func (*EBigInt) isExpr()               {}
+func (*EObject) isExpr()               {}
+func (*ESpread) isExpr()               {}
+func (*EString) isExpr()               {}
+func (*ETemplate) isExpr()             {}
+func (*ERegExp) isExpr()               {}
+func (*EInlinedEnum) isExpr()          {}
+func (*EAnnotation) isExpr()           {}
+func (*EAwait) isExpr()                {}
+func (*EYield) isExpr()                {}
+func (*EIf) isExpr()                   {}
+func (*ERequireString) isExpr()        {}
+func (*ERequireResolveString) isExpr() {}
+func (*EImportString) isExpr()         {}
+func (*EImportCall) isExpr()           {}
+
+type EArray struct {
+	Items            []Expr
+	CommaAfterSpread logger.Loc
+	CloseBracketLoc  logger.Loc
+	IsSingleLine     bool
+	IsParenthesized  bool
+}
+
+type EUnary struct {
+	Value Expr
+	Op    OpCode
+
+	// The expression "typeof (0, x)" must not become "typeof x" if "x"
+	// is unbound because that could suppress a ReferenceError from "x".
+	//
+	// Also if we know a typeof operator was originally an identifier, then
+	// we know that this typeof operator always has no side effects (even if
+	// we consider the identifier by itself to have a side effect).
+	//
+	// Note that there *is* actually a case where "typeof x" can throw an error:
+	// when "x" is being referenced inside of its TDZ (temporal dead zone). TDZ
+	// checks are not yet handled correctly by esbuild, so this possibility is
+	// currently ignored.
+	WasOriginallyTypeofIdentifier bool
+
+	// Similarly the expression "delete (0, x)" must not become "delete x"
+	// because that syntax is invalid in strict mode. We also need to make sure
+	// we don't accidentally change the return value:
+	//
+	//   Returns false:
+	//     "var a; delete (a)"
+	//     "var a = Object.freeze({b: 1}); delete (a.b)"
+	//     "var a = Object.freeze({b: 1}); delete (a?.b)"
+	//     "var a = Object.freeze({b: 1}); delete (a['b'])"
+	//     "var a = Object.freeze({b: 1}); delete (a?.['b'])"
+	//
+	//   Returns true:
+	//     "var a; delete (0, a)"
+	//     "var a = Object.freeze({b: 1}); delete (true && a.b)"
+	//     "var a = Object.freeze({b: 1}); delete (false || a?.b)"
+	//     "var a = Object.freeze({b: 1}); delete (null ?? a?.['b'])"
+	//     "var a = Object.freeze({b: 1}); delete (true ? a['b'] : a['b'])"
+	//
+	WasOriginallyDeleteOfIdentifierOrPropertyAccess bool
+}
+
+type EBinary struct {
+	Left  Expr
+	Right Expr
+	Op    OpCode
+}
+
+type EBoolean struct{ Value bool }
+
+type EMissing struct{}
+
+type ESuper struct{}
+
+type ENull struct{}
+
+type EUndefined struct{}
+
+type EThis struct{}
+
+type ENewTarget struct {
+	Range logger.Range
+}
+
+type EImportMeta struct {
+	RangeLen int32
+}
+
+// These help reduce unnecessary memory allocations
+var BMissingShared = &BMissing{}
+var EMissingShared = &EMissing{}
+var ENullShared = &ENull{}
+var ESuperShared = &ESuper{}
+var EThisShared = &EThis{}
+var EUndefinedShared = &EUndefined{}
+var SDebuggerShared = &SDebugger{}
+var SEmptyShared = &SEmpty{}
+var STypeScriptShared = &STypeScript{}
+var STypeScriptSharedWasDeclareClass = &STypeScript{WasDeclareClass: true}
+
+type ENew struct {
+	Target Expr
+	Args   []Expr
+
+	CloseParenLoc logger.Loc
+	IsMultiLine   bool
+
+	// True if there is a comment containing "@__PURE__" or "#__PURE__" preceding
+	// this call expression. See the comment inside ECall for more details.
+	CanBeUnwrappedIfUnused bool
+}
+
+type CallKind uint8
+
+const (
+	NormalCall CallKind = iota
+	DirectEval
+	TargetWasOriginallyPropertyAccess
+)
+
+type OptionalChain uint8
+
+const (
+	// "a.b"
+	OptionalChainNone OptionalChain = iota
+
+	// "a?.b"
+	OptionalChainStart
+
+	// "a?.b.c" => ".c" is OptionalChainContinue
+	// "(a?.b).c" => ".c" is OptionalChainNone
+	OptionalChainContinue
+)
+
+type ECall struct {
+	Target        Expr
+	Args          []Expr
+	CloseParenLoc logger.Loc
+	OptionalChain OptionalChain
+	Kind          CallKind
+	IsMultiLine   bool
+
+	// True if there is a comment containing "@__PURE__" or "#__PURE__" preceding
+	// this call expression. This is an annotation used for tree shaking, and
+	// means that the call can be removed if it's unused. It does not mean the
+	// call is pure (e.g. it may still return something different if called twice).
+	//
+	// Note that the arguments are not considered to be part of the call. If the
+	// call itself is removed due to this annotation, the arguments must remain
+	// if they have side effects.
+	CanBeUnwrappedIfUnused bool
+}
+
+func (a *ECall) HasSameFlagsAs(b *ECall) bool {
+	return a.OptionalChain == b.OptionalChain &&
+		a.Kind == b.Kind &&
+		a.CanBeUnwrappedIfUnused == b.CanBeUnwrappedIfUnused
+}
+
+type EDot struct {
+	Target        Expr
+	Name          string
+	NameLoc       logger.Loc
+	OptionalChain OptionalChain
+
+	// If true, this property access is known to be free of side-effects. That
+	// means it can be removed if the resulting value isn't used.
+	CanBeRemovedIfUnused bool
+
+	// If true, this property access is a function that, when called, can be
+	// unwrapped if the resulting value is unused. Unwrapping means discarding
+	// the call target but keeping any arguments with side effects.
+	CallCanBeUnwrappedIfUnused bool
+
+	// Symbol values are known to not have side effects when used as property
+	// names in class declarations and object literals.
+	IsSymbolInstance bool
+}
+
+func (a *EDot) HasSameFlagsAs(b *EDot) bool {
+	return a.OptionalChain == b.OptionalChain &&
+		a.CanBeRemovedIfUnused == b.CanBeRemovedIfUnused &&
+		a.CallCanBeUnwrappedIfUnused == b.CallCanBeUnwrappedIfUnused &&
+		a.IsSymbolInstance == b.IsSymbolInstance
+}
+
+type EIndex struct {
+	Target          Expr
+	Index           Expr
+	CloseBracketLoc logger.Loc
+	OptionalChain   OptionalChain
+
+	// If true, this property access is known to be free of side-effects. That
+	// means it can be removed if the resulting value isn't used.
+	CanBeRemovedIfUnused bool
+
+	// If true, this property access is a function that, when called, can be
+	// unwrapped if the resulting value is unused. Unwrapping means discarding
+	// the call target but keeping any arguments with side effects.
+	CallCanBeUnwrappedIfUnused bool
+
+	// Symbol values are known to not have side effects when used as property
+	// names in class declarations and object literals.
+	IsSymbolInstance bool
+}
+
+func (a *EIndex) HasSameFlagsAs(b *EIndex) bool {
+	return a.OptionalChain == b.OptionalChain &&
+		a.CanBeRemovedIfUnused == b.CanBeRemovedIfUnused &&
+		a.CallCanBeUnwrappedIfUnused == b.CallCanBeUnwrappedIfUnused &&
+		a.IsSymbolInstance == b.IsSymbolInstance
+}
+
+type EArrow struct {
+	Args []Arg
+	Body FnBody
+
+	IsAsync    bool
+	HasRestArg bool
+	PreferExpr bool // Use shorthand if true and "Body" is a single return statement
+
+	// See: https://github.com/rollup/rollup/pull/5024
+	HasNoSideEffectsComment bool
+}
+
+type EFunction struct{ Fn Fn }
+
+type EClass struct{ Class Class }
+
+type EIdentifier struct {
+	Ref ast.Ref
+
+	// If we're inside a "with" statement, this identifier may be a property
+	// access. In that case it would be incorrect to remove this identifier since
+	// the property access may be a getter or setter with side effects.
+	MustKeepDueToWithStmt bool
+
+	// If true, this identifier is known to not have a side effect (i.e. to not
+	// throw an exception) when referenced. If false, this identifier may or may
+	// not have side effects when referenced. This is used to allow the removal
+	// of known globals such as "Object" if they aren't used.
+	CanBeRemovedIfUnused bool
+
+	// If true, this identifier represents a function that, when called, can be
+	// unwrapped if the resulting value is unused. Unwrapping means discarding
+	// the call target but keeping any arguments with side effects.
+	CallCanBeUnwrappedIfUnused bool
+}
+
+// This is similar to an EIdentifier but it represents a reference to an ES6
+// import item.
+//
+// Depending on how the code is linked, the file containing this EImportIdentifier
+// may or may not be in the same module group as the file it was imported from.
+//
+// If it's the same module group than we can just merge the import item symbol
+// with the corresponding symbol that was imported, effectively renaming them
+// to be the same thing and statically binding them together.
+//
+// But if it's a different module group, then the import must be dynamically
+// evaluated using a property access off the corresponding namespace symbol,
+// which represents the result of a require() call.
+//
+// It's stored as a separate type so it's not easy to confuse with a plain
+// identifier. For example, it'd be bad if code trying to convert "{x: x}" into
+// "{x}" shorthand syntax wasn't aware that the "x" in this case is actually
+// "{x: importedNamespace.x}". This separate type forces code to opt-in to
+// doing this instead of opt-out.
+type EImportIdentifier struct {
+	Ref             ast.Ref
+	PreferQuotedKey bool
+
+	// If true, this was originally an identifier expression such as "foo". If
+	// false, this could potentially have been a member access expression such
+	// as "ns.foo" off of an imported namespace object.
+	WasOriginallyIdentifier bool
+}
+
+// This is similar to EIdentifier but it represents class-private fields and
+// methods. It can be used where computed properties can be used, such as
+// EIndex and Property.
+type EPrivateIdentifier struct {
+	Ref ast.Ref
+}
+
+// This represents an internal property name that can be mangled. The symbol
+// referenced by this expression should be a "SymbolMangledProp" symbol.
+type ENameOfSymbol struct {
+	Ref                   ast.Ref
+	HasPropertyKeyComment bool // If true, a preceding comment contains "@__KEY__"
+}
+
+type EJSXElement struct {
+	TagOrNil   Expr
+	Properties []Property
+
+	// Note: This array may contain nil entries. Be careful about nil entries
+	// when iterating over this array.
+	//
+	// Each nil entry corresponds to the "JSXChildExpression_opt" part of the
+	// grammar (https://facebook.github.io/jsx/#prod-JSXChild):
+	//
+	//   JSXChild :
+	//       JSXText
+	//       JSXElement
+	//       JSXFragment
+	//       { JSXChildExpression_opt }
+	//
+	// This is the "{}" part in "<a>{}</a>". We allow this because some people
+	// put comments there and then expect to be able to process them from
+	// esbuild's output. These absent AST nodes are completely omitted when
+	// JSX is transformed to JS. They are only present when JSX preservation is
+	// enabled.
+	NullableChildren []Expr
+
+	CloseLoc        logger.Loc
+	IsTagSingleLine bool
+}
+
+// The JSX specification doesn't say how JSX text is supposed to be interpreted
+// so our "preserve" JSX transform should reproduce the original source code
+// verbatim. One reason why this matters is because there is no canonical way
+// to interpret JSX text (Babel and TypeScript differ in what newlines mean).
+// Another reason is that some people want to do custom things such as this:
+// https://github.com/evanw/esbuild/issues/3605
+type EJSXText struct {
+	Raw string
+}
+
+type ENumber struct{ Value float64 }
+
+type EBigInt struct{ Value string }
+
+type EObject struct {
+	Properties       []Property
+	CommaAfterSpread logger.Loc
+	CloseBraceLoc    logger.Loc
+	IsSingleLine     bool
+	IsParenthesized  bool
+}
+
+type ESpread struct{ Value Expr }
+
+// This is used for both strings and no-substitution template literals to reduce
+// the number of cases that need to be checked for string optimization code
+type EString struct {
+	Value                 []uint16
+	LegacyOctalLoc        logger.Loc
+	PreferTemplate        bool
+	HasPropertyKeyComment bool // If true, a preceding comment contains "@__KEY__"
+	ContainsUniqueKey     bool // If true, this string must not be wrapped
+}
+
+type TemplatePart struct {
+	Value      Expr
+	TailRaw    string   // Only use when "TagOrNil" is not nil
+	TailCooked []uint16 // Only use when "TagOrNil" is nil
+	TailLoc    logger.Loc
+}
+
+type ETemplate struct {
+	TagOrNil       Expr
+	HeadRaw        string   // Only use when "TagOrNil" is not nil
+	HeadCooked     []uint16 // Only use when "TagOrNil" is nil
+	Parts          []TemplatePart
+	HeadLoc        logger.Loc
+	LegacyOctalLoc logger.Loc
+
+	// True if this is a tagged template literal with a comment that indicates
+	// this function call can be removed if the result is unused. Note that the
+	// arguments are not considered to be part of the call. If the call itself
+	// is removed due to this annotation, the arguments must remain if they have
+	// side effects (including the string conversions).
+	CanBeUnwrappedIfUnused bool
+
+	// If the tag is present, it is expected to be a function and is called. If
+	// the tag is a syntactic property access, then the value for "this" in the
+	// function call is the object whose property was accessed (e.g. in "a.b``"
+	// the value for "this" in "a.b" is "a"). We need to ensure that if "a``"
+	// ever becomes "b.c``" later on due to optimizations, it is written as
+	// "(0, b.c)``" to avoid a behavior change.
+	TagWasOriginallyPropertyAccess bool
+}
+
+type ERegExp struct{ Value string }
+
+type EInlinedEnum struct {
+	Value   Expr
+	Comment string
+}
+
+type AnnotationFlags uint8
+
+const (
+	// This is sort of like an IIFE with a "/* @__PURE__ */" comment except it's an
+	// inline annotation on an expression itself without the nested scope. Sometimes
+	// we can't easily introduce a new scope (e.g. if the expression uses "await").
+	CanBeRemovedIfUnusedFlag AnnotationFlags = 1 << iota
+)
+
+func (flags AnnotationFlags) Has(flag AnnotationFlags) bool {
+	return (flags & flag) != 0
+}
+
+type EAnnotation struct {
+	Value Expr
+	Flags AnnotationFlags
+}
+
+type EAwait struct {
+	Value Expr
+}
+
+type EYield struct {
+	ValueOrNil Expr
+	IsStar     bool
+}
+
+type EIf struct {
+	Test Expr
+	Yes  Expr
+	No   Expr
+}
+
+type ERequireString struct {
+	ImportRecordIndex uint32
+	CloseParenLoc     logger.Loc
+}
+
+type ERequireResolveString struct {
+	ImportRecordIndex uint32
+	CloseParenLoc     logger.Loc
+}
+
+type EImportString struct {
+	ImportRecordIndex uint32
+	CloseParenLoc     logger.Loc
+}
+
+type EImportCall struct {
+	Expr          Expr
+	OptionsOrNil  Expr
+	CloseParenLoc logger.Loc
+}
+
+type Stmt struct {
+	Data S
+	Loc  logger.Loc
+}
+
+// This interface is never called. Its purpose is to encode a variant type in
+// Go's type system.
+type S interface{ isStmt() }
+
+func (*SBlock) isStmt()         {}
+func (*SComment) isStmt()       {}
+func (*SDebugger) isStmt()      {}
+func (*SDirective) isStmt()     {}
+func (*SEmpty) isStmt()         {}
+func (*STypeScript) isStmt()    {}
+func (*SExportClause) isStmt()  {}
+func (*SExportFrom) isStmt()    {}
+func (*SExportDefault) isStmt() {}
+func (*SExportStar) isStmt()    {}
+func (*SExportEquals) isStmt()  {}
+func (*SLazyExport) isStmt()    {}
+func (*SExpr) isStmt()          {}
+func (*SEnum) isStmt()          {}
+func (*SNamespace) isStmt()     {}
+func (*SFunction) isStmt()      {}
+func (*SClass) isStmt()         {}
+func (*SLabel) isStmt()         {}
+func (*SIf) isStmt()            {}
+func (*SFor) isStmt()           {}
+func (*SForIn) isStmt()         {}
+func (*SForOf) isStmt()         {}
+func (*SDoWhile) isStmt()       {}
+func (*SWhile) isStmt()         {}
+func (*SWith) isStmt()          {}
+func (*STry) isStmt()           {}
+func (*SSwitch) isStmt()        {}
+func (*SImport) isStmt()        {}
+func (*SReturn) isStmt()        {}
+func (*SThrow) isStmt()         {}
+func (*SLocal) isStmt()         {}
+func (*SBreak) isStmt()         {}
+func (*SContinue) isStmt()      {}
+
+type SBlock struct {
+	Stmts         []Stmt
+	CloseBraceLoc logger.Loc
+}
+
+type SEmpty struct{}
+
+// This is a stand-in for a TypeScript type declaration
+type STypeScript struct {
+	WasDeclareClass bool
+}
+
+type SComment struct {
+	Text           string
+	IsLegalComment bool
+}
+
+type SDebugger struct{}
+
+type SDirective struct {
+	Value          []uint16
+	LegacyOctalLoc logger.Loc
+}
+
+type SExportClause struct {
+	Items        []ClauseItem
+	IsSingleLine bool
+}
+
+type SExportFrom struct {
+	Items             []ClauseItem
+	NamespaceRef      ast.Ref
+	ImportRecordIndex uint32
+	IsSingleLine      bool
+}
+
+type SExportDefault struct {
+	Value       Stmt // May be a SExpr or SFunction or SClass
+	DefaultName ast.LocRef
+}
+
+type ExportStarAlias struct {
+	// Although this alias name starts off as being the same as the statement's
+	// namespace symbol, it may diverge if the namespace symbol name is minified.
+	// The original alias name is preserved here to avoid this scenario.
+	OriginalName string
+
+	Loc logger.Loc
+}
+
+type SExportStar struct {
+	Alias             *ExportStarAlias
+	NamespaceRef      ast.Ref
+	ImportRecordIndex uint32
+}
+
+// This is an "export = value;" statement in TypeScript
+type SExportEquals struct {
+	Value Expr
+}
+
+// The decision of whether to export an expression using "module.exports" or
+// "export default" is deferred until linking using this statement kind
+type SLazyExport struct {
+	Value Expr
+}
+
+type SExpr struct {
+	Value Expr
+
+	// This is set to true for automatically-generated expressions that are part
+	// of class syntax lowering. A single class declaration may end up with many
+	// generated expressions after it (e.g. class field initializations, a call
+	// to keep the original value of the "name" property). When this happens we
+	// can't tell that the class is side-effect free anymore because all of these
+	// methods mutate the class. We use this annotation for that instead.
+	IsFromClassOrFnThatCanBeRemovedIfUnused bool
+}
+
+type EnumValue struct {
+	ValueOrNil Expr
+	Name       []uint16
+	Ref        ast.Ref
+	Loc        logger.Loc
+}
+
+type SEnum struct {
+	Values   []EnumValue
+	Name     ast.LocRef
+	Arg      ast.Ref
+	IsExport bool
+}
+
+type SNamespace struct {
+	Stmts    []Stmt
+	Name     ast.LocRef
+	Arg      ast.Ref
+	IsExport bool
+}
+
+type SFunction struct {
+	Fn       Fn
+	IsExport bool
+}
+
+type SClass struct {
+	Class    Class
+	IsExport bool
+}
+
+type SLabel struct {
+	Stmt             Stmt
+	Name             ast.LocRef
+	IsSingleLineStmt bool
+}
+
+type SIf struct {
+	Test            Expr
+	Yes             Stmt
+	NoOrNil         Stmt
+	IsSingleLineYes bool
+	IsSingleLineNo  bool
+}
+
+type SFor struct {
+	InitOrNil        Stmt // May be a SConst, SLet, SVar, or SExpr
+	TestOrNil        Expr
+	UpdateOrNil      Expr
+	Body             Stmt
+	IsSingleLineBody bool
+}
+
+type SForIn struct {
+	Init             Stmt // May be a SConst, SLet, SVar, or SExpr
+	Value            Expr
+	Body             Stmt
+	IsSingleLineBody bool
+}
+
+type SForOf struct {
+	Init             Stmt // May be a SConst, SLet, SVar, or SExpr
+	Value            Expr
+	Body             Stmt
+	Await            logger.Range
+	IsSingleLineBody bool
+}
+
+type SDoWhile struct {
+	Body Stmt
+	Test Expr
+}
+
+type SWhile struct {
+	Test             Expr
+	Body             Stmt
+	IsSingleLineBody bool
+}
+
+type SWith struct {
+	Value            Expr
+	Body             Stmt
+	BodyLoc          logger.Loc
+	IsSingleLineBody bool
+}
+
+type Catch struct {
+	BindingOrNil Binding
+	Block        SBlock
+	Loc          logger.Loc
+	BlockLoc     logger.Loc
+}
+
+type Finally struct {
+	Block SBlock
+	Loc   logger.Loc
+}
+
+type STry struct {
+	Catch    *Catch
+	Finally  *Finally
+	Block    SBlock
+	BlockLoc logger.Loc
+}
+
+type Case struct {
+	ValueOrNil Expr // If this is nil, this is "default" instead of "case"
+	Body       []Stmt
+	Loc        logger.Loc
+}
+
+type SSwitch struct {
+	Test          Expr
+	Cases         []Case
+	BodyLoc       logger.Loc
+	CloseBraceLoc logger.Loc
+}
+
+// This object represents all of these types of import statements:
+//
+//	import 'path'
+//	import {item1, item2} from 'path'
+//	import * as ns from 'path'
+//	import defaultItem, {item1, item2} from 'path'
+//	import defaultItem, * as ns from 'path'
+//
+// Many parts are optional and can be combined in different ways. The only
+// restriction is that you cannot have both a clause and a star namespace.
+type SImport struct {
+	DefaultName *ast.LocRef
+	Items       *[]ClauseItem
+	StarNameLoc *logger.Loc
+
+	// If this is a star import: This is a Ref for the namespace symbol. The Loc
+	// for the symbol is StarLoc.
+	//
+	// Otherwise: This is an auto-generated Ref for the namespace representing
+	// the imported file. In this case StarLoc is nil. The NamespaceRef is used
+	// when converting this module to a CommonJS module.
+	NamespaceRef ast.Ref
+
+	ImportRecordIndex uint32
+	IsSingleLine      bool
+}
+
+type SReturn struct {
+	ValueOrNil Expr
+}
+
+type SThrow struct {
+	Value Expr
+}
+
+type LocalKind uint8
+
+const (
+	LocalVar LocalKind = iota
+	LocalLet
+	LocalConst
+	LocalUsing
+	LocalAwaitUsing
+)
+
+func (kind LocalKind) IsUsing() bool {
+	return kind >= LocalUsing
+}
+
+type SLocal struct {
+	Decls    []Decl
+	Kind     LocalKind
+	IsExport bool
+
+	// The TypeScript compiler doesn't generate code for "import foo = bar"
+	// statements where the import is never used.
+	WasTSImportEquals bool
+}
+
+type SBreak struct {
+	Label *ast.LocRef
+}
+
+type SContinue struct {
+	Label *ast.LocRef
+}
+
+type ClauseItem struct {
+	Alias string
+
+	// This is the original name of the symbol stored in "Name". It's needed for
+	// "SExportClause" statements such as this:
+	//
+	//   export {foo as bar} from 'path'
+	//
+	// In this case both "foo" and "bar" are aliases because it's a re-export.
+	// We need to preserve both aliases in case the symbol is renamed. In this
+	// example, "foo" is "OriginalName" and "bar" is "Alias".
+	OriginalName string
+
+	AliasLoc logger.Loc
+	Name     ast.LocRef
+}
+
+type Decl struct {
+	Binding    Binding
+	ValueOrNil Expr
+}
+
+type ScopeKind uint8
+
+const (
+	ScopeBlock ScopeKind = iota
+	ScopeWith
+	ScopeLabel
+	ScopeClassName
+	ScopeClassBody
+	ScopeCatchBinding
+
+	// The scopes below stop hoisted variables from extending into parent scopes
+	ScopeEntry // This is a module, TypeScript enum, or TypeScript namespace
+	ScopeFunctionArgs
+	ScopeFunctionBody
+	ScopeClassStaticInit
+)
+
+func (kind ScopeKind) StopsHoisting() bool {
+	return kind >= ScopeEntry
+}
+
+type ScopeMember struct {
+	Ref ast.Ref
+	Loc logger.Loc
+}
+
+type Scope struct {
+	// This will be non-nil if this is a TypeScript "namespace" or "enum"
+	TSNamespace *TSNamespaceScope
+
+	Parent    *Scope
+	Children  []*Scope
+	Members   map[string]ScopeMember
+	Replaced  []ScopeMember
+	Generated []ast.Ref
+
+	// The location of the "use strict" directive for ExplicitStrictMode
+	UseStrictLoc logger.Loc
+
+	// This is used to store the ref of the label symbol for ScopeLabel scopes.
+	Label           ast.LocRef
+	LabelStmtIsLoop bool
+
+	// If a scope contains a direct eval() expression, then none of the symbols
+	// inside that scope can be renamed. We conservatively assume that the
+	// evaluated code might reference anything that it has access to.
+	ContainsDirectEval bool
+
+	// This is to help forbid "arguments" inside class body scopes
+	ForbidArguments bool
+
+	// As a special case, we enable constant propagation for any chain of "const"
+	// declarations at the start of a statement list. This special case doesn't
+	// have any TDZ considerations because no other statements come before it.
+	IsAfterConstLocalPrefix bool
+
+	StrictMode StrictModeKind
+	Kind       ScopeKind
+}
+
+type StrictModeKind uint8
+
+const (
+	SloppyMode StrictModeKind = iota
+	ExplicitStrictMode
+	ImplicitStrictModeClass
+	ImplicitStrictModeESM
+	ImplicitStrictModeTSAlwaysStrict
+	ImplicitStrictModeJSXAutomaticRuntime
+)
+
+func (s *Scope) RecursiveSetStrictMode(kind StrictModeKind) {
+	if s.StrictMode == SloppyMode {
+		s.StrictMode = kind
+		for _, child := range s.Children {
+			child.RecursiveSetStrictMode(kind)
+		}
+	}
+}
+
+// This is for TypeScript "enum" and "namespace" blocks. Each block can
+// potentially be instantiated multiple times. The exported members of each
+// block are merged into a single namespace while the non-exported code is
+// still scoped to just within that block:
+//
+//	let x = 1;
+//	namespace Foo {
+//	  let x = 2;
+//	  export let y = 3;
+//	}
+//	namespace Foo {
+//	  console.log(x); // 1
+//	  console.log(y); // 3
+//	}
+//
+// Doing this also works inside an enum:
+//
+//	enum Foo {
+//	  A = 3,
+//	  B = A + 1,
+//	}
+//	enum Foo {
+//	  C = A + 2,
+//	}
+//	console.log(Foo.B) // 4
+//	console.log(Foo.C) // 5
+//
+// This is a form of identifier lookup that works differently than the
+// hierarchical scope-based identifier lookup in JavaScript. Lookup now needs
+// to search sibling scopes in addition to parent scopes. This is accomplished
+// by sharing the map of exported members between all matching sibling scopes.
+type TSNamespaceScope struct {
+	// This is shared between all sibling namespace blocks
+	ExportedMembers TSNamespaceMembers
+
+	// This is a lazily-generated map of identifiers that actually represent
+	// property accesses to this namespace's properties. For example:
+	//
+	//   namespace x {
+	//     export let y = 123
+	//   }
+	//   namespace x {
+	//     export let z = y
+	//   }
+	//
+	// This should be compiled into the following code:
+	//
+	//   var x;
+	//   (function(x2) {
+	//     x2.y = 123;
+	//   })(x || (x = {}));
+	//   (function(x3) {
+	//     x3.z = x3.y;
+	//   })(x || (x = {}));
+	//
+	// When we try to find the symbol "y", we instead return one of these lazily
+	// generated proxy symbols that represent the property access "x3.y". This
+	// map is unique per namespace block because "x3" is the argument symbol that
+	// is specific to that particular namespace block.
+	LazilyGeneratedProperyAccesses map[string]ast.Ref
+
+	// This is specific to this namespace block. It's the argument of the
+	// immediately-invoked function expression that the namespace block is
+	// compiled into:
+	//
+	//   var ns;
+	//   (function (ns2) {
+	//     ns2.x = 123;
+	//   })(ns || (ns = {}));
+	//
+	// This variable is "ns2" in the above example. It's the symbol to use when
+	// generating property accesses off of this namespace when it's in scope.
+	ArgRef ast.Ref
+
+	// Even though enums are like namespaces and both enums and namespaces allow
+	// implicit references to properties of sibling scopes, they behave like
+	// separate, er, namespaces. Implicit references only work namespace-to-
+	// namespace and enum-to-enum. They do not work enum-to-namespace. And I'm
+	// not sure what's supposed to happen for the namespace-to-enum case because
+	// the compiler crashes: https://github.com/microsoft/TypeScript/issues/46891.
+	// So basically these both work:
+	//
+	//   enum a { b = 1 }
+	//   enum a { c = b }
+	//
+	//   namespace x { export let y = 1 }
+	//   namespace x { export let z = y }
+	//
+	// This doesn't work:
+	//
+	//   enum a { b = 1 }
+	//   namespace a { export let c = b }
+	//
+	// And this crashes the TypeScript compiler:
+	//
+	//   namespace a { export let b = 1 }
+	//   enum a { c = b }
+	//
+	// Therefore we only allow enum/enum and namespace/namespace interactions.
+	IsEnumScope bool
+}
+
+type TSNamespaceMembers map[string]TSNamespaceMember
+
+type TSNamespaceMember struct {
+	Data        TSNamespaceMemberData
+	Loc         logger.Loc
+	IsEnumValue bool
+}
+
+type TSNamespaceMemberData interface {
+	isTSNamespaceMember()
+}
+
+func (TSNamespaceMemberProperty) isTSNamespaceMember()   {}
+func (TSNamespaceMemberNamespace) isTSNamespaceMember()  {}
+func (TSNamespaceMemberEnumNumber) isTSNamespaceMember() {}
+func (TSNamespaceMemberEnumString) isTSNamespaceMember() {}
+
+// "namespace ns { export let it }"
+type TSNamespaceMemberProperty struct{}
+
+// "namespace ns { export namespace it {} }"
+type TSNamespaceMemberNamespace struct {
+	ExportedMembers TSNamespaceMembers
+}
+
+// "enum ns { it }"
+type TSNamespaceMemberEnumNumber struct {
+	Value float64
+}
+
+// "enum ns { it = 'it' }"
+type TSNamespaceMemberEnumString struct {
+	Value []uint16
+}
+
+type ExportsKind uint8
+
+const (
+	// This file doesn't have any kind of export, so it's impossible to say what
+	// kind of file this is. An empty file is in this category, for example.
+	ExportsNone ExportsKind = iota
+
+	// The exports are stored on "module" and/or "exports". Calling "require()"
+	// on this module returns "module.exports". All imports to this module are
+	// allowed but may return undefined.
+	ExportsCommonJS
+
+	// All export names are known explicitly. Calling "require()" on this module
+	// generates an exports object (stored in "exports") with getters for the
+	// export names. Named imports to this module are only allowed if they are
+	// in the set of export names.
+	ExportsESM
+
+	// Some export names are known explicitly, but others fall back to a dynamic
+	// run-time object. This is necessary when using the "export * from" syntax
+	// with either a CommonJS module or an external module (i.e. a module whose
+	// export names are not known at compile-time).
+	//
+	// Calling "require()" on this module generates an exports object (stored in
+	// "exports") with getters for the export names. All named imports to this
+	// module are allowed. Direct named imports reference the corresponding export
+	// directly. Other imports go through property accesses on "exports".
+	ExportsESMWithDynamicFallback
+)
+
+func (kind ExportsKind) IsDynamic() bool {
+	return kind == ExportsCommonJS || kind == ExportsESMWithDynamicFallback
+}
+
+type ModuleType uint8
+
+const (
+	ModuleUnknown ModuleType = iota
+
+	// ".cjs" or ".cts" or "type: commonjs" in package.json
+	ModuleCommonJS_CJS
+	ModuleCommonJS_CTS
+	ModuleCommonJS_PackageJSON
+
+	// ".mjs" or ".mts" or "type: module" in package.json
+	ModuleESM_MJS
+	ModuleESM_MTS
+	ModuleESM_PackageJSON
+)
+
+func (mt ModuleType) IsCommonJS() bool {
+	return mt >= ModuleCommonJS_CJS && mt <= ModuleCommonJS_PackageJSON
+}
+
+func (mt ModuleType) IsESM() bool {
+	return mt >= ModuleESM_MJS && mt <= ModuleESM_PackageJSON
+}
+
+type ModuleTypeData struct {
+	Source *logger.Source
+	Range  logger.Range
+	Type   ModuleType
+}
+
+// This is the index to the automatically-generated part containing code that
+// calls "__export(exports, { ... getters ... })". This is used to generate
+// getters on an exports object for ES6 export statements, and is both for
+// ES6 star imports and CommonJS-style modules. All files have one of these,
+// although it may contain no statements if there is nothing to export.
+const NSExportPartIndex = uint32(0)
+
+type AST struct {
+	ModuleTypeData ModuleTypeData
+	Parts          []Part
+	Symbols        []ast.Symbol
+	ExprComments   map[logger.Loc][]string
+	ModuleScope    *Scope
+	CharFreq       *ast.CharFreq
+
+	// This is internal-only data used for the implementation of Yarn PnP
+	ManifestForYarnPnP Expr
+
+	Hashbang   string
+	Directives []string
+	URLForCSS  string
+
+	// Note: If you're in the linker, do not use this map directly. This map is
+	// filled in by the parser and is considered immutable. For performance reasons,
+	// the linker doesn't mutate this map (cloning a map is slow in Go). Instead the
+	// linker super-imposes relevant information on top in a method call. You should
+	// call "TopLevelSymbolToParts" instead.
+	TopLevelSymbolToPartsFromParser map[ast.Ref][]uint32
+
+	// This contains all top-level exported TypeScript enum constants. It exists
+	// to enable cross-module inlining of constant enums.
+	TSEnums map[ast.Ref]map[string]TSEnumValue
+
+	// This contains the values of all detected inlinable constants. It exists
+	// to enable cross-module inlining of these constants.
+	ConstValues map[ast.Ref]ConstValue
+
+	// Properties in here are represented as symbols instead of strings, which
+	// allows them to be renamed to smaller names.
+	MangledProps map[string]ast.Ref
+
+	// Properties in here are existing non-mangled properties in the source code
+	// and must not be used when generating mangled names to avoid a collision.
+	ReservedProps map[string]bool
+
+	// These are stored at the AST level instead of on individual AST nodes so
+	// they can be manipulated efficiently without a full AST traversal
+	ImportRecords []ast.ImportRecord
+
+	// These are used when bundling. They are filled in during the parser pass
+	// since we already have to traverse the AST then anyway and the parser pass
+	// is conveniently fully parallelized.
+	NamedImports            map[ast.Ref]NamedImport
+	NamedExports            map[string]NamedExport
+	ExportStarImportRecords []uint32
+
+	SourceMapComment logger.Span
+
+	// This is a list of ES6 features. They are ranges instead of booleans so
+	// that they can be used in log messages. Check to see if "Len > 0".
+	ExportKeyword            logger.Range // Does not include TypeScript-specific syntax
+	TopLevelAwaitKeyword     logger.Range
+	LiveTopLevelAwaitKeyword logger.Range // Excludes top-level await in dead branches
+
+	ExportsRef ast.Ref
+	ModuleRef  ast.Ref
+	WrapperRef ast.Ref
+
+	ApproximateLineCount  int32
+	NestedScopeSlotCounts ast.SlotCounts
+	HasLazyExport         bool
+
+	// This is a list of CommonJS features. When a file uses CommonJS features,
+	// it's not a candidate for "flat bundling" and must be wrapped in its own
+	// closure. Note that this also includes top-level "return" but these aren't
+	// here because only the parser checks those.
+	UsesExportsRef bool
+	UsesModuleRef  bool
+	ExportsKind    ExportsKind
+}
+
+type TSEnumValue struct {
+	String []uint16 // Use this if it's not nil
+	Number float64  // Use this if "String" is nil
+}
+
+type ConstValueKind uint8
+
+const (
+	ConstValueNone ConstValueKind = iota
+	ConstValueNull
+	ConstValueUndefined
+	ConstValueTrue
+	ConstValueFalse
+	ConstValueNumber
+)
+
+type ConstValue struct {
+	Number float64 // Use this for "ConstValueNumber"
+	Kind   ConstValueKind
+}
+
+func ExprToConstValue(expr Expr) ConstValue {
+	switch v := expr.Data.(type) {
+	case *ENull:
+		return ConstValue{Kind: ConstValueNull}
+
+	case *EUndefined:
+		return ConstValue{Kind: ConstValueUndefined}
+
+	case *EBoolean:
+		if v.Value {
+			return ConstValue{Kind: ConstValueTrue}
+		} else {
+			return ConstValue{Kind: ConstValueFalse}
+		}
+
+	case *ENumber:
+		// Inline integers and other small numbers. Don't inline large
+		// real numbers because people may not want them to be inlined
+		// as it will increase the minified code size by too much.
+		if asInt := int64(v.Value); v.Value == float64(asInt) || len(strconv.FormatFloat(v.Value, 'g', -1, 64)) <= 8 {
+			return ConstValue{Kind: ConstValueNumber, Number: v.Value}
+		}
+
+	case *EString:
+		// I'm deliberately not inlining strings here. It seems more likely that
+		// people won't want them to be inlined since they can be arbitrarily long.
+
+	case *EBigInt:
+		// I'm deliberately not inlining bigints here for the same reason (they can
+		// be arbitrarily long).
+	}
+
+	return ConstValue{}
+}
+
+func ConstValueToExpr(loc logger.Loc, value ConstValue) Expr {
+	switch value.Kind {
+	case ConstValueNull:
+		return Expr{Loc: loc, Data: ENullShared}
+
+	case ConstValueUndefined:
+		return Expr{Loc: loc, Data: EUndefinedShared}
+
+	case ConstValueTrue:
+		return Expr{Loc: loc, Data: &EBoolean{Value: true}}
+
+	case ConstValueFalse:
+		return Expr{Loc: loc, Data: &EBoolean{Value: false}}
+
+	case ConstValueNumber:
+		return Expr{Loc: loc, Data: &ENumber{Value: value.Number}}
+	}
+
+	panic("Internal error: invalid constant value")
+}
+
+type NamedImport struct {
+	Alias string
+
+	// Parts within this file that use this import
+	LocalPartsWithUses []uint32
+
+	AliasLoc          logger.Loc
+	NamespaceRef      ast.Ref
+	ImportRecordIndex uint32
+
+	// If true, the alias refers to the entire export namespace object of a
+	// module. This is no longer represented as an alias called "*" because of
+	// the upcoming "Arbitrary module namespace identifier names" feature:
+	// https://github.com/tc39/ecma262/pull/2154
+	AliasIsStar bool
+
+	// It's useful to flag exported imports because if they are in a TypeScript
+	// file, we can't tell if they are a type or a value.
+	IsExported bool
+}
+
+type NamedExport struct {
+	Ref      ast.Ref
+	AliasLoc logger.Loc
+}
+
+// Each file is made up of multiple parts, and each part consists of one or
+// more top-level statements. Parts are used for tree shaking and code
+// splitting analysis. Individual parts of a file can be discarded by tree
+// shaking and can be assigned to separate chunks (i.e. output files) by code
+// splitting.
+type Part struct {
+	Stmts  []Stmt
+	Scopes []*Scope
+
+	// Each is an index into the file-level import record list
+	ImportRecordIndices []uint32
+
+	// All symbols that are declared in this part. Note that a given symbol may
+	// have multiple declarations, and so may end up being declared in multiple
+	// parts (e.g. multiple "var" declarations with the same name). Also note
+	// that this list isn't deduplicated and may contain duplicates.
+	DeclaredSymbols []DeclaredSymbol
+
+	// An estimate of the number of uses of all symbols used within this part.
+	SymbolUses map[ast.Ref]SymbolUse
+
+	// An estimate of the number of uses of all symbols used as the target of
+	// function calls within this part.
+	SymbolCallUses map[ast.Ref]SymbolCallUse
+
+	// This tracks property accesses off of imported symbols. We don't know
+	// during parsing if an imported symbol is going to be an inlined enum
+	// value or not. This is only known during linking. So we defer adding
+	// a dependency on these imported symbols until we know whether the
+	// property access is an inlined enum value or not.
+	ImportSymbolPropertyUses map[ast.Ref]map[string]SymbolUse
+
+	// The indices of the other parts in this file that are needed if this part
+	// is needed.
+	Dependencies []Dependency
+
+	// If true, this part can be removed if none of the declared symbols are
+	// used. If the file containing this part is imported, then all parts that
+	// don't have this flag enabled must be included.
+	CanBeRemovedIfUnused bool
+
+	// This is used for generated parts that we don't want to be present if they
+	// aren't needed. This enables tree shaking for these parts even if global
+	// tree shaking isn't enabled.
+	ForceTreeShaking bool
+
+	// This is true if this file has been marked as live by the tree shaking
+	// algorithm.
+	IsLive bool
+}
+
+type Dependency struct {
+	SourceIndex uint32
+	PartIndex   uint32
+}
+
+type DeclaredSymbol struct {
+	Ref        ast.Ref
+	IsTopLevel bool
+}
+
+type SymbolUse struct {
+	CountEstimate uint32
+}
+
+type SymbolCallUse struct {
+	CallCountEstimate                   uint32
+	SingleArgNonSpreadCallCountEstimate uint32
+}
+
+// For readability, the names of certain automatically-generated symbols are
+// derived from the file name. For example, instead of the CommonJS wrapper for
+// a file being called something like "require273" it can be called something
+// like "require_react" instead. This function generates the part of these
+// identifiers that's specific to the file path. It can take both an absolute
+// path (OS-specific) and a path in the source code (OS-independent).
+//
+// Note that these generated names do not at all relate to the correctness of
+// the code as far as avoiding symbol name collisions. These names still go
+// through the renaming logic that all other symbols go through to avoid name
+// collisions.
+func GenerateNonUniqueNameFromPath(path string) string {
+	// Get the file name without the extension
+	dir, base, _ := logger.PlatformIndependentPathDirBaseExt(path)
+
+	// If the name is "index", use the directory name instead. This is because
+	// many packages in npm use the file name "index.js" because it triggers
+	// node's implicit module resolution rules that allows you to import it by
+	// just naming the directory.
+	if base == "index" {
+		_, dirBase, _ := logger.PlatformIndependentPathDirBaseExt(dir)
+		if dirBase != "" {
+			base = dirBase
+		}
+	}
+
+	return EnsureValidIdentifier(base)
+}
+
+func EnsureValidIdentifier(base string) string {
+	// Convert it to an ASCII identifier. Note: If you change this to a non-ASCII
+	// identifier, you're going to potentially cause trouble with non-BMP code
+	// points in target environments that don't support bracketed Unicode escapes.
+	bytes := []byte{}
+	needsGap := false
+	for _, c := range base {
+		if (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || (len(bytes) > 0 && c >= '0' && c <= '9') {
+			if needsGap {
+				bytes = append(bytes, '_')
+				needsGap = false
+			}
+			bytes = append(bytes, byte(c))
+		} else if len(bytes) > 0 {
+			needsGap = true
+		}
+	}
+
+	// Make sure the name isn't empty
+	if len(bytes) == 0 {
+		return "_"
+	}
+	return string(bytes)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/js_ast/js_ast_helpers.go b/source/vendor/github.com/evanw/esbuild/internal/js_ast/js_ast_helpers.go
new file mode 100644
index 0000000..da78ea7
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/js_ast/js_ast_helpers.go
@@ -0,0 +1,2973 @@
+package js_ast
+
+import (
+	"math"
+	"strconv"
+	"strings"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type HelperContext struct {
+	isUnbound func(ast.Ref) bool
+}
+
+func MakeHelperContext(isUnbound func(ast.Ref) bool) HelperContext {
+	return HelperContext{
+		isUnbound: isUnbound,
+	}
+}
+
+// If this returns true, then calling this expression captures the target of
+// the property access as "this" when calling the function in the property.
+func IsPropertyAccess(expr Expr) bool {
+	switch expr.Data.(type) {
+	case *EDot, *EIndex:
+		return true
+	}
+	return false
+}
+
+func IsOptionalChain(value Expr) bool {
+	switch e := value.Data.(type) {
+	case *EDot:
+		return e.OptionalChain != OptionalChainNone
+	case *EIndex:
+		return e.OptionalChain != OptionalChainNone
+	case *ECall:
+		return e.OptionalChain != OptionalChainNone
+	}
+	return false
+}
+
+func Assign(a Expr, b Expr) Expr {
+	return Expr{Loc: a.Loc, Data: &EBinary{Op: BinOpAssign, Left: a, Right: b}}
+}
+
+func AssignStmt(a Expr, b Expr) Stmt {
+	return Stmt{Loc: a.Loc, Data: &SExpr{Value: Assign(a, b)}}
+}
+
+// Wraps the provided expression in the "!" prefix operator. The expression
+// will potentially be simplified to avoid generating unnecessary extra "!"
+// operators. For example, calling this with "!!x" will return "!x" instead
+// of returning "!!!x".
+func Not(expr Expr) Expr {
+	if result, ok := MaybeSimplifyNot(expr); ok {
+		return result
+	}
+	return Expr{Loc: expr.Loc, Data: &EUnary{Op: UnOpNot, Value: expr}}
+}
+
+// The given "expr" argument should be the operand of a "!" prefix operator
+// (i.e. the "x" in "!x"). This returns a simplified expression for the
+// whole operator (i.e. the "!x") if it can be simplified, or false if not.
+// It's separate from "Not()" above to avoid allocation on failure in case
+// that is undesired.
+//
+// This function intentionally avoids mutating the input AST so it can be
+// called after the AST has been frozen (i.e. after parsing ends).
+func MaybeSimplifyNot(expr Expr) (Expr, bool) {
+	switch e := expr.Data.(type) {
+	case *EAnnotation:
+		return MaybeSimplifyNot(e.Value)
+
+	case *EInlinedEnum:
+		if value, ok := MaybeSimplifyNot(e.Value); ok {
+			return value, true
+		}
+
+	case *ENull, *EUndefined:
+		return Expr{Loc: expr.Loc, Data: &EBoolean{Value: true}}, true
+
+	case *EBoolean:
+		return Expr{Loc: expr.Loc, Data: &EBoolean{Value: !e.Value}}, true
+
+	case *ENumber:
+		return Expr{Loc: expr.Loc, Data: &EBoolean{Value: e.Value == 0 || math.IsNaN(e.Value)}}, true
+
+	case *EBigInt:
+		if equal, ok := CheckEqualityBigInt(e.Value, "0"); ok {
+			return Expr{Loc: expr.Loc, Data: &EBoolean{Value: equal}}, true
+		}
+
+	case *EString:
+		return Expr{Loc: expr.Loc, Data: &EBoolean{Value: len(e.Value) == 0}}, true
+
+	case *EFunction, *EArrow, *ERegExp:
+		return Expr{Loc: expr.Loc, Data: &EBoolean{Value: false}}, true
+
+	case *EUnary:
+		// "!!!a" => "!a"
+		if e.Op == UnOpNot && KnownPrimitiveType(e.Value.Data) == PrimitiveBoolean {
+			return e.Value, true
+		}
+
+	case *EBinary:
+		// Make sure that these transformations are all safe for special values.
+		// For example, "!(a < b)" is not the same as "a >= b" if a and/or b are
+		// NaN (or undefined, or null, or possibly other problem cases too).
+		switch e.Op {
+		case BinOpLooseEq:
+			// "!(a == b)" => "a != b"
+			return Expr{Loc: expr.Loc, Data: &EBinary{Op: BinOpLooseNe, Left: e.Left, Right: e.Right}}, true
+
+		case BinOpLooseNe:
+			// "!(a != b)" => "a == b"
+			return Expr{Loc: expr.Loc, Data: &EBinary{Op: BinOpLooseEq, Left: e.Left, Right: e.Right}}, true
+
+		case BinOpStrictEq:
+			// "!(a === b)" => "a !== b"
+			return Expr{Loc: expr.Loc, Data: &EBinary{Op: BinOpStrictNe, Left: e.Left, Right: e.Right}}, true
+
+		case BinOpStrictNe:
+			// "!(a !== b)" => "a === b"
+			return Expr{Loc: expr.Loc, Data: &EBinary{Op: BinOpStrictEq, Left: e.Left, Right: e.Right}}, true
+
+		case BinOpComma:
+			// "!(a, b)" => "a, !b"
+			return Expr{Loc: expr.Loc, Data: &EBinary{Op: BinOpComma, Left: e.Left, Right: Not(e.Right)}}, true
+		}
+	}
+
+	return Expr{}, false
+}
+
+// This function intentionally avoids mutating the input AST so it can be
+// called after the AST has been frozen (i.e. after parsing ends).
+func MaybeSimplifyEqualityComparison(loc logger.Loc, e *EBinary, unsupportedFeatures compat.JSFeature) (Expr, bool) {
+	value, primitive := e.Left, e.Right
+
+	// Detect when the primitive comes first and flip the order of our checks
+	if IsPrimitiveLiteral(value.Data) {
+		value, primitive = primitive, value
+	}
+
+	// "!x === true" => "!x"
+	// "!x === false" => "!!x"
+	// "!x !== true" => "!!x"
+	// "!x !== false" => "!x"
+	if boolean, ok := primitive.Data.(*EBoolean); ok && KnownPrimitiveType(value.Data) == PrimitiveBoolean {
+		if boolean.Value == (e.Op == BinOpLooseNe || e.Op == BinOpStrictNe) {
+			return Not(value), true
+		} else {
+			return value, true
+		}
+	}
+
+	// "typeof x != 'undefined'" => "typeof x < 'u'"
+	// "typeof x == 'undefined'" => "typeof x > 'u'"
+	if !unsupportedFeatures.Has(compat.TypeofExoticObjectIsObject) {
+		// Only do this optimization if we know that the "typeof" operator won't
+		// return something random. The only case of this happening was Internet
+		// Explorer returning "unknown" for some objects, which messes with this
+		// optimization. So we don't do this when targeting Internet Explorer.
+		if typeof, ok := value.Data.(*EUnary); ok && typeof.Op == UnOpTypeof {
+			if str, ok := primitive.Data.(*EString); ok && helpers.UTF16EqualsString(str.Value, "undefined") {
+				flip := value == e.Right
+				op := BinOpLt
+				if (e.Op == BinOpLooseEq || e.Op == BinOpStrictEq) != flip {
+					op = BinOpGt
+				}
+				primitive.Data = &EString{Value: []uint16{'u'}}
+				if flip {
+					value, primitive = primitive, value
+				}
+				return Expr{Loc: loc, Data: &EBinary{Op: op, Left: value, Right: primitive}}, true
+			}
+		}
+	}
+
+	return Expr{}, false
+}
+
+func IsSymbolInstance(data E) bool {
+	switch e := data.(type) {
+	case *EDot:
+		return e.IsSymbolInstance
+
+	case *EIndex:
+		return e.IsSymbolInstance
+	}
+	return false
+}
+
+func IsPrimitiveLiteral(data E) bool {
+	switch e := data.(type) {
+	case *EAnnotation:
+		return IsPrimitiveLiteral(e.Value.Data)
+
+	case *EInlinedEnum:
+		return IsPrimitiveLiteral(e.Value.Data)
+
+	case *ENull, *EUndefined, *EString, *EBoolean, *ENumber, *EBigInt:
+		return true
+	}
+	return false
+}
+
+type PrimitiveType uint8
+
+const (
+	PrimitiveUnknown PrimitiveType = iota
+	PrimitiveMixed
+	PrimitiveNull
+	PrimitiveUndefined
+	PrimitiveBoolean
+	PrimitiveNumber
+	PrimitiveString
+	PrimitiveBigInt
+)
+
+// This can be used when the returned type is either one or the other
+func MergedKnownPrimitiveTypes(a Expr, b Expr) PrimitiveType {
+	x := KnownPrimitiveType(a.Data)
+	if x == PrimitiveUnknown {
+		return PrimitiveUnknown
+	}
+
+	y := KnownPrimitiveType(b.Data)
+	if y == PrimitiveUnknown {
+		return PrimitiveUnknown
+	}
+
+	if x == y {
+		return x
+	}
+	return PrimitiveMixed // Definitely some kind of primitive
+}
+
+// Note: This function does not say whether the expression is side-effect free
+// or not. For example, the expression "++x" always returns a primitive.
+func KnownPrimitiveType(expr E) PrimitiveType {
+	switch e := expr.(type) {
+	case *EAnnotation:
+		return KnownPrimitiveType(e.Value.Data)
+
+	case *EInlinedEnum:
+		return KnownPrimitiveType(e.Value.Data)
+
+	case *ENull:
+		return PrimitiveNull
+
+	case *EUndefined:
+		return PrimitiveUndefined
+
+	case *EBoolean:
+		return PrimitiveBoolean
+
+	case *ENumber:
+		return PrimitiveNumber
+
+	case *EString:
+		return PrimitiveString
+
+	case *EBigInt:
+		return PrimitiveBigInt
+
+	case *ETemplate:
+		if e.TagOrNil.Data == nil {
+			return PrimitiveString
+		}
+
+	case *EIf:
+		return MergedKnownPrimitiveTypes(e.Yes, e.No)
+
+	case *EUnary:
+		switch e.Op {
+		case UnOpVoid:
+			return PrimitiveUndefined
+
+		case UnOpTypeof:
+			return PrimitiveString
+
+		case UnOpNot, UnOpDelete:
+			return PrimitiveBoolean
+
+		case UnOpPos:
+			return PrimitiveNumber // Cannot be bigint because that throws an exception
+
+		case UnOpNeg, UnOpCpl:
+			value := KnownPrimitiveType(e.Value.Data)
+			if value == PrimitiveBigInt {
+				return PrimitiveBigInt
+			}
+			if value != PrimitiveUnknown && value != PrimitiveMixed {
+				return PrimitiveNumber
+			}
+			return PrimitiveMixed // Can be number or bigint
+
+		case UnOpPreDec, UnOpPreInc, UnOpPostDec, UnOpPostInc:
+			return PrimitiveMixed // Can be number or bigint
+		}
+
+	case *EBinary:
+		switch e.Op {
+		case BinOpStrictEq, BinOpStrictNe, BinOpLooseEq, BinOpLooseNe,
+			BinOpLt, BinOpGt, BinOpLe, BinOpGe,
+			BinOpInstanceof, BinOpIn:
+			return PrimitiveBoolean
+
+		case BinOpLogicalOr, BinOpLogicalAnd:
+			return MergedKnownPrimitiveTypes(e.Left, e.Right)
+
+		case BinOpNullishCoalescing:
+			left := KnownPrimitiveType(e.Left.Data)
+			right := KnownPrimitiveType(e.Right.Data)
+			if left == PrimitiveNull || left == PrimitiveUndefined {
+				return right
+			}
+			if left != PrimitiveUnknown {
+				if left != PrimitiveMixed {
+					return left // Definitely not null or undefined
+				}
+				if right != PrimitiveUnknown {
+					return PrimitiveMixed // Definitely some kind of primitive
+				}
+			}
+
+		case BinOpAdd:
+			left := KnownPrimitiveType(e.Left.Data)
+			right := KnownPrimitiveType(e.Right.Data)
+			if left == PrimitiveString || right == PrimitiveString {
+				return PrimitiveString
+			}
+			if left == PrimitiveBigInt && right == PrimitiveBigInt {
+				return PrimitiveBigInt
+			}
+			if left != PrimitiveUnknown && left != PrimitiveMixed && left != PrimitiveBigInt &&
+				right != PrimitiveUnknown && right != PrimitiveMixed && right != PrimitiveBigInt {
+				return PrimitiveNumber
+			}
+			return PrimitiveMixed // Can be number or bigint or string (or an exception)
+
+		case BinOpAddAssign:
+			right := KnownPrimitiveType(e.Right.Data)
+			if right == PrimitiveString {
+				return PrimitiveString
+			}
+			return PrimitiveMixed // Can be number or bigint or string (or an exception)
+
+		case
+			BinOpSub, BinOpSubAssign,
+			BinOpMul, BinOpMulAssign,
+			BinOpDiv, BinOpDivAssign,
+			BinOpRem, BinOpRemAssign,
+			BinOpPow, BinOpPowAssign,
+			BinOpBitwiseAnd, BinOpBitwiseAndAssign,
+			BinOpBitwiseOr, BinOpBitwiseOrAssign,
+			BinOpBitwiseXor, BinOpBitwiseXorAssign,
+			BinOpShl, BinOpShlAssign,
+			BinOpShr, BinOpShrAssign,
+			BinOpUShr, BinOpUShrAssign:
+			return PrimitiveMixed // Can be number or bigint (or an exception)
+
+		case BinOpAssign, BinOpComma:
+			return KnownPrimitiveType(e.Right.Data)
+		}
+	}
+
+	return PrimitiveUnknown
+}
+
+func CanChangeStrictToLoose(a Expr, b Expr) bool {
+	x := KnownPrimitiveType(a.Data)
+	y := KnownPrimitiveType(b.Data)
+	return x == y && x != PrimitiveUnknown && x != PrimitiveMixed
+}
+
+// Returns true if the result of the "typeof" operator on this expression is
+// statically determined and this expression has no side effects (i.e. can be
+// removed without consequence).
+func TypeofWithoutSideEffects(data E) (string, bool) {
+	switch e := data.(type) {
+	case *EAnnotation:
+		if e.Flags.Has(CanBeRemovedIfUnusedFlag) {
+			return TypeofWithoutSideEffects(e.Value.Data)
+		}
+
+	case *EInlinedEnum:
+		return TypeofWithoutSideEffects(e.Value.Data)
+
+	case *ENull:
+		return "object", true
+
+	case *EUndefined:
+		return "undefined", true
+
+	case *EBoolean:
+		return "boolean", true
+
+	case *ENumber:
+		return "number", true
+
+	case *EBigInt:
+		return "bigint", true
+
+	case *EString:
+		return "string", true
+
+	case *EFunction, *EArrow:
+		return "function", true
+	}
+
+	return "", false
+}
+
+// The goal of this function is to "rotate" the AST if it's possible to use the
+// left-associative property of the operator to avoid unnecessary parentheses.
+//
+// When using this, make absolutely sure that the operator is actually
+// associative. For example, the "+" operator is not associative for
+// floating-point numbers.
+//
+// This function intentionally avoids mutating the input AST so it can be
+// called after the AST has been frozen (i.e. after parsing ends).
+func JoinWithLeftAssociativeOp(op OpCode, a Expr, b Expr) Expr {
+	// "(a, b) op c" => "a, b op c"
+	if comma, ok := a.Data.(*EBinary); ok && comma.Op == BinOpComma {
+		// Don't mutate the original AST
+		clone := *comma
+		clone.Right = JoinWithLeftAssociativeOp(op, clone.Right, b)
+		return Expr{Loc: a.Loc, Data: &clone}
+	}
+
+	// "a op (b op c)" => "(a op b) op c"
+	// "a op (b op (c op d))" => "((a op b) op c) op d"
+	for {
+		if binary, ok := b.Data.(*EBinary); ok && binary.Op == op {
+			a = JoinWithLeftAssociativeOp(op, a, binary.Left)
+			b = binary.Right
+		} else {
+			break
+		}
+	}
+
+	// "a op b" => "a op b"
+	// "(a op b) op c" => "(a op b) op c"
+	return Expr{Loc: a.Loc, Data: &EBinary{Op: op, Left: a, Right: b}}
+}
+
+func JoinWithComma(a Expr, b Expr) Expr {
+	if a.Data == nil {
+		return b
+	}
+	if b.Data == nil {
+		return a
+	}
+	return Expr{Loc: a.Loc, Data: &EBinary{Op: BinOpComma, Left: a, Right: b}}
+}
+
+func JoinAllWithComma(all []Expr) (result Expr) {
+	for _, value := range all {
+		result = JoinWithComma(result, value)
+	}
+	return
+}
+
+func ConvertBindingToExpr(binding Binding, wrapIdentifier func(logger.Loc, ast.Ref) Expr) Expr {
+	loc := binding.Loc
+
+	switch b := binding.Data.(type) {
+	case *BMissing:
+		return Expr{Loc: loc, Data: &EMissing{}}
+
+	case *BIdentifier:
+		if wrapIdentifier != nil {
+			return wrapIdentifier(loc, b.Ref)
+		}
+		return Expr{Loc: loc, Data: &EIdentifier{Ref: b.Ref}}
+
+	case *BArray:
+		exprs := make([]Expr, len(b.Items))
+		for i, item := range b.Items {
+			expr := ConvertBindingToExpr(item.Binding, wrapIdentifier)
+			if b.HasSpread && i+1 == len(b.Items) {
+				expr = Expr{Loc: expr.Loc, Data: &ESpread{Value: expr}}
+			} else if item.DefaultValueOrNil.Data != nil {
+				expr = Assign(expr, item.DefaultValueOrNil)
+			}
+			exprs[i] = expr
+		}
+		return Expr{Loc: loc, Data: &EArray{
+			Items:        exprs,
+			IsSingleLine: b.IsSingleLine,
+		}}
+
+	case *BObject:
+		properties := make([]Property, len(b.Properties))
+		for i, property := range b.Properties {
+			value := ConvertBindingToExpr(property.Value, wrapIdentifier)
+			kind := PropertyField
+			if property.IsSpread {
+				kind = PropertySpread
+			}
+			var flags PropertyFlags
+			if property.IsComputed {
+				flags |= PropertyIsComputed
+			}
+			properties[i] = Property{
+				Kind:             kind,
+				Flags:            flags,
+				Key:              property.Key,
+				ValueOrNil:       value,
+				InitializerOrNil: property.DefaultValueOrNil,
+			}
+		}
+		return Expr{Loc: loc, Data: &EObject{
+			Properties:   properties,
+			IsSingleLine: b.IsSingleLine,
+		}}
+
+	default:
+		panic("Internal error")
+	}
+}
+
+// This will return a nil expression if the expression can be totally removed.
+//
+// This function intentionally avoids mutating the input AST so it can be
+// called after the AST has been frozen (i.e. after parsing ends).
+func (ctx HelperContext) SimplifyUnusedExpr(expr Expr, unsupportedFeatures compat.JSFeature) Expr {
+	switch e := expr.Data.(type) {
+	case *EAnnotation:
+		if e.Flags.Has(CanBeRemovedIfUnusedFlag) {
+			return Expr{}
+		}
+
+	case *EInlinedEnum:
+		return ctx.SimplifyUnusedExpr(e.Value, unsupportedFeatures)
+
+	case *ENull, *EUndefined, *EMissing, *EBoolean, *ENumber, *EBigInt,
+		*EString, *EThis, *ERegExp, *EFunction, *EArrow, *EImportMeta:
+		return Expr{}
+
+	case *EDot:
+		if e.CanBeRemovedIfUnused {
+			return Expr{}
+		}
+
+	case *EIdentifier:
+		if e.MustKeepDueToWithStmt {
+			break
+		}
+		if e.CanBeRemovedIfUnused || !ctx.isUnbound(e.Ref) {
+			return Expr{}
+		}
+
+	case *ETemplate:
+		if e.TagOrNil.Data == nil {
+			var comma Expr
+			var templateLoc logger.Loc
+			var template *ETemplate
+			for _, part := range e.Parts {
+				// If we know this value is some kind of primitive, then we know that
+				// "ToString" has no side effects and can be avoided.
+				if KnownPrimitiveType(part.Value.Data) != PrimitiveUnknown {
+					if template != nil {
+						comma = JoinWithComma(comma, Expr{Loc: templateLoc, Data: template})
+						template = nil
+					}
+					comma = JoinWithComma(comma, ctx.SimplifyUnusedExpr(part.Value, unsupportedFeatures))
+					continue
+				}
+
+				// Make sure "ToString" is still evaluated on the value. We can't use
+				// string addition here because that may evaluate "ValueOf" instead.
+				if template == nil {
+					template = &ETemplate{}
+					templateLoc = part.Value.Loc
+				}
+				template.Parts = append(template.Parts, TemplatePart{Value: part.Value})
+			}
+			if template != nil {
+				comma = JoinWithComma(comma, Expr{Loc: templateLoc, Data: template})
+			}
+			return comma
+		} else if e.CanBeUnwrappedIfUnused {
+			// If the function call was annotated as being able to be removed if the
+			// result is unused, then we can remove it and just keep the arguments.
+			// Note that there are no implicit "ToString" operations for tagged
+			// template literals.
+			var comma Expr
+			for _, part := range e.Parts {
+				comma = JoinWithComma(comma, ctx.SimplifyUnusedExpr(part.Value, unsupportedFeatures))
+			}
+			return comma
+		}
+
+	case *EArray:
+		// Arrays with "..." spread expressions can't be unwrapped because the
+		// "..." triggers code evaluation via iterators. In that case, just trim
+		// the other items instead and leave the array expression there.
+		for _, spread := range e.Items {
+			if _, ok := spread.Data.(*ESpread); ok {
+				items := make([]Expr, 0, len(e.Items))
+				for _, item := range e.Items {
+					item = ctx.SimplifyUnusedExpr(item, unsupportedFeatures)
+					if item.Data != nil {
+						items = append(items, item)
+					}
+				}
+
+				// Don't mutate the original AST
+				clone := *e
+				clone.Items = items
+				return Expr{Loc: expr.Loc, Data: &clone}
+			}
+		}
+
+		// Otherwise, the array can be completely removed. We only need to keep any
+		// array items with side effects. Apply this simplification recursively.
+		var result Expr
+		for _, item := range e.Items {
+			result = JoinWithComma(result, ctx.SimplifyUnusedExpr(item, unsupportedFeatures))
+		}
+		return result
+
+	case *EObject:
+		// Objects with "..." spread expressions can't be unwrapped because the
+		// "..." triggers code evaluation via getters. In that case, just trim
+		// the other items instead and leave the object expression there.
+		for _, spread := range e.Properties {
+			if spread.Kind == PropertySpread {
+				properties := make([]Property, 0, len(e.Properties))
+				for _, property := range e.Properties {
+					// Spread properties must always be evaluated
+					if property.Kind != PropertySpread {
+						value := ctx.SimplifyUnusedExpr(property.ValueOrNil, unsupportedFeatures)
+						if value.Data != nil {
+							// Keep the value
+							property.ValueOrNil = value
+						} else if !property.Flags.Has(PropertyIsComputed) {
+							// Skip this property if the key doesn't need to be computed
+							continue
+						} else {
+							// Replace values without side effects with "0" because it's short
+							property.ValueOrNil.Data = &ENumber{}
+						}
+					}
+					properties = append(properties, property)
+				}
+
+				// Don't mutate the original AST
+				clone := *e
+				clone.Properties = properties
+				return Expr{Loc: expr.Loc, Data: &clone}
+			}
+		}
+
+		// Otherwise, the object can be completely removed. We only need to keep any
+		// object properties with side effects. Apply this simplification recursively.
+		var result Expr
+		for _, property := range e.Properties {
+			if property.Flags.Has(PropertyIsComputed) {
+				// Make sure "ToString" is still evaluated on the key
+				result = JoinWithComma(result, Expr{Loc: property.Key.Loc, Data: &EBinary{
+					Op:    BinOpAdd,
+					Left:  property.Key,
+					Right: Expr{Loc: property.Key.Loc, Data: &EString{}},
+				}})
+			}
+			result = JoinWithComma(result, ctx.SimplifyUnusedExpr(property.ValueOrNil, unsupportedFeatures))
+		}
+		return result
+
+	case *EIf:
+		yes := ctx.SimplifyUnusedExpr(e.Yes, unsupportedFeatures)
+		no := ctx.SimplifyUnusedExpr(e.No, unsupportedFeatures)
+
+		// "foo() ? 1 : 2" => "foo()"
+		if yes.Data == nil && no.Data == nil {
+			return ctx.SimplifyUnusedExpr(e.Test, unsupportedFeatures)
+		}
+
+		// "foo() ? 1 : bar()" => "foo() || bar()"
+		if yes.Data == nil {
+			return JoinWithLeftAssociativeOp(BinOpLogicalOr, e.Test, no)
+		}
+
+		// "foo() ? bar() : 2" => "foo() && bar()"
+		if no.Data == nil {
+			return JoinWithLeftAssociativeOp(BinOpLogicalAnd, e.Test, yes)
+		}
+
+		if yes != e.Yes || no != e.No {
+			return Expr{Loc: expr.Loc, Data: &EIf{Test: e.Test, Yes: yes, No: no}}
+		}
+
+	case *EUnary:
+		switch e.Op {
+		// These operators must not have any type conversions that can execute code
+		// such as "toString" or "valueOf". They must also never throw any exceptions.
+		case UnOpVoid, UnOpNot:
+			return ctx.SimplifyUnusedExpr(e.Value, unsupportedFeatures)
+
+		case UnOpTypeof:
+			if _, ok := e.Value.Data.(*EIdentifier); ok && e.WasOriginallyTypeofIdentifier {
+				// "typeof x" must not be transformed into if "x" since doing so could
+				// cause an exception to be thrown. Instead we can just remove it since
+				// "typeof x" is special-cased in the standard to never throw.
+				return Expr{}
+			}
+			return ctx.SimplifyUnusedExpr(e.Value, unsupportedFeatures)
+		}
+
+	case *EBinary:
+		left := e.Left
+		right := e.Right
+
+		switch e.Op {
+		// These operators must not have any type conversions that can execute code
+		// such as "toString" or "valueOf". They must also never throw any exceptions.
+		case BinOpStrictEq, BinOpStrictNe, BinOpComma:
+			return JoinWithComma(ctx.SimplifyUnusedExpr(left, unsupportedFeatures), ctx.SimplifyUnusedExpr(right, unsupportedFeatures))
+
+		// We can simplify "==" and "!=" even though they can call "toString" and/or
+		// "valueOf" if we can statically determine that the types of both sides are
+		// primitives. In that case there won't be any chance for user-defined
+		// "toString" and/or "valueOf" to be called.
+		case BinOpLooseEq, BinOpLooseNe:
+			if MergedKnownPrimitiveTypes(left, right) != PrimitiveUnknown {
+				return JoinWithComma(ctx.SimplifyUnusedExpr(left, unsupportedFeatures), ctx.SimplifyUnusedExpr(right, unsupportedFeatures))
+			}
+
+		case BinOpLogicalAnd, BinOpLogicalOr, BinOpNullishCoalescing:
+			// If this is a boolean logical operation and the result is unused, then
+			// we know the left operand will only be used for its boolean value and
+			// can be simplified under that assumption
+			if e.Op != BinOpNullishCoalescing {
+				left = ctx.SimplifyBooleanExpr(left)
+			}
+
+			// Preserve short-circuit behavior: the left expression is only unused if
+			// the right expression can be completely removed. Otherwise, the left
+			// expression is important for the branch.
+			right = ctx.SimplifyUnusedExpr(right, unsupportedFeatures)
+			if right.Data == nil {
+				return ctx.SimplifyUnusedExpr(left, unsupportedFeatures)
+			}
+
+			// Try to take advantage of the optional chain operator to shorten code
+			if !unsupportedFeatures.Has(compat.OptionalChain) {
+				if binary, ok := left.Data.(*EBinary); ok {
+					// "a != null && a.b()" => "a?.b()"
+					// "a == null || a.b()" => "a?.b()"
+					if (binary.Op == BinOpLooseNe && e.Op == BinOpLogicalAnd) || (binary.Op == BinOpLooseEq && e.Op == BinOpLogicalOr) {
+						var test Expr
+						if _, ok := binary.Right.Data.(*ENull); ok {
+							test = binary.Left
+						} else if _, ok := binary.Left.Data.(*ENull); ok {
+							test = binary.Right
+						}
+
+						// Note: Technically unbound identifiers can refer to a getter on
+						// the global object and that getter can have side effects that can
+						// be observed if we run that getter once instead of twice. But this
+						// seems like terrible coding practice and very unlikely to come up
+						// in real software, so we deliberately ignore this possibility and
+						// optimize for size instead of for this obscure edge case.
+						//
+						// If this is ever changed, then we must also pessimize the lowering
+						// of "foo?.bar" to save the value of "foo" to ensure that it's only
+						// evaluated once. Specifically "foo?.bar" would have to expand to:
+						//
+						//   var _a;
+						//   (_a = foo) == null ? void 0 : _a.bar;
+						//
+						// instead of:
+						//
+						//   foo == null ? void 0 : foo.bar;
+						//
+						// Babel does the first one while TypeScript does the second one.
+						// Since TypeScript doesn't handle this extreme edge case and
+						// TypeScript is very widely used, I think it's fine for us to not
+						// handle this edge case either.
+						if id, ok := test.Data.(*EIdentifier); ok && !id.MustKeepDueToWithStmt && TryToInsertOptionalChain(test, right) {
+							return right
+						}
+					}
+				}
+			}
+
+		case BinOpAdd:
+			if result, isStringAddition := simplifyUnusedStringAdditionChain(expr); isStringAddition {
+				return result
+			}
+		}
+
+		if left != e.Left || right != e.Right {
+			return Expr{Loc: expr.Loc, Data: &EBinary{Op: e.Op, Left: left, Right: right}}
+		}
+
+	case *ECall:
+		// A call that has been marked "__PURE__" can be removed if all arguments
+		// can be removed. The annotation causes us to ignore the target.
+		if e.CanBeUnwrappedIfUnused {
+			var result Expr
+			for _, arg := range e.Args {
+				if _, ok := arg.Data.(*ESpread); ok {
+					arg.Data = &EArray{Items: []Expr{arg}, IsSingleLine: true}
+				}
+				result = JoinWithComma(result, ctx.SimplifyUnusedExpr(arg, unsupportedFeatures))
+			}
+			return result
+		}
+
+		// Attempt to shorten IIFEs
+		if len(e.Args) == 0 {
+			switch target := e.Target.Data.(type) {
+			case *EFunction:
+				if len(target.Fn.Args) != 0 {
+					break
+				}
+
+				// Just delete "(function() {})()" completely
+				if len(target.Fn.Body.Block.Stmts) == 0 {
+					return Expr{}
+				}
+
+			case *EArrow:
+				if len(target.Args) != 0 {
+					break
+				}
+
+				// Just delete "(() => {})()" completely
+				if len(target.Body.Block.Stmts) == 0 {
+					return Expr{}
+				}
+
+				if len(target.Body.Block.Stmts) == 1 {
+					switch s := target.Body.Block.Stmts[0].Data.(type) {
+					case *SExpr:
+						if !target.IsAsync {
+							// Replace "(() => { foo() })()" with "foo()"
+							return s.Value
+						} else {
+							// Replace "(async () => { foo() })()" with "(async () => foo())()"
+							clone := *target
+							clone.Body.Block.Stmts[0].Data = &SReturn{ValueOrNil: s.Value}
+							clone.PreferExpr = true
+							return Expr{Loc: expr.Loc, Data: &ECall{Target: Expr{Loc: e.Target.Loc, Data: &clone}}}
+						}
+
+					case *SReturn:
+						if !target.IsAsync {
+							// Replace "(() => foo())()" with "foo()"
+							return s.ValueOrNil
+						}
+					}
+				}
+			}
+		}
+
+	case *ENew:
+		// A constructor call that has been marked "__PURE__" can be removed if all
+		// arguments can be removed. The annotation causes us to ignore the target.
+		if e.CanBeUnwrappedIfUnused {
+			var result Expr
+			for _, arg := range e.Args {
+				if _, ok := arg.Data.(*ESpread); ok {
+					arg.Data = &EArray{Items: []Expr{arg}, IsSingleLine: true}
+				}
+				result = JoinWithComma(result, ctx.SimplifyUnusedExpr(arg, unsupportedFeatures))
+			}
+			return result
+		}
+	}
+
+	return expr
+}
+
+// This function intentionally avoids mutating the input AST so it can be
+// called after the AST has been frozen (i.e. after parsing ends).
+func simplifyUnusedStringAdditionChain(expr Expr) (Expr, bool) {
+	switch e := expr.Data.(type) {
+	case *EString:
+		// "'x' + y" => "'' + y"
+		return Expr{Loc: expr.Loc, Data: &EString{}}, true
+
+	case *EBinary:
+		if e.Op == BinOpAdd {
+			left, leftIsStringAddition := simplifyUnusedStringAdditionChain(e.Left)
+
+			if right, rightIsString := e.Right.Data.(*EString); rightIsString {
+				// "('' + x) + 'y'" => "'' + x"
+				if leftIsStringAddition {
+					return left, true
+				}
+
+				// "x + 'y'" => "x + ''"
+				if !leftIsStringAddition && len(right.Value) > 0 {
+					return Expr{Loc: expr.Loc, Data: &EBinary{
+						Op:    BinOpAdd,
+						Left:  left,
+						Right: Expr{Loc: e.Right.Loc, Data: &EString{}},
+					}}, true
+				}
+			}
+
+			// Don't mutate the original AST
+			if left != e.Left {
+				expr.Data = &EBinary{Op: BinOpAdd, Left: left, Right: e.Right}
+			}
+
+			return expr, leftIsStringAddition
+		}
+	}
+
+	return expr, false
+}
+
+func ToInt32(f float64) int32 {
+	// The easy way
+	i := int32(f)
+	if float64(i) == f {
+		return i
+	}
+
+	// Special-case non-finite numbers (casting them is unspecified behavior in Go)
+	if math.IsNaN(f) || math.IsInf(f, 0) {
+		return 0
+	}
+
+	// The hard way
+	i = int32(uint32(math.Mod(math.Abs(f), 4294967296)))
+	if math.Signbit(f) {
+		return -i
+	}
+	return i
+}
+
+func ToUint32(f float64) uint32 {
+	return uint32(ToInt32(f))
+}
+
+func isInt32OrUint32(data E) bool {
+	switch e := data.(type) {
+	case *EUnary:
+		return e.Op == UnOpCpl
+
+	case *EBinary:
+		switch e.Op {
+		case BinOpBitwiseAnd, BinOpBitwiseOr, BinOpBitwiseXor, BinOpShl, BinOpShr, BinOpUShr:
+			return true
+
+		case BinOpLogicalOr, BinOpLogicalAnd:
+			return isInt32OrUint32(e.Left.Data) && isInt32OrUint32(e.Right.Data)
+		}
+
+	case *EIf:
+		return isInt32OrUint32(e.Yes.Data) && isInt32OrUint32(e.No.Data)
+	}
+	return false
+}
+
+func ToNumberWithoutSideEffects(data E) (float64, bool) {
+	switch e := data.(type) {
+	case *EAnnotation:
+		return ToNumberWithoutSideEffects(e.Value.Data)
+
+	case *EInlinedEnum:
+		return ToNumberWithoutSideEffects(e.Value.Data)
+
+	case *ENull:
+		return 0, true
+
+	case *EUndefined, *ERegExp:
+		return math.NaN(), true
+
+	case *EArray:
+		if len(e.Items) == 0 {
+			// "+[]" => "0"
+			return 0, true
+		}
+
+	case *EObject:
+		if len(e.Properties) == 0 {
+			// "+{}" => "NaN"
+			return math.NaN(), true
+		}
+
+	case *EBoolean:
+		if e.Value {
+			return 1, true
+		} else {
+			return 0, true
+		}
+
+	case *ENumber:
+		return e.Value, true
+
+	case *EString:
+		// "+''" => "0"
+		if len(e.Value) == 0 {
+			return 0, true
+		}
+
+		// "+'1'" => "1"
+		if num, ok := StringToEquivalentNumberValue(e.Value); ok {
+			return num, true
+		}
+	}
+
+	return 0, false
+}
+
+func ToStringWithoutSideEffects(data E) (string, bool) {
+	switch e := data.(type) {
+	case *ENull:
+		return "null", true
+
+	case *EUndefined:
+		return "undefined", true
+
+	case *EBoolean:
+		if e.Value {
+			return "true", true
+		} else {
+			return "false", true
+		}
+
+	case *EBigInt:
+		// Only do this if there is no radix
+		if len(e.Value) < 2 || e.Value[0] != '0' {
+			return e.Value, true
+		}
+
+	case *ENumber:
+		if str, ok := TryToStringOnNumberSafely(e.Value, 10); ok {
+			return str, true
+		}
+
+	case *ERegExp:
+		return e.Value, true
+
+	case *EDot:
+		// This is dumb but some JavaScript obfuscators use this to generate string literals
+		if e.Name == "constructor" {
+			switch e.Target.Data.(type) {
+			case *EString:
+				return "function String() { [native code] }", true
+
+			case *ERegExp:
+				return "function RegExp() { [native code] }", true
+			}
+		}
+	}
+
+	return "", false
+}
+
+func extractNumericValue(data E) (float64, bool) {
+	switch e := data.(type) {
+	case *EAnnotation:
+		return extractNumericValue(e.Value.Data)
+
+	case *EInlinedEnum:
+		return extractNumericValue(e.Value.Data)
+
+	case *ENumber:
+		return e.Value, true
+	}
+
+	return 0, false
+}
+
+func extractNumericValues(left Expr, right Expr) (float64, float64, bool) {
+	if a, ok := extractNumericValue(left.Data); ok {
+		if b, ok := extractNumericValue(right.Data); ok {
+			return a, b, true
+		}
+	}
+	return 0, 0, false
+}
+
+func extractStringValue(data E) ([]uint16, bool) {
+	switch e := data.(type) {
+	case *EAnnotation:
+		return extractStringValue(e.Value.Data)
+
+	case *EInlinedEnum:
+		return extractStringValue(e.Value.Data)
+
+	case *EString:
+		return e.Value, true
+	}
+
+	return nil, false
+}
+
+func extractStringValues(left Expr, right Expr) ([]uint16, []uint16, bool) {
+	if a, ok := extractStringValue(left.Data); ok {
+		if b, ok := extractStringValue(right.Data); ok {
+			return a, b, true
+		}
+	}
+	return nil, nil, false
+}
+
+func stringCompareUCS2(a []uint16, b []uint16) int {
+	var n int
+	if len(a) < len(b) {
+		n = len(a)
+	} else {
+		n = len(b)
+	}
+	for i := 0; i < n; i++ {
+		if delta := int(a[i]) - int(b[i]); delta != 0 {
+			return delta
+		}
+	}
+	return len(a) - len(b)
+}
+
+func approximatePrintedIntCharCount(intValue float64) int {
+	count := 1 + (int)(math.Max(0, math.Floor(math.Log10(math.Abs(intValue)))))
+	if intValue < 0 {
+		count++
+	}
+	return count
+}
+
+func ShouldFoldBinaryOperatorWhenMinifying(binary *EBinary) bool {
+	switch binary.Op {
+	case
+		// Equality tests should always result in smaller code when folded
+		BinOpLooseEq,
+		BinOpLooseNe,
+		BinOpStrictEq,
+		BinOpStrictNe,
+
+		// Minification always folds right signed shift operations since they are
+		// unlikely to result in larger output. Note: ">>>" could result in
+		// bigger output such as "-1 >>> 0" becoming "4294967295".
+		BinOpShr,
+
+		// Minification always folds the following bitwise operations since they
+		// are unlikely to result in larger output.
+		BinOpBitwiseAnd,
+		BinOpBitwiseOr,
+		BinOpBitwiseXor,
+		BinOpLt,
+		BinOpGt,
+		BinOpLe,
+		BinOpGe:
+		return true
+
+	case BinOpAdd:
+		// Addition of small-ish integers can definitely be folded without issues
+		// "1 + 2" => "3"
+		if left, right, ok := extractNumericValues(binary.Left, binary.Right); ok &&
+			left == math.Trunc(left) && math.Abs(left) <= 0xFFFF_FFFF &&
+			right == math.Trunc(right) && math.Abs(right) <= 0xFFFF_FFFF {
+			return true
+		}
+
+		// String addition should pretty much always be more compact when folded
+		if _, _, ok := extractStringValues(binary.Left, binary.Right); ok {
+			return true
+		}
+
+	case BinOpSub:
+		// Subtraction of small-ish integers can definitely be folded without issues
+		// "3 - 1" => "2"
+		if left, right, ok := extractNumericValues(binary.Left, binary.Right); ok &&
+			left == math.Trunc(left) && math.Abs(left) <= 0xFFFF_FFFF &&
+			right == math.Trunc(right) && math.Abs(right) <= 0xFFFF_FFFF {
+			return true
+		}
+
+	case BinOpDiv:
+		// "0/0" => "NaN"
+		// "1/0" => "Infinity"
+		// "1/-0" => "-Infinity"
+		if _, right, ok := extractNumericValues(binary.Left, binary.Right); ok && right == 0 {
+			return true
+		}
+
+	case BinOpShl:
+		// "1 << 3" => "8"
+		// "1 << 24" => "1 << 24" (since "1<<24" is shorter than "16777216")
+		if left, right, ok := extractNumericValues(binary.Left, binary.Right); ok {
+			leftLen := approximatePrintedIntCharCount(left)
+			rightLen := approximatePrintedIntCharCount(right)
+			resultLen := approximatePrintedIntCharCount(float64(ToInt32(left) << (ToUint32(right) & 31)))
+			return resultLen <= leftLen+2+rightLen
+		}
+
+	case BinOpUShr:
+		// "10 >>> 1" => "5"
+		// "-1 >>> 0" => "-1 >>> 0" (since "-1>>>0" is shorter than "4294967295")
+		if left, right, ok := extractNumericValues(binary.Left, binary.Right); ok {
+			leftLen := approximatePrintedIntCharCount(left)
+			rightLen := approximatePrintedIntCharCount(right)
+			resultLen := approximatePrintedIntCharCount(float64(ToUint32(left) >> (ToUint32(right) & 31)))
+			return resultLen <= leftLen+3+rightLen
+		}
+
+	case BinOpLogicalAnd, BinOpLogicalOr, BinOpNullishCoalescing:
+		if IsPrimitiveLiteral(binary.Left.Data) {
+			return true
+		}
+	}
+	return false
+}
+
+// This function intentionally avoids mutating the input AST so it can be
+// called after the AST has been frozen (i.e. after parsing ends).
+func FoldBinaryOperator(loc logger.Loc, e *EBinary) Expr {
+	switch e.Op {
+	case BinOpAdd:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &ENumber{Value: left + right}}
+		}
+		if left, right, ok := extractStringValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &EString{Value: joinStrings(left, right)}}
+		}
+
+	case BinOpSub:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &ENumber{Value: left - right}}
+		}
+
+	case BinOpMul:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &ENumber{Value: left * right}}
+		}
+
+	case BinOpDiv:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &ENumber{Value: left / right}}
+		}
+
+	case BinOpRem:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &ENumber{Value: math.Mod(left, right)}}
+		}
+
+	case BinOpPow:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &ENumber{Value: math.Pow(left, right)}}
+		}
+
+	case BinOpShl:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &ENumber{Value: float64(ToInt32(left) << (ToUint32(right) & 31))}}
+		}
+
+	case BinOpShr:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &ENumber{Value: float64(ToInt32(left) >> (ToUint32(right) & 31))}}
+		}
+
+	case BinOpUShr:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &ENumber{Value: float64(ToUint32(left) >> (ToUint32(right) & 31))}}
+		}
+
+	case BinOpBitwiseAnd:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &ENumber{Value: float64(ToInt32(left) & ToInt32(right))}}
+		}
+
+	case BinOpBitwiseOr:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &ENumber{Value: float64(ToInt32(left) | ToInt32(right))}}
+		}
+
+	case BinOpBitwiseXor:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &ENumber{Value: float64(ToInt32(left) ^ ToInt32(right))}}
+		}
+
+	case BinOpLt:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &EBoolean{Value: left < right}}
+		}
+		if left, right, ok := extractStringValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &EBoolean{Value: stringCompareUCS2(left, right) < 0}}
+		}
+
+	case BinOpGt:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &EBoolean{Value: left > right}}
+		}
+		if left, right, ok := extractStringValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &EBoolean{Value: stringCompareUCS2(left, right) > 0}}
+		}
+
+	case BinOpLe:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &EBoolean{Value: left <= right}}
+		}
+		if left, right, ok := extractStringValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &EBoolean{Value: stringCompareUCS2(left, right) <= 0}}
+		}
+
+	case BinOpGe:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &EBoolean{Value: left >= right}}
+		}
+		if left, right, ok := extractStringValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &EBoolean{Value: stringCompareUCS2(left, right) >= 0}}
+		}
+
+	case BinOpLooseEq, BinOpStrictEq:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &EBoolean{Value: left == right}}
+		}
+		if left, right, ok := extractStringValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &EBoolean{Value: stringCompareUCS2(left, right) == 0}}
+		}
+
+	case BinOpLooseNe, BinOpStrictNe:
+		if left, right, ok := extractNumericValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &EBoolean{Value: left != right}}
+		}
+		if left, right, ok := extractStringValues(e.Left, e.Right); ok {
+			return Expr{Loc: loc, Data: &EBoolean{Value: stringCompareUCS2(left, right) != 0}}
+		}
+
+	case BinOpLogicalAnd:
+		if boolean, sideEffects, ok := ToBooleanWithSideEffects(e.Left.Data); ok {
+			if !boolean {
+				return e.Left
+			} else if sideEffects == NoSideEffects {
+				return e.Right
+			}
+		}
+
+	case BinOpLogicalOr:
+		if boolean, sideEffects, ok := ToBooleanWithSideEffects(e.Left.Data); ok {
+			if boolean {
+				return e.Left
+			} else if sideEffects == NoSideEffects {
+				return e.Right
+			}
+		}
+
+	case BinOpNullishCoalescing:
+		if isNullOrUndefined, sideEffects, ok := ToNullOrUndefinedWithSideEffects(e.Left.Data); ok {
+			if !isNullOrUndefined {
+				return e.Left
+			} else if sideEffects == NoSideEffects {
+				return e.Right
+			}
+		}
+	}
+
+	return Expr{}
+}
+
+func IsBinaryNullAndUndefined(left Expr, right Expr, op OpCode) (Expr, Expr, bool) {
+	if a, ok := left.Data.(*EBinary); ok && a.Op == op {
+		if b, ok := right.Data.(*EBinary); ok && b.Op == op {
+			idA, eqA := a.Left, a.Right
+			idB, eqB := b.Left, b.Right
+
+			// Detect when the identifier comes second and flip the order of our checks
+			if _, ok := eqA.Data.(*EIdentifier); ok {
+				idA, eqA = eqA, idA
+			}
+			if _, ok := eqB.Data.(*EIdentifier); ok {
+				idB, eqB = eqB, idB
+			}
+
+			if idA, ok := idA.Data.(*EIdentifier); ok {
+				if idB, ok := idB.Data.(*EIdentifier); ok && idA.Ref == idB.Ref {
+					// "a === null || a === void 0"
+					if _, ok := eqA.Data.(*ENull); ok {
+						if _, ok := eqB.Data.(*EUndefined); ok {
+							return a.Left, a.Right, true
+						}
+					}
+
+					// "a === void 0 || a === null"
+					if _, ok := eqA.Data.(*EUndefined); ok {
+						if _, ok := eqB.Data.(*ENull); ok {
+							return b.Left, b.Right, true
+						}
+					}
+				}
+			}
+		}
+	}
+
+	return Expr{}, Expr{}, false
+}
+
+func CheckEqualityBigInt(a string, b string) (equal bool, ok bool) {
+	// Equal literals are always equal
+	if a == b {
+		return true, true
+	}
+
+	// Unequal literals are unequal if neither has a radix. Leading zeros are
+	// disallowed in bigint literals without a radix, so in this case we know
+	// each value is in canonical form.
+	if (len(a) < 2 || a[0] != '0') && (len(b) < 2 || b[0] != '0') {
+		return false, true
+	}
+
+	return false, false
+}
+
+type EqualityKind uint8
+
+const (
+	LooseEquality EqualityKind = iota
+	StrictEquality
+)
+
+// Returns "equal, ok". If "ok" is false, then nothing is known about the two
+// values. If "ok" is true, the equality or inequality of the two values is
+// stored in "equal".
+func CheckEqualityIfNoSideEffects(left E, right E, kind EqualityKind) (equal bool, ok bool) {
+	if r, ok := right.(*EInlinedEnum); ok {
+		return CheckEqualityIfNoSideEffects(left, r.Value.Data, kind)
+	}
+
+	switch l := left.(type) {
+	case *EInlinedEnum:
+		return CheckEqualityIfNoSideEffects(l.Value.Data, right, kind)
+
+	case *ENull:
+		switch right.(type) {
+		case *ENull:
+			// "null === null" is true
+			return true, true
+
+		case *EUndefined:
+			// "null == undefined" is true
+			// "null === undefined" is false
+			return kind == LooseEquality, true
+
+		default:
+			if IsPrimitiveLiteral(right) {
+				// "null == (not null or undefined)" is false
+				return false, true
+			}
+		}
+
+	case *EUndefined:
+		switch right.(type) {
+		case *EUndefined:
+			// "undefined === undefined" is true
+			return true, true
+
+		case *ENull:
+			// "undefined == null" is true
+			// "undefined === null" is false
+			return kind == LooseEquality, true
+
+		default:
+			if IsPrimitiveLiteral(right) {
+				// "undefined == (not null or undefined)" is false
+				return false, true
+			}
+		}
+
+	case *EBoolean:
+		switch r := right.(type) {
+		case *EBoolean:
+			// "false === false" is true
+			// "false === true" is false
+			return l.Value == r.Value, true
+
+		case *ENumber:
+			if kind == LooseEquality {
+				if l.Value {
+					// "true == 1" is true
+					return r.Value == 1, true
+				} else {
+					// "false == 0" is true
+					return r.Value == 0, true
+				}
+			} else {
+				// "true === 1" is false
+				// "false === 0" is false
+				return false, true
+			}
+
+		case *ENull, *EUndefined:
+			// "(not null or undefined) == undefined" is false
+			return false, true
+		}
+
+	case *ENumber:
+		switch r := right.(type) {
+		case *ENumber:
+			// "0 === 0" is true
+			// "0 === 1" is false
+			return l.Value == r.Value, true
+
+		case *EBoolean:
+			if kind == LooseEquality {
+				if r.Value {
+					// "1 == true" is true
+					return l.Value == 1, true
+				} else {
+					// "0 == false" is true
+					return l.Value == 0, true
+				}
+			} else {
+				// "1 === true" is false
+				// "0 === false" is false
+				return false, true
+			}
+
+		case *ENull, *EUndefined:
+			// "(not null or undefined) == undefined" is false
+			return false, true
+		}
+
+	case *EBigInt:
+		switch r := right.(type) {
+		case *EBigInt:
+			// "0n === 0n" is true
+			// "0n === 1n" is false
+			return CheckEqualityBigInt(l.Value, r.Value)
+
+		case *ENull, *EUndefined:
+			// "(not null or undefined) == undefined" is false
+			return false, true
+		}
+
+	case *EString:
+		switch r := right.(type) {
+		case *EString:
+			// "'a' === 'a'" is true
+			// "'a' === 'b'" is false
+			return helpers.UTF16EqualsUTF16(l.Value, r.Value), true
+
+		case *ENull, *EUndefined:
+			// "(not null or undefined) == undefined" is false
+			return false, true
+		}
+	}
+
+	return false, false
+}
+
+func ValuesLookTheSame(left E, right E) bool {
+	if b, ok := right.(*EInlinedEnum); ok {
+		return ValuesLookTheSame(left, b.Value.Data)
+	}
+
+	switch a := left.(type) {
+	case *EInlinedEnum:
+		return ValuesLookTheSame(a.Value.Data, right)
+
+	case *EIdentifier:
+		if b, ok := right.(*EIdentifier); ok && a.Ref == b.Ref {
+			return true
+		}
+
+	case *EDot:
+		if b, ok := right.(*EDot); ok && a.HasSameFlagsAs(b) &&
+			a.Name == b.Name && ValuesLookTheSame(a.Target.Data, b.Target.Data) {
+			return true
+		}
+
+	case *EIndex:
+		if b, ok := right.(*EIndex); ok && a.HasSameFlagsAs(b) &&
+			ValuesLookTheSame(a.Target.Data, b.Target.Data) && ValuesLookTheSame(a.Index.Data, b.Index.Data) {
+			return true
+		}
+
+	case *EIf:
+		if b, ok := right.(*EIf); ok && ValuesLookTheSame(a.Test.Data, b.Test.Data) &&
+			ValuesLookTheSame(a.Yes.Data, b.Yes.Data) && ValuesLookTheSame(a.No.Data, b.No.Data) {
+			return true
+		}
+
+	case *EUnary:
+		if b, ok := right.(*EUnary); ok && a.Op == b.Op && ValuesLookTheSame(a.Value.Data, b.Value.Data) {
+			return true
+		}
+
+	case *EBinary:
+		if b, ok := right.(*EBinary); ok && a.Op == b.Op && ValuesLookTheSame(a.Left.Data, b.Left.Data) &&
+			ValuesLookTheSame(a.Right.Data, b.Right.Data) {
+			return true
+		}
+
+	case *ECall:
+		if b, ok := right.(*ECall); ok && a.HasSameFlagsAs(b) &&
+			len(a.Args) == len(b.Args) && ValuesLookTheSame(a.Target.Data, b.Target.Data) {
+			for i := range a.Args {
+				if !ValuesLookTheSame(a.Args[i].Data, b.Args[i].Data) {
+					return false
+				}
+			}
+			return true
+		}
+
+	// Special-case to distinguish between negative an non-negative zero when mangling
+	// "a ? -0 : 0" => "a ? -0 : 0"
+	// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Equality_comparisons_and_sameness
+	case *ENumber:
+		b, ok := right.(*ENumber)
+		if ok && a.Value == 0 && b.Value == 0 && math.Signbit(a.Value) != math.Signbit(b.Value) {
+			return false
+		}
+	}
+
+	equal, ok := CheckEqualityIfNoSideEffects(left, right, StrictEquality)
+	return ok && equal
+}
+
+func TryToInsertOptionalChain(test Expr, expr Expr) bool {
+	switch e := expr.Data.(type) {
+	case *EDot:
+		if ValuesLookTheSame(test.Data, e.Target.Data) {
+			e.OptionalChain = OptionalChainStart
+			return true
+		}
+		if TryToInsertOptionalChain(test, e.Target) {
+			if e.OptionalChain == OptionalChainNone {
+				e.OptionalChain = OptionalChainContinue
+			}
+			return true
+		}
+
+	case *EIndex:
+		if ValuesLookTheSame(test.Data, e.Target.Data) {
+			e.OptionalChain = OptionalChainStart
+			return true
+		}
+		if TryToInsertOptionalChain(test, e.Target) {
+			if e.OptionalChain == OptionalChainNone {
+				e.OptionalChain = OptionalChainContinue
+			}
+			return true
+		}
+
+	case *ECall:
+		if ValuesLookTheSame(test.Data, e.Target.Data) {
+			e.OptionalChain = OptionalChainStart
+			return true
+		}
+		if TryToInsertOptionalChain(test, e.Target) {
+			if e.OptionalChain == OptionalChainNone {
+				e.OptionalChain = OptionalChainContinue
+			}
+			return true
+		}
+	}
+
+	return false
+}
+
+func joinStrings(a []uint16, b []uint16) []uint16 {
+	data := make([]uint16, len(a)+len(b))
+	copy(data[:len(a)], a)
+	copy(data[len(a):], b)
+	return data
+}
+
+// String concatenation with numbers is required by the TypeScript compiler for
+// "constant expression" handling in enums. However, we don't want to introduce
+// correctness bugs by accidentally stringifying a number differently than how
+// a real JavaScript VM would do it. So we are conservative and we only do this
+// when we know it'll be the same result.
+func TryToStringOnNumberSafely(n float64, radix int) (string, bool) {
+	if i := int32(n); float64(i) == n {
+		return strconv.FormatInt(int64(i), radix), true
+	}
+	if math.IsNaN(n) {
+		return "NaN", true
+	}
+	if math.IsInf(n, 1) {
+		return "Infinity", true
+	}
+	if math.IsInf(n, -1) {
+		return "-Infinity", true
+	}
+	return "", false
+}
+
+// Note: We don't know if this is string addition yet at this point
+func foldAdditionPreProcess(expr Expr) Expr {
+	switch e := expr.Data.(type) {
+	case *EInlinedEnum:
+		// "See through" inline enum constants
+		expr = e.Value
+
+	case *EArray:
+		// "[] + x" => "'' + x"
+		// "[1,2] + x" => "'1,2' + x"
+		items := make([]string, 0, len(e.Items))
+		for _, item := range e.Items {
+			switch item.Data.(type) {
+			case *EUndefined, *ENull:
+				items = append(items, "")
+				continue
+			}
+			if str, ok := ToStringWithoutSideEffects(item.Data); ok {
+				item.Data = &EString{Value: helpers.StringToUTF16(str)}
+			}
+			str, ok := item.Data.(*EString)
+			if !ok {
+				break
+			}
+			items = append(items, helpers.UTF16ToString(str.Value))
+		}
+		if len(items) == len(e.Items) {
+			expr.Data = &EString{Value: helpers.StringToUTF16(strings.Join(items, ","))}
+		}
+
+	case *EObject:
+		// "{} + x" => "'[object Object]' + x"
+		if len(e.Properties) == 0 {
+			expr.Data = &EString{Value: helpers.StringToUTF16("[object Object]")}
+		}
+	}
+	return expr
+}
+
+type StringAdditionKind uint8
+
+const (
+	StringAdditionNormal StringAdditionKind = iota
+	StringAdditionWithNestedLeft
+)
+
+// This function intentionally avoids mutating the input AST so it can be
+// called after the AST has been frozen (i.e. after parsing ends).
+func FoldStringAddition(left Expr, right Expr, kind StringAdditionKind) Expr {
+	left = foldAdditionPreProcess(left)
+	right = foldAdditionPreProcess(right)
+
+	// Transforming the left operand into a string is not safe if it comes from
+	// a nested AST node. The following transforms are invalid:
+	//
+	//   "0 + 1 + 'x'" => "0 + '1x'"
+	//   "0 + 1 + `${x}`" => "0 + `1${x}`"
+	//
+	if kind != StringAdditionWithNestedLeft {
+		switch right.Data.(type) {
+		case *EString, *ETemplate:
+			if str, ok := ToStringWithoutSideEffects(left.Data); ok {
+				left.Data = &EString{Value: helpers.StringToUTF16(str)}
+			}
+		}
+	}
+
+	switch l := left.Data.(type) {
+	case *EString:
+		// "'x' + 0" => "'x' + '0'"
+		if str, ok := ToStringWithoutSideEffects(right.Data); ok {
+			right.Data = &EString{Value: helpers.StringToUTF16(str)}
+		}
+
+		switch r := right.Data.(type) {
+		case *EString:
+			// "'x' + 'y'" => "'xy'"
+			return Expr{Loc: left.Loc, Data: &EString{
+				Value:          joinStrings(l.Value, r.Value),
+				PreferTemplate: l.PreferTemplate || r.PreferTemplate,
+			}}
+
+		case *ETemplate:
+			if r.TagOrNil.Data == nil {
+				// "'x' + `y${z}`" => "`xy${z}`"
+				return Expr{Loc: left.Loc, Data: &ETemplate{
+					HeadLoc:    left.Loc,
+					HeadCooked: joinStrings(l.Value, r.HeadCooked),
+					Parts:      r.Parts,
+				}}
+			}
+		}
+
+		// "'' + typeof x" => "typeof x"
+		if len(l.Value) == 0 && KnownPrimitiveType(right.Data) == PrimitiveString {
+			return right
+		}
+
+	case *ETemplate:
+		if l.TagOrNil.Data == nil {
+			// "`${x}` + 0" => "`${x}` + '0'"
+			if str, ok := ToStringWithoutSideEffects(right.Data); ok {
+				right.Data = &EString{Value: helpers.StringToUTF16(str)}
+			}
+
+			switch r := right.Data.(type) {
+			case *EString:
+				// "`${x}y` + 'z'" => "`${x}yz`"
+				n := len(l.Parts)
+				head := l.HeadCooked
+				parts := make([]TemplatePart, n)
+				if n == 0 {
+					head = joinStrings(head, r.Value)
+				} else {
+					copy(parts, l.Parts)
+					parts[n-1].TailCooked = joinStrings(parts[n-1].TailCooked, r.Value)
+				}
+				return Expr{Loc: left.Loc, Data: &ETemplate{
+					HeadLoc:    l.HeadLoc,
+					HeadCooked: head,
+					Parts:      parts,
+				}}
+
+			case *ETemplate:
+				if r.TagOrNil.Data == nil {
+					// "`${a}b` + `x${y}`" => "`${a}bx${y}`"
+					n := len(l.Parts)
+					head := l.HeadCooked
+					parts := make([]TemplatePart, n+len(r.Parts))
+					copy(parts[n:], r.Parts)
+					if n == 0 {
+						head = joinStrings(head, r.HeadCooked)
+					} else {
+						copy(parts[:n], l.Parts)
+						parts[n-1].TailCooked = joinStrings(parts[n-1].TailCooked, r.HeadCooked)
+					}
+					return Expr{Loc: left.Loc, Data: &ETemplate{
+						HeadLoc:    l.HeadLoc,
+						HeadCooked: head,
+						Parts:      parts,
+					}}
+				}
+			}
+		}
+	}
+
+	// "typeof x + ''" => "typeof x"
+	if r, ok := right.Data.(*EString); ok && len(r.Value) == 0 && KnownPrimitiveType(left.Data) == PrimitiveString {
+		return left
+	}
+
+	return Expr{}
+}
+
+// "`a${'b'}c`" => "`abc`"
+//
+// This function intentionally avoids mutating the input AST so it can be
+// called after the AST has been frozen (i.e. after parsing ends).
+func InlinePrimitivesIntoTemplate(loc logger.Loc, e *ETemplate) Expr {
+	// Can't inline strings if there's a custom template tag
+	if e.TagOrNil.Data != nil {
+		return Expr{Loc: loc, Data: e}
+	}
+
+	headCooked := e.HeadCooked
+	parts := make([]TemplatePart, 0, len(e.Parts))
+
+	for _, part := range e.Parts {
+		if value, ok := part.Value.Data.(*EInlinedEnum); ok {
+			part.Value = value.Value
+		}
+		if str, ok := ToStringWithoutSideEffects(part.Value.Data); ok {
+			part.Value.Data = &EString{Value: helpers.StringToUTF16(str)}
+		}
+		if str, ok := part.Value.Data.(*EString); ok {
+			if len(parts) == 0 {
+				headCooked = append(append(headCooked, str.Value...), part.TailCooked...)
+			} else {
+				prevPart := &parts[len(parts)-1]
+				prevPart.TailCooked = append(append(prevPart.TailCooked, str.Value...), part.TailCooked...)
+			}
+		} else {
+			parts = append(parts, part)
+		}
+	}
+
+	// Become a plain string if there are no substitutions
+	if len(parts) == 0 {
+		return Expr{Loc: loc, Data: &EString{
+			Value:          headCooked,
+			PreferTemplate: true,
+		}}
+	}
+
+	return Expr{Loc: loc, Data: &ETemplate{
+		HeadLoc:    e.HeadLoc,
+		HeadCooked: headCooked,
+		Parts:      parts,
+	}}
+}
+
+type SideEffects uint8
+
+const (
+	CouldHaveSideEffects SideEffects = iota
+	NoSideEffects
+)
+
+func ToNullOrUndefinedWithSideEffects(data E) (isNullOrUndefined bool, sideEffects SideEffects, ok bool) {
+	switch e := data.(type) {
+	case *EAnnotation:
+		isNullOrUndefined, sideEffects, ok = ToNullOrUndefinedWithSideEffects(e.Value.Data)
+		if e.Flags.Has(CanBeRemovedIfUnusedFlag) {
+			sideEffects = NoSideEffects
+		}
+		return
+
+	case *EInlinedEnum:
+		return ToNullOrUndefinedWithSideEffects(e.Value.Data)
+
+		// Never null or undefined
+	case *EBoolean, *ENumber, *EString, *ERegExp,
+		*EFunction, *EArrow, *EBigInt:
+		return false, NoSideEffects, true
+
+	// Never null or undefined
+	case *EObject, *EArray, *EClass:
+		return false, CouldHaveSideEffects, true
+
+	// Always null or undefined
+	case *ENull, *EUndefined:
+		return true, NoSideEffects, true
+
+	case *EUnary:
+		switch e.Op {
+		case
+			// Always number or bigint
+			UnOpPos, UnOpNeg, UnOpCpl,
+			UnOpPreDec, UnOpPreInc, UnOpPostDec, UnOpPostInc,
+			// Always boolean
+			UnOpNot, UnOpDelete:
+			return false, CouldHaveSideEffects, true
+
+		// Always boolean
+		case UnOpTypeof:
+			if e.WasOriginallyTypeofIdentifier {
+				// Expressions such as "typeof x" never have any side effects
+				return false, NoSideEffects, true
+			}
+			return false, CouldHaveSideEffects, true
+
+		// Always undefined
+		case UnOpVoid:
+			return true, CouldHaveSideEffects, true
+		}
+
+	case *EBinary:
+		switch e.Op {
+		case
+			// Always string or number or bigint
+			BinOpAdd, BinOpAddAssign,
+			// Always number or bigint
+			BinOpSub, BinOpMul, BinOpDiv, BinOpRem, BinOpPow,
+			BinOpSubAssign, BinOpMulAssign, BinOpDivAssign, BinOpRemAssign, BinOpPowAssign,
+			BinOpShl, BinOpShr, BinOpUShr,
+			BinOpShlAssign, BinOpShrAssign, BinOpUShrAssign,
+			BinOpBitwiseOr, BinOpBitwiseAnd, BinOpBitwiseXor,
+			BinOpBitwiseOrAssign, BinOpBitwiseAndAssign, BinOpBitwiseXorAssign,
+			// Always boolean
+			BinOpLt, BinOpLe, BinOpGt, BinOpGe, BinOpIn, BinOpInstanceof,
+			BinOpLooseEq, BinOpLooseNe, BinOpStrictEq, BinOpStrictNe:
+			return false, CouldHaveSideEffects, true
+
+		case BinOpComma:
+			if isNullOrUndefined, _, ok := ToNullOrUndefinedWithSideEffects(e.Right.Data); ok {
+				return isNullOrUndefined, CouldHaveSideEffects, true
+			}
+		}
+	}
+
+	return false, NoSideEffects, false
+}
+
+func ToBooleanWithSideEffects(data E) (boolean bool, sideEffects SideEffects, ok bool) {
+	switch e := data.(type) {
+	case *EAnnotation:
+		boolean, sideEffects, ok = ToBooleanWithSideEffects(e.Value.Data)
+		if e.Flags.Has(CanBeRemovedIfUnusedFlag) {
+			sideEffects = NoSideEffects
+		}
+		return
+
+	case *EInlinedEnum:
+		return ToBooleanWithSideEffects(e.Value.Data)
+
+	case *ENull, *EUndefined:
+		return false, NoSideEffects, true
+
+	case *EBoolean:
+		return e.Value, NoSideEffects, true
+
+	case *ENumber:
+		return e.Value != 0 && !math.IsNaN(e.Value), NoSideEffects, true
+
+	case *EBigInt:
+		equal, ok := CheckEqualityBigInt(e.Value, "0")
+		return !equal, NoSideEffects, ok
+
+	case *EString:
+		return len(e.Value) > 0, NoSideEffects, true
+
+	case *EFunction, *EArrow, *ERegExp:
+		return true, NoSideEffects, true
+
+	case *EObject, *EArray, *EClass:
+		return true, CouldHaveSideEffects, true
+
+	case *EUnary:
+		switch e.Op {
+		case UnOpVoid:
+			return false, CouldHaveSideEffects, true
+
+		case UnOpTypeof:
+			// Never an empty string
+			if e.WasOriginallyTypeofIdentifier {
+				// Expressions such as "typeof x" never have any side effects
+				return true, NoSideEffects, true
+			}
+			return true, CouldHaveSideEffects, true
+
+		case UnOpNot:
+			if boolean, SideEffects, ok := ToBooleanWithSideEffects(e.Value.Data); ok {
+				return !boolean, SideEffects, true
+			}
+		}
+
+	case *EBinary:
+		switch e.Op {
+		case BinOpLogicalOr:
+			// "anything || truthy" is truthy
+			if boolean, _, ok := ToBooleanWithSideEffects(e.Right.Data); ok && boolean {
+				return true, CouldHaveSideEffects, true
+			}
+
+		case BinOpLogicalAnd:
+			// "anything && falsy" is falsy
+			if boolean, _, ok := ToBooleanWithSideEffects(e.Right.Data); ok && !boolean {
+				return false, CouldHaveSideEffects, true
+			}
+
+		case BinOpComma:
+			// "anything, truthy/falsy" is truthy/falsy
+			if boolean, _, ok := ToBooleanWithSideEffects(e.Right.Data); ok {
+				return boolean, CouldHaveSideEffects, true
+			}
+		}
+	}
+
+	return false, CouldHaveSideEffects, false
+}
+
+// Simplify syntax when we know it's used inside a boolean context
+//
+// This function intentionally avoids mutating the input AST so it can be
+// called after the AST has been frozen (i.e. after parsing ends).
+func (ctx HelperContext) SimplifyBooleanExpr(expr Expr) Expr {
+	switch e := expr.Data.(type) {
+	case *EUnary:
+		if e.Op == UnOpNot {
+			// "!!a" => "a"
+			if e2, ok2 := e.Value.Data.(*EUnary); ok2 && e2.Op == UnOpNot {
+				return ctx.SimplifyBooleanExpr(e2.Value)
+			}
+
+			// "!!!a" => "!a"
+			return Expr{Loc: expr.Loc, Data: &EUnary{Op: UnOpNot, Value: ctx.SimplifyBooleanExpr(e.Value)}}
+		}
+
+	case *EBinary:
+		left := e.Left
+		right := e.Right
+
+		switch e.Op {
+		case BinOpStrictEq, BinOpStrictNe, BinOpLooseEq, BinOpLooseNe:
+			if r, ok := extractNumericValue(right.Data); ok && r == 0 && isInt32OrUint32(left.Data) {
+				// If the left is guaranteed to be an integer (e.g. not NaN,
+				// Infinity, or a non-numeric value) then a test against zero
+				// in a boolean context is unnecessary because the value is
+				// only truthy if it's not zero.
+				if e.Op == BinOpStrictNe || e.Op == BinOpLooseNe {
+					// "if ((a | b) !== 0)" => "if (a | b)"
+					return left
+				} else {
+					// "if ((a | b) === 0)" => "if (!(a | b))"
+					return Not(left)
+				}
+			}
+
+		case BinOpLogicalAnd:
+			// "if (!!a && !!b)" => "if (a && b)"
+			left = ctx.SimplifyBooleanExpr(left)
+			right = ctx.SimplifyBooleanExpr(right)
+
+			if boolean, SideEffects, ok := ToBooleanWithSideEffects(right.Data); ok && boolean && SideEffects == NoSideEffects {
+				// "if (anything && truthyNoSideEffects)" => "if (anything)"
+				return left
+			}
+
+		case BinOpLogicalOr:
+			// "if (!!a || !!b)" => "if (a || b)"
+			left = ctx.SimplifyBooleanExpr(left)
+			right = ctx.SimplifyBooleanExpr(right)
+
+			if boolean, SideEffects, ok := ToBooleanWithSideEffects(right.Data); ok && !boolean && SideEffects == NoSideEffects {
+				// "if (anything || falsyNoSideEffects)" => "if (anything)"
+				return left
+			}
+		}
+
+		if left != e.Left || right != e.Right {
+			return Expr{Loc: expr.Loc, Data: &EBinary{Op: e.Op, Left: left, Right: right}}
+		}
+
+	case *EIf:
+		// "if (a ? !!b : !!c)" => "if (a ? b : c)"
+		yes := ctx.SimplifyBooleanExpr(e.Yes)
+		no := ctx.SimplifyBooleanExpr(e.No)
+
+		if boolean, SideEffects, ok := ToBooleanWithSideEffects(yes.Data); ok && SideEffects == NoSideEffects {
+			if boolean {
+				// "if (anything1 ? truthyNoSideEffects : anything2)" => "if (anything1 || anything2)"
+				return JoinWithLeftAssociativeOp(BinOpLogicalOr, e.Test, no)
+			} else {
+				// "if (anything1 ? falsyNoSideEffects : anything2)" => "if (!anything1 || anything2)"
+				return JoinWithLeftAssociativeOp(BinOpLogicalAnd, Not(e.Test), no)
+			}
+		}
+
+		if boolean, SideEffects, ok := ToBooleanWithSideEffects(no.Data); ok && SideEffects == NoSideEffects {
+			if boolean {
+				// "if (anything1 ? anything2 : truthyNoSideEffects)" => "if (!anything1 || anything2)"
+				return JoinWithLeftAssociativeOp(BinOpLogicalOr, Not(e.Test), yes)
+			} else {
+				// "if (anything1 ? anything2 : falsyNoSideEffects)" => "if (anything1 && anything2)"
+				return JoinWithLeftAssociativeOp(BinOpLogicalAnd, e.Test, yes)
+			}
+		}
+
+		if yes != e.Yes || no != e.No {
+			return Expr{Loc: expr.Loc, Data: &EIf{Test: e.Test, Yes: yes, No: no}}
+		}
+
+	default:
+		// "!![]" => "true"
+		if boolean, sideEffects, ok := ToBooleanWithSideEffects(expr.Data); ok && (sideEffects == NoSideEffects || ctx.ExprCanBeRemovedIfUnused(expr)) {
+			return Expr{Loc: expr.Loc, Data: &EBoolean{Value: boolean}}
+		}
+	}
+
+	return expr
+}
+
+type StmtsCanBeRemovedIfUnusedFlags uint8
+
+const (
+	KeepExportClauses StmtsCanBeRemovedIfUnusedFlags = 1 << iota
+	ReturnCanBeRemovedIfUnused
+)
+
+func (ctx HelperContext) StmtsCanBeRemovedIfUnused(stmts []Stmt, flags StmtsCanBeRemovedIfUnusedFlags) bool {
+	for _, stmt := range stmts {
+		switch s := stmt.Data.(type) {
+		case *SFunction, *SEmpty:
+			// These never have side effects
+
+		case *SImport:
+			// Let these be removed if they are unused. Note that we also need to
+			// check if the imported file is marked as "sideEffects: false" before we
+			// can remove a SImport statement. Otherwise the import must be kept for
+			// its side effects.
+
+		case *SClass:
+			if !ctx.ClassCanBeRemovedIfUnused(s.Class) {
+				return false
+			}
+
+		case *SReturn:
+			if (flags&ReturnCanBeRemovedIfUnused) == 0 || (s.ValueOrNil.Data != nil && !ctx.ExprCanBeRemovedIfUnused(s.ValueOrNil)) {
+				return false
+			}
+
+		case *SExpr:
+			if !ctx.ExprCanBeRemovedIfUnused(s.Value) {
+				if s.IsFromClassOrFnThatCanBeRemovedIfUnused {
+					// This statement was automatically generated when lowering a class
+					// or function that we were able to analyze as having no side effects
+					// before lowering. So we consider it to be removable. The assumption
+					// here is that we are seeing at least all of the statements from the
+					// class lowering operation all at once (although we may possibly be
+					// seeing even more statements than that). Since we're making a binary
+					// all-or-nothing decision about the side effects of these statements,
+					// we can safely consider these to be side-effect free because we
+					// aren't in danger of partially dropping some of the class setup code.
+				} else {
+					return false
+				}
+			}
+
+		case *SLocal:
+			// "await" is a side effect because it affects code timing
+			if s.Kind == LocalAwaitUsing {
+				return false
+			}
+
+			for _, decl := range s.Decls {
+				// Check that the bindings are side-effect free
+				switch binding := decl.Binding.Data.(type) {
+				case *BIdentifier:
+					// An identifier binding has no side effects
+
+				case *BArray:
+					// Destructuring the initializer has no side effects if the
+					// initializer is an array, since we assume the iterator is then
+					// the built-in side-effect free array iterator.
+					if _, ok := decl.ValueOrNil.Data.(*EArray); ok {
+						for _, item := range binding.Items {
+							if item.DefaultValueOrNil.Data != nil && !ctx.ExprCanBeRemovedIfUnused(item.DefaultValueOrNil) {
+								return false
+							}
+
+							switch item.Binding.Data.(type) {
+							case *BIdentifier, *BMissing:
+								// Right now we only handle an array pattern with identifier
+								// bindings or with empty holes (i.e. "missing" elements)
+							default:
+								return false
+							}
+						}
+						break
+					}
+					return false
+
+				default:
+					// Consider anything else to potentially have side effects
+					return false
+				}
+
+				// Check that the initializer is side-effect free
+				if decl.ValueOrNil.Data != nil {
+					if !ctx.ExprCanBeRemovedIfUnused(decl.ValueOrNil) {
+						return false
+					}
+
+					// "using" declarations are only side-effect free if they are initialized to null or undefined
+					if s.Kind.IsUsing() {
+						if t := KnownPrimitiveType(decl.ValueOrNil.Data); t != PrimitiveNull && t != PrimitiveUndefined {
+							return false
+						}
+					}
+				}
+			}
+
+		case *STry:
+			if !ctx.StmtsCanBeRemovedIfUnused(s.Block.Stmts, 0) || (s.Finally != nil && !ctx.StmtsCanBeRemovedIfUnused(s.Finally.Block.Stmts, 0)) {
+				return false
+			}
+
+		case *SExportFrom:
+			// Exports are tracked separately, so this isn't necessary
+
+		case *SExportClause:
+			if (flags & KeepExportClauses) != 0 {
+				return false
+			}
+
+		case *SExportDefault:
+			switch s2 := s.Value.Data.(type) {
+			case *SExpr:
+				if !ctx.ExprCanBeRemovedIfUnused(s2.Value) {
+					return false
+				}
+
+			case *SFunction:
+				// These never have side effects
+
+			case *SClass:
+				if !ctx.ClassCanBeRemovedIfUnused(s2.Class) {
+					return false
+				}
+
+			default:
+				panic("Internal error")
+			}
+
+		default:
+			// Assume that all statements not explicitly special-cased here have side
+			// effects, and cannot be removed even if unused
+			return false
+		}
+	}
+
+	return true
+}
+
+func (ctx HelperContext) ClassCanBeRemovedIfUnused(class Class) bool {
+	if len(class.Decorators) > 0 {
+		return false
+	}
+
+	// Note: This check is incorrect. Extending a non-constructible object can
+	// throw an error, which is a side effect:
+	//
+	//   async function x() {}
+	//   class y extends x {}
+	//
+	// But refusing to tree-shake every class with a base class is not a useful
+	// thing for a bundler to do. So we pretend that this edge case doesn't
+	// exist. At the time of writing, both Rollup and Terser don't consider this
+	// to be a side effect either.
+	if class.ExtendsOrNil.Data != nil && !ctx.ExprCanBeRemovedIfUnused(class.ExtendsOrNil) {
+		return false
+	}
+
+	for _, property := range class.Properties {
+		if property.Kind == PropertyClassStaticBlock {
+			if !ctx.StmtsCanBeRemovedIfUnused(property.ClassStaticBlock.Block.Stmts, 0) {
+				return false
+			}
+			continue
+		}
+
+		if len(property.Decorators) > 0 {
+			return false
+		}
+
+		if property.Flags.Has(PropertyIsComputed) && !IsPrimitiveLiteral(property.Key.Data) && !IsSymbolInstance(property.Key.Data) {
+			return false
+		}
+
+		if property.Kind.IsMethodDefinition() {
+			if fn, ok := property.ValueOrNil.Data.(*EFunction); ok {
+				for _, arg := range fn.Fn.Args {
+					if len(arg.Decorators) > 0 {
+						return false
+					}
+				}
+			}
+		}
+
+		if property.Flags.Has(PropertyIsStatic) {
+			if property.ValueOrNil.Data != nil && !ctx.ExprCanBeRemovedIfUnused(property.ValueOrNil) {
+				return false
+			}
+
+			if property.InitializerOrNil.Data != nil && !ctx.ExprCanBeRemovedIfUnused(property.InitializerOrNil) {
+				return false
+			}
+
+			// Legacy TypeScript static class fields are considered to have side
+			// effects because they use assign semantics, not define semantics, and
+			// that can trigger getters. For example:
+			//
+			//   class Foo {
+			//     static set foo(x) { importantSideEffect(x) }
+			//   }
+			//   class Bar extends Foo {
+			//     foo = 1
+			//   }
+			//
+			// This happens in TypeScript when "useDefineForClassFields" is disabled
+			// because TypeScript (and esbuild) transforms the above class into this:
+			//
+			//   class Foo {
+			//     static set foo(x) { importantSideEffect(x); }
+			//   }
+			//   class Bar extends Foo {
+			//   }
+			//   Bar.foo = 1;
+			//
+			// Note that it's not possible to analyze the base class to determine that
+			// these assignments are side-effect free. For example:
+			//
+			//   // Some code that already ran before your code
+			//   Object.defineProperty(Object.prototype, 'foo', {
+			//     set(x) { imporantSideEffect(x) }
+			//   })
+			//
+			//   // Your code
+			//   class Foo {
+			//     static foo = 1
+			//   }
+			//
+			if property.Kind == PropertyField && !class.UseDefineForClassFields {
+				return false
+			}
+		}
+	}
+
+	return true
+}
+
+func (ctx HelperContext) ExprCanBeRemovedIfUnused(expr Expr) bool {
+	switch e := expr.Data.(type) {
+	case *EAnnotation:
+		return e.Flags.Has(CanBeRemovedIfUnusedFlag)
+
+	case *EInlinedEnum:
+		return ctx.ExprCanBeRemovedIfUnused(e.Value)
+
+	case *ENull, *EUndefined, *EMissing, *EBoolean, *ENumber, *EBigInt,
+		*EString, *EThis, *ERegExp, *EFunction, *EArrow, *EImportMeta:
+		return true
+
+	case *EDot:
+		return e.CanBeRemovedIfUnused
+
+	case *EClass:
+		return ctx.ClassCanBeRemovedIfUnused(e.Class)
+
+	case *EIdentifier:
+		if e.MustKeepDueToWithStmt {
+			return false
+		}
+
+		// Unbound identifiers cannot be removed because they can have side effects.
+		// One possible side effect is throwing a ReferenceError if they don't exist.
+		// Another one is a getter with side effects on the global object:
+		//
+		//   Object.defineProperty(globalThis, 'x', {
+		//     get() {
+		//       sideEffect();
+		//     },
+		//   });
+		//
+		// Be very careful about this possibility. It's tempting to treat all
+		// identifier expressions as not having side effects but that's wrong. We
+		// must make sure they have been declared by the code we are currently
+		// compiling before we can tell that they have no side effects.
+		//
+		// Note that we currently ignore ReferenceErrors due to TDZ access. This is
+		// incorrect but proper TDZ analysis is very complicated and would have to
+		// be very conservative, which would inhibit a lot of optimizations of code
+		// inside closures. This may need to be revisited if it proves problematic.
+		if e.CanBeRemovedIfUnused || !ctx.isUnbound(e.Ref) {
+			return true
+		}
+
+	case *EImportIdentifier:
+		// References to an ES6 import item are always side-effect free in an
+		// ECMAScript environment.
+		//
+		// They could technically have side effects if the imported module is a
+		// CommonJS module and the import item was translated to a property access
+		// (which esbuild's bundler does) and the property has a getter with side
+		// effects.
+		//
+		// But this is very unlikely and respecting this edge case would mean
+		// disabling tree shaking of all code that references an export from a
+		// CommonJS module. It would also likely violate the expectations of some
+		// developers because the code *looks* like it should be able to be tree
+		// shaken.
+		//
+		// So we deliberately ignore this edge case and always treat import item
+		// references as being side-effect free.
+		return true
+
+	case *EIf:
+		return ctx.ExprCanBeRemovedIfUnused(e.Test) &&
+			((ctx.isSideEffectFreeUnboundIdentifierRef(e.Yes, e.Test, true) || ctx.ExprCanBeRemovedIfUnused(e.Yes)) &&
+				(ctx.isSideEffectFreeUnboundIdentifierRef(e.No, e.Test, false) || ctx.ExprCanBeRemovedIfUnused(e.No)))
+
+	case *EArray:
+		for _, item := range e.Items {
+			if spread, ok := item.Data.(*ESpread); ok {
+				if _, ok := spread.Value.Data.(*EArray); ok {
+					// Spread of an inline array such as "[...[x]]" is side-effect free
+					item = spread.Value
+				}
+			}
+
+			if !ctx.ExprCanBeRemovedIfUnused(item) {
+				return false
+			}
+		}
+		return true
+
+	case *EObject:
+		for _, property := range e.Properties {
+			// The key must still be evaluated if it's computed or a spread
+			if property.Kind == PropertySpread {
+				return false
+			}
+			if property.Flags.Has(PropertyIsComputed) && !IsPrimitiveLiteral(property.Key.Data) && !IsSymbolInstance(property.Key.Data) {
+				return false
+			}
+			if property.ValueOrNil.Data != nil && !ctx.ExprCanBeRemovedIfUnused(property.ValueOrNil) {
+				return false
+			}
+		}
+		return true
+
+	case *ECall:
+		canCallBeRemoved := e.CanBeUnwrappedIfUnused
+
+		// A call that has been marked "__PURE__" can be removed if all arguments
+		// can be removed. The annotation causes us to ignore the target.
+		if canCallBeRemoved {
+			for _, arg := range e.Args {
+				if !ctx.ExprCanBeRemovedIfUnused(arg) {
+					return false
+				}
+			}
+			return true
+		}
+
+	case *ENew:
+		// A constructor call that has been marked "__PURE__" can be removed if all
+		// arguments can be removed. The annotation causes us to ignore the target.
+		if e.CanBeUnwrappedIfUnused {
+			for _, arg := range e.Args {
+				if !ctx.ExprCanBeRemovedIfUnused(arg) {
+					return false
+				}
+			}
+			return true
+		}
+
+	case *EUnary:
+		switch e.Op {
+		// These operators must not have any type conversions that can execute code
+		// such as "toString" or "valueOf". They must also never throw any exceptions.
+		case UnOpVoid, UnOpNot:
+			return ctx.ExprCanBeRemovedIfUnused(e.Value)
+
+		// The "typeof" operator doesn't do any type conversions so it can be removed
+		// if the result is unused and the operand has no side effects. However, it
+		// has a special case where if the operand is an identifier expression such
+		// as "typeof x" and "x" doesn't exist, no reference error is thrown so the
+		// operation has no side effects.
+		case UnOpTypeof:
+			if _, ok := e.Value.Data.(*EIdentifier); ok && e.WasOriginallyTypeofIdentifier {
+				// Expressions such as "typeof x" never have any side effects
+				return true
+			}
+			return ctx.ExprCanBeRemovedIfUnused(e.Value)
+		}
+
+	case *EBinary:
+		switch e.Op {
+		// These operators must not have any type conversions that can execute code
+		// such as "toString" or "valueOf". They must also never throw any exceptions.
+		case BinOpStrictEq, BinOpStrictNe, BinOpComma, BinOpNullishCoalescing:
+			return ctx.ExprCanBeRemovedIfUnused(e.Left) && ctx.ExprCanBeRemovedIfUnused(e.Right)
+
+		// Special-case "||" to make sure "typeof x === 'undefined' || x" can be removed
+		case BinOpLogicalOr:
+			return ctx.ExprCanBeRemovedIfUnused(e.Left) &&
+				(ctx.isSideEffectFreeUnboundIdentifierRef(e.Right, e.Left, false) || ctx.ExprCanBeRemovedIfUnused(e.Right))
+
+		// Special-case "&&" to make sure "typeof x !== 'undefined' && x" can be removed
+		case BinOpLogicalAnd:
+			return ctx.ExprCanBeRemovedIfUnused(e.Left) &&
+				(ctx.isSideEffectFreeUnboundIdentifierRef(e.Right, e.Left, true) || ctx.ExprCanBeRemovedIfUnused(e.Right))
+
+		// For "==" and "!=", pretend the operator was actually "===" or "!==". If
+		// we know that we can convert it to "==" or "!=", then we can consider the
+		// operator itself to have no side effects. This matters because our mangle
+		// logic will convert "typeof x === 'object'" into "typeof x == 'object'"
+		// and since "typeof x === 'object'" is considered to be side-effect free,
+		// we must also consider "typeof x == 'object'" to be side-effect free.
+		case BinOpLooseEq, BinOpLooseNe:
+			return CanChangeStrictToLoose(e.Left, e.Right) && ctx.ExprCanBeRemovedIfUnused(e.Left) && ctx.ExprCanBeRemovedIfUnused(e.Right)
+
+		// Special-case "<" and ">" with string, number, or bigint arguments
+		case BinOpLt, BinOpGt, BinOpLe, BinOpGe:
+			left := KnownPrimitiveType(e.Left.Data)
+			switch left {
+			case PrimitiveString, PrimitiveNumber, PrimitiveBigInt:
+				return KnownPrimitiveType(e.Right.Data) == left && ctx.ExprCanBeRemovedIfUnused(e.Left) && ctx.ExprCanBeRemovedIfUnused(e.Right)
+			}
+		}
+
+	case *ETemplate:
+		// A template can be removed if it has no tag and every value has no side
+		// effects and results in some kind of primitive, since all primitives
+		// have a "ToString" operation with no side effects.
+		if e.TagOrNil.Data == nil || e.CanBeUnwrappedIfUnused {
+			for _, part := range e.Parts {
+				if !ctx.ExprCanBeRemovedIfUnused(part.Value) || KnownPrimitiveType(part.Value.Data) == PrimitiveUnknown {
+					return false
+				}
+			}
+			return true
+		}
+	}
+
+	// Assume all other expression types have side effects and cannot be removed
+	return false
+}
+
+func (ctx HelperContext) isSideEffectFreeUnboundIdentifierRef(value Expr, guardCondition Expr, isYesBranch bool) bool {
+	if id, ok := value.Data.(*EIdentifier); ok && ctx.isUnbound(id.Ref) {
+		if binary, ok := guardCondition.Data.(*EBinary); ok {
+			switch binary.Op {
+			case BinOpStrictEq, BinOpStrictNe, BinOpLooseEq, BinOpLooseNe:
+				// Pattern match for "typeof x !== <string>"
+				typeof, string := binary.Left, binary.Right
+				if _, ok := typeof.Data.(*EString); ok {
+					typeof, string = string, typeof
+				}
+				if typeof, ok := typeof.Data.(*EUnary); ok && typeof.Op == UnOpTypeof && typeof.WasOriginallyTypeofIdentifier {
+					if text, ok := string.Data.(*EString); ok {
+						// In "typeof x !== 'undefined' ? x : null", the reference to "x" is side-effect free
+						// In "typeof x === 'object' ? x : null", the reference to "x" is side-effect free
+						if (helpers.UTF16EqualsString(text.Value, "undefined") == isYesBranch) ==
+							(binary.Op == BinOpStrictNe || binary.Op == BinOpLooseNe) {
+							if id2, ok := typeof.Value.Data.(*EIdentifier); ok && id2.Ref == id.Ref {
+								return true
+							}
+						}
+					}
+				}
+
+			case BinOpLt, BinOpGt, BinOpLe, BinOpGe:
+				// Pattern match for "typeof x < <string>"
+				typeof, string := binary.Left, binary.Right
+				if _, ok := typeof.Data.(*EString); ok {
+					typeof, string = string, typeof
+					isYesBranch = !isYesBranch
+				}
+				if typeof, ok := typeof.Data.(*EUnary); ok && typeof.Op == UnOpTypeof && typeof.WasOriginallyTypeofIdentifier {
+					if text, ok := string.Data.(*EString); ok && helpers.UTF16EqualsString(text.Value, "u") {
+						// In "typeof x < 'u' ? x : null", the reference to "x" is side-effect free
+						// In "typeof x > 'u' ? x : null", the reference to "x" is side-effect free
+						if isYesBranch == (binary.Op == BinOpLt || binary.Op == BinOpLe) {
+							if id2, ok := typeof.Value.Data.(*EIdentifier); ok && id2.Ref == id.Ref {
+								return true
+							}
+						}
+					}
+				}
+			}
+		}
+	}
+	return false
+}
+
+func StringToEquivalentNumberValue(value []uint16) (float64, bool) {
+	if len(value) > 0 {
+		var intValue int32
+		isNegative := false
+		start := 0
+
+		if value[0] == '-' && len(value) > 1 {
+			isNegative = true
+			start++
+		}
+
+		for _, c := range value[start:] {
+			if c < '0' || c > '9' {
+				return 0, false
+			}
+			intValue = intValue*10 + int32(c) - '0'
+		}
+
+		if isNegative {
+			intValue = -intValue
+		}
+
+		if helpers.UTF16EqualsString(value, strconv.FormatInt(int64(intValue), 10)) {
+			return float64(intValue), true
+		}
+	}
+
+	return 0, false
+}
+
+// This function intentionally avoids mutating the input AST so it can be
+// called after the AST has been frozen (i.e. after parsing ends).
+func InlineSpreadsOfArrayLiterals(values []Expr) (results []Expr) {
+	for _, value := range values {
+		if spread, ok := value.Data.(*ESpread); ok {
+			if array, ok := spread.Value.Data.(*EArray); ok {
+				for _, item := range array.Items {
+					if _, ok := item.Data.(*EMissing); ok {
+						results = append(results, Expr{Loc: item.Loc, Data: EUndefinedShared})
+					} else {
+						results = append(results, item)
+					}
+				}
+				continue
+			}
+		}
+		results = append(results, value)
+	}
+	return
+}
+
+// This function intentionally avoids mutating the input AST so it can be
+// called after the AST has been frozen (i.e. after parsing ends).
+func MangleObjectSpread(properties []Property) []Property {
+	var result []Property
+	for _, property := range properties {
+		if property.Kind == PropertySpread {
+			switch v := property.ValueOrNil.Data.(type) {
+			case *EBoolean, *ENull, *EUndefined, *ENumber,
+				*EBigInt, *ERegExp, *EFunction, *EArrow:
+				// This value is ignored because it doesn't have any of its own properties
+				continue
+
+			case *EObject:
+				for i, p := range v.Properties {
+					// Getters are evaluated at iteration time. The property
+					// descriptor is not inlined into the caller. Since we are not
+					// evaluating code at compile time, just bail if we hit one
+					// and preserve the spread with the remaining properties.
+					if p.Kind == PropertyGetter || p.Kind == PropertySetter {
+						// Don't mutate the original AST
+						clone := *v
+						clone.Properties = v.Properties[i:]
+						property.ValueOrNil.Data = &clone
+						result = append(result, property)
+						break
+					}
+
+					// Also bail if we hit a verbatim "__proto__" key. This will
+					// actually set the prototype of the object being spread so
+					// inlining it is not correct.
+					if p.Kind == PropertyField && !p.Flags.Has(PropertyIsComputed) {
+						if str, ok := p.Key.Data.(*EString); ok && helpers.UTF16EqualsString(str.Value, "__proto__") {
+							// Don't mutate the original AST
+							clone := *v
+							clone.Properties = v.Properties[i:]
+							property.ValueOrNil.Data = &clone
+							result = append(result, property)
+							break
+						}
+					}
+
+					result = append(result, p)
+				}
+				continue
+			}
+		}
+		result = append(result, property)
+	}
+	return result
+}
+
+// This function intentionally avoids mutating the input AST so it can be
+// called after the AST has been frozen (i.e. after parsing ends).
+func (ctx HelperContext) MangleIfExpr(loc logger.Loc, e *EIf, unsupportedFeatures compat.JSFeature) Expr {
+	test := e.Test
+	yes := e.Yes
+	no := e.No
+
+	// "(a, b) ? c : d" => "a, b ? c : d"
+	if comma, ok := test.Data.(*EBinary); ok && comma.Op == BinOpComma {
+		return JoinWithComma(comma.Left, ctx.MangleIfExpr(comma.Right.Loc, &EIf{
+			Test: comma.Right,
+			Yes:  yes,
+			No:   no,
+		}, unsupportedFeatures))
+	}
+
+	// "!a ? b : c" => "a ? c : b"
+	if not, ok := test.Data.(*EUnary); ok && not.Op == UnOpNot {
+		test = not.Value
+		yes, no = no, yes
+	}
+
+	if ValuesLookTheSame(yes.Data, no.Data) {
+		// "/* @__PURE__ */ a() ? b : b" => "b"
+		if ctx.ExprCanBeRemovedIfUnused(test) {
+			return yes
+		}
+
+		// "a ? b : b" => "a, b"
+		return JoinWithComma(test, yes)
+	}
+
+	// "a ? true : false" => "!!a"
+	// "a ? false : true" => "!a"
+	if y, ok := yes.Data.(*EBoolean); ok {
+		if n, ok := no.Data.(*EBoolean); ok {
+			if y.Value && !n.Value {
+				return Not(Not(test))
+			}
+			if !y.Value && n.Value {
+				return Not(test)
+			}
+		}
+	}
+
+	if id, ok := test.Data.(*EIdentifier); ok {
+		// "a ? a : b" => "a || b"
+		if id2, ok := yes.Data.(*EIdentifier); ok && id.Ref == id2.Ref {
+			return JoinWithLeftAssociativeOp(BinOpLogicalOr, test, no)
+		}
+
+		// "a ? b : a" => "a && b"
+		if id2, ok := no.Data.(*EIdentifier); ok && id.Ref == id2.Ref {
+			return JoinWithLeftAssociativeOp(BinOpLogicalAnd, test, yes)
+		}
+	}
+
+	// "a ? b ? c : d : d" => "a && b ? c : d"
+	if yesIf, ok := yes.Data.(*EIf); ok && ValuesLookTheSame(yesIf.No.Data, no.Data) {
+		return Expr{Loc: loc, Data: &EIf{Test: JoinWithLeftAssociativeOp(BinOpLogicalAnd, test, yesIf.Test), Yes: yesIf.Yes, No: no}}
+	}
+
+	// "a ? b : c ? b : d" => "a || c ? b : d"
+	if noIf, ok := no.Data.(*EIf); ok && ValuesLookTheSame(yes.Data, noIf.Yes.Data) {
+		return Expr{Loc: loc, Data: &EIf{Test: JoinWithLeftAssociativeOp(BinOpLogicalOr, test, noIf.Test), Yes: yes, No: noIf.No}}
+	}
+
+	// "a ? c : (b, c)" => "(a || b), c"
+	if comma, ok := no.Data.(*EBinary); ok && comma.Op == BinOpComma && ValuesLookTheSame(yes.Data, comma.Right.Data) {
+		return JoinWithComma(
+			JoinWithLeftAssociativeOp(BinOpLogicalOr, test, comma.Left),
+			comma.Right,
+		)
+	}
+
+	// "a ? (b, c) : c" => "(a && b), c"
+	if comma, ok := yes.Data.(*EBinary); ok && comma.Op == BinOpComma && ValuesLookTheSame(comma.Right.Data, no.Data) {
+		return JoinWithComma(
+			JoinWithLeftAssociativeOp(BinOpLogicalAnd, test, comma.Left),
+			comma.Right,
+		)
+	}
+
+	// "a ? b || c : c" => "(a && b) || c"
+	if binary, ok := yes.Data.(*EBinary); ok && binary.Op == BinOpLogicalOr &&
+		ValuesLookTheSame(binary.Right.Data, no.Data) {
+		return Expr{Loc: loc, Data: &EBinary{
+			Op:    BinOpLogicalOr,
+			Left:  JoinWithLeftAssociativeOp(BinOpLogicalAnd, test, binary.Left),
+			Right: binary.Right,
+		}}
+	}
+
+	// "a ? c : b && c" => "(a || b) && c"
+	if binary, ok := no.Data.(*EBinary); ok && binary.Op == BinOpLogicalAnd &&
+		ValuesLookTheSame(yes.Data, binary.Right.Data) {
+		return Expr{Loc: loc, Data: &EBinary{
+			Op:    BinOpLogicalAnd,
+			Left:  JoinWithLeftAssociativeOp(BinOpLogicalOr, test, binary.Left),
+			Right: binary.Right,
+		}}
+	}
+
+	// "a ? b(c, d) : b(e, d)" => "b(a ? c : e, d)"
+	if y, ok := yes.Data.(*ECall); ok && len(y.Args) > 0 {
+		if n, ok := no.Data.(*ECall); ok && len(n.Args) == len(y.Args) &&
+			y.HasSameFlagsAs(n) && ValuesLookTheSame(y.Target.Data, n.Target.Data) {
+			// Only do this if the condition can be reordered past the call target
+			// without side effects. For example, if the test or the call target is
+			// an unbound identifier, reordering could potentially mean evaluating
+			// the code could throw a different ReferenceError.
+			if ctx.ExprCanBeRemovedIfUnused(test) && ctx.ExprCanBeRemovedIfUnused(y.Target) {
+				sameTailArgs := true
+				for i, count := 1, len(y.Args); i < count; i++ {
+					if !ValuesLookTheSame(y.Args[i].Data, n.Args[i].Data) {
+						sameTailArgs = false
+						break
+					}
+				}
+				if sameTailArgs {
+					yesSpread, yesIsSpread := y.Args[0].Data.(*ESpread)
+					noSpread, noIsSpread := n.Args[0].Data.(*ESpread)
+
+					// "a ? b(...c) : b(...e)" => "b(...a ? c : e)"
+					if yesIsSpread && noIsSpread {
+						// Don't mutate the original AST
+						temp := EIf{Test: test, Yes: yesSpread.Value, No: noSpread.Value}
+						clone := *y
+						clone.Args = append([]Expr{}, clone.Args...)
+						clone.Args[0] = Expr{Loc: loc, Data: &ESpread{Value: ctx.MangleIfExpr(loc, &temp, unsupportedFeatures)}}
+						return Expr{Loc: loc, Data: &clone}
+					}
+
+					// "a ? b(c) : b(e)" => "b(a ? c : e)"
+					if !yesIsSpread && !noIsSpread {
+						// Don't mutate the original AST
+						temp := EIf{Test: test, Yes: y.Args[0], No: n.Args[0]}
+						clone := *y
+						clone.Args = append([]Expr{}, clone.Args...)
+						clone.Args[0] = ctx.MangleIfExpr(loc, &temp, unsupportedFeatures)
+						return Expr{Loc: loc, Data: &clone}
+					}
+				}
+			}
+		}
+	}
+
+	// Try using the "??" or "?." operators
+	if binary, ok := test.Data.(*EBinary); ok {
+		var check Expr
+		var whenNull Expr
+		var whenNonNull Expr
+
+		switch binary.Op {
+		case BinOpLooseEq:
+			if _, ok := binary.Right.Data.(*ENull); ok {
+				// "a == null ? _ : _"
+				check = binary.Left
+				whenNull = yes
+				whenNonNull = no
+			} else if _, ok := binary.Left.Data.(*ENull); ok {
+				// "null == a ? _ : _"
+				check = binary.Right
+				whenNull = yes
+				whenNonNull = no
+			}
+
+		case BinOpLooseNe:
+			if _, ok := binary.Right.Data.(*ENull); ok {
+				// "a != null ? _ : _"
+				check = binary.Left
+				whenNonNull = yes
+				whenNull = no
+			} else if _, ok := binary.Left.Data.(*ENull); ok {
+				// "null != a ? _ : _"
+				check = binary.Right
+				whenNonNull = yes
+				whenNull = no
+			}
+		}
+
+		if ctx.ExprCanBeRemovedIfUnused(check) {
+			// "a != null ? a : b" => "a ?? b"
+			if !unsupportedFeatures.Has(compat.NullishCoalescing) && ValuesLookTheSame(check.Data, whenNonNull.Data) {
+				return JoinWithLeftAssociativeOp(BinOpNullishCoalescing, check, whenNull)
+			}
+
+			// "a != null ? a.b.c[d](e) : undefined" => "a?.b.c[d](e)"
+			if !unsupportedFeatures.Has(compat.OptionalChain) {
+				if _, ok := whenNull.Data.(*EUndefined); ok && TryToInsertOptionalChain(check, whenNonNull) {
+					return whenNonNull
+				}
+			}
+		}
+	}
+
+	// Don't mutate the original AST
+	if test != e.Test || yes != e.Yes || no != e.No {
+		return Expr{Loc: loc, Data: &EIf{Test: test, Yes: yes, No: no}}
+	}
+
+	return Expr{Loc: loc, Data: e}
+}
+
+func ForEachIdentifierBindingInDecls(decls []Decl, callback func(loc logger.Loc, b *BIdentifier)) {
+	for _, decl := range decls {
+		ForEachIdentifierBinding(decl.Binding, callback)
+	}
+}
+
+func ForEachIdentifierBinding(binding Binding, callback func(loc logger.Loc, b *BIdentifier)) {
+	switch b := binding.Data.(type) {
+	case *BMissing:
+
+	case *BIdentifier:
+		callback(binding.Loc, b)
+
+	case *BArray:
+		for _, item := range b.Items {
+			ForEachIdentifierBinding(item.Binding, callback)
+		}
+
+	case *BObject:
+		for _, property := range b.Properties {
+			ForEachIdentifierBinding(property.Value, callback)
+		}
+
+	default:
+		panic("Internal error")
+	}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/js_ast/js_ident.go b/source/vendor/github.com/evanw/esbuild/internal/js_ast/js_ident.go
new file mode 100644
index 0000000..b1ff22d
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/js_ast/js_ident.go
@@ -0,0 +1,247 @@
+package js_ast
+
+import (
+	"strings"
+	"unicode"
+	"unicode/utf8"
+)
+
+func IsIdentifier(text string) bool {
+	if len(text) == 0 {
+		return false
+	}
+	for i, codePoint := range text {
+		if i == 0 {
+			if !IsIdentifierStart(codePoint) {
+				return false
+			}
+		} else {
+			if !IsIdentifierContinue(codePoint) {
+				return false
+			}
+		}
+	}
+	return true
+}
+
+func IsIdentifierES5AndESNext(text string) bool {
+	if len(text) == 0 {
+		return false
+	}
+	for i, codePoint := range text {
+		if i == 0 {
+			if !IsIdentifierStartES5AndESNext(codePoint) {
+				return false
+			}
+		} else {
+			if !IsIdentifierContinueES5AndESNext(codePoint) {
+				return false
+			}
+		}
+	}
+	return true
+}
+
+func ForceValidIdentifier(prefix string, text string) string {
+	sb := strings.Builder{}
+
+	// Private identifiers must be prefixed by "#"
+	if prefix != "" {
+		sb.WriteString(prefix)
+	}
+
+	// Identifier start
+	c, width := utf8.DecodeRuneInString(text)
+	text = text[width:]
+	if IsIdentifierStart(c) {
+		sb.WriteRune(c)
+	} else {
+		sb.WriteRune('_')
+	}
+
+	// Identifier continue
+	for text != "" {
+		c, width := utf8.DecodeRuneInString(text)
+		text = text[width:]
+		if IsIdentifierContinue(c) {
+			sb.WriteRune(c)
+		} else {
+			sb.WriteRune('_')
+		}
+	}
+
+	return sb.String()
+}
+
+// This does "IsIdentifier(UTF16ToString(text))" without any allocations
+func IsIdentifierUTF16(text []uint16) bool {
+	n := len(text)
+	if n == 0 {
+		return false
+	}
+	for i := 0; i < n; i++ {
+		isStart := i == 0
+		r1 := rune(text[i])
+		if r1 >= 0xD800 && r1 <= 0xDBFF && i+1 < n {
+			if r2 := rune(text[i+1]); r2 >= 0xDC00 && r2 <= 0xDFFF {
+				r1 = (r1 << 10) + r2 + (0x10000 - (0xD800 << 10) - 0xDC00)
+				i++
+			}
+		}
+		if isStart {
+			if !IsIdentifierStart(r1) {
+				return false
+			}
+		} else {
+			if !IsIdentifierContinue(r1) {
+				return false
+			}
+		}
+	}
+	return true
+}
+
+// This does "IsIdentifierES5AndESNext(UTF16ToString(text))" without any allocations
+func IsIdentifierES5AndESNextUTF16(text []uint16) bool {
+	n := len(text)
+	if n == 0 {
+		return false
+	}
+	for i := 0; i < n; i++ {
+		isStart := i == 0
+		r1 := rune(text[i])
+		if r1 >= 0xD800 && r1 <= 0xDBFF && i+1 < n {
+			if r2 := rune(text[i+1]); r2 >= 0xDC00 && r2 <= 0xDFFF {
+				r1 = (r1 << 10) + r2 + (0x10000 - (0xD800 << 10) - 0xDC00)
+				i++
+			}
+		}
+		if isStart {
+			if !IsIdentifierStartES5AndESNext(r1) {
+				return false
+			}
+		} else {
+			if !IsIdentifierContinueES5AndESNext(r1) {
+				return false
+			}
+		}
+	}
+	return true
+}
+
+func IsIdentifierStart(codePoint rune) bool {
+	switch codePoint {
+	case '_', '$',
+		'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm',
+		'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z',
+		'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M',
+		'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z':
+		return true
+	}
+
+	// All ASCII identifier start code points are listed above
+	if codePoint < 0x7F {
+		return false
+	}
+
+	return unicode.Is(idStartES5OrESNext, codePoint)
+}
+
+func IsIdentifierContinue(codePoint rune) bool {
+	switch codePoint {
+	case '_', '$', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
+		'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm',
+		'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z',
+		'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M',
+		'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z':
+		return true
+	}
+
+	// All ASCII identifier start code points are listed above
+	if codePoint < 0x7F {
+		return false
+	}
+
+	// ZWNJ and ZWJ are allowed in identifiers
+	if codePoint == 0x200C || codePoint == 0x200D {
+		return true
+	}
+
+	return unicode.Is(idContinueES5OrESNext, codePoint)
+}
+
+func IsIdentifierStartES5AndESNext(codePoint rune) bool {
+	switch codePoint {
+	case '_', '$',
+		'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm',
+		'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z',
+		'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M',
+		'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z':
+		return true
+	}
+
+	// All ASCII identifier start code points are listed above
+	if codePoint < 0x7F {
+		return false
+	}
+
+	return unicode.Is(idStartES5AndESNext, codePoint)
+}
+
+func IsIdentifierContinueES5AndESNext(codePoint rune) bool {
+	switch codePoint {
+	case '_', '$', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
+		'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm',
+		'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z',
+		'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M',
+		'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z':
+		return true
+	}
+
+	// All ASCII identifier start code points are listed above
+	if codePoint < 0x7F {
+		return false
+	}
+
+	// ZWNJ and ZWJ are allowed in identifiers
+	if codePoint == 0x200C || codePoint == 0x200D {
+		return true
+	}
+
+	return unicode.Is(idContinueES5AndESNext, codePoint)
+}
+
+// See the "White Space Code Points" table in the ECMAScript standard
+func IsWhitespace(codePoint rune) bool {
+	switch codePoint {
+	case
+		'\u0009', // character tabulation
+		'\u000B', // line tabulation
+		'\u000C', // form feed
+		'\u0020', // space
+		'\u00A0', // no-break space
+
+		// Unicode "Space_Separator" code points
+		'\u1680', // ogham space mark
+		'\u2000', // en quad
+		'\u2001', // em quad
+		'\u2002', // en space
+		'\u2003', // em space
+		'\u2004', // three-per-em space
+		'\u2005', // four-per-em space
+		'\u2006', // six-per-em space
+		'\u2007', // figure space
+		'\u2008', // punctuation space
+		'\u2009', // thin space
+		'\u200A', // hair space
+		'\u202F', // narrow no-break space
+		'\u205F', // medium mathematical space
+		'\u3000', // ideographic space
+
+		'\uFEFF': // zero width non-breaking space
+		return true
+
+	default:
+		return false
+	}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/js_ast/unicode.go b/source/vendor/github.com/evanw/esbuild/internal/js_ast/unicode.go
new file mode 100644
index 0000000..f1d6720
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/js_ast/unicode.go
@@ -0,0 +1,2065 @@
+// This file was automatically generated by gen-unicode-table.js. Do not edit.
+package js_ast
+
+import "unicode"
+
+var idStartES5AndESNext = &unicode.RangeTable{
+	LatinOffset: 117,
+	R16: []unicode.Range16{
+		{Lo: 0x41, Hi: 0x5a, Stride: 1},
+		{Lo: 0x61, Hi: 0x7a, Stride: 1},
+		{Lo: 0xaa, Hi: 0xaa, Stride: 1},
+		{Lo: 0xb5, Hi: 0xb5, Stride: 1},
+		{Lo: 0xba, Hi: 0xba, Stride: 1},
+		{Lo: 0xc0, Hi: 0xd6, Stride: 1},
+		{Lo: 0xd8, Hi: 0xf6, Stride: 1},
+		{Lo: 0xf8, Hi: 0x21f, Stride: 1},
+		{Lo: 0x222, Hi: 0x233, Stride: 1},
+		{Lo: 0x250, Hi: 0x2ad, Stride: 1},
+		{Lo: 0x2b0, Hi: 0x2b8, Stride: 1},
+		{Lo: 0x2bb, Hi: 0x2c1, Stride: 1},
+		{Lo: 0x2d0, Hi: 0x2d1, Stride: 1},
+		{Lo: 0x2e0, Hi: 0x2e4, Stride: 1},
+		{Lo: 0x2ee, Hi: 0x2ee, Stride: 1},
+		{Lo: 0x37a, Hi: 0x37a, Stride: 1},
+		{Lo: 0x386, Hi: 0x386, Stride: 1},
+		{Lo: 0x388, Hi: 0x38a, Stride: 1},
+		{Lo: 0x38c, Hi: 0x38c, Stride: 1},
+		{Lo: 0x38e, Hi: 0x3a1, Stride: 1},
+		{Lo: 0x3a3, Hi: 0x3ce, Stride: 1},
+		{Lo: 0x3d0, Hi: 0x3d7, Stride: 1},
+		{Lo: 0x3da, Hi: 0x3f3, Stride: 1},
+		{Lo: 0x400, Hi: 0x481, Stride: 1},
+		{Lo: 0x48c, Hi: 0x4c4, Stride: 1},
+		{Lo: 0x4c7, Hi: 0x4c8, Stride: 1},
+		{Lo: 0x4cb, Hi: 0x4cc, Stride: 1},
+		{Lo: 0x4d0, Hi: 0x4f5, Stride: 1},
+		{Lo: 0x4f8, Hi: 0x4f9, Stride: 1},
+		{Lo: 0x531, Hi: 0x556, Stride: 1},
+		{Lo: 0x559, Hi: 0x559, Stride: 1},
+		{Lo: 0x561, Hi: 0x587, Stride: 1},
+		{Lo: 0x5d0, Hi: 0x5ea, Stride: 1},
+		{Lo: 0x5f0, Hi: 0x5f2, Stride: 1},
+		{Lo: 0x621, Hi: 0x63a, Stride: 1},
+		{Lo: 0x640, Hi: 0x64a, Stride: 1},
+		{Lo: 0x671, Hi: 0x6d3, Stride: 1},
+		{Lo: 0x6d5, Hi: 0x6d5, Stride: 1},
+		{Lo: 0x6e5, Hi: 0x6e6, Stride: 1},
+		{Lo: 0x6fa, Hi: 0x6fc, Stride: 1},
+		{Lo: 0x710, Hi: 0x710, Stride: 1},
+		{Lo: 0x712, Hi: 0x72c, Stride: 1},
+		{Lo: 0x780, Hi: 0x7a5, Stride: 1},
+		{Lo: 0x905, Hi: 0x939, Stride: 1},
+		{Lo: 0x93d, Hi: 0x93d, Stride: 1},
+		{Lo: 0x950, Hi: 0x950, Stride: 1},
+		{Lo: 0x958, Hi: 0x961, Stride: 1},
+		{Lo: 0x985, Hi: 0x98c, Stride: 1},
+		{Lo: 0x98f, Hi: 0x990, Stride: 1},
+		{Lo: 0x993, Hi: 0x9a8, Stride: 1},
+		{Lo: 0x9aa, Hi: 0x9b0, Stride: 1},
+		{Lo: 0x9b2, Hi: 0x9b2, Stride: 1},
+		{Lo: 0x9b6, Hi: 0x9b9, Stride: 1},
+		{Lo: 0x9dc, Hi: 0x9dd, Stride: 1},
+		{Lo: 0x9df, Hi: 0x9e1, Stride: 1},
+		{Lo: 0x9f0, Hi: 0x9f1, Stride: 1},
+		{Lo: 0xa05, Hi: 0xa0a, Stride: 1},
+		{Lo: 0xa0f, Hi: 0xa10, Stride: 1},
+		{Lo: 0xa13, Hi: 0xa28, Stride: 1},
+		{Lo: 0xa2a, Hi: 0xa30, Stride: 1},
+		{Lo: 0xa32, Hi: 0xa33, Stride: 1},
+		{Lo: 0xa35, Hi: 0xa36, Stride: 1},
+		{Lo: 0xa38, Hi: 0xa39, Stride: 1},
+		{Lo: 0xa59, Hi: 0xa5c, Stride: 1},
+		{Lo: 0xa5e, Hi: 0xa5e, Stride: 1},
+		{Lo: 0xa72, Hi: 0xa74, Stride: 1},
+		{Lo: 0xa85, Hi: 0xa8b, Stride: 1},
+		{Lo: 0xa8d, Hi: 0xa8d, Stride: 1},
+		{Lo: 0xa8f, Hi: 0xa91, Stride: 1},
+		{Lo: 0xa93, Hi: 0xaa8, Stride: 1},
+		{Lo: 0xaaa, Hi: 0xab0, Stride: 1},
+		{Lo: 0xab2, Hi: 0xab3, Stride: 1},
+		{Lo: 0xab5, Hi: 0xab9, Stride: 1},
+		{Lo: 0xabd, Hi: 0xabd, Stride: 1},
+		{Lo: 0xad0, Hi: 0xad0, Stride: 1},
+		{Lo: 0xae0, Hi: 0xae0, Stride: 1},
+		{Lo: 0xb05, Hi: 0xb0c, Stride: 1},
+		{Lo: 0xb0f, Hi: 0xb10, Stride: 1},
+		{Lo: 0xb13, Hi: 0xb28, Stride: 1},
+		{Lo: 0xb2a, Hi: 0xb30, Stride: 1},
+		{Lo: 0xb32, Hi: 0xb33, Stride: 1},
+		{Lo: 0xb36, Hi: 0xb39, Stride: 1},
+		{Lo: 0xb3d, Hi: 0xb3d, Stride: 1},
+		{Lo: 0xb5c, Hi: 0xb5d, Stride: 1},
+		{Lo: 0xb5f, Hi: 0xb61, Stride: 1},
+		{Lo: 0xb85, Hi: 0xb8a, Stride: 1},
+		{Lo: 0xb8e, Hi: 0xb90, Stride: 1},
+		{Lo: 0xb92, Hi: 0xb95, Stride: 1},
+		{Lo: 0xb99, Hi: 0xb9a, Stride: 1},
+		{Lo: 0xb9c, Hi: 0xb9c, Stride: 1},
+		{Lo: 0xb9e, Hi: 0xb9f, Stride: 1},
+		{Lo: 0xba3, Hi: 0xba4, Stride: 1},
+		{Lo: 0xba8, Hi: 0xbaa, Stride: 1},
+		{Lo: 0xbae, Hi: 0xbb5, Stride: 1},
+		{Lo: 0xbb7, Hi: 0xbb9, Stride: 1},
+		{Lo: 0xc05, Hi: 0xc0c, Stride: 1},
+		{Lo: 0xc0e, Hi: 0xc10, Stride: 1},
+		{Lo: 0xc12, Hi: 0xc28, Stride: 1},
+		{Lo: 0xc2a, Hi: 0xc33, Stride: 1},
+		{Lo: 0xc35, Hi: 0xc39, Stride: 1},
+		{Lo: 0xc60, Hi: 0xc61, Stride: 1},
+		{Lo: 0xc85, Hi: 0xc8c, Stride: 1},
+		{Lo: 0xc8e, Hi: 0xc90, Stride: 1},
+		{Lo: 0xc92, Hi: 0xca8, Stride: 1},
+		{Lo: 0xcaa, Hi: 0xcb3, Stride: 1},
+		{Lo: 0xcb5, Hi: 0xcb9, Stride: 1},
+		{Lo: 0xcde, Hi: 0xcde, Stride: 1},
+		{Lo: 0xce0, Hi: 0xce1, Stride: 1},
+		{Lo: 0xd05, Hi: 0xd0c, Stride: 1},
+		{Lo: 0xd0e, Hi: 0xd10, Stride: 1},
+		{Lo: 0xd12, Hi: 0xd28, Stride: 1},
+		{Lo: 0xd2a, Hi: 0xd39, Stride: 1},
+		{Lo: 0xd60, Hi: 0xd61, Stride: 1},
+		{Lo: 0xd85, Hi: 0xd96, Stride: 1},
+		{Lo: 0xd9a, Hi: 0xdb1, Stride: 1},
+		{Lo: 0xdb3, Hi: 0xdbb, Stride: 1},
+		{Lo: 0xdbd, Hi: 0xdbd, Stride: 1},
+		{Lo: 0xdc0, Hi: 0xdc6, Stride: 1},
+		{Lo: 0xe01, Hi: 0xe30, Stride: 1},
+		{Lo: 0xe32, Hi: 0xe33, Stride: 1},
+		{Lo: 0xe40, Hi: 0xe46, Stride: 1},
+		{Lo: 0xe81, Hi: 0xe82, Stride: 1},
+		{Lo: 0xe84, Hi: 0xe84, Stride: 1},
+		{Lo: 0xe87, Hi: 0xe88, Stride: 1},
+		{Lo: 0xe8a, Hi: 0xe8a, Stride: 1},
+		{Lo: 0xe8d, Hi: 0xe8d, Stride: 1},
+		{Lo: 0xe94, Hi: 0xe97, Stride: 1},
+		{Lo: 0xe99, Hi: 0xe9f, Stride: 1},
+		{Lo: 0xea1, Hi: 0xea3, Stride: 1},
+		{Lo: 0xea5, Hi: 0xea5, Stride: 1},
+		{Lo: 0xea7, Hi: 0xea7, Stride: 1},
+		{Lo: 0xeaa, Hi: 0xeab, Stride: 1},
+		{Lo: 0xead, Hi: 0xeb0, Stride: 1},
+		{Lo: 0xeb2, Hi: 0xeb3, Stride: 1},
+		{Lo: 0xebd, Hi: 0xebd, Stride: 1},
+		{Lo: 0xec0, Hi: 0xec4, Stride: 1},
+		{Lo: 0xec6, Hi: 0xec6, Stride: 1},
+		{Lo: 0xedc, Hi: 0xedd, Stride: 1},
+		{Lo: 0xf00, Hi: 0xf00, Stride: 1},
+		{Lo: 0xf40, Hi: 0xf47, Stride: 1},
+		{Lo: 0xf49, Hi: 0xf6a, Stride: 1},
+		{Lo: 0xf88, Hi: 0xf8b, Stride: 1},
+	},
+	R32: []unicode.Range32{
+		{Lo: 0x1000, Hi: 0x1021, Stride: 1},
+		{Lo: 0x1023, Hi: 0x1027, Stride: 1},
+		{Lo: 0x1029, Hi: 0x102a, Stride: 1},
+		{Lo: 0x1050, Hi: 0x1055, Stride: 1},
+		{Lo: 0x10a0, Hi: 0x10c5, Stride: 1},
+		{Lo: 0x10d0, Hi: 0x10f6, Stride: 1},
+		{Lo: 0x1100, Hi: 0x1159, Stride: 1},
+		{Lo: 0x115f, Hi: 0x11a2, Stride: 1},
+		{Lo: 0x11a8, Hi: 0x11f9, Stride: 1},
+		{Lo: 0x1200, Hi: 0x1206, Stride: 1},
+		{Lo: 0x1208, Hi: 0x1246, Stride: 1},
+		{Lo: 0x1248, Hi: 0x1248, Stride: 1},
+		{Lo: 0x124a, Hi: 0x124d, Stride: 1},
+		{Lo: 0x1250, Hi: 0x1256, Stride: 1},
+		{Lo: 0x1258, Hi: 0x1258, Stride: 1},
+		{Lo: 0x125a, Hi: 0x125d, Stride: 1},
+		{Lo: 0x1260, Hi: 0x1286, Stride: 1},
+		{Lo: 0x1288, Hi: 0x1288, Stride: 1},
+		{Lo: 0x128a, Hi: 0x128d, Stride: 1},
+		{Lo: 0x1290, Hi: 0x12ae, Stride: 1},
+		{Lo: 0x12b0, Hi: 0x12b0, Stride: 1},
+		{Lo: 0x12b2, Hi: 0x12b5, Stride: 1},
+		{Lo: 0x12b8, Hi: 0x12be, Stride: 1},
+		{Lo: 0x12c0, Hi: 0x12c0, Stride: 1},
+		{Lo: 0x12c2, Hi: 0x12c5, Stride: 1},
+		{Lo: 0x12c8, Hi: 0x12ce, Stride: 1},
+		{Lo: 0x12d0, Hi: 0x12d6, Stride: 1},
+		{Lo: 0x12d8, Hi: 0x12ee, Stride: 1},
+		{Lo: 0x12f0, Hi: 0x130e, Stride: 1},
+		{Lo: 0x1310, Hi: 0x1310, Stride: 1},
+		{Lo: 0x1312, Hi: 0x1315, Stride: 1},
+		{Lo: 0x1318, Hi: 0x131e, Stride: 1},
+		{Lo: 0x1320, Hi: 0x1346, Stride: 1},
+		{Lo: 0x1348, Hi: 0x135a, Stride: 1},
+		{Lo: 0x13a0, Hi: 0x13f4, Stride: 1},
+		{Lo: 0x1401, Hi: 0x166c, Stride: 1},
+		{Lo: 0x166f, Hi: 0x1676, Stride: 1},
+		{Lo: 0x1681, Hi: 0x169a, Stride: 1},
+		{Lo: 0x16a0, Hi: 0x16ea, Stride: 1},
+		{Lo: 0x1780, Hi: 0x17b3, Stride: 1},
+		{Lo: 0x1820, Hi: 0x1877, Stride: 1},
+		{Lo: 0x1880, Hi: 0x18a8, Stride: 1},
+		{Lo: 0x1e00, Hi: 0x1e9b, Stride: 1},
+		{Lo: 0x1ea0, Hi: 0x1ef9, Stride: 1},
+		{Lo: 0x1f00, Hi: 0x1f15, Stride: 1},
+		{Lo: 0x1f18, Hi: 0x1f1d, Stride: 1},
+		{Lo: 0x1f20, Hi: 0x1f45, Stride: 1},
+		{Lo: 0x1f48, Hi: 0x1f4d, Stride: 1},
+		{Lo: 0x1f50, Hi: 0x1f57, Stride: 1},
+		{Lo: 0x1f59, Hi: 0x1f59, Stride: 1},
+		{Lo: 0x1f5b, Hi: 0x1f5b, Stride: 1},
+		{Lo: 0x1f5d, Hi: 0x1f5d, Stride: 1},
+		{Lo: 0x1f5f, Hi: 0x1f7d, Stride: 1},
+		{Lo: 0x1f80, Hi: 0x1fb4, Stride: 1},
+		{Lo: 0x1fb6, Hi: 0x1fbc, Stride: 1},
+		{Lo: 0x1fbe, Hi: 0x1fbe, Stride: 1},
+		{Lo: 0x1fc2, Hi: 0x1fc4, Stride: 1},
+		{Lo: 0x1fc6, Hi: 0x1fcc, Stride: 1},
+		{Lo: 0x1fd0, Hi: 0x1fd3, Stride: 1},
+		{Lo: 0x1fd6, Hi: 0x1fdb, Stride: 1},
+		{Lo: 0x1fe0, Hi: 0x1fec, Stride: 1},
+		{Lo: 0x1ff2, Hi: 0x1ff4, Stride: 1},
+		{Lo: 0x1ff6, Hi: 0x1ffc, Stride: 1},
+		{Lo: 0x207f, Hi: 0x207f, Stride: 1},
+		{Lo: 0x2102, Hi: 0x2102, Stride: 1},
+		{Lo: 0x2107, Hi: 0x2107, Stride: 1},
+		{Lo: 0x210a, Hi: 0x2113, Stride: 1},
+		{Lo: 0x2115, Hi: 0x2115, Stride: 1},
+		{Lo: 0x2119, Hi: 0x211d, Stride: 1},
+		{Lo: 0x2124, Hi: 0x2124, Stride: 1},
+		{Lo: 0x2126, Hi: 0x2126, Stride: 1},
+		{Lo: 0x2128, Hi: 0x2128, Stride: 1},
+		{Lo: 0x212a, Hi: 0x212d, Stride: 1},
+		{Lo: 0x212f, Hi: 0x2131, Stride: 1},
+		{Lo: 0x2133, Hi: 0x2139, Stride: 1},
+		{Lo: 0x3005, Hi: 0x3006, Stride: 1},
+		{Lo: 0x3031, Hi: 0x3035, Stride: 1},
+		{Lo: 0x3041, Hi: 0x3094, Stride: 1},
+		{Lo: 0x309d, Hi: 0x309e, Stride: 1},
+		{Lo: 0x30a1, Hi: 0x30fa, Stride: 1},
+		{Lo: 0x30fc, Hi: 0x30fe, Stride: 1},
+		{Lo: 0x3105, Hi: 0x312c, Stride: 1},
+		{Lo: 0x3131, Hi: 0x318e, Stride: 1},
+		{Lo: 0x31a0, Hi: 0x31b7, Stride: 1},
+		{Lo: 0x3400, Hi: 0x4db5, Stride: 1},
+		{Lo: 0x4e00, Hi: 0x9fa5, Stride: 1},
+		{Lo: 0xa000, Hi: 0xa48c, Stride: 1},
+		{Lo: 0xac00, Hi: 0xd7a3, Stride: 1},
+		{Lo: 0xf900, Hi: 0xfa2d, Stride: 1},
+		{Lo: 0xfb00, Hi: 0xfb06, Stride: 1},
+		{Lo: 0xfb13, Hi: 0xfb17, Stride: 1},
+		{Lo: 0xfb1d, Hi: 0xfb1d, Stride: 1},
+		{Lo: 0xfb1f, Hi: 0xfb28, Stride: 1},
+		{Lo: 0xfb2a, Hi: 0xfb36, Stride: 1},
+		{Lo: 0xfb38, Hi: 0xfb3c, Stride: 1},
+		{Lo: 0xfb3e, Hi: 0xfb3e, Stride: 1},
+		{Lo: 0xfb40, Hi: 0xfb41, Stride: 1},
+		{Lo: 0xfb43, Hi: 0xfb44, Stride: 1},
+		{Lo: 0xfb46, Hi: 0xfbb1, Stride: 1},
+		{Lo: 0xfbd3, Hi: 0xfd3d, Stride: 1},
+		{Lo: 0xfd50, Hi: 0xfd8f, Stride: 1},
+		{Lo: 0xfd92, Hi: 0xfdc7, Stride: 1},
+		{Lo: 0xfdf0, Hi: 0xfdfb, Stride: 1},
+		{Lo: 0xfe70, Hi: 0xfe72, Stride: 1},
+		{Lo: 0xfe74, Hi: 0xfe74, Stride: 1},
+		{Lo: 0xfe76, Hi: 0xfefc, Stride: 1},
+		{Lo: 0xff21, Hi: 0xff3a, Stride: 1},
+		{Lo: 0xff41, Hi: 0xff5a, Stride: 1},
+		{Lo: 0xff66, Hi: 0xffbe, Stride: 1},
+		{Lo: 0xffc2, Hi: 0xffc7, Stride: 1},
+		{Lo: 0xffca, Hi: 0xffcf, Stride: 1},
+		{Lo: 0xffd2, Hi: 0xffd7, Stride: 1},
+		{Lo: 0xffda, Hi: 0xffdc, Stride: 1},
+	},
+}
+
+var idContinueES5AndESNext = &unicode.RangeTable{
+	LatinOffset: 128,
+	R16: []unicode.Range16{
+		{Lo: 0x30, Hi: 0x39, Stride: 1},
+		{Lo: 0x41, Hi: 0x5a, Stride: 1},
+		{Lo: 0x5f, Hi: 0x5f, Stride: 1},
+		{Lo: 0x61, Hi: 0x7a, Stride: 1},
+		{Lo: 0xaa, Hi: 0xaa, Stride: 1},
+		{Lo: 0xb5, Hi: 0xb5, Stride: 1},
+		{Lo: 0xba, Hi: 0xba, Stride: 1},
+		{Lo: 0xc0, Hi: 0xd6, Stride: 1},
+		{Lo: 0xd8, Hi: 0xf6, Stride: 1},
+		{Lo: 0xf8, Hi: 0x21f, Stride: 1},
+		{Lo: 0x222, Hi: 0x233, Stride: 1},
+		{Lo: 0x250, Hi: 0x2ad, Stride: 1},
+		{Lo: 0x2b0, Hi: 0x2b8, Stride: 1},
+		{Lo: 0x2bb, Hi: 0x2c1, Stride: 1},
+		{Lo: 0x2d0, Hi: 0x2d1, Stride: 1},
+		{Lo: 0x2e0, Hi: 0x2e4, Stride: 1},
+		{Lo: 0x2ee, Hi: 0x2ee, Stride: 1},
+		{Lo: 0x300, Hi: 0x34e, Stride: 1},
+		{Lo: 0x360, Hi: 0x362, Stride: 1},
+		{Lo: 0x37a, Hi: 0x37a, Stride: 1},
+		{Lo: 0x386, Hi: 0x386, Stride: 1},
+		{Lo: 0x388, Hi: 0x38a, Stride: 1},
+		{Lo: 0x38c, Hi: 0x38c, Stride: 1},
+		{Lo: 0x38e, Hi: 0x3a1, Stride: 1},
+		{Lo: 0x3a3, Hi: 0x3ce, Stride: 1},
+		{Lo: 0x3d0, Hi: 0x3d7, Stride: 1},
+		{Lo: 0x3da, Hi: 0x3f3, Stride: 1},
+		{Lo: 0x400, Hi: 0x481, Stride: 1},
+		{Lo: 0x483, Hi: 0x486, Stride: 1},
+		{Lo: 0x48c, Hi: 0x4c4, Stride: 1},
+		{Lo: 0x4c7, Hi: 0x4c8, Stride: 1},
+		{Lo: 0x4cb, Hi: 0x4cc, Stride: 1},
+		{Lo: 0x4d0, Hi: 0x4f5, Stride: 1},
+		{Lo: 0x4f8, Hi: 0x4f9, Stride: 1},
+		{Lo: 0x531, Hi: 0x556, Stride: 1},
+		{Lo: 0x559, Hi: 0x559, Stride: 1},
+		{Lo: 0x561, Hi: 0x587, Stride: 1},
+		{Lo: 0x591, Hi: 0x5a1, Stride: 1},
+		{Lo: 0x5a3, Hi: 0x5b9, Stride: 1},
+		{Lo: 0x5bb, Hi: 0x5bd, Stride: 1},
+		{Lo: 0x5bf, Hi: 0x5bf, Stride: 1},
+		{Lo: 0x5c1, Hi: 0x5c2, Stride: 1},
+		{Lo: 0x5c4, Hi: 0x5c4, Stride: 1},
+		{Lo: 0x5d0, Hi: 0x5ea, Stride: 1},
+		{Lo: 0x5f0, Hi: 0x5f2, Stride: 1},
+		{Lo: 0x621, Hi: 0x63a, Stride: 1},
+		{Lo: 0x640, Hi: 0x655, Stride: 1},
+		{Lo: 0x660, Hi: 0x669, Stride: 1},
+		{Lo: 0x670, Hi: 0x6d3, Stride: 1},
+		{Lo: 0x6d5, Hi: 0x6dc, Stride: 1},
+		{Lo: 0x6df, Hi: 0x6e8, Stride: 1},
+		{Lo: 0x6ea, Hi: 0x6ed, Stride: 1},
+		{Lo: 0x6f0, Hi: 0x6fc, Stride: 1},
+		{Lo: 0x710, Hi: 0x72c, Stride: 1},
+		{Lo: 0x730, Hi: 0x74a, Stride: 1},
+		{Lo: 0x780, Hi: 0x7b0, Stride: 1},
+		{Lo: 0x901, Hi: 0x903, Stride: 1},
+		{Lo: 0x905, Hi: 0x939, Stride: 1},
+		{Lo: 0x93c, Hi: 0x94d, Stride: 1},
+		{Lo: 0x950, Hi: 0x954, Stride: 1},
+		{Lo: 0x958, Hi: 0x963, Stride: 1},
+		{Lo: 0x966, Hi: 0x96f, Stride: 1},
+		{Lo: 0x981, Hi: 0x983, Stride: 1},
+		{Lo: 0x985, Hi: 0x98c, Stride: 1},
+		{Lo: 0x98f, Hi: 0x990, Stride: 1},
+		{Lo: 0x993, Hi: 0x9a8, Stride: 1},
+		{Lo: 0x9aa, Hi: 0x9b0, Stride: 1},
+		{Lo: 0x9b2, Hi: 0x9b2, Stride: 1},
+		{Lo: 0x9b6, Hi: 0x9b9, Stride: 1},
+		{Lo: 0x9bc, Hi: 0x9bc, Stride: 1},
+		{Lo: 0x9be, Hi: 0x9c4, Stride: 1},
+		{Lo: 0x9c7, Hi: 0x9c8, Stride: 1},
+		{Lo: 0x9cb, Hi: 0x9cd, Stride: 1},
+		{Lo: 0x9d7, Hi: 0x9d7, Stride: 1},
+		{Lo: 0x9dc, Hi: 0x9dd, Stride: 1},
+		{Lo: 0x9df, Hi: 0x9e3, Stride: 1},
+		{Lo: 0x9e6, Hi: 0x9f1, Stride: 1},
+		{Lo: 0xa02, Hi: 0xa02, Stride: 1},
+		{Lo: 0xa05, Hi: 0xa0a, Stride: 1},
+		{Lo: 0xa0f, Hi: 0xa10, Stride: 1},
+		{Lo: 0xa13, Hi: 0xa28, Stride: 1},
+		{Lo: 0xa2a, Hi: 0xa30, Stride: 1},
+		{Lo: 0xa32, Hi: 0xa33, Stride: 1},
+		{Lo: 0xa35, Hi: 0xa36, Stride: 1},
+		{Lo: 0xa38, Hi: 0xa39, Stride: 1},
+		{Lo: 0xa3c, Hi: 0xa3c, Stride: 1},
+		{Lo: 0xa3e, Hi: 0xa42, Stride: 1},
+		{Lo: 0xa47, Hi: 0xa48, Stride: 1},
+		{Lo: 0xa4b, Hi: 0xa4d, Stride: 1},
+		{Lo: 0xa59, Hi: 0xa5c, Stride: 1},
+		{Lo: 0xa5e, Hi: 0xa5e, Stride: 1},
+		{Lo: 0xa66, Hi: 0xa74, Stride: 1},
+		{Lo: 0xa81, Hi: 0xa83, Stride: 1},
+		{Lo: 0xa85, Hi: 0xa8b, Stride: 1},
+		{Lo: 0xa8d, Hi: 0xa8d, Stride: 1},
+		{Lo: 0xa8f, Hi: 0xa91, Stride: 1},
+		{Lo: 0xa93, Hi: 0xaa8, Stride: 1},
+		{Lo: 0xaaa, Hi: 0xab0, Stride: 1},
+		{Lo: 0xab2, Hi: 0xab3, Stride: 1},
+		{Lo: 0xab5, Hi: 0xab9, Stride: 1},
+		{Lo: 0xabc, Hi: 0xac5, Stride: 1},
+		{Lo: 0xac7, Hi: 0xac9, Stride: 1},
+		{Lo: 0xacb, Hi: 0xacd, Stride: 1},
+		{Lo: 0xad0, Hi: 0xad0, Stride: 1},
+		{Lo: 0xae0, Hi: 0xae0, Stride: 1},
+		{Lo: 0xae6, Hi: 0xaef, Stride: 1},
+		{Lo: 0xb01, Hi: 0xb03, Stride: 1},
+		{Lo: 0xb05, Hi: 0xb0c, Stride: 1},
+		{Lo: 0xb0f, Hi: 0xb10, Stride: 1},
+		{Lo: 0xb13, Hi: 0xb28, Stride: 1},
+		{Lo: 0xb2a, Hi: 0xb30, Stride: 1},
+		{Lo: 0xb32, Hi: 0xb33, Stride: 1},
+		{Lo: 0xb36, Hi: 0xb39, Stride: 1},
+		{Lo: 0xb3c, Hi: 0xb43, Stride: 1},
+		{Lo: 0xb47, Hi: 0xb48, Stride: 1},
+		{Lo: 0xb4b, Hi: 0xb4d, Stride: 1},
+		{Lo: 0xb56, Hi: 0xb57, Stride: 1},
+		{Lo: 0xb5c, Hi: 0xb5d, Stride: 1},
+		{Lo: 0xb5f, Hi: 0xb61, Stride: 1},
+		{Lo: 0xb66, Hi: 0xb6f, Stride: 1},
+		{Lo: 0xb82, Hi: 0xb83, Stride: 1},
+		{Lo: 0xb85, Hi: 0xb8a, Stride: 1},
+		{Lo: 0xb8e, Hi: 0xb90, Stride: 1},
+		{Lo: 0xb92, Hi: 0xb95, Stride: 1},
+		{Lo: 0xb99, Hi: 0xb9a, Stride: 1},
+		{Lo: 0xb9c, Hi: 0xb9c, Stride: 1},
+		{Lo: 0xb9e, Hi: 0xb9f, Stride: 1},
+		{Lo: 0xba3, Hi: 0xba4, Stride: 1},
+		{Lo: 0xba8, Hi: 0xbaa, Stride: 1},
+		{Lo: 0xbae, Hi: 0xbb5, Stride: 1},
+		{Lo: 0xbb7, Hi: 0xbb9, Stride: 1},
+		{Lo: 0xbbe, Hi: 0xbc2, Stride: 1},
+		{Lo: 0xbc6, Hi: 0xbc8, Stride: 1},
+		{Lo: 0xbca, Hi: 0xbcd, Stride: 1},
+		{Lo: 0xbd7, Hi: 0xbd7, Stride: 1},
+		{Lo: 0xbe7, Hi: 0xbef, Stride: 1},
+		{Lo: 0xc01, Hi: 0xc03, Stride: 1},
+		{Lo: 0xc05, Hi: 0xc0c, Stride: 1},
+		{Lo: 0xc0e, Hi: 0xc10, Stride: 1},
+		{Lo: 0xc12, Hi: 0xc28, Stride: 1},
+		{Lo: 0xc2a, Hi: 0xc33, Stride: 1},
+		{Lo: 0xc35, Hi: 0xc39, Stride: 1},
+		{Lo: 0xc3e, Hi: 0xc44, Stride: 1},
+		{Lo: 0xc46, Hi: 0xc48, Stride: 1},
+		{Lo: 0xc4a, Hi: 0xc4d, Stride: 1},
+		{Lo: 0xc55, Hi: 0xc56, Stride: 1},
+		{Lo: 0xc60, Hi: 0xc61, Stride: 1},
+		{Lo: 0xc66, Hi: 0xc6f, Stride: 1},
+		{Lo: 0xc82, Hi: 0xc83, Stride: 1},
+		{Lo: 0xc85, Hi: 0xc8c, Stride: 1},
+		{Lo: 0xc8e, Hi: 0xc90, Stride: 1},
+		{Lo: 0xc92, Hi: 0xca8, Stride: 1},
+		{Lo: 0xcaa, Hi: 0xcb3, Stride: 1},
+		{Lo: 0xcb5, Hi: 0xcb9, Stride: 1},
+		{Lo: 0xcbe, Hi: 0xcc4, Stride: 1},
+		{Lo: 0xcc6, Hi: 0xcc8, Stride: 1},
+		{Lo: 0xcca, Hi: 0xccd, Stride: 1},
+		{Lo: 0xcd5, Hi: 0xcd6, Stride: 1},
+		{Lo: 0xcde, Hi: 0xcde, Stride: 1},
+		{Lo: 0xce0, Hi: 0xce1, Stride: 1},
+		{Lo: 0xce6, Hi: 0xcef, Stride: 1},
+		{Lo: 0xd02, Hi: 0xd03, Stride: 1},
+		{Lo: 0xd05, Hi: 0xd0c, Stride: 1},
+		{Lo: 0xd0e, Hi: 0xd10, Stride: 1},
+		{Lo: 0xd12, Hi: 0xd28, Stride: 1},
+		{Lo: 0xd2a, Hi: 0xd39, Stride: 1},
+		{Lo: 0xd3e, Hi: 0xd43, Stride: 1},
+		{Lo: 0xd46, Hi: 0xd48, Stride: 1},
+		{Lo: 0xd4a, Hi: 0xd4d, Stride: 1},
+		{Lo: 0xd57, Hi: 0xd57, Stride: 1},
+		{Lo: 0xd60, Hi: 0xd61, Stride: 1},
+		{Lo: 0xd66, Hi: 0xd6f, Stride: 1},
+		{Lo: 0xd82, Hi: 0xd83, Stride: 1},
+		{Lo: 0xd85, Hi: 0xd96, Stride: 1},
+		{Lo: 0xd9a, Hi: 0xdb1, Stride: 1},
+		{Lo: 0xdb3, Hi: 0xdbb, Stride: 1},
+		{Lo: 0xdbd, Hi: 0xdbd, Stride: 1},
+		{Lo: 0xdc0, Hi: 0xdc6, Stride: 1},
+		{Lo: 0xdca, Hi: 0xdca, Stride: 1},
+		{Lo: 0xdcf, Hi: 0xdd4, Stride: 1},
+		{Lo: 0xdd6, Hi: 0xdd6, Stride: 1},
+		{Lo: 0xdd8, Hi: 0xddf, Stride: 1},
+		{Lo: 0xdf2, Hi: 0xdf3, Stride: 1},
+		{Lo: 0xe01, Hi: 0xe3a, Stride: 1},
+		{Lo: 0xe40, Hi: 0xe4e, Stride: 1},
+		{Lo: 0xe50, Hi: 0xe59, Stride: 1},
+		{Lo: 0xe81, Hi: 0xe82, Stride: 1},
+		{Lo: 0xe84, Hi: 0xe84, Stride: 1},
+		{Lo: 0xe87, Hi: 0xe88, Stride: 1},
+		{Lo: 0xe8a, Hi: 0xe8a, Stride: 1},
+		{Lo: 0xe8d, Hi: 0xe8d, Stride: 1},
+		{Lo: 0xe94, Hi: 0xe97, Stride: 1},
+		{Lo: 0xe99, Hi: 0xe9f, Stride: 1},
+		{Lo: 0xea1, Hi: 0xea3, Stride: 1},
+		{Lo: 0xea5, Hi: 0xea5, Stride: 1},
+		{Lo: 0xea7, Hi: 0xea7, Stride: 1},
+		{Lo: 0xeaa, Hi: 0xeab, Stride: 1},
+		{Lo: 0xead, Hi: 0xeb9, Stride: 1},
+		{Lo: 0xebb, Hi: 0xebd, Stride: 1},
+		{Lo: 0xec0, Hi: 0xec4, Stride: 1},
+		{Lo: 0xec6, Hi: 0xec6, Stride: 1},
+		{Lo: 0xec8, Hi: 0xecd, Stride: 1},
+		{Lo: 0xed0, Hi: 0xed9, Stride: 1},
+		{Lo: 0xedc, Hi: 0xedd, Stride: 1},
+		{Lo: 0xf00, Hi: 0xf00, Stride: 1},
+		{Lo: 0xf18, Hi: 0xf19, Stride: 1},
+		{Lo: 0xf20, Hi: 0xf29, Stride: 1},
+		{Lo: 0xf35, Hi: 0xf35, Stride: 1},
+		{Lo: 0xf37, Hi: 0xf37, Stride: 1},
+		{Lo: 0xf39, Hi: 0xf39, Stride: 1},
+		{Lo: 0xf3e, Hi: 0xf47, Stride: 1},
+		{Lo: 0xf49, Hi: 0xf6a, Stride: 1},
+		{Lo: 0xf71, Hi: 0xf84, Stride: 1},
+		{Lo: 0xf86, Hi: 0xf8b, Stride: 1},
+		{Lo: 0xf90, Hi: 0xf97, Stride: 1},
+		{Lo: 0xf99, Hi: 0xfbc, Stride: 1},
+		{Lo: 0xfc6, Hi: 0xfc6, Stride: 1},
+	},
+	R32: []unicode.Range32{
+		{Lo: 0x1000, Hi: 0x1021, Stride: 1},
+		{Lo: 0x1023, Hi: 0x1027, Stride: 1},
+		{Lo: 0x1029, Hi: 0x102a, Stride: 1},
+		{Lo: 0x102c, Hi: 0x1032, Stride: 1},
+		{Lo: 0x1036, Hi: 0x1039, Stride: 1},
+		{Lo: 0x1040, Hi: 0x1049, Stride: 1},
+		{Lo: 0x1050, Hi: 0x1059, Stride: 1},
+		{Lo: 0x10a0, Hi: 0x10c5, Stride: 1},
+		{Lo: 0x10d0, Hi: 0x10f6, Stride: 1},
+		{Lo: 0x1100, Hi: 0x1159, Stride: 1},
+		{Lo: 0x115f, Hi: 0x11a2, Stride: 1},
+		{Lo: 0x11a8, Hi: 0x11f9, Stride: 1},
+		{Lo: 0x1200, Hi: 0x1206, Stride: 1},
+		{Lo: 0x1208, Hi: 0x1246, Stride: 1},
+		{Lo: 0x1248, Hi: 0x1248, Stride: 1},
+		{Lo: 0x124a, Hi: 0x124d, Stride: 1},
+		{Lo: 0x1250, Hi: 0x1256, Stride: 1},
+		{Lo: 0x1258, Hi: 0x1258, Stride: 1},
+		{Lo: 0x125a, Hi: 0x125d, Stride: 1},
+		{Lo: 0x1260, Hi: 0x1286, Stride: 1},
+		{Lo: 0x1288, Hi: 0x1288, Stride: 1},
+		{Lo: 0x128a, Hi: 0x128d, Stride: 1},
+		{Lo: 0x1290, Hi: 0x12ae, Stride: 1},
+		{Lo: 0x12b0, Hi: 0x12b0, Stride: 1},
+		{Lo: 0x12b2, Hi: 0x12b5, Stride: 1},
+		{Lo: 0x12b8, Hi: 0x12be, Stride: 1},
+		{Lo: 0x12c0, Hi: 0x12c0, Stride: 1},
+		{Lo: 0x12c2, Hi: 0x12c5, Stride: 1},
+		{Lo: 0x12c8, Hi: 0x12ce, Stride: 1},
+		{Lo: 0x12d0, Hi: 0x12d6, Stride: 1},
+		{Lo: 0x12d8, Hi: 0x12ee, Stride: 1},
+		{Lo: 0x12f0, Hi: 0x130e, Stride: 1},
+		{Lo: 0x1310, Hi: 0x1310, Stride: 1},
+		{Lo: 0x1312, Hi: 0x1315, Stride: 1},
+		{Lo: 0x1318, Hi: 0x131e, Stride: 1},
+		{Lo: 0x1320, Hi: 0x1346, Stride: 1},
+		{Lo: 0x1348, Hi: 0x135a, Stride: 1},
+		{Lo: 0x1369, Hi: 0x1371, Stride: 1},
+		{Lo: 0x13a0, Hi: 0x13f4, Stride: 1},
+		{Lo: 0x1401, Hi: 0x166c, Stride: 1},
+		{Lo: 0x166f, Hi: 0x1676, Stride: 1},
+		{Lo: 0x1681, Hi: 0x169a, Stride: 1},
+		{Lo: 0x16a0, Hi: 0x16ea, Stride: 1},
+		{Lo: 0x1780, Hi: 0x17d3, Stride: 1},
+		{Lo: 0x17e0, Hi: 0x17e9, Stride: 1},
+		{Lo: 0x1810, Hi: 0x1819, Stride: 1},
+		{Lo: 0x1820, Hi: 0x1877, Stride: 1},
+		{Lo: 0x1880, Hi: 0x18a9, Stride: 1},
+		{Lo: 0x1e00, Hi: 0x1e9b, Stride: 1},
+		{Lo: 0x1ea0, Hi: 0x1ef9, Stride: 1},
+		{Lo: 0x1f00, Hi: 0x1f15, Stride: 1},
+		{Lo: 0x1f18, Hi: 0x1f1d, Stride: 1},
+		{Lo: 0x1f20, Hi: 0x1f45, Stride: 1},
+		{Lo: 0x1f48, Hi: 0x1f4d, Stride: 1},
+		{Lo: 0x1f50, Hi: 0x1f57, Stride: 1},
+		{Lo: 0x1f59, Hi: 0x1f59, Stride: 1},
+		{Lo: 0x1f5b, Hi: 0x1f5b, Stride: 1},
+		{Lo: 0x1f5d, Hi: 0x1f5d, Stride: 1},
+		{Lo: 0x1f5f, Hi: 0x1f7d, Stride: 1},
+		{Lo: 0x1f80, Hi: 0x1fb4, Stride: 1},
+		{Lo: 0x1fb6, Hi: 0x1fbc, Stride: 1},
+		{Lo: 0x1fbe, Hi: 0x1fbe, Stride: 1},
+		{Lo: 0x1fc2, Hi: 0x1fc4, Stride: 1},
+		{Lo: 0x1fc6, Hi: 0x1fcc, Stride: 1},
+		{Lo: 0x1fd0, Hi: 0x1fd3, Stride: 1},
+		{Lo: 0x1fd6, Hi: 0x1fdb, Stride: 1},
+		{Lo: 0x1fe0, Hi: 0x1fec, Stride: 1},
+		{Lo: 0x1ff2, Hi: 0x1ff4, Stride: 1},
+		{Lo: 0x1ff6, Hi: 0x1ffc, Stride: 1},
+		{Lo: 0x203f, Hi: 0x2040, Stride: 1},
+		{Lo: 0x207f, Hi: 0x207f, Stride: 1},
+		{Lo: 0x20d0, Hi: 0x20dc, Stride: 1},
+		{Lo: 0x20e1, Hi: 0x20e1, Stride: 1},
+		{Lo: 0x2102, Hi: 0x2102, Stride: 1},
+		{Lo: 0x2107, Hi: 0x2107, Stride: 1},
+		{Lo: 0x210a, Hi: 0x2113, Stride: 1},
+		{Lo: 0x2115, Hi: 0x2115, Stride: 1},
+		{Lo: 0x2119, Hi: 0x211d, Stride: 1},
+		{Lo: 0x2124, Hi: 0x2124, Stride: 1},
+		{Lo: 0x2126, Hi: 0x2126, Stride: 1},
+		{Lo: 0x2128, Hi: 0x2128, Stride: 1},
+		{Lo: 0x212a, Hi: 0x212d, Stride: 1},
+		{Lo: 0x212f, Hi: 0x2131, Stride: 1},
+		{Lo: 0x2133, Hi: 0x2139, Stride: 1},
+		{Lo: 0x3005, Hi: 0x3006, Stride: 1},
+		{Lo: 0x302a, Hi: 0x302f, Stride: 1},
+		{Lo: 0x3031, Hi: 0x3035, Stride: 1},
+		{Lo: 0x3041, Hi: 0x3094, Stride: 1},
+		{Lo: 0x3099, Hi: 0x309a, Stride: 1},
+		{Lo: 0x309d, Hi: 0x309e, Stride: 1},
+		{Lo: 0x30a1, Hi: 0x30fa, Stride: 1},
+		{Lo: 0x30fc, Hi: 0x30fe, Stride: 1},
+		{Lo: 0x3105, Hi: 0x312c, Stride: 1},
+		{Lo: 0x3131, Hi: 0x318e, Stride: 1},
+		{Lo: 0x31a0, Hi: 0x31b7, Stride: 1},
+		{Lo: 0x3400, Hi: 0x4db5, Stride: 1},
+		{Lo: 0x4e00, Hi: 0x9fa5, Stride: 1},
+		{Lo: 0xa000, Hi: 0xa48c, Stride: 1},
+		{Lo: 0xac00, Hi: 0xd7a3, Stride: 1},
+		{Lo: 0xf900, Hi: 0xfa2d, Stride: 1},
+		{Lo: 0xfb00, Hi: 0xfb06, Stride: 1},
+		{Lo: 0xfb13, Hi: 0xfb17, Stride: 1},
+		{Lo: 0xfb1d, Hi: 0xfb28, Stride: 1},
+		{Lo: 0xfb2a, Hi: 0xfb36, Stride: 1},
+		{Lo: 0xfb38, Hi: 0xfb3c, Stride: 1},
+		{Lo: 0xfb3e, Hi: 0xfb3e, Stride: 1},
+		{Lo: 0xfb40, Hi: 0xfb41, Stride: 1},
+		{Lo: 0xfb43, Hi: 0xfb44, Stride: 1},
+		{Lo: 0xfb46, Hi: 0xfbb1, Stride: 1},
+		{Lo: 0xfbd3, Hi: 0xfd3d, Stride: 1},
+		{Lo: 0xfd50, Hi: 0xfd8f, Stride: 1},
+		{Lo: 0xfd92, Hi: 0xfdc7, Stride: 1},
+		{Lo: 0xfdf0, Hi: 0xfdfb, Stride: 1},
+		{Lo: 0xfe20, Hi: 0xfe23, Stride: 1},
+		{Lo: 0xfe33, Hi: 0xfe34, Stride: 1},
+		{Lo: 0xfe4d, Hi: 0xfe4f, Stride: 1},
+		{Lo: 0xfe70, Hi: 0xfe72, Stride: 1},
+		{Lo: 0xfe74, Hi: 0xfe74, Stride: 1},
+		{Lo: 0xfe76, Hi: 0xfefc, Stride: 1},
+		{Lo: 0xff10, Hi: 0xff19, Stride: 1},
+		{Lo: 0xff21, Hi: 0xff3a, Stride: 1},
+		{Lo: 0xff3f, Hi: 0xff3f, Stride: 1},
+		{Lo: 0xff41, Hi: 0xff5a, Stride: 1},
+		{Lo: 0xff66, Hi: 0xffbe, Stride: 1},
+		{Lo: 0xffc2, Hi: 0xffc7, Stride: 1},
+		{Lo: 0xffca, Hi: 0xffcf, Stride: 1},
+		{Lo: 0xffd2, Hi: 0xffd7, Stride: 1},
+		{Lo: 0xffda, Hi: 0xffdc, Stride: 1},
+	},
+}
+
+var idStartES5OrESNext = &unicode.RangeTable{
+	LatinOffset: 117,
+	R16: []unicode.Range16{
+		{Lo: 0x41, Hi: 0x5a, Stride: 1},
+		{Lo: 0x61, Hi: 0x7a, Stride: 1},
+		{Lo: 0xaa, Hi: 0xaa, Stride: 1},
+		{Lo: 0xb5, Hi: 0xb5, Stride: 1},
+		{Lo: 0xba, Hi: 0xba, Stride: 1},
+		{Lo: 0xc0, Hi: 0xd6, Stride: 1},
+		{Lo: 0xd8, Hi: 0xf6, Stride: 1},
+		{Lo: 0xf8, Hi: 0x2c1, Stride: 1},
+		{Lo: 0x2c6, Hi: 0x2d1, Stride: 1},
+		{Lo: 0x2e0, Hi: 0x2e4, Stride: 1},
+		{Lo: 0x2ec, Hi: 0x2ec, Stride: 1},
+		{Lo: 0x2ee, Hi: 0x2ee, Stride: 1},
+		{Lo: 0x370, Hi: 0x374, Stride: 1},
+		{Lo: 0x376, Hi: 0x377, Stride: 1},
+		{Lo: 0x37a, Hi: 0x37d, Stride: 1},
+		{Lo: 0x37f, Hi: 0x37f, Stride: 1},
+		{Lo: 0x386, Hi: 0x386, Stride: 1},
+		{Lo: 0x388, Hi: 0x38a, Stride: 1},
+		{Lo: 0x38c, Hi: 0x38c, Stride: 1},
+		{Lo: 0x38e, Hi: 0x3a1, Stride: 1},
+		{Lo: 0x3a3, Hi: 0x3f5, Stride: 1},
+		{Lo: 0x3f7, Hi: 0x481, Stride: 1},
+		{Lo: 0x48a, Hi: 0x52f, Stride: 1},
+		{Lo: 0x531, Hi: 0x556, Stride: 1},
+		{Lo: 0x559, Hi: 0x559, Stride: 1},
+		{Lo: 0x560, Hi: 0x588, Stride: 1},
+		{Lo: 0x5d0, Hi: 0x5ea, Stride: 1},
+		{Lo: 0x5ef, Hi: 0x5f2, Stride: 1},
+		{Lo: 0x620, Hi: 0x64a, Stride: 1},
+		{Lo: 0x66e, Hi: 0x66f, Stride: 1},
+		{Lo: 0x671, Hi: 0x6d3, Stride: 1},
+		{Lo: 0x6d5, Hi: 0x6d5, Stride: 1},
+		{Lo: 0x6e5, Hi: 0x6e6, Stride: 1},
+		{Lo: 0x6ee, Hi: 0x6ef, Stride: 1},
+		{Lo: 0x6fa, Hi: 0x6fc, Stride: 1},
+		{Lo: 0x6ff, Hi: 0x6ff, Stride: 1},
+		{Lo: 0x710, Hi: 0x710, Stride: 1},
+		{Lo: 0x712, Hi: 0x72f, Stride: 1},
+		{Lo: 0x74d, Hi: 0x7a5, Stride: 1},
+		{Lo: 0x7b1, Hi: 0x7b1, Stride: 1},
+		{Lo: 0x7ca, Hi: 0x7ea, Stride: 1},
+		{Lo: 0x7f4, Hi: 0x7f5, Stride: 1},
+		{Lo: 0x7fa, Hi: 0x7fa, Stride: 1},
+		{Lo: 0x800, Hi: 0x815, Stride: 1},
+		{Lo: 0x81a, Hi: 0x81a, Stride: 1},
+		{Lo: 0x824, Hi: 0x824, Stride: 1},
+		{Lo: 0x828, Hi: 0x828, Stride: 1},
+		{Lo: 0x840, Hi: 0x858, Stride: 1},
+		{Lo: 0x860, Hi: 0x86a, Stride: 1},
+		{Lo: 0x870, Hi: 0x887, Stride: 1},
+		{Lo: 0x889, Hi: 0x88e, Stride: 1},
+		{Lo: 0x8a0, Hi: 0x8c9, Stride: 1},
+		{Lo: 0x904, Hi: 0x939, Stride: 1},
+		{Lo: 0x93d, Hi: 0x93d, Stride: 1},
+		{Lo: 0x950, Hi: 0x950, Stride: 1},
+		{Lo: 0x958, Hi: 0x961, Stride: 1},
+		{Lo: 0x971, Hi: 0x980, Stride: 1},
+		{Lo: 0x985, Hi: 0x98c, Stride: 1},
+		{Lo: 0x98f, Hi: 0x990, Stride: 1},
+		{Lo: 0x993, Hi: 0x9a8, Stride: 1},
+		{Lo: 0x9aa, Hi: 0x9b0, Stride: 1},
+		{Lo: 0x9b2, Hi: 0x9b2, Stride: 1},
+		{Lo: 0x9b6, Hi: 0x9b9, Stride: 1},
+		{Lo: 0x9bd, Hi: 0x9bd, Stride: 1},
+		{Lo: 0x9ce, Hi: 0x9ce, Stride: 1},
+		{Lo: 0x9dc, Hi: 0x9dd, Stride: 1},
+		{Lo: 0x9df, Hi: 0x9e1, Stride: 1},
+		{Lo: 0x9f0, Hi: 0x9f1, Stride: 1},
+		{Lo: 0x9fc, Hi: 0x9fc, Stride: 1},
+		{Lo: 0xa05, Hi: 0xa0a, Stride: 1},
+		{Lo: 0xa0f, Hi: 0xa10, Stride: 1},
+		{Lo: 0xa13, Hi: 0xa28, Stride: 1},
+		{Lo: 0xa2a, Hi: 0xa30, Stride: 1},
+		{Lo: 0xa32, Hi: 0xa33, Stride: 1},
+		{Lo: 0xa35, Hi: 0xa36, Stride: 1},
+		{Lo: 0xa38, Hi: 0xa39, Stride: 1},
+		{Lo: 0xa59, Hi: 0xa5c, Stride: 1},
+		{Lo: 0xa5e, Hi: 0xa5e, Stride: 1},
+		{Lo: 0xa72, Hi: 0xa74, Stride: 1},
+		{Lo: 0xa85, Hi: 0xa8d, Stride: 1},
+		{Lo: 0xa8f, Hi: 0xa91, Stride: 1},
+		{Lo: 0xa93, Hi: 0xaa8, Stride: 1},
+		{Lo: 0xaaa, Hi: 0xab0, Stride: 1},
+		{Lo: 0xab2, Hi: 0xab3, Stride: 1},
+		{Lo: 0xab5, Hi: 0xab9, Stride: 1},
+		{Lo: 0xabd, Hi: 0xabd, Stride: 1},
+		{Lo: 0xad0, Hi: 0xad0, Stride: 1},
+		{Lo: 0xae0, Hi: 0xae1, Stride: 1},
+		{Lo: 0xaf9, Hi: 0xaf9, Stride: 1},
+		{Lo: 0xb05, Hi: 0xb0c, Stride: 1},
+		{Lo: 0xb0f, Hi: 0xb10, Stride: 1},
+		{Lo: 0xb13, Hi: 0xb28, Stride: 1},
+		{Lo: 0xb2a, Hi: 0xb30, Stride: 1},
+		{Lo: 0xb32, Hi: 0xb33, Stride: 1},
+		{Lo: 0xb35, Hi: 0xb39, Stride: 1},
+		{Lo: 0xb3d, Hi: 0xb3d, Stride: 1},
+		{Lo: 0xb5c, Hi: 0xb5d, Stride: 1},
+		{Lo: 0xb5f, Hi: 0xb61, Stride: 1},
+		{Lo: 0xb71, Hi: 0xb71, Stride: 1},
+		{Lo: 0xb83, Hi: 0xb83, Stride: 1},
+		{Lo: 0xb85, Hi: 0xb8a, Stride: 1},
+		{Lo: 0xb8e, Hi: 0xb90, Stride: 1},
+		{Lo: 0xb92, Hi: 0xb95, Stride: 1},
+		{Lo: 0xb99, Hi: 0xb9a, Stride: 1},
+		{Lo: 0xb9c, Hi: 0xb9c, Stride: 1},
+		{Lo: 0xb9e, Hi: 0xb9f, Stride: 1},
+		{Lo: 0xba3, Hi: 0xba4, Stride: 1},
+		{Lo: 0xba8, Hi: 0xbaa, Stride: 1},
+		{Lo: 0xbae, Hi: 0xbb9, Stride: 1},
+		{Lo: 0xbd0, Hi: 0xbd0, Stride: 1},
+		{Lo: 0xc05, Hi: 0xc0c, Stride: 1},
+		{Lo: 0xc0e, Hi: 0xc10, Stride: 1},
+		{Lo: 0xc12, Hi: 0xc28, Stride: 1},
+		{Lo: 0xc2a, Hi: 0xc39, Stride: 1},
+		{Lo: 0xc3d, Hi: 0xc3d, Stride: 1},
+		{Lo: 0xc58, Hi: 0xc5a, Stride: 1},
+		{Lo: 0xc5d, Hi: 0xc5d, Stride: 1},
+		{Lo: 0xc60, Hi: 0xc61, Stride: 1},
+		{Lo: 0xc80, Hi: 0xc80, Stride: 1},
+		{Lo: 0xc85, Hi: 0xc8c, Stride: 1},
+		{Lo: 0xc8e, Hi: 0xc90, Stride: 1},
+		{Lo: 0xc92, Hi: 0xca8, Stride: 1},
+		{Lo: 0xcaa, Hi: 0xcb3, Stride: 1},
+		{Lo: 0xcb5, Hi: 0xcb9, Stride: 1},
+		{Lo: 0xcbd, Hi: 0xcbd, Stride: 1},
+		{Lo: 0xcdd, Hi: 0xcde, Stride: 1},
+		{Lo: 0xce0, Hi: 0xce1, Stride: 1},
+		{Lo: 0xcf1, Hi: 0xcf2, Stride: 1},
+		{Lo: 0xd04, Hi: 0xd0c, Stride: 1},
+		{Lo: 0xd0e, Hi: 0xd10, Stride: 1},
+		{Lo: 0xd12, Hi: 0xd3a, Stride: 1},
+		{Lo: 0xd3d, Hi: 0xd3d, Stride: 1},
+		{Lo: 0xd4e, Hi: 0xd4e, Stride: 1},
+		{Lo: 0xd54, Hi: 0xd56, Stride: 1},
+		{Lo: 0xd5f, Hi: 0xd61, Stride: 1},
+		{Lo: 0xd7a, Hi: 0xd7f, Stride: 1},
+		{Lo: 0xd85, Hi: 0xd96, Stride: 1},
+		{Lo: 0xd9a, Hi: 0xdb1, Stride: 1},
+		{Lo: 0xdb3, Hi: 0xdbb, Stride: 1},
+		{Lo: 0xdbd, Hi: 0xdbd, Stride: 1},
+		{Lo: 0xdc0, Hi: 0xdc6, Stride: 1},
+		{Lo: 0xe01, Hi: 0xe30, Stride: 1},
+		{Lo: 0xe32, Hi: 0xe33, Stride: 1},
+		{Lo: 0xe40, Hi: 0xe46, Stride: 1},
+		{Lo: 0xe81, Hi: 0xe82, Stride: 1},
+		{Lo: 0xe84, Hi: 0xe84, Stride: 1},
+		{Lo: 0xe86, Hi: 0xe8a, Stride: 1},
+		{Lo: 0xe8c, Hi: 0xea3, Stride: 1},
+		{Lo: 0xea5, Hi: 0xea5, Stride: 1},
+		{Lo: 0xea7, Hi: 0xeb0, Stride: 1},
+		{Lo: 0xeb2, Hi: 0xeb3, Stride: 1},
+		{Lo: 0xebd, Hi: 0xebd, Stride: 1},
+		{Lo: 0xec0, Hi: 0xec4, Stride: 1},
+		{Lo: 0xec6, Hi: 0xec6, Stride: 1},
+		{Lo: 0xedc, Hi: 0xedf, Stride: 1},
+		{Lo: 0xf00, Hi: 0xf00, Stride: 1},
+		{Lo: 0xf40, Hi: 0xf47, Stride: 1},
+		{Lo: 0xf49, Hi: 0xf6c, Stride: 1},
+		{Lo: 0xf88, Hi: 0xf8c, Stride: 1},
+	},
+	R32: []unicode.Range32{
+		{Lo: 0x1000, Hi: 0x102a, Stride: 1},
+		{Lo: 0x103f, Hi: 0x103f, Stride: 1},
+		{Lo: 0x1050, Hi: 0x1055, Stride: 1},
+		{Lo: 0x105a, Hi: 0x105d, Stride: 1},
+		{Lo: 0x1061, Hi: 0x1061, Stride: 1},
+		{Lo: 0x1065, Hi: 0x1066, Stride: 1},
+		{Lo: 0x106e, Hi: 0x1070, Stride: 1},
+		{Lo: 0x1075, Hi: 0x1081, Stride: 1},
+		{Lo: 0x108e, Hi: 0x108e, Stride: 1},
+		{Lo: 0x10a0, Hi: 0x10c5, Stride: 1},
+		{Lo: 0x10c7, Hi: 0x10c7, Stride: 1},
+		{Lo: 0x10cd, Hi: 0x10cd, Stride: 1},
+		{Lo: 0x10d0, Hi: 0x10fa, Stride: 1},
+		{Lo: 0x10fc, Hi: 0x1248, Stride: 1},
+		{Lo: 0x124a, Hi: 0x124d, Stride: 1},
+		{Lo: 0x1250, Hi: 0x1256, Stride: 1},
+		{Lo: 0x1258, Hi: 0x1258, Stride: 1},
+		{Lo: 0x125a, Hi: 0x125d, Stride: 1},
+		{Lo: 0x1260, Hi: 0x1288, Stride: 1},
+		{Lo: 0x128a, Hi: 0x128d, Stride: 1},
+		{Lo: 0x1290, Hi: 0x12b0, Stride: 1},
+		{Lo: 0x12b2, Hi: 0x12b5, Stride: 1},
+		{Lo: 0x12b8, Hi: 0x12be, Stride: 1},
+		{Lo: 0x12c0, Hi: 0x12c0, Stride: 1},
+		{Lo: 0x12c2, Hi: 0x12c5, Stride: 1},
+		{Lo: 0x12c8, Hi: 0x12d6, Stride: 1},
+		{Lo: 0x12d8, Hi: 0x1310, Stride: 1},
+		{Lo: 0x1312, Hi: 0x1315, Stride: 1},
+		{Lo: 0x1318, Hi: 0x135a, Stride: 1},
+		{Lo: 0x1380, Hi: 0x138f, Stride: 1},
+		{Lo: 0x13a0, Hi: 0x13f5, Stride: 1},
+		{Lo: 0x13f8, Hi: 0x13fd, Stride: 1},
+		{Lo: 0x1401, Hi: 0x166c, Stride: 1},
+		{Lo: 0x166f, Hi: 0x167f, Stride: 1},
+		{Lo: 0x1681, Hi: 0x169a, Stride: 1},
+		{Lo: 0x16a0, Hi: 0x16ea, Stride: 1},
+		{Lo: 0x16ee, Hi: 0x16f8, Stride: 1},
+		{Lo: 0x1700, Hi: 0x1711, Stride: 1},
+		{Lo: 0x171f, Hi: 0x1731, Stride: 1},
+		{Lo: 0x1740, Hi: 0x1751, Stride: 1},
+		{Lo: 0x1760, Hi: 0x176c, Stride: 1},
+		{Lo: 0x176e, Hi: 0x1770, Stride: 1},
+		{Lo: 0x1780, Hi: 0x17b3, Stride: 1},
+		{Lo: 0x17d7, Hi: 0x17d7, Stride: 1},
+		{Lo: 0x17dc, Hi: 0x17dc, Stride: 1},
+		{Lo: 0x1820, Hi: 0x1878, Stride: 1},
+		{Lo: 0x1880, Hi: 0x18a8, Stride: 1},
+		{Lo: 0x18aa, Hi: 0x18aa, Stride: 1},
+		{Lo: 0x18b0, Hi: 0x18f5, Stride: 1},
+		{Lo: 0x1900, Hi: 0x191e, Stride: 1},
+		{Lo: 0x1950, Hi: 0x196d, Stride: 1},
+		{Lo: 0x1970, Hi: 0x1974, Stride: 1},
+		{Lo: 0x1980, Hi: 0x19ab, Stride: 1},
+		{Lo: 0x19b0, Hi: 0x19c9, Stride: 1},
+		{Lo: 0x1a00, Hi: 0x1a16, Stride: 1},
+		{Lo: 0x1a20, Hi: 0x1a54, Stride: 1},
+		{Lo: 0x1aa7, Hi: 0x1aa7, Stride: 1},
+		{Lo: 0x1b05, Hi: 0x1b33, Stride: 1},
+		{Lo: 0x1b45, Hi: 0x1b4c, Stride: 1},
+		{Lo: 0x1b83, Hi: 0x1ba0, Stride: 1},
+		{Lo: 0x1bae, Hi: 0x1baf, Stride: 1},
+		{Lo: 0x1bba, Hi: 0x1be5, Stride: 1},
+		{Lo: 0x1c00, Hi: 0x1c23, Stride: 1},
+		{Lo: 0x1c4d, Hi: 0x1c4f, Stride: 1},
+		{Lo: 0x1c5a, Hi: 0x1c7d, Stride: 1},
+		{Lo: 0x1c80, Hi: 0x1c88, Stride: 1},
+		{Lo: 0x1c90, Hi: 0x1cba, Stride: 1},
+		{Lo: 0x1cbd, Hi: 0x1cbf, Stride: 1},
+		{Lo: 0x1ce9, Hi: 0x1cec, Stride: 1},
+		{Lo: 0x1cee, Hi: 0x1cf3, Stride: 1},
+		{Lo: 0x1cf5, Hi: 0x1cf6, Stride: 1},
+		{Lo: 0x1cfa, Hi: 0x1cfa, Stride: 1},
+		{Lo: 0x1d00, Hi: 0x1dbf, Stride: 1},
+		{Lo: 0x1e00, Hi: 0x1f15, Stride: 1},
+		{Lo: 0x1f18, Hi: 0x1f1d, Stride: 1},
+		{Lo: 0x1f20, Hi: 0x1f45, Stride: 1},
+		{Lo: 0x1f48, Hi: 0x1f4d, Stride: 1},
+		{Lo: 0x1f50, Hi: 0x1f57, Stride: 1},
+		{Lo: 0x1f59, Hi: 0x1f59, Stride: 1},
+		{Lo: 0x1f5b, Hi: 0x1f5b, Stride: 1},
+		{Lo: 0x1f5d, Hi: 0x1f5d, Stride: 1},
+		{Lo: 0x1f5f, Hi: 0x1f7d, Stride: 1},
+		{Lo: 0x1f80, Hi: 0x1fb4, Stride: 1},
+		{Lo: 0x1fb6, Hi: 0x1fbc, Stride: 1},
+		{Lo: 0x1fbe, Hi: 0x1fbe, Stride: 1},
+		{Lo: 0x1fc2, Hi: 0x1fc4, Stride: 1},
+		{Lo: 0x1fc6, Hi: 0x1fcc, Stride: 1},
+		{Lo: 0x1fd0, Hi: 0x1fd3, Stride: 1},
+		{Lo: 0x1fd6, Hi: 0x1fdb, Stride: 1},
+		{Lo: 0x1fe0, Hi: 0x1fec, Stride: 1},
+		{Lo: 0x1ff2, Hi: 0x1ff4, Stride: 1},
+		{Lo: 0x1ff6, Hi: 0x1ffc, Stride: 1},
+		{Lo: 0x2071, Hi: 0x2071, Stride: 1},
+		{Lo: 0x207f, Hi: 0x207f, Stride: 1},
+		{Lo: 0x2090, Hi: 0x209c, Stride: 1},
+		{Lo: 0x2102, Hi: 0x2102, Stride: 1},
+		{Lo: 0x2107, Hi: 0x2107, Stride: 1},
+		{Lo: 0x210a, Hi: 0x2113, Stride: 1},
+		{Lo: 0x2115, Hi: 0x2115, Stride: 1},
+		{Lo: 0x2118, Hi: 0x211d, Stride: 1},
+		{Lo: 0x2124, Hi: 0x2124, Stride: 1},
+		{Lo: 0x2126, Hi: 0x2126, Stride: 1},
+		{Lo: 0x2128, Hi: 0x2128, Stride: 1},
+		{Lo: 0x212a, Hi: 0x2139, Stride: 1},
+		{Lo: 0x213c, Hi: 0x213f, Stride: 1},
+		{Lo: 0x2145, Hi: 0x2149, Stride: 1},
+		{Lo: 0x214e, Hi: 0x214e, Stride: 1},
+		{Lo: 0x2160, Hi: 0x2188, Stride: 1},
+		{Lo: 0x2c00, Hi: 0x2ce4, Stride: 1},
+		{Lo: 0x2ceb, Hi: 0x2cee, Stride: 1},
+		{Lo: 0x2cf2, Hi: 0x2cf3, Stride: 1},
+		{Lo: 0x2d00, Hi: 0x2d25, Stride: 1},
+		{Lo: 0x2d27, Hi: 0x2d27, Stride: 1},
+		{Lo: 0x2d2d, Hi: 0x2d2d, Stride: 1},
+		{Lo: 0x2d30, Hi: 0x2d67, Stride: 1},
+		{Lo: 0x2d6f, Hi: 0x2d6f, Stride: 1},
+		{Lo: 0x2d80, Hi: 0x2d96, Stride: 1},
+		{Lo: 0x2da0, Hi: 0x2da6, Stride: 1},
+		{Lo: 0x2da8, Hi: 0x2dae, Stride: 1},
+		{Lo: 0x2db0, Hi: 0x2db6, Stride: 1},
+		{Lo: 0x2db8, Hi: 0x2dbe, Stride: 1},
+		{Lo: 0x2dc0, Hi: 0x2dc6, Stride: 1},
+		{Lo: 0x2dc8, Hi: 0x2dce, Stride: 1},
+		{Lo: 0x2dd0, Hi: 0x2dd6, Stride: 1},
+		{Lo: 0x2dd8, Hi: 0x2dde, Stride: 1},
+		{Lo: 0x3005, Hi: 0x3007, Stride: 1},
+		{Lo: 0x3021, Hi: 0x3029, Stride: 1},
+		{Lo: 0x3031, Hi: 0x3035, Stride: 1},
+		{Lo: 0x3038, Hi: 0x303c, Stride: 1},
+		{Lo: 0x3041, Hi: 0x3096, Stride: 1},
+		{Lo: 0x309b, Hi: 0x309f, Stride: 1},
+		{Lo: 0x30a1, Hi: 0x30fa, Stride: 1},
+		{Lo: 0x30fc, Hi: 0x30ff, Stride: 1},
+		{Lo: 0x3105, Hi: 0x312f, Stride: 1},
+		{Lo: 0x3131, Hi: 0x318e, Stride: 1},
+		{Lo: 0x31a0, Hi: 0x31bf, Stride: 1},
+		{Lo: 0x31f0, Hi: 0x31ff, Stride: 1},
+		{Lo: 0x3400, Hi: 0x4dbf, Stride: 1},
+		{Lo: 0x4e00, Hi: 0xa48c, Stride: 1},
+		{Lo: 0xa4d0, Hi: 0xa4fd, Stride: 1},
+		{Lo: 0xa500, Hi: 0xa60c, Stride: 1},
+		{Lo: 0xa610, Hi: 0xa61f, Stride: 1},
+		{Lo: 0xa62a, Hi: 0xa62b, Stride: 1},
+		{Lo: 0xa640, Hi: 0xa66e, Stride: 1},
+		{Lo: 0xa67f, Hi: 0xa69d, Stride: 1},
+		{Lo: 0xa6a0, Hi: 0xa6ef, Stride: 1},
+		{Lo: 0xa717, Hi: 0xa71f, Stride: 1},
+		{Lo: 0xa722, Hi: 0xa788, Stride: 1},
+		{Lo: 0xa78b, Hi: 0xa7ca, Stride: 1},
+		{Lo: 0xa7d0, Hi: 0xa7d1, Stride: 1},
+		{Lo: 0xa7d3, Hi: 0xa7d3, Stride: 1},
+		{Lo: 0xa7d5, Hi: 0xa7d9, Stride: 1},
+		{Lo: 0xa7f2, Hi: 0xa801, Stride: 1},
+		{Lo: 0xa803, Hi: 0xa805, Stride: 1},
+		{Lo: 0xa807, Hi: 0xa80a, Stride: 1},
+		{Lo: 0xa80c, Hi: 0xa822, Stride: 1},
+		{Lo: 0xa840, Hi: 0xa873, Stride: 1},
+		{Lo: 0xa882, Hi: 0xa8b3, Stride: 1},
+		{Lo: 0xa8f2, Hi: 0xa8f7, Stride: 1},
+		{Lo: 0xa8fb, Hi: 0xa8fb, Stride: 1},
+		{Lo: 0xa8fd, Hi: 0xa8fe, Stride: 1},
+		{Lo: 0xa90a, Hi: 0xa925, Stride: 1},
+		{Lo: 0xa930, Hi: 0xa946, Stride: 1},
+		{Lo: 0xa960, Hi: 0xa97c, Stride: 1},
+		{Lo: 0xa984, Hi: 0xa9b2, Stride: 1},
+		{Lo: 0xa9cf, Hi: 0xa9cf, Stride: 1},
+		{Lo: 0xa9e0, Hi: 0xa9e4, Stride: 1},
+		{Lo: 0xa9e6, Hi: 0xa9ef, Stride: 1},
+		{Lo: 0xa9fa, Hi: 0xa9fe, Stride: 1},
+		{Lo: 0xaa00, Hi: 0xaa28, Stride: 1},
+		{Lo: 0xaa40, Hi: 0xaa42, Stride: 1},
+		{Lo: 0xaa44, Hi: 0xaa4b, Stride: 1},
+		{Lo: 0xaa60, Hi: 0xaa76, Stride: 1},
+		{Lo: 0xaa7a, Hi: 0xaa7a, Stride: 1},
+		{Lo: 0xaa7e, Hi: 0xaaaf, Stride: 1},
+		{Lo: 0xaab1, Hi: 0xaab1, Stride: 1},
+		{Lo: 0xaab5, Hi: 0xaab6, Stride: 1},
+		{Lo: 0xaab9, Hi: 0xaabd, Stride: 1},
+		{Lo: 0xaac0, Hi: 0xaac0, Stride: 1},
+		{Lo: 0xaac2, Hi: 0xaac2, Stride: 1},
+		{Lo: 0xaadb, Hi: 0xaadd, Stride: 1},
+		{Lo: 0xaae0, Hi: 0xaaea, Stride: 1},
+		{Lo: 0xaaf2, Hi: 0xaaf4, Stride: 1},
+		{Lo: 0xab01, Hi: 0xab06, Stride: 1},
+		{Lo: 0xab09, Hi: 0xab0e, Stride: 1},
+		{Lo: 0xab11, Hi: 0xab16, Stride: 1},
+		{Lo: 0xab20, Hi: 0xab26, Stride: 1},
+		{Lo: 0xab28, Hi: 0xab2e, Stride: 1},
+		{Lo: 0xab30, Hi: 0xab5a, Stride: 1},
+		{Lo: 0xab5c, Hi: 0xab69, Stride: 1},
+		{Lo: 0xab70, Hi: 0xabe2, Stride: 1},
+		{Lo: 0xac00, Hi: 0xd7a3, Stride: 1},
+		{Lo: 0xd7b0, Hi: 0xd7c6, Stride: 1},
+		{Lo: 0xd7cb, Hi: 0xd7fb, Stride: 1},
+		{Lo: 0xf900, Hi: 0xfa6d, Stride: 1},
+		{Lo: 0xfa70, Hi: 0xfad9, Stride: 1},
+		{Lo: 0xfb00, Hi: 0xfb06, Stride: 1},
+		{Lo: 0xfb13, Hi: 0xfb17, Stride: 1},
+		{Lo: 0xfb1d, Hi: 0xfb1d, Stride: 1},
+		{Lo: 0xfb1f, Hi: 0xfb28, Stride: 1},
+		{Lo: 0xfb2a, Hi: 0xfb36, Stride: 1},
+		{Lo: 0xfb38, Hi: 0xfb3c, Stride: 1},
+		{Lo: 0xfb3e, Hi: 0xfb3e, Stride: 1},
+		{Lo: 0xfb40, Hi: 0xfb41, Stride: 1},
+		{Lo: 0xfb43, Hi: 0xfb44, Stride: 1},
+		{Lo: 0xfb46, Hi: 0xfbb1, Stride: 1},
+		{Lo: 0xfbd3, Hi: 0xfd3d, Stride: 1},
+		{Lo: 0xfd50, Hi: 0xfd8f, Stride: 1},
+		{Lo: 0xfd92, Hi: 0xfdc7, Stride: 1},
+		{Lo: 0xfdf0, Hi: 0xfdfb, Stride: 1},
+		{Lo: 0xfe70, Hi: 0xfe74, Stride: 1},
+		{Lo: 0xfe76, Hi: 0xfefc, Stride: 1},
+		{Lo: 0xff21, Hi: 0xff3a, Stride: 1},
+		{Lo: 0xff41, Hi: 0xff5a, Stride: 1},
+		{Lo: 0xff66, Hi: 0xffbe, Stride: 1},
+		{Lo: 0xffc2, Hi: 0xffc7, Stride: 1},
+		{Lo: 0xffca, Hi: 0xffcf, Stride: 1},
+		{Lo: 0xffd2, Hi: 0xffd7, Stride: 1},
+		{Lo: 0xffda, Hi: 0xffdc, Stride: 1},
+		{Lo: 0x10000, Hi: 0x1000b, Stride: 1},
+		{Lo: 0x1000d, Hi: 0x10026, Stride: 1},
+		{Lo: 0x10028, Hi: 0x1003a, Stride: 1},
+		{Lo: 0x1003c, Hi: 0x1003d, Stride: 1},
+		{Lo: 0x1003f, Hi: 0x1004d, Stride: 1},
+		{Lo: 0x10050, Hi: 0x1005d, Stride: 1},
+		{Lo: 0x10080, Hi: 0x100fa, Stride: 1},
+		{Lo: 0x10140, Hi: 0x10174, Stride: 1},
+		{Lo: 0x10280, Hi: 0x1029c, Stride: 1},
+		{Lo: 0x102a0, Hi: 0x102d0, Stride: 1},
+		{Lo: 0x10300, Hi: 0x1031f, Stride: 1},
+		{Lo: 0x1032d, Hi: 0x1034a, Stride: 1},
+		{Lo: 0x10350, Hi: 0x10375, Stride: 1},
+		{Lo: 0x10380, Hi: 0x1039d, Stride: 1},
+		{Lo: 0x103a0, Hi: 0x103c3, Stride: 1},
+		{Lo: 0x103c8, Hi: 0x103cf, Stride: 1},
+		{Lo: 0x103d1, Hi: 0x103d5, Stride: 1},
+		{Lo: 0x10400, Hi: 0x1049d, Stride: 1},
+		{Lo: 0x104b0, Hi: 0x104d3, Stride: 1},
+		{Lo: 0x104d8, Hi: 0x104fb, Stride: 1},
+		{Lo: 0x10500, Hi: 0x10527, Stride: 1},
+		{Lo: 0x10530, Hi: 0x10563, Stride: 1},
+		{Lo: 0x10570, Hi: 0x1057a, Stride: 1},
+		{Lo: 0x1057c, Hi: 0x1058a, Stride: 1},
+		{Lo: 0x1058c, Hi: 0x10592, Stride: 1},
+		{Lo: 0x10594, Hi: 0x10595, Stride: 1},
+		{Lo: 0x10597, Hi: 0x105a1, Stride: 1},
+		{Lo: 0x105a3, Hi: 0x105b1, Stride: 1},
+		{Lo: 0x105b3, Hi: 0x105b9, Stride: 1},
+		{Lo: 0x105bb, Hi: 0x105bc, Stride: 1},
+		{Lo: 0x10600, Hi: 0x10736, Stride: 1},
+		{Lo: 0x10740, Hi: 0x10755, Stride: 1},
+		{Lo: 0x10760, Hi: 0x10767, Stride: 1},
+		{Lo: 0x10780, Hi: 0x10785, Stride: 1},
+		{Lo: 0x10787, Hi: 0x107b0, Stride: 1},
+		{Lo: 0x107b2, Hi: 0x107ba, Stride: 1},
+		{Lo: 0x10800, Hi: 0x10805, Stride: 1},
+		{Lo: 0x10808, Hi: 0x10808, Stride: 1},
+		{Lo: 0x1080a, Hi: 0x10835, Stride: 1},
+		{Lo: 0x10837, Hi: 0x10838, Stride: 1},
+		{Lo: 0x1083c, Hi: 0x1083c, Stride: 1},
+		{Lo: 0x1083f, Hi: 0x10855, Stride: 1},
+		{Lo: 0x10860, Hi: 0x10876, Stride: 1},
+		{Lo: 0x10880, Hi: 0x1089e, Stride: 1},
+		{Lo: 0x108e0, Hi: 0x108f2, Stride: 1},
+		{Lo: 0x108f4, Hi: 0x108f5, Stride: 1},
+		{Lo: 0x10900, Hi: 0x10915, Stride: 1},
+		{Lo: 0x10920, Hi: 0x10939, Stride: 1},
+		{Lo: 0x10980, Hi: 0x109b7, Stride: 1},
+		{Lo: 0x109be, Hi: 0x109bf, Stride: 1},
+		{Lo: 0x10a00, Hi: 0x10a00, Stride: 1},
+		{Lo: 0x10a10, Hi: 0x10a13, Stride: 1},
+		{Lo: 0x10a15, Hi: 0x10a17, Stride: 1},
+		{Lo: 0x10a19, Hi: 0x10a35, Stride: 1},
+		{Lo: 0x10a60, Hi: 0x10a7c, Stride: 1},
+		{Lo: 0x10a80, Hi: 0x10a9c, Stride: 1},
+		{Lo: 0x10ac0, Hi: 0x10ac7, Stride: 1},
+		{Lo: 0x10ac9, Hi: 0x10ae4, Stride: 1},
+		{Lo: 0x10b00, Hi: 0x10b35, Stride: 1},
+		{Lo: 0x10b40, Hi: 0x10b55, Stride: 1},
+		{Lo: 0x10b60, Hi: 0x10b72, Stride: 1},
+		{Lo: 0x10b80, Hi: 0x10b91, Stride: 1},
+		{Lo: 0x10c00, Hi: 0x10c48, Stride: 1},
+		{Lo: 0x10c80, Hi: 0x10cb2, Stride: 1},
+		{Lo: 0x10cc0, Hi: 0x10cf2, Stride: 1},
+		{Lo: 0x10d00, Hi: 0x10d23, Stride: 1},
+		{Lo: 0x10e80, Hi: 0x10ea9, Stride: 1},
+		{Lo: 0x10eb0, Hi: 0x10eb1, Stride: 1},
+		{Lo: 0x10f00, Hi: 0x10f1c, Stride: 1},
+		{Lo: 0x10f27, Hi: 0x10f27, Stride: 1},
+		{Lo: 0x10f30, Hi: 0x10f45, Stride: 1},
+		{Lo: 0x10f70, Hi: 0x10f81, Stride: 1},
+		{Lo: 0x10fb0, Hi: 0x10fc4, Stride: 1},
+		{Lo: 0x10fe0, Hi: 0x10ff6, Stride: 1},
+		{Lo: 0x11003, Hi: 0x11037, Stride: 1},
+		{Lo: 0x11071, Hi: 0x11072, Stride: 1},
+		{Lo: 0x11075, Hi: 0x11075, Stride: 1},
+		{Lo: 0x11083, Hi: 0x110af, Stride: 1},
+		{Lo: 0x110d0, Hi: 0x110e8, Stride: 1},
+		{Lo: 0x11103, Hi: 0x11126, Stride: 1},
+		{Lo: 0x11144, Hi: 0x11144, Stride: 1},
+		{Lo: 0x11147, Hi: 0x11147, Stride: 1},
+		{Lo: 0x11150, Hi: 0x11172, Stride: 1},
+		{Lo: 0x11176, Hi: 0x11176, Stride: 1},
+		{Lo: 0x11183, Hi: 0x111b2, Stride: 1},
+		{Lo: 0x111c1, Hi: 0x111c4, Stride: 1},
+		{Lo: 0x111da, Hi: 0x111da, Stride: 1},
+		{Lo: 0x111dc, Hi: 0x111dc, Stride: 1},
+		{Lo: 0x11200, Hi: 0x11211, Stride: 1},
+		{Lo: 0x11213, Hi: 0x1122b, Stride: 1},
+		{Lo: 0x1123f, Hi: 0x11240, Stride: 1},
+		{Lo: 0x11280, Hi: 0x11286, Stride: 1},
+		{Lo: 0x11288, Hi: 0x11288, Stride: 1},
+		{Lo: 0x1128a, Hi: 0x1128d, Stride: 1},
+		{Lo: 0x1128f, Hi: 0x1129d, Stride: 1},
+		{Lo: 0x1129f, Hi: 0x112a8, Stride: 1},
+		{Lo: 0x112b0, Hi: 0x112de, Stride: 1},
+		{Lo: 0x11305, Hi: 0x1130c, Stride: 1},
+		{Lo: 0x1130f, Hi: 0x11310, Stride: 1},
+		{Lo: 0x11313, Hi: 0x11328, Stride: 1},
+		{Lo: 0x1132a, Hi: 0x11330, Stride: 1},
+		{Lo: 0x11332, Hi: 0x11333, Stride: 1},
+		{Lo: 0x11335, Hi: 0x11339, Stride: 1},
+		{Lo: 0x1133d, Hi: 0x1133d, Stride: 1},
+		{Lo: 0x11350, Hi: 0x11350, Stride: 1},
+		{Lo: 0x1135d, Hi: 0x11361, Stride: 1},
+		{Lo: 0x11400, Hi: 0x11434, Stride: 1},
+		{Lo: 0x11447, Hi: 0x1144a, Stride: 1},
+		{Lo: 0x1145f, Hi: 0x11461, Stride: 1},
+		{Lo: 0x11480, Hi: 0x114af, Stride: 1},
+		{Lo: 0x114c4, Hi: 0x114c5, Stride: 1},
+		{Lo: 0x114c7, Hi: 0x114c7, Stride: 1},
+		{Lo: 0x11580, Hi: 0x115ae, Stride: 1},
+		{Lo: 0x115d8, Hi: 0x115db, Stride: 1},
+		{Lo: 0x11600, Hi: 0x1162f, Stride: 1},
+		{Lo: 0x11644, Hi: 0x11644, Stride: 1},
+		{Lo: 0x11680, Hi: 0x116aa, Stride: 1},
+		{Lo: 0x116b8, Hi: 0x116b8, Stride: 1},
+		{Lo: 0x11700, Hi: 0x1171a, Stride: 1},
+		{Lo: 0x11740, Hi: 0x11746, Stride: 1},
+		{Lo: 0x11800, Hi: 0x1182b, Stride: 1},
+		{Lo: 0x118a0, Hi: 0x118df, Stride: 1},
+		{Lo: 0x118ff, Hi: 0x11906, Stride: 1},
+		{Lo: 0x11909, Hi: 0x11909, Stride: 1},
+		{Lo: 0x1190c, Hi: 0x11913, Stride: 1},
+		{Lo: 0x11915, Hi: 0x11916, Stride: 1},
+		{Lo: 0x11918, Hi: 0x1192f, Stride: 1},
+		{Lo: 0x1193f, Hi: 0x1193f, Stride: 1},
+		{Lo: 0x11941, Hi: 0x11941, Stride: 1},
+		{Lo: 0x119a0, Hi: 0x119a7, Stride: 1},
+		{Lo: 0x119aa, Hi: 0x119d0, Stride: 1},
+		{Lo: 0x119e1, Hi: 0x119e1, Stride: 1},
+		{Lo: 0x119e3, Hi: 0x119e3, Stride: 1},
+		{Lo: 0x11a00, Hi: 0x11a00, Stride: 1},
+		{Lo: 0x11a0b, Hi: 0x11a32, Stride: 1},
+		{Lo: 0x11a3a, Hi: 0x11a3a, Stride: 1},
+		{Lo: 0x11a50, Hi: 0x11a50, Stride: 1},
+		{Lo: 0x11a5c, Hi: 0x11a89, Stride: 1},
+		{Lo: 0x11a9d, Hi: 0x11a9d, Stride: 1},
+		{Lo: 0x11ab0, Hi: 0x11af8, Stride: 1},
+		{Lo: 0x11c00, Hi: 0x11c08, Stride: 1},
+		{Lo: 0x11c0a, Hi: 0x11c2e, Stride: 1},
+		{Lo: 0x11c40, Hi: 0x11c40, Stride: 1},
+		{Lo: 0x11c72, Hi: 0x11c8f, Stride: 1},
+		{Lo: 0x11d00, Hi: 0x11d06, Stride: 1},
+		{Lo: 0x11d08, Hi: 0x11d09, Stride: 1},
+		{Lo: 0x11d0b, Hi: 0x11d30, Stride: 1},
+		{Lo: 0x11d46, Hi: 0x11d46, Stride: 1},
+		{Lo: 0x11d60, Hi: 0x11d65, Stride: 1},
+		{Lo: 0x11d67, Hi: 0x11d68, Stride: 1},
+		{Lo: 0x11d6a, Hi: 0x11d89, Stride: 1},
+		{Lo: 0x11d98, Hi: 0x11d98, Stride: 1},
+		{Lo: 0x11ee0, Hi: 0x11ef2, Stride: 1},
+		{Lo: 0x11f02, Hi: 0x11f02, Stride: 1},
+		{Lo: 0x11f04, Hi: 0x11f10, Stride: 1},
+		{Lo: 0x11f12, Hi: 0x11f33, Stride: 1},
+		{Lo: 0x11fb0, Hi: 0x11fb0, Stride: 1},
+		{Lo: 0x12000, Hi: 0x12399, Stride: 1},
+		{Lo: 0x12400, Hi: 0x1246e, Stride: 1},
+		{Lo: 0x12480, Hi: 0x12543, Stride: 1},
+		{Lo: 0x12f90, Hi: 0x12ff0, Stride: 1},
+		{Lo: 0x13000, Hi: 0x1342f, Stride: 1},
+		{Lo: 0x13441, Hi: 0x13446, Stride: 1},
+		{Lo: 0x14400, Hi: 0x14646, Stride: 1},
+		{Lo: 0x16800, Hi: 0x16a38, Stride: 1},
+		{Lo: 0x16a40, Hi: 0x16a5e, Stride: 1},
+		{Lo: 0x16a70, Hi: 0x16abe, Stride: 1},
+		{Lo: 0x16ad0, Hi: 0x16aed, Stride: 1},
+		{Lo: 0x16b00, Hi: 0x16b2f, Stride: 1},
+		{Lo: 0x16b40, Hi: 0x16b43, Stride: 1},
+		{Lo: 0x16b63, Hi: 0x16b77, Stride: 1},
+		{Lo: 0x16b7d, Hi: 0x16b8f, Stride: 1},
+		{Lo: 0x16e40, Hi: 0x16e7f, Stride: 1},
+		{Lo: 0x16f00, Hi: 0x16f4a, Stride: 1},
+		{Lo: 0x16f50, Hi: 0x16f50, Stride: 1},
+		{Lo: 0x16f93, Hi: 0x16f9f, Stride: 1},
+		{Lo: 0x16fe0, Hi: 0x16fe1, Stride: 1},
+		{Lo: 0x16fe3, Hi: 0x16fe3, Stride: 1},
+		{Lo: 0x17000, Hi: 0x187f7, Stride: 1},
+		{Lo: 0x18800, Hi: 0x18cd5, Stride: 1},
+		{Lo: 0x18d00, Hi: 0x18d08, Stride: 1},
+		{Lo: 0x1aff0, Hi: 0x1aff3, Stride: 1},
+		{Lo: 0x1aff5, Hi: 0x1affb, Stride: 1},
+		{Lo: 0x1affd, Hi: 0x1affe, Stride: 1},
+		{Lo: 0x1b000, Hi: 0x1b122, Stride: 1},
+		{Lo: 0x1b132, Hi: 0x1b132, Stride: 1},
+		{Lo: 0x1b150, Hi: 0x1b152, Stride: 1},
+		{Lo: 0x1b155, Hi: 0x1b155, Stride: 1},
+		{Lo: 0x1b164, Hi: 0x1b167, Stride: 1},
+		{Lo: 0x1b170, Hi: 0x1b2fb, Stride: 1},
+		{Lo: 0x1bc00, Hi: 0x1bc6a, Stride: 1},
+		{Lo: 0x1bc70, Hi: 0x1bc7c, Stride: 1},
+		{Lo: 0x1bc80, Hi: 0x1bc88, Stride: 1},
+		{Lo: 0x1bc90, Hi: 0x1bc99, Stride: 1},
+		{Lo: 0x1d400, Hi: 0x1d454, Stride: 1},
+		{Lo: 0x1d456, Hi: 0x1d49c, Stride: 1},
+		{Lo: 0x1d49e, Hi: 0x1d49f, Stride: 1},
+		{Lo: 0x1d4a2, Hi: 0x1d4a2, Stride: 1},
+		{Lo: 0x1d4a5, Hi: 0x1d4a6, Stride: 1},
+		{Lo: 0x1d4a9, Hi: 0x1d4ac, Stride: 1},
+		{Lo: 0x1d4ae, Hi: 0x1d4b9, Stride: 1},
+		{Lo: 0x1d4bb, Hi: 0x1d4bb, Stride: 1},
+		{Lo: 0x1d4bd, Hi: 0x1d4c3, Stride: 1},
+		{Lo: 0x1d4c5, Hi: 0x1d505, Stride: 1},
+		{Lo: 0x1d507, Hi: 0x1d50a, Stride: 1},
+		{Lo: 0x1d50d, Hi: 0x1d514, Stride: 1},
+		{Lo: 0x1d516, Hi: 0x1d51c, Stride: 1},
+		{Lo: 0x1d51e, Hi: 0x1d539, Stride: 1},
+		{Lo: 0x1d53b, Hi: 0x1d53e, Stride: 1},
+		{Lo: 0x1d540, Hi: 0x1d544, Stride: 1},
+		{Lo: 0x1d546, Hi: 0x1d546, Stride: 1},
+		{Lo: 0x1d54a, Hi: 0x1d550, Stride: 1},
+		{Lo: 0x1d552, Hi: 0x1d6a5, Stride: 1},
+		{Lo: 0x1d6a8, Hi: 0x1d6c0, Stride: 1},
+		{Lo: 0x1d6c2, Hi: 0x1d6da, Stride: 1},
+		{Lo: 0x1d6dc, Hi: 0x1d6fa, Stride: 1},
+		{Lo: 0x1d6fc, Hi: 0x1d714, Stride: 1},
+		{Lo: 0x1d716, Hi: 0x1d734, Stride: 1},
+		{Lo: 0x1d736, Hi: 0x1d74e, Stride: 1},
+		{Lo: 0x1d750, Hi: 0x1d76e, Stride: 1},
+		{Lo: 0x1d770, Hi: 0x1d788, Stride: 1},
+		{Lo: 0x1d78a, Hi: 0x1d7a8, Stride: 1},
+		{Lo: 0x1d7aa, Hi: 0x1d7c2, Stride: 1},
+		{Lo: 0x1d7c4, Hi: 0x1d7cb, Stride: 1},
+		{Lo: 0x1df00, Hi: 0x1df1e, Stride: 1},
+		{Lo: 0x1df25, Hi: 0x1df2a, Stride: 1},
+		{Lo: 0x1e030, Hi: 0x1e06d, Stride: 1},
+		{Lo: 0x1e100, Hi: 0x1e12c, Stride: 1},
+		{Lo: 0x1e137, Hi: 0x1e13d, Stride: 1},
+		{Lo: 0x1e14e, Hi: 0x1e14e, Stride: 1},
+		{Lo: 0x1e290, Hi: 0x1e2ad, Stride: 1},
+		{Lo: 0x1e2c0, Hi: 0x1e2eb, Stride: 1},
+		{Lo: 0x1e4d0, Hi: 0x1e4eb, Stride: 1},
+		{Lo: 0x1e7e0, Hi: 0x1e7e6, Stride: 1},
+		{Lo: 0x1e7e8, Hi: 0x1e7eb, Stride: 1},
+		{Lo: 0x1e7ed, Hi: 0x1e7ee, Stride: 1},
+		{Lo: 0x1e7f0, Hi: 0x1e7fe, Stride: 1},
+		{Lo: 0x1e800, Hi: 0x1e8c4, Stride: 1},
+		{Lo: 0x1e900, Hi: 0x1e943, Stride: 1},
+		{Lo: 0x1e94b, Hi: 0x1e94b, Stride: 1},
+		{Lo: 0x1ee00, Hi: 0x1ee03, Stride: 1},
+		{Lo: 0x1ee05, Hi: 0x1ee1f, Stride: 1},
+		{Lo: 0x1ee21, Hi: 0x1ee22, Stride: 1},
+		{Lo: 0x1ee24, Hi: 0x1ee24, Stride: 1},
+		{Lo: 0x1ee27, Hi: 0x1ee27, Stride: 1},
+		{Lo: 0x1ee29, Hi: 0x1ee32, Stride: 1},
+		{Lo: 0x1ee34, Hi: 0x1ee37, Stride: 1},
+		{Lo: 0x1ee39, Hi: 0x1ee39, Stride: 1},
+		{Lo: 0x1ee3b, Hi: 0x1ee3b, Stride: 1},
+		{Lo: 0x1ee42, Hi: 0x1ee42, Stride: 1},
+		{Lo: 0x1ee47, Hi: 0x1ee47, Stride: 1},
+		{Lo: 0x1ee49, Hi: 0x1ee49, Stride: 1},
+		{Lo: 0x1ee4b, Hi: 0x1ee4b, Stride: 1},
+		{Lo: 0x1ee4d, Hi: 0x1ee4f, Stride: 1},
+		{Lo: 0x1ee51, Hi: 0x1ee52, Stride: 1},
+		{Lo: 0x1ee54, Hi: 0x1ee54, Stride: 1},
+		{Lo: 0x1ee57, Hi: 0x1ee57, Stride: 1},
+		{Lo: 0x1ee59, Hi: 0x1ee59, Stride: 1},
+		{Lo: 0x1ee5b, Hi: 0x1ee5b, Stride: 1},
+		{Lo: 0x1ee5d, Hi: 0x1ee5d, Stride: 1},
+		{Lo: 0x1ee5f, Hi: 0x1ee5f, Stride: 1},
+		{Lo: 0x1ee61, Hi: 0x1ee62, Stride: 1},
+		{Lo: 0x1ee64, Hi: 0x1ee64, Stride: 1},
+		{Lo: 0x1ee67, Hi: 0x1ee6a, Stride: 1},
+		{Lo: 0x1ee6c, Hi: 0x1ee72, Stride: 1},
+		{Lo: 0x1ee74, Hi: 0x1ee77, Stride: 1},
+		{Lo: 0x1ee79, Hi: 0x1ee7c, Stride: 1},
+		{Lo: 0x1ee7e, Hi: 0x1ee7e, Stride: 1},
+		{Lo: 0x1ee80, Hi: 0x1ee89, Stride: 1},
+		{Lo: 0x1ee8b, Hi: 0x1ee9b, Stride: 1},
+		{Lo: 0x1eea1, Hi: 0x1eea3, Stride: 1},
+		{Lo: 0x1eea5, Hi: 0x1eea9, Stride: 1},
+		{Lo: 0x1eeab, Hi: 0x1eebb, Stride: 1},
+		{Lo: 0x20000, Hi: 0x2a6df, Stride: 1},
+		{Lo: 0x2a700, Hi: 0x2b739, Stride: 1},
+		{Lo: 0x2b740, Hi: 0x2b81d, Stride: 1},
+		{Lo: 0x2b820, Hi: 0x2cea1, Stride: 1},
+		{Lo: 0x2ceb0, Hi: 0x2ebe0, Stride: 1},
+		{Lo: 0x2ebf0, Hi: 0x2ee5d, Stride: 1},
+		{Lo: 0x2f800, Hi: 0x2fa1d, Stride: 1},
+		{Lo: 0x30000, Hi: 0x3134a, Stride: 1},
+		{Lo: 0x31350, Hi: 0x323af, Stride: 1},
+	},
+}
+
+var idContinueES5OrESNext = &unicode.RangeTable{
+	LatinOffset: 129,
+	R16: []unicode.Range16{
+		{Lo: 0x30, Hi: 0x39, Stride: 1},
+		{Lo: 0x41, Hi: 0x5a, Stride: 1},
+		{Lo: 0x5f, Hi: 0x5f, Stride: 1},
+		{Lo: 0x61, Hi: 0x7a, Stride: 1},
+		{Lo: 0xaa, Hi: 0xaa, Stride: 1},
+		{Lo: 0xb5, Hi: 0xb5, Stride: 1},
+		{Lo: 0xb7, Hi: 0xb7, Stride: 1},
+		{Lo: 0xba, Hi: 0xba, Stride: 1},
+		{Lo: 0xc0, Hi: 0xd6, Stride: 1},
+		{Lo: 0xd8, Hi: 0xf6, Stride: 1},
+		{Lo: 0xf8, Hi: 0x2c1, Stride: 1},
+		{Lo: 0x2c6, Hi: 0x2d1, Stride: 1},
+		{Lo: 0x2e0, Hi: 0x2e4, Stride: 1},
+		{Lo: 0x2ec, Hi: 0x2ec, Stride: 1},
+		{Lo: 0x2ee, Hi: 0x2ee, Stride: 1},
+		{Lo: 0x300, Hi: 0x374, Stride: 1},
+		{Lo: 0x376, Hi: 0x377, Stride: 1},
+		{Lo: 0x37a, Hi: 0x37d, Stride: 1},
+		{Lo: 0x37f, Hi: 0x37f, Stride: 1},
+		{Lo: 0x386, Hi: 0x38a, Stride: 1},
+		{Lo: 0x38c, Hi: 0x38c, Stride: 1},
+		{Lo: 0x38e, Hi: 0x3a1, Stride: 1},
+		{Lo: 0x3a3, Hi: 0x3f5, Stride: 1},
+		{Lo: 0x3f7, Hi: 0x481, Stride: 1},
+		{Lo: 0x483, Hi: 0x487, Stride: 1},
+		{Lo: 0x48a, Hi: 0x52f, Stride: 1},
+		{Lo: 0x531, Hi: 0x556, Stride: 1},
+		{Lo: 0x559, Hi: 0x559, Stride: 1},
+		{Lo: 0x560, Hi: 0x588, Stride: 1},
+		{Lo: 0x591, Hi: 0x5bd, Stride: 1},
+		{Lo: 0x5bf, Hi: 0x5bf, Stride: 1},
+		{Lo: 0x5c1, Hi: 0x5c2, Stride: 1},
+		{Lo: 0x5c4, Hi: 0x5c5, Stride: 1},
+		{Lo: 0x5c7, Hi: 0x5c7, Stride: 1},
+		{Lo: 0x5d0, Hi: 0x5ea, Stride: 1},
+		{Lo: 0x5ef, Hi: 0x5f2, Stride: 1},
+		{Lo: 0x610, Hi: 0x61a, Stride: 1},
+		{Lo: 0x620, Hi: 0x669, Stride: 1},
+		{Lo: 0x66e, Hi: 0x6d3, Stride: 1},
+		{Lo: 0x6d5, Hi: 0x6dc, Stride: 1},
+		{Lo: 0x6df, Hi: 0x6e8, Stride: 1},
+		{Lo: 0x6ea, Hi: 0x6fc, Stride: 1},
+		{Lo: 0x6ff, Hi: 0x6ff, Stride: 1},
+		{Lo: 0x710, Hi: 0x74a, Stride: 1},
+		{Lo: 0x74d, Hi: 0x7b1, Stride: 1},
+		{Lo: 0x7c0, Hi: 0x7f5, Stride: 1},
+		{Lo: 0x7fa, Hi: 0x7fa, Stride: 1},
+		{Lo: 0x7fd, Hi: 0x7fd, Stride: 1},
+		{Lo: 0x800, Hi: 0x82d, Stride: 1},
+		{Lo: 0x840, Hi: 0x85b, Stride: 1},
+		{Lo: 0x860, Hi: 0x86a, Stride: 1},
+		{Lo: 0x870, Hi: 0x887, Stride: 1},
+		{Lo: 0x889, Hi: 0x88e, Stride: 1},
+		{Lo: 0x898, Hi: 0x8e1, Stride: 1},
+		{Lo: 0x8e3, Hi: 0x963, Stride: 1},
+		{Lo: 0x966, Hi: 0x96f, Stride: 1},
+		{Lo: 0x971, Hi: 0x983, Stride: 1},
+		{Lo: 0x985, Hi: 0x98c, Stride: 1},
+		{Lo: 0x98f, Hi: 0x990, Stride: 1},
+		{Lo: 0x993, Hi: 0x9a8, Stride: 1},
+		{Lo: 0x9aa, Hi: 0x9b0, Stride: 1},
+		{Lo: 0x9b2, Hi: 0x9b2, Stride: 1},
+		{Lo: 0x9b6, Hi: 0x9b9, Stride: 1},
+		{Lo: 0x9bc, Hi: 0x9c4, Stride: 1},
+		{Lo: 0x9c7, Hi: 0x9c8, Stride: 1},
+		{Lo: 0x9cb, Hi: 0x9ce, Stride: 1},
+		{Lo: 0x9d7, Hi: 0x9d7, Stride: 1},
+		{Lo: 0x9dc, Hi: 0x9dd, Stride: 1},
+		{Lo: 0x9df, Hi: 0x9e3, Stride: 1},
+		{Lo: 0x9e6, Hi: 0x9f1, Stride: 1},
+		{Lo: 0x9fc, Hi: 0x9fc, Stride: 1},
+		{Lo: 0x9fe, Hi: 0x9fe, Stride: 1},
+		{Lo: 0xa01, Hi: 0xa03, Stride: 1},
+		{Lo: 0xa05, Hi: 0xa0a, Stride: 1},
+		{Lo: 0xa0f, Hi: 0xa10, Stride: 1},
+		{Lo: 0xa13, Hi: 0xa28, Stride: 1},
+		{Lo: 0xa2a, Hi: 0xa30, Stride: 1},
+		{Lo: 0xa32, Hi: 0xa33, Stride: 1},
+		{Lo: 0xa35, Hi: 0xa36, Stride: 1},
+		{Lo: 0xa38, Hi: 0xa39, Stride: 1},
+		{Lo: 0xa3c, Hi: 0xa3c, Stride: 1},
+		{Lo: 0xa3e, Hi: 0xa42, Stride: 1},
+		{Lo: 0xa47, Hi: 0xa48, Stride: 1},
+		{Lo: 0xa4b, Hi: 0xa4d, Stride: 1},
+		{Lo: 0xa51, Hi: 0xa51, Stride: 1},
+		{Lo: 0xa59, Hi: 0xa5c, Stride: 1},
+		{Lo: 0xa5e, Hi: 0xa5e, Stride: 1},
+		{Lo: 0xa66, Hi: 0xa75, Stride: 1},
+		{Lo: 0xa81, Hi: 0xa83, Stride: 1},
+		{Lo: 0xa85, Hi: 0xa8d, Stride: 1},
+		{Lo: 0xa8f, Hi: 0xa91, Stride: 1},
+		{Lo: 0xa93, Hi: 0xaa8, Stride: 1},
+		{Lo: 0xaaa, Hi: 0xab0, Stride: 1},
+		{Lo: 0xab2, Hi: 0xab3, Stride: 1},
+		{Lo: 0xab5, Hi: 0xab9, Stride: 1},
+		{Lo: 0xabc, Hi: 0xac5, Stride: 1},
+		{Lo: 0xac7, Hi: 0xac9, Stride: 1},
+		{Lo: 0xacb, Hi: 0xacd, Stride: 1},
+		{Lo: 0xad0, Hi: 0xad0, Stride: 1},
+		{Lo: 0xae0, Hi: 0xae3, Stride: 1},
+		{Lo: 0xae6, Hi: 0xaef, Stride: 1},
+		{Lo: 0xaf9, Hi: 0xaff, Stride: 1},
+		{Lo: 0xb01, Hi: 0xb03, Stride: 1},
+		{Lo: 0xb05, Hi: 0xb0c, Stride: 1},
+		{Lo: 0xb0f, Hi: 0xb10, Stride: 1},
+		{Lo: 0xb13, Hi: 0xb28, Stride: 1},
+		{Lo: 0xb2a, Hi: 0xb30, Stride: 1},
+		{Lo: 0xb32, Hi: 0xb33, Stride: 1},
+		{Lo: 0xb35, Hi: 0xb39, Stride: 1},
+		{Lo: 0xb3c, Hi: 0xb44, Stride: 1},
+		{Lo: 0xb47, Hi: 0xb48, Stride: 1},
+		{Lo: 0xb4b, Hi: 0xb4d, Stride: 1},
+		{Lo: 0xb55, Hi: 0xb57, Stride: 1},
+		{Lo: 0xb5c, Hi: 0xb5d, Stride: 1},
+		{Lo: 0xb5f, Hi: 0xb63, Stride: 1},
+		{Lo: 0xb66, Hi: 0xb6f, Stride: 1},
+		{Lo: 0xb71, Hi: 0xb71, Stride: 1},
+		{Lo: 0xb82, Hi: 0xb83, Stride: 1},
+		{Lo: 0xb85, Hi: 0xb8a, Stride: 1},
+		{Lo: 0xb8e, Hi: 0xb90, Stride: 1},
+		{Lo: 0xb92, Hi: 0xb95, Stride: 1},
+		{Lo: 0xb99, Hi: 0xb9a, Stride: 1},
+		{Lo: 0xb9c, Hi: 0xb9c, Stride: 1},
+		{Lo: 0xb9e, Hi: 0xb9f, Stride: 1},
+		{Lo: 0xba3, Hi: 0xba4, Stride: 1},
+		{Lo: 0xba8, Hi: 0xbaa, Stride: 1},
+		{Lo: 0xbae, Hi: 0xbb9, Stride: 1},
+		{Lo: 0xbbe, Hi: 0xbc2, Stride: 1},
+		{Lo: 0xbc6, Hi: 0xbc8, Stride: 1},
+		{Lo: 0xbca, Hi: 0xbcd, Stride: 1},
+		{Lo: 0xbd0, Hi: 0xbd0, Stride: 1},
+		{Lo: 0xbd7, Hi: 0xbd7, Stride: 1},
+		{Lo: 0xbe6, Hi: 0xbef, Stride: 1},
+		{Lo: 0xc00, Hi: 0xc0c, Stride: 1},
+		{Lo: 0xc0e, Hi: 0xc10, Stride: 1},
+		{Lo: 0xc12, Hi: 0xc28, Stride: 1},
+		{Lo: 0xc2a, Hi: 0xc39, Stride: 1},
+		{Lo: 0xc3c, Hi: 0xc44, Stride: 1},
+		{Lo: 0xc46, Hi: 0xc48, Stride: 1},
+		{Lo: 0xc4a, Hi: 0xc4d, Stride: 1},
+		{Lo: 0xc55, Hi: 0xc56, Stride: 1},
+		{Lo: 0xc58, Hi: 0xc5a, Stride: 1},
+		{Lo: 0xc5d, Hi: 0xc5d, Stride: 1},
+		{Lo: 0xc60, Hi: 0xc63, Stride: 1},
+		{Lo: 0xc66, Hi: 0xc6f, Stride: 1},
+		{Lo: 0xc80, Hi: 0xc83, Stride: 1},
+		{Lo: 0xc85, Hi: 0xc8c, Stride: 1},
+		{Lo: 0xc8e, Hi: 0xc90, Stride: 1},
+		{Lo: 0xc92, Hi: 0xca8, Stride: 1},
+		{Lo: 0xcaa, Hi: 0xcb3, Stride: 1},
+		{Lo: 0xcb5, Hi: 0xcb9, Stride: 1},
+		{Lo: 0xcbc, Hi: 0xcc4, Stride: 1},
+		{Lo: 0xcc6, Hi: 0xcc8, Stride: 1},
+		{Lo: 0xcca, Hi: 0xccd, Stride: 1},
+		{Lo: 0xcd5, Hi: 0xcd6, Stride: 1},
+		{Lo: 0xcdd, Hi: 0xcde, Stride: 1},
+		{Lo: 0xce0, Hi: 0xce3, Stride: 1},
+		{Lo: 0xce6, Hi: 0xcef, Stride: 1},
+		{Lo: 0xcf1, Hi: 0xcf3, Stride: 1},
+		{Lo: 0xd00, Hi: 0xd0c, Stride: 1},
+		{Lo: 0xd0e, Hi: 0xd10, Stride: 1},
+		{Lo: 0xd12, Hi: 0xd44, Stride: 1},
+		{Lo: 0xd46, Hi: 0xd48, Stride: 1},
+		{Lo: 0xd4a, Hi: 0xd4e, Stride: 1},
+		{Lo: 0xd54, Hi: 0xd57, Stride: 1},
+		{Lo: 0xd5f, Hi: 0xd63, Stride: 1},
+		{Lo: 0xd66, Hi: 0xd6f, Stride: 1},
+		{Lo: 0xd7a, Hi: 0xd7f, Stride: 1},
+		{Lo: 0xd81, Hi: 0xd83, Stride: 1},
+		{Lo: 0xd85, Hi: 0xd96, Stride: 1},
+		{Lo: 0xd9a, Hi: 0xdb1, Stride: 1},
+		{Lo: 0xdb3, Hi: 0xdbb, Stride: 1},
+		{Lo: 0xdbd, Hi: 0xdbd, Stride: 1},
+		{Lo: 0xdc0, Hi: 0xdc6, Stride: 1},
+		{Lo: 0xdca, Hi: 0xdca, Stride: 1},
+		{Lo: 0xdcf, Hi: 0xdd4, Stride: 1},
+		{Lo: 0xdd6, Hi: 0xdd6, Stride: 1},
+		{Lo: 0xdd8, Hi: 0xddf, Stride: 1},
+		{Lo: 0xde6, Hi: 0xdef, Stride: 1},
+		{Lo: 0xdf2, Hi: 0xdf3, Stride: 1},
+		{Lo: 0xe01, Hi: 0xe3a, Stride: 1},
+		{Lo: 0xe40, Hi: 0xe4e, Stride: 1},
+		{Lo: 0xe50, Hi: 0xe59, Stride: 1},
+		{Lo: 0xe81, Hi: 0xe82, Stride: 1},
+		{Lo: 0xe84, Hi: 0xe84, Stride: 1},
+		{Lo: 0xe86, Hi: 0xe8a, Stride: 1},
+		{Lo: 0xe8c, Hi: 0xea3, Stride: 1},
+		{Lo: 0xea5, Hi: 0xea5, Stride: 1},
+		{Lo: 0xea7, Hi: 0xebd, Stride: 1},
+		{Lo: 0xec0, Hi: 0xec4, Stride: 1},
+		{Lo: 0xec6, Hi: 0xec6, Stride: 1},
+		{Lo: 0xec8, Hi: 0xece, Stride: 1},
+		{Lo: 0xed0, Hi: 0xed9, Stride: 1},
+		{Lo: 0xedc, Hi: 0xedf, Stride: 1},
+		{Lo: 0xf00, Hi: 0xf00, Stride: 1},
+		{Lo: 0xf18, Hi: 0xf19, Stride: 1},
+		{Lo: 0xf20, Hi: 0xf29, Stride: 1},
+		{Lo: 0xf35, Hi: 0xf35, Stride: 1},
+		{Lo: 0xf37, Hi: 0xf37, Stride: 1},
+		{Lo: 0xf39, Hi: 0xf39, Stride: 1},
+		{Lo: 0xf3e, Hi: 0xf47, Stride: 1},
+		{Lo: 0xf49, Hi: 0xf6c, Stride: 1},
+		{Lo: 0xf71, Hi: 0xf84, Stride: 1},
+		{Lo: 0xf86, Hi: 0xf97, Stride: 1},
+		{Lo: 0xf99, Hi: 0xfbc, Stride: 1},
+		{Lo: 0xfc6, Hi: 0xfc6, Stride: 1},
+	},
+	R32: []unicode.Range32{
+		{Lo: 0x1000, Hi: 0x1049, Stride: 1},
+		{Lo: 0x1050, Hi: 0x109d, Stride: 1},
+		{Lo: 0x10a0, Hi: 0x10c5, Stride: 1},
+		{Lo: 0x10c7, Hi: 0x10c7, Stride: 1},
+		{Lo: 0x10cd, Hi: 0x10cd, Stride: 1},
+		{Lo: 0x10d0, Hi: 0x10fa, Stride: 1},
+		{Lo: 0x10fc, Hi: 0x1248, Stride: 1},
+		{Lo: 0x124a, Hi: 0x124d, Stride: 1},
+		{Lo: 0x1250, Hi: 0x1256, Stride: 1},
+		{Lo: 0x1258, Hi: 0x1258, Stride: 1},
+		{Lo: 0x125a, Hi: 0x125d, Stride: 1},
+		{Lo: 0x1260, Hi: 0x1288, Stride: 1},
+		{Lo: 0x128a, Hi: 0x128d, Stride: 1},
+		{Lo: 0x1290, Hi: 0x12b0, Stride: 1},
+		{Lo: 0x12b2, Hi: 0x12b5, Stride: 1},
+		{Lo: 0x12b8, Hi: 0x12be, Stride: 1},
+		{Lo: 0x12c0, Hi: 0x12c0, Stride: 1},
+		{Lo: 0x12c2, Hi: 0x12c5, Stride: 1},
+		{Lo: 0x12c8, Hi: 0x12d6, Stride: 1},
+		{Lo: 0x12d8, Hi: 0x1310, Stride: 1},
+		{Lo: 0x1312, Hi: 0x1315, Stride: 1},
+		{Lo: 0x1318, Hi: 0x135a, Stride: 1},
+		{Lo: 0x135d, Hi: 0x135f, Stride: 1},
+		{Lo: 0x1369, Hi: 0x1371, Stride: 1},
+		{Lo: 0x1380, Hi: 0x138f, Stride: 1},
+		{Lo: 0x13a0, Hi: 0x13f5, Stride: 1},
+		{Lo: 0x13f8, Hi: 0x13fd, Stride: 1},
+		{Lo: 0x1401, Hi: 0x166c, Stride: 1},
+		{Lo: 0x166f, Hi: 0x167f, Stride: 1},
+		{Lo: 0x1681, Hi: 0x169a, Stride: 1},
+		{Lo: 0x16a0, Hi: 0x16ea, Stride: 1},
+		{Lo: 0x16ee, Hi: 0x16f8, Stride: 1},
+		{Lo: 0x1700, Hi: 0x1715, Stride: 1},
+		{Lo: 0x171f, Hi: 0x1734, Stride: 1},
+		{Lo: 0x1740, Hi: 0x1753, Stride: 1},
+		{Lo: 0x1760, Hi: 0x176c, Stride: 1},
+		{Lo: 0x176e, Hi: 0x1770, Stride: 1},
+		{Lo: 0x1772, Hi: 0x1773, Stride: 1},
+		{Lo: 0x1780, Hi: 0x17d3, Stride: 1},
+		{Lo: 0x17d7, Hi: 0x17d7, Stride: 1},
+		{Lo: 0x17dc, Hi: 0x17dd, Stride: 1},
+		{Lo: 0x17e0, Hi: 0x17e9, Stride: 1},
+		{Lo: 0x180b, Hi: 0x180d, Stride: 1},
+		{Lo: 0x180f, Hi: 0x1819, Stride: 1},
+		{Lo: 0x1820, Hi: 0x1878, Stride: 1},
+		{Lo: 0x1880, Hi: 0x18aa, Stride: 1},
+		{Lo: 0x18b0, Hi: 0x18f5, Stride: 1},
+		{Lo: 0x1900, Hi: 0x191e, Stride: 1},
+		{Lo: 0x1920, Hi: 0x192b, Stride: 1},
+		{Lo: 0x1930, Hi: 0x193b, Stride: 1},
+		{Lo: 0x1946, Hi: 0x196d, Stride: 1},
+		{Lo: 0x1970, Hi: 0x1974, Stride: 1},
+		{Lo: 0x1980, Hi: 0x19ab, Stride: 1},
+		{Lo: 0x19b0, Hi: 0x19c9, Stride: 1},
+		{Lo: 0x19d0, Hi: 0x19da, Stride: 1},
+		{Lo: 0x1a00, Hi: 0x1a1b, Stride: 1},
+		{Lo: 0x1a20, Hi: 0x1a5e, Stride: 1},
+		{Lo: 0x1a60, Hi: 0x1a7c, Stride: 1},
+		{Lo: 0x1a7f, Hi: 0x1a89, Stride: 1},
+		{Lo: 0x1a90, Hi: 0x1a99, Stride: 1},
+		{Lo: 0x1aa7, Hi: 0x1aa7, Stride: 1},
+		{Lo: 0x1ab0, Hi: 0x1abd, Stride: 1},
+		{Lo: 0x1abf, Hi: 0x1ace, Stride: 1},
+		{Lo: 0x1b00, Hi: 0x1b4c, Stride: 1},
+		{Lo: 0x1b50, Hi: 0x1b59, Stride: 1},
+		{Lo: 0x1b6b, Hi: 0x1b73, Stride: 1},
+		{Lo: 0x1b80, Hi: 0x1bf3, Stride: 1},
+		{Lo: 0x1c00, Hi: 0x1c37, Stride: 1},
+		{Lo: 0x1c40, Hi: 0x1c49, Stride: 1},
+		{Lo: 0x1c4d, Hi: 0x1c7d, Stride: 1},
+		{Lo: 0x1c80, Hi: 0x1c88, Stride: 1},
+		{Lo: 0x1c90, Hi: 0x1cba, Stride: 1},
+		{Lo: 0x1cbd, Hi: 0x1cbf, Stride: 1},
+		{Lo: 0x1cd0, Hi: 0x1cd2, Stride: 1},
+		{Lo: 0x1cd4, Hi: 0x1cfa, Stride: 1},
+		{Lo: 0x1d00, Hi: 0x1f15, Stride: 1},
+		{Lo: 0x1f18, Hi: 0x1f1d, Stride: 1},
+		{Lo: 0x1f20, Hi: 0x1f45, Stride: 1},
+		{Lo: 0x1f48, Hi: 0x1f4d, Stride: 1},
+		{Lo: 0x1f50, Hi: 0x1f57, Stride: 1},
+		{Lo: 0x1f59, Hi: 0x1f59, Stride: 1},
+		{Lo: 0x1f5b, Hi: 0x1f5b, Stride: 1},
+		{Lo: 0x1f5d, Hi: 0x1f5d, Stride: 1},
+		{Lo: 0x1f5f, Hi: 0x1f7d, Stride: 1},
+		{Lo: 0x1f80, Hi: 0x1fb4, Stride: 1},
+		{Lo: 0x1fb6, Hi: 0x1fbc, Stride: 1},
+		{Lo: 0x1fbe, Hi: 0x1fbe, Stride: 1},
+		{Lo: 0x1fc2, Hi: 0x1fc4, Stride: 1},
+		{Lo: 0x1fc6, Hi: 0x1fcc, Stride: 1},
+		{Lo: 0x1fd0, Hi: 0x1fd3, Stride: 1},
+		{Lo: 0x1fd6, Hi: 0x1fdb, Stride: 1},
+		{Lo: 0x1fe0, Hi: 0x1fec, Stride: 1},
+		{Lo: 0x1ff2, Hi: 0x1ff4, Stride: 1},
+		{Lo: 0x1ff6, Hi: 0x1ffc, Stride: 1},
+		{Lo: 0x200c, Hi: 0x200d, Stride: 1},
+		{Lo: 0x203f, Hi: 0x2040, Stride: 1},
+		{Lo: 0x2054, Hi: 0x2054, Stride: 1},
+		{Lo: 0x2071, Hi: 0x2071, Stride: 1},
+		{Lo: 0x207f, Hi: 0x207f, Stride: 1},
+		{Lo: 0x2090, Hi: 0x209c, Stride: 1},
+		{Lo: 0x20d0, Hi: 0x20dc, Stride: 1},
+		{Lo: 0x20e1, Hi: 0x20e1, Stride: 1},
+		{Lo: 0x20e5, Hi: 0x20f0, Stride: 1},
+		{Lo: 0x2102, Hi: 0x2102, Stride: 1},
+		{Lo: 0x2107, Hi: 0x2107, Stride: 1},
+		{Lo: 0x210a, Hi: 0x2113, Stride: 1},
+		{Lo: 0x2115, Hi: 0x2115, Stride: 1},
+		{Lo: 0x2118, Hi: 0x211d, Stride: 1},
+		{Lo: 0x2124, Hi: 0x2124, Stride: 1},
+		{Lo: 0x2126, Hi: 0x2126, Stride: 1},
+		{Lo: 0x2128, Hi: 0x2128, Stride: 1},
+		{Lo: 0x212a, Hi: 0x2139, Stride: 1},
+		{Lo: 0x213c, Hi: 0x213f, Stride: 1},
+		{Lo: 0x2145, Hi: 0x2149, Stride: 1},
+		{Lo: 0x214e, Hi: 0x214e, Stride: 1},
+		{Lo: 0x2160, Hi: 0x2188, Stride: 1},
+		{Lo: 0x2c00, Hi: 0x2ce4, Stride: 1},
+		{Lo: 0x2ceb, Hi: 0x2cf3, Stride: 1},
+		{Lo: 0x2d00, Hi: 0x2d25, Stride: 1},
+		{Lo: 0x2d27, Hi: 0x2d27, Stride: 1},
+		{Lo: 0x2d2d, Hi: 0x2d2d, Stride: 1},
+		{Lo: 0x2d30, Hi: 0x2d67, Stride: 1},
+		{Lo: 0x2d6f, Hi: 0x2d6f, Stride: 1},
+		{Lo: 0x2d7f, Hi: 0x2d96, Stride: 1},
+		{Lo: 0x2da0, Hi: 0x2da6, Stride: 1},
+		{Lo: 0x2da8, Hi: 0x2dae, Stride: 1},
+		{Lo: 0x2db0, Hi: 0x2db6, Stride: 1},
+		{Lo: 0x2db8, Hi: 0x2dbe, Stride: 1},
+		{Lo: 0x2dc0, Hi: 0x2dc6, Stride: 1},
+		{Lo: 0x2dc8, Hi: 0x2dce, Stride: 1},
+		{Lo: 0x2dd0, Hi: 0x2dd6, Stride: 1},
+		{Lo: 0x2dd8, Hi: 0x2dde, Stride: 1},
+		{Lo: 0x2de0, Hi: 0x2dff, Stride: 1},
+		{Lo: 0x3005, Hi: 0x3007, Stride: 1},
+		{Lo: 0x3021, Hi: 0x302f, Stride: 1},
+		{Lo: 0x3031, Hi: 0x3035, Stride: 1},
+		{Lo: 0x3038, Hi: 0x303c, Stride: 1},
+		{Lo: 0x3041, Hi: 0x3096, Stride: 1},
+		{Lo: 0x3099, Hi: 0x309f, Stride: 1},
+		{Lo: 0x30a1, Hi: 0x30ff, Stride: 1},
+		{Lo: 0x3105, Hi: 0x312f, Stride: 1},
+		{Lo: 0x3131, Hi: 0x318e, Stride: 1},
+		{Lo: 0x31a0, Hi: 0x31bf, Stride: 1},
+		{Lo: 0x31f0, Hi: 0x31ff, Stride: 1},
+		{Lo: 0x3400, Hi: 0x4dbf, Stride: 1},
+		{Lo: 0x4e00, Hi: 0xa48c, Stride: 1},
+		{Lo: 0xa4d0, Hi: 0xa4fd, Stride: 1},
+		{Lo: 0xa500, Hi: 0xa60c, Stride: 1},
+		{Lo: 0xa610, Hi: 0xa62b, Stride: 1},
+		{Lo: 0xa640, Hi: 0xa66f, Stride: 1},
+		{Lo: 0xa674, Hi: 0xa67d, Stride: 1},
+		{Lo: 0xa67f, Hi: 0xa6f1, Stride: 1},
+		{Lo: 0xa717, Hi: 0xa71f, Stride: 1},
+		{Lo: 0xa722, Hi: 0xa788, Stride: 1},
+		{Lo: 0xa78b, Hi: 0xa7ca, Stride: 1},
+		{Lo: 0xa7d0, Hi: 0xa7d1, Stride: 1},
+		{Lo: 0xa7d3, Hi: 0xa7d3, Stride: 1},
+		{Lo: 0xa7d5, Hi: 0xa7d9, Stride: 1},
+		{Lo: 0xa7f2, Hi: 0xa827, Stride: 1},
+		{Lo: 0xa82c, Hi: 0xa82c, Stride: 1},
+		{Lo: 0xa840, Hi: 0xa873, Stride: 1},
+		{Lo: 0xa880, Hi: 0xa8c5, Stride: 1},
+		{Lo: 0xa8d0, Hi: 0xa8d9, Stride: 1},
+		{Lo: 0xa8e0, Hi: 0xa8f7, Stride: 1},
+		{Lo: 0xa8fb, Hi: 0xa8fb, Stride: 1},
+		{Lo: 0xa8fd, Hi: 0xa92d, Stride: 1},
+		{Lo: 0xa930, Hi: 0xa953, Stride: 1},
+		{Lo: 0xa960, Hi: 0xa97c, Stride: 1},
+		{Lo: 0xa980, Hi: 0xa9c0, Stride: 1},
+		{Lo: 0xa9cf, Hi: 0xa9d9, Stride: 1},
+		{Lo: 0xa9e0, Hi: 0xa9fe, Stride: 1},
+		{Lo: 0xaa00, Hi: 0xaa36, Stride: 1},
+		{Lo: 0xaa40, Hi: 0xaa4d, Stride: 1},
+		{Lo: 0xaa50, Hi: 0xaa59, Stride: 1},
+		{Lo: 0xaa60, Hi: 0xaa76, Stride: 1},
+		{Lo: 0xaa7a, Hi: 0xaac2, Stride: 1},
+		{Lo: 0xaadb, Hi: 0xaadd, Stride: 1},
+		{Lo: 0xaae0, Hi: 0xaaef, Stride: 1},
+		{Lo: 0xaaf2, Hi: 0xaaf6, Stride: 1},
+		{Lo: 0xab01, Hi: 0xab06, Stride: 1},
+		{Lo: 0xab09, Hi: 0xab0e, Stride: 1},
+		{Lo: 0xab11, Hi: 0xab16, Stride: 1},
+		{Lo: 0xab20, Hi: 0xab26, Stride: 1},
+		{Lo: 0xab28, Hi: 0xab2e, Stride: 1},
+		{Lo: 0xab30, Hi: 0xab5a, Stride: 1},
+		{Lo: 0xab5c, Hi: 0xab69, Stride: 1},
+		{Lo: 0xab70, Hi: 0xabea, Stride: 1},
+		{Lo: 0xabec, Hi: 0xabed, Stride: 1},
+		{Lo: 0xabf0, Hi: 0xabf9, Stride: 1},
+		{Lo: 0xac00, Hi: 0xd7a3, Stride: 1},
+		{Lo: 0xd7b0, Hi: 0xd7c6, Stride: 1},
+		{Lo: 0xd7cb, Hi: 0xd7fb, Stride: 1},
+		{Lo: 0xf900, Hi: 0xfa6d, Stride: 1},
+		{Lo: 0xfa70, Hi: 0xfad9, Stride: 1},
+		{Lo: 0xfb00, Hi: 0xfb06, Stride: 1},
+		{Lo: 0xfb13, Hi: 0xfb17, Stride: 1},
+		{Lo: 0xfb1d, Hi: 0xfb28, Stride: 1},
+		{Lo: 0xfb2a, Hi: 0xfb36, Stride: 1},
+		{Lo: 0xfb38, Hi: 0xfb3c, Stride: 1},
+		{Lo: 0xfb3e, Hi: 0xfb3e, Stride: 1},
+		{Lo: 0xfb40, Hi: 0xfb41, Stride: 1},
+		{Lo: 0xfb43, Hi: 0xfb44, Stride: 1},
+		{Lo: 0xfb46, Hi: 0xfbb1, Stride: 1},
+		{Lo: 0xfbd3, Hi: 0xfd3d, Stride: 1},
+		{Lo: 0xfd50, Hi: 0xfd8f, Stride: 1},
+		{Lo: 0xfd92, Hi: 0xfdc7, Stride: 1},
+		{Lo: 0xfdf0, Hi: 0xfdfb, Stride: 1},
+		{Lo: 0xfe00, Hi: 0xfe0f, Stride: 1},
+		{Lo: 0xfe20, Hi: 0xfe2f, Stride: 1},
+		{Lo: 0xfe33, Hi: 0xfe34, Stride: 1},
+		{Lo: 0xfe4d, Hi: 0xfe4f, Stride: 1},
+		{Lo: 0xfe70, Hi: 0xfe74, Stride: 1},
+		{Lo: 0xfe76, Hi: 0xfefc, Stride: 1},
+		{Lo: 0xff10, Hi: 0xff19, Stride: 1},
+		{Lo: 0xff21, Hi: 0xff3a, Stride: 1},
+		{Lo: 0xff3f, Hi: 0xff3f, Stride: 1},
+		{Lo: 0xff41, Hi: 0xff5a, Stride: 1},
+		{Lo: 0xff65, Hi: 0xffbe, Stride: 1},
+		{Lo: 0xffc2, Hi: 0xffc7, Stride: 1},
+		{Lo: 0xffca, Hi: 0xffcf, Stride: 1},
+		{Lo: 0xffd2, Hi: 0xffd7, Stride: 1},
+		{Lo: 0xffda, Hi: 0xffdc, Stride: 1},
+		{Lo: 0x10000, Hi: 0x1000b, Stride: 1},
+		{Lo: 0x1000d, Hi: 0x10026, Stride: 1},
+		{Lo: 0x10028, Hi: 0x1003a, Stride: 1},
+		{Lo: 0x1003c, Hi: 0x1003d, Stride: 1},
+		{Lo: 0x1003f, Hi: 0x1004d, Stride: 1},
+		{Lo: 0x10050, Hi: 0x1005d, Stride: 1},
+		{Lo: 0x10080, Hi: 0x100fa, Stride: 1},
+		{Lo: 0x10140, Hi: 0x10174, Stride: 1},
+		{Lo: 0x101fd, Hi: 0x101fd, Stride: 1},
+		{Lo: 0x10280, Hi: 0x1029c, Stride: 1},
+		{Lo: 0x102a0, Hi: 0x102d0, Stride: 1},
+		{Lo: 0x102e0, Hi: 0x102e0, Stride: 1},
+		{Lo: 0x10300, Hi: 0x1031f, Stride: 1},
+		{Lo: 0x1032d, Hi: 0x1034a, Stride: 1},
+		{Lo: 0x10350, Hi: 0x1037a, Stride: 1},
+		{Lo: 0x10380, Hi: 0x1039d, Stride: 1},
+		{Lo: 0x103a0, Hi: 0x103c3, Stride: 1},
+		{Lo: 0x103c8, Hi: 0x103cf, Stride: 1},
+		{Lo: 0x103d1, Hi: 0x103d5, Stride: 1},
+		{Lo: 0x10400, Hi: 0x1049d, Stride: 1},
+		{Lo: 0x104a0, Hi: 0x104a9, Stride: 1},
+		{Lo: 0x104b0, Hi: 0x104d3, Stride: 1},
+		{Lo: 0x104d8, Hi: 0x104fb, Stride: 1},
+		{Lo: 0x10500, Hi: 0x10527, Stride: 1},
+		{Lo: 0x10530, Hi: 0x10563, Stride: 1},
+		{Lo: 0x10570, Hi: 0x1057a, Stride: 1},
+		{Lo: 0x1057c, Hi: 0x1058a, Stride: 1},
+		{Lo: 0x1058c, Hi: 0x10592, Stride: 1},
+		{Lo: 0x10594, Hi: 0x10595, Stride: 1},
+		{Lo: 0x10597, Hi: 0x105a1, Stride: 1},
+		{Lo: 0x105a3, Hi: 0x105b1, Stride: 1},
+		{Lo: 0x105b3, Hi: 0x105b9, Stride: 1},
+		{Lo: 0x105bb, Hi: 0x105bc, Stride: 1},
+		{Lo: 0x10600, Hi: 0x10736, Stride: 1},
+		{Lo: 0x10740, Hi: 0x10755, Stride: 1},
+		{Lo: 0x10760, Hi: 0x10767, Stride: 1},
+		{Lo: 0x10780, Hi: 0x10785, Stride: 1},
+		{Lo: 0x10787, Hi: 0x107b0, Stride: 1},
+		{Lo: 0x107b2, Hi: 0x107ba, Stride: 1},
+		{Lo: 0x10800, Hi: 0x10805, Stride: 1},
+		{Lo: 0x10808, Hi: 0x10808, Stride: 1},
+		{Lo: 0x1080a, Hi: 0x10835, Stride: 1},
+		{Lo: 0x10837, Hi: 0x10838, Stride: 1},
+		{Lo: 0x1083c, Hi: 0x1083c, Stride: 1},
+		{Lo: 0x1083f, Hi: 0x10855, Stride: 1},
+		{Lo: 0x10860, Hi: 0x10876, Stride: 1},
+		{Lo: 0x10880, Hi: 0x1089e, Stride: 1},
+		{Lo: 0x108e0, Hi: 0x108f2, Stride: 1},
+		{Lo: 0x108f4, Hi: 0x108f5, Stride: 1},
+		{Lo: 0x10900, Hi: 0x10915, Stride: 1},
+		{Lo: 0x10920, Hi: 0x10939, Stride: 1},
+		{Lo: 0x10980, Hi: 0x109b7, Stride: 1},
+		{Lo: 0x109be, Hi: 0x109bf, Stride: 1},
+		{Lo: 0x10a00, Hi: 0x10a03, Stride: 1},
+		{Lo: 0x10a05, Hi: 0x10a06, Stride: 1},
+		{Lo: 0x10a0c, Hi: 0x10a13, Stride: 1},
+		{Lo: 0x10a15, Hi: 0x10a17, Stride: 1},
+		{Lo: 0x10a19, Hi: 0x10a35, Stride: 1},
+		{Lo: 0x10a38, Hi: 0x10a3a, Stride: 1},
+		{Lo: 0x10a3f, Hi: 0x10a3f, Stride: 1},
+		{Lo: 0x10a60, Hi: 0x10a7c, Stride: 1},
+		{Lo: 0x10a80, Hi: 0x10a9c, Stride: 1},
+		{Lo: 0x10ac0, Hi: 0x10ac7, Stride: 1},
+		{Lo: 0x10ac9, Hi: 0x10ae6, Stride: 1},
+		{Lo: 0x10b00, Hi: 0x10b35, Stride: 1},
+		{Lo: 0x10b40, Hi: 0x10b55, Stride: 1},
+		{Lo: 0x10b60, Hi: 0x10b72, Stride: 1},
+		{Lo: 0x10b80, Hi: 0x10b91, Stride: 1},
+		{Lo: 0x10c00, Hi: 0x10c48, Stride: 1},
+		{Lo: 0x10c80, Hi: 0x10cb2, Stride: 1},
+		{Lo: 0x10cc0, Hi: 0x10cf2, Stride: 1},
+		{Lo: 0x10d00, Hi: 0x10d27, Stride: 1},
+		{Lo: 0x10d30, Hi: 0x10d39, Stride: 1},
+		{Lo: 0x10e80, Hi: 0x10ea9, Stride: 1},
+		{Lo: 0x10eab, Hi: 0x10eac, Stride: 1},
+		{Lo: 0x10eb0, Hi: 0x10eb1, Stride: 1},
+		{Lo: 0x10efd, Hi: 0x10f1c, Stride: 1},
+		{Lo: 0x10f27, Hi: 0x10f27, Stride: 1},
+		{Lo: 0x10f30, Hi: 0x10f50, Stride: 1},
+		{Lo: 0x10f70, Hi: 0x10f85, Stride: 1},
+		{Lo: 0x10fb0, Hi: 0x10fc4, Stride: 1},
+		{Lo: 0x10fe0, Hi: 0x10ff6, Stride: 1},
+		{Lo: 0x11000, Hi: 0x11046, Stride: 1},
+		{Lo: 0x11066, Hi: 0x11075, Stride: 1},
+		{Lo: 0x1107f, Hi: 0x110ba, Stride: 1},
+		{Lo: 0x110c2, Hi: 0x110c2, Stride: 1},
+		{Lo: 0x110d0, Hi: 0x110e8, Stride: 1},
+		{Lo: 0x110f0, Hi: 0x110f9, Stride: 1},
+		{Lo: 0x11100, Hi: 0x11134, Stride: 1},
+		{Lo: 0x11136, Hi: 0x1113f, Stride: 1},
+		{Lo: 0x11144, Hi: 0x11147, Stride: 1},
+		{Lo: 0x11150, Hi: 0x11173, Stride: 1},
+		{Lo: 0x11176, Hi: 0x11176, Stride: 1},
+		{Lo: 0x11180, Hi: 0x111c4, Stride: 1},
+		{Lo: 0x111c9, Hi: 0x111cc, Stride: 1},
+		{Lo: 0x111ce, Hi: 0x111da, Stride: 1},
+		{Lo: 0x111dc, Hi: 0x111dc, Stride: 1},
+		{Lo: 0x11200, Hi: 0x11211, Stride: 1},
+		{Lo: 0x11213, Hi: 0x11237, Stride: 1},
+		{Lo: 0x1123e, Hi: 0x11241, Stride: 1},
+		{Lo: 0x11280, Hi: 0x11286, Stride: 1},
+		{Lo: 0x11288, Hi: 0x11288, Stride: 1},
+		{Lo: 0x1128a, Hi: 0x1128d, Stride: 1},
+		{Lo: 0x1128f, Hi: 0x1129d, Stride: 1},
+		{Lo: 0x1129f, Hi: 0x112a8, Stride: 1},
+		{Lo: 0x112b0, Hi: 0x112ea, Stride: 1},
+		{Lo: 0x112f0, Hi: 0x112f9, Stride: 1},
+		{Lo: 0x11300, Hi: 0x11303, Stride: 1},
+		{Lo: 0x11305, Hi: 0x1130c, Stride: 1},
+		{Lo: 0x1130f, Hi: 0x11310, Stride: 1},
+		{Lo: 0x11313, Hi: 0x11328, Stride: 1},
+		{Lo: 0x1132a, Hi: 0x11330, Stride: 1},
+		{Lo: 0x11332, Hi: 0x11333, Stride: 1},
+		{Lo: 0x11335, Hi: 0x11339, Stride: 1},
+		{Lo: 0x1133b, Hi: 0x11344, Stride: 1},
+		{Lo: 0x11347, Hi: 0x11348, Stride: 1},
+		{Lo: 0x1134b, Hi: 0x1134d, Stride: 1},
+		{Lo: 0x11350, Hi: 0x11350, Stride: 1},
+		{Lo: 0x11357, Hi: 0x11357, Stride: 1},
+		{Lo: 0x1135d, Hi: 0x11363, Stride: 1},
+		{Lo: 0x11366, Hi: 0x1136c, Stride: 1},
+		{Lo: 0x11370, Hi: 0x11374, Stride: 1},
+		{Lo: 0x11400, Hi: 0x1144a, Stride: 1},
+		{Lo: 0x11450, Hi: 0x11459, Stride: 1},
+		{Lo: 0x1145e, Hi: 0x11461, Stride: 1},
+		{Lo: 0x11480, Hi: 0x114c5, Stride: 1},
+		{Lo: 0x114c7, Hi: 0x114c7, Stride: 1},
+		{Lo: 0x114d0, Hi: 0x114d9, Stride: 1},
+		{Lo: 0x11580, Hi: 0x115b5, Stride: 1},
+		{Lo: 0x115b8, Hi: 0x115c0, Stride: 1},
+		{Lo: 0x115d8, Hi: 0x115dd, Stride: 1},
+		{Lo: 0x11600, Hi: 0x11640, Stride: 1},
+		{Lo: 0x11644, Hi: 0x11644, Stride: 1},
+		{Lo: 0x11650, Hi: 0x11659, Stride: 1},
+		{Lo: 0x11680, Hi: 0x116b8, Stride: 1},
+		{Lo: 0x116c0, Hi: 0x116c9, Stride: 1},
+		{Lo: 0x11700, Hi: 0x1171a, Stride: 1},
+		{Lo: 0x1171d, Hi: 0x1172b, Stride: 1},
+		{Lo: 0x11730, Hi: 0x11739, Stride: 1},
+		{Lo: 0x11740, Hi: 0x11746, Stride: 1},
+		{Lo: 0x11800, Hi: 0x1183a, Stride: 1},
+		{Lo: 0x118a0, Hi: 0x118e9, Stride: 1},
+		{Lo: 0x118ff, Hi: 0x11906, Stride: 1},
+		{Lo: 0x11909, Hi: 0x11909, Stride: 1},
+		{Lo: 0x1190c, Hi: 0x11913, Stride: 1},
+		{Lo: 0x11915, Hi: 0x11916, Stride: 1},
+		{Lo: 0x11918, Hi: 0x11935, Stride: 1},
+		{Lo: 0x11937, Hi: 0x11938, Stride: 1},
+		{Lo: 0x1193b, Hi: 0x11943, Stride: 1},
+		{Lo: 0x11950, Hi: 0x11959, Stride: 1},
+		{Lo: 0x119a0, Hi: 0x119a7, Stride: 1},
+		{Lo: 0x119aa, Hi: 0x119d7, Stride: 1},
+		{Lo: 0x119da, Hi: 0x119e1, Stride: 1},
+		{Lo: 0x119e3, Hi: 0x119e4, Stride: 1},
+		{Lo: 0x11a00, Hi: 0x11a3e, Stride: 1},
+		{Lo: 0x11a47, Hi: 0x11a47, Stride: 1},
+		{Lo: 0x11a50, Hi: 0x11a99, Stride: 1},
+		{Lo: 0x11a9d, Hi: 0x11a9d, Stride: 1},
+		{Lo: 0x11ab0, Hi: 0x11af8, Stride: 1},
+		{Lo: 0x11c00, Hi: 0x11c08, Stride: 1},
+		{Lo: 0x11c0a, Hi: 0x11c36, Stride: 1},
+		{Lo: 0x11c38, Hi: 0x11c40, Stride: 1},
+		{Lo: 0x11c50, Hi: 0x11c59, Stride: 1},
+		{Lo: 0x11c72, Hi: 0x11c8f, Stride: 1},
+		{Lo: 0x11c92, Hi: 0x11ca7, Stride: 1},
+		{Lo: 0x11ca9, Hi: 0x11cb6, Stride: 1},
+		{Lo: 0x11d00, Hi: 0x11d06, Stride: 1},
+		{Lo: 0x11d08, Hi: 0x11d09, Stride: 1},
+		{Lo: 0x11d0b, Hi: 0x11d36, Stride: 1},
+		{Lo: 0x11d3a, Hi: 0x11d3a, Stride: 1},
+		{Lo: 0x11d3c, Hi: 0x11d3d, Stride: 1},
+		{Lo: 0x11d3f, Hi: 0x11d47, Stride: 1},
+		{Lo: 0x11d50, Hi: 0x11d59, Stride: 1},
+		{Lo: 0x11d60, Hi: 0x11d65, Stride: 1},
+		{Lo: 0x11d67, Hi: 0x11d68, Stride: 1},
+		{Lo: 0x11d6a, Hi: 0x11d8e, Stride: 1},
+		{Lo: 0x11d90, Hi: 0x11d91, Stride: 1},
+		{Lo: 0x11d93, Hi: 0x11d98, Stride: 1},
+		{Lo: 0x11da0, Hi: 0x11da9, Stride: 1},
+		{Lo: 0x11ee0, Hi: 0x11ef6, Stride: 1},
+		{Lo: 0x11f00, Hi: 0x11f10, Stride: 1},
+		{Lo: 0x11f12, Hi: 0x11f3a, Stride: 1},
+		{Lo: 0x11f3e, Hi: 0x11f42, Stride: 1},
+		{Lo: 0x11f50, Hi: 0x11f59, Stride: 1},
+		{Lo: 0x11fb0, Hi: 0x11fb0, Stride: 1},
+		{Lo: 0x12000, Hi: 0x12399, Stride: 1},
+		{Lo: 0x12400, Hi: 0x1246e, Stride: 1},
+		{Lo: 0x12480, Hi: 0x12543, Stride: 1},
+		{Lo: 0x12f90, Hi: 0x12ff0, Stride: 1},
+		{Lo: 0x13000, Hi: 0x1342f, Stride: 1},
+		{Lo: 0x13440, Hi: 0x13455, Stride: 1},
+		{Lo: 0x14400, Hi: 0x14646, Stride: 1},
+		{Lo: 0x16800, Hi: 0x16a38, Stride: 1},
+		{Lo: 0x16a40, Hi: 0x16a5e, Stride: 1},
+		{Lo: 0x16a60, Hi: 0x16a69, Stride: 1},
+		{Lo: 0x16a70, Hi: 0x16abe, Stride: 1},
+		{Lo: 0x16ac0, Hi: 0x16ac9, Stride: 1},
+		{Lo: 0x16ad0, Hi: 0x16aed, Stride: 1},
+		{Lo: 0x16af0, Hi: 0x16af4, Stride: 1},
+		{Lo: 0x16b00, Hi: 0x16b36, Stride: 1},
+		{Lo: 0x16b40, Hi: 0x16b43, Stride: 1},
+		{Lo: 0x16b50, Hi: 0x16b59, Stride: 1},
+		{Lo: 0x16b63, Hi: 0x16b77, Stride: 1},
+		{Lo: 0x16b7d, Hi: 0x16b8f, Stride: 1},
+		{Lo: 0x16e40, Hi: 0x16e7f, Stride: 1},
+		{Lo: 0x16f00, Hi: 0x16f4a, Stride: 1},
+		{Lo: 0x16f4f, Hi: 0x16f87, Stride: 1},
+		{Lo: 0x16f8f, Hi: 0x16f9f, Stride: 1},
+		{Lo: 0x16fe0, Hi: 0x16fe1, Stride: 1},
+		{Lo: 0x16fe3, Hi: 0x16fe4, Stride: 1},
+		{Lo: 0x16ff0, Hi: 0x16ff1, Stride: 1},
+		{Lo: 0x17000, Hi: 0x187f7, Stride: 1},
+		{Lo: 0x18800, Hi: 0x18cd5, Stride: 1},
+		{Lo: 0x18d00, Hi: 0x18d08, Stride: 1},
+		{Lo: 0x1aff0, Hi: 0x1aff3, Stride: 1},
+		{Lo: 0x1aff5, Hi: 0x1affb, Stride: 1},
+		{Lo: 0x1affd, Hi: 0x1affe, Stride: 1},
+		{Lo: 0x1b000, Hi: 0x1b122, Stride: 1},
+		{Lo: 0x1b132, Hi: 0x1b132, Stride: 1},
+		{Lo: 0x1b150, Hi: 0x1b152, Stride: 1},
+		{Lo: 0x1b155, Hi: 0x1b155, Stride: 1},
+		{Lo: 0x1b164, Hi: 0x1b167, Stride: 1},
+		{Lo: 0x1b170, Hi: 0x1b2fb, Stride: 1},
+		{Lo: 0x1bc00, Hi: 0x1bc6a, Stride: 1},
+		{Lo: 0x1bc70, Hi: 0x1bc7c, Stride: 1},
+		{Lo: 0x1bc80, Hi: 0x1bc88, Stride: 1},
+		{Lo: 0x1bc90, Hi: 0x1bc99, Stride: 1},
+		{Lo: 0x1bc9d, Hi: 0x1bc9e, Stride: 1},
+		{Lo: 0x1cf00, Hi: 0x1cf2d, Stride: 1},
+		{Lo: 0x1cf30, Hi: 0x1cf46, Stride: 1},
+		{Lo: 0x1d165, Hi: 0x1d169, Stride: 1},
+		{Lo: 0x1d16d, Hi: 0x1d172, Stride: 1},
+		{Lo: 0x1d17b, Hi: 0x1d182, Stride: 1},
+		{Lo: 0x1d185, Hi: 0x1d18b, Stride: 1},
+		{Lo: 0x1d1aa, Hi: 0x1d1ad, Stride: 1},
+		{Lo: 0x1d242, Hi: 0x1d244, Stride: 1},
+		{Lo: 0x1d400, Hi: 0x1d454, Stride: 1},
+		{Lo: 0x1d456, Hi: 0x1d49c, Stride: 1},
+		{Lo: 0x1d49e, Hi: 0x1d49f, Stride: 1},
+		{Lo: 0x1d4a2, Hi: 0x1d4a2, Stride: 1},
+		{Lo: 0x1d4a5, Hi: 0x1d4a6, Stride: 1},
+		{Lo: 0x1d4a9, Hi: 0x1d4ac, Stride: 1},
+		{Lo: 0x1d4ae, Hi: 0x1d4b9, Stride: 1},
+		{Lo: 0x1d4bb, Hi: 0x1d4bb, Stride: 1},
+		{Lo: 0x1d4bd, Hi: 0x1d4c3, Stride: 1},
+		{Lo: 0x1d4c5, Hi: 0x1d505, Stride: 1},
+		{Lo: 0x1d507, Hi: 0x1d50a, Stride: 1},
+		{Lo: 0x1d50d, Hi: 0x1d514, Stride: 1},
+		{Lo: 0x1d516, Hi: 0x1d51c, Stride: 1},
+		{Lo: 0x1d51e, Hi: 0x1d539, Stride: 1},
+		{Lo: 0x1d53b, Hi: 0x1d53e, Stride: 1},
+		{Lo: 0x1d540, Hi: 0x1d544, Stride: 1},
+		{Lo: 0x1d546, Hi: 0x1d546, Stride: 1},
+		{Lo: 0x1d54a, Hi: 0x1d550, Stride: 1},
+		{Lo: 0x1d552, Hi: 0x1d6a5, Stride: 1},
+		{Lo: 0x1d6a8, Hi: 0x1d6c0, Stride: 1},
+		{Lo: 0x1d6c2, Hi: 0x1d6da, Stride: 1},
+		{Lo: 0x1d6dc, Hi: 0x1d6fa, Stride: 1},
+		{Lo: 0x1d6fc, Hi: 0x1d714, Stride: 1},
+		{Lo: 0x1d716, Hi: 0x1d734, Stride: 1},
+		{Lo: 0x1d736, Hi: 0x1d74e, Stride: 1},
+		{Lo: 0x1d750, Hi: 0x1d76e, Stride: 1},
+		{Lo: 0x1d770, Hi: 0x1d788, Stride: 1},
+		{Lo: 0x1d78a, Hi: 0x1d7a8, Stride: 1},
+		{Lo: 0x1d7aa, Hi: 0x1d7c2, Stride: 1},
+		{Lo: 0x1d7c4, Hi: 0x1d7cb, Stride: 1},
+		{Lo: 0x1d7ce, Hi: 0x1d7ff, Stride: 1},
+		{Lo: 0x1da00, Hi: 0x1da36, Stride: 1},
+		{Lo: 0x1da3b, Hi: 0x1da6c, Stride: 1},
+		{Lo: 0x1da75, Hi: 0x1da75, Stride: 1},
+		{Lo: 0x1da84, Hi: 0x1da84, Stride: 1},
+		{Lo: 0x1da9b, Hi: 0x1da9f, Stride: 1},
+		{Lo: 0x1daa1, Hi: 0x1daaf, Stride: 1},
+		{Lo: 0x1df00, Hi: 0x1df1e, Stride: 1},
+		{Lo: 0x1df25, Hi: 0x1df2a, Stride: 1},
+		{Lo: 0x1e000, Hi: 0x1e006, Stride: 1},
+		{Lo: 0x1e008, Hi: 0x1e018, Stride: 1},
+		{Lo: 0x1e01b, Hi: 0x1e021, Stride: 1},
+		{Lo: 0x1e023, Hi: 0x1e024, Stride: 1},
+		{Lo: 0x1e026, Hi: 0x1e02a, Stride: 1},
+		{Lo: 0x1e030, Hi: 0x1e06d, Stride: 1},
+		{Lo: 0x1e08f, Hi: 0x1e08f, Stride: 1},
+		{Lo: 0x1e100, Hi: 0x1e12c, Stride: 1},
+		{Lo: 0x1e130, Hi: 0x1e13d, Stride: 1},
+		{Lo: 0x1e140, Hi: 0x1e149, Stride: 1},
+		{Lo: 0x1e14e, Hi: 0x1e14e, Stride: 1},
+		{Lo: 0x1e290, Hi: 0x1e2ae, Stride: 1},
+		{Lo: 0x1e2c0, Hi: 0x1e2f9, Stride: 1},
+		{Lo: 0x1e4d0, Hi: 0x1e4f9, Stride: 1},
+		{Lo: 0x1e7e0, Hi: 0x1e7e6, Stride: 1},
+		{Lo: 0x1e7e8, Hi: 0x1e7eb, Stride: 1},
+		{Lo: 0x1e7ed, Hi: 0x1e7ee, Stride: 1},
+		{Lo: 0x1e7f0, Hi: 0x1e7fe, Stride: 1},
+		{Lo: 0x1e800, Hi: 0x1e8c4, Stride: 1},
+		{Lo: 0x1e8d0, Hi: 0x1e8d6, Stride: 1},
+		{Lo: 0x1e900, Hi: 0x1e94b, Stride: 1},
+		{Lo: 0x1e950, Hi: 0x1e959, Stride: 1},
+		{Lo: 0x1ee00, Hi: 0x1ee03, Stride: 1},
+		{Lo: 0x1ee05, Hi: 0x1ee1f, Stride: 1},
+		{Lo: 0x1ee21, Hi: 0x1ee22, Stride: 1},
+		{Lo: 0x1ee24, Hi: 0x1ee24, Stride: 1},
+		{Lo: 0x1ee27, Hi: 0x1ee27, Stride: 1},
+		{Lo: 0x1ee29, Hi: 0x1ee32, Stride: 1},
+		{Lo: 0x1ee34, Hi: 0x1ee37, Stride: 1},
+		{Lo: 0x1ee39, Hi: 0x1ee39, Stride: 1},
+		{Lo: 0x1ee3b, Hi: 0x1ee3b, Stride: 1},
+		{Lo: 0x1ee42, Hi: 0x1ee42, Stride: 1},
+		{Lo: 0x1ee47, Hi: 0x1ee47, Stride: 1},
+		{Lo: 0x1ee49, Hi: 0x1ee49, Stride: 1},
+		{Lo: 0x1ee4b, Hi: 0x1ee4b, Stride: 1},
+		{Lo: 0x1ee4d, Hi: 0x1ee4f, Stride: 1},
+		{Lo: 0x1ee51, Hi: 0x1ee52, Stride: 1},
+		{Lo: 0x1ee54, Hi: 0x1ee54, Stride: 1},
+		{Lo: 0x1ee57, Hi: 0x1ee57, Stride: 1},
+		{Lo: 0x1ee59, Hi: 0x1ee59, Stride: 1},
+		{Lo: 0x1ee5b, Hi: 0x1ee5b, Stride: 1},
+		{Lo: 0x1ee5d, Hi: 0x1ee5d, Stride: 1},
+		{Lo: 0x1ee5f, Hi: 0x1ee5f, Stride: 1},
+		{Lo: 0x1ee61, Hi: 0x1ee62, Stride: 1},
+		{Lo: 0x1ee64, Hi: 0x1ee64, Stride: 1},
+		{Lo: 0x1ee67, Hi: 0x1ee6a, Stride: 1},
+		{Lo: 0x1ee6c, Hi: 0x1ee72, Stride: 1},
+		{Lo: 0x1ee74, Hi: 0x1ee77, Stride: 1},
+		{Lo: 0x1ee79, Hi: 0x1ee7c, Stride: 1},
+		{Lo: 0x1ee7e, Hi: 0x1ee7e, Stride: 1},
+		{Lo: 0x1ee80, Hi: 0x1ee89, Stride: 1},
+		{Lo: 0x1ee8b, Hi: 0x1ee9b, Stride: 1},
+		{Lo: 0x1eea1, Hi: 0x1eea3, Stride: 1},
+		{Lo: 0x1eea5, Hi: 0x1eea9, Stride: 1},
+		{Lo: 0x1eeab, Hi: 0x1eebb, Stride: 1},
+		{Lo: 0x1fbf0, Hi: 0x1fbf9, Stride: 1},
+		{Lo: 0x20000, Hi: 0x2a6df, Stride: 1},
+		{Lo: 0x2a700, Hi: 0x2b739, Stride: 1},
+		{Lo: 0x2b740, Hi: 0x2b81d, Stride: 1},
+		{Lo: 0x2b820, Hi: 0x2cea1, Stride: 1},
+		{Lo: 0x2ceb0, Hi: 0x2ebe0, Stride: 1},
+		{Lo: 0x2ebf0, Hi: 0x2ee5d, Stride: 1},
+		{Lo: 0x2f800, Hi: 0x2fa1d, Stride: 1},
+		{Lo: 0x30000, Hi: 0x3134a, Stride: 1},
+		{Lo: 0x31350, Hi: 0x323af, Stride: 1},
+		{Lo: 0xe0100, Hi: 0xe01ef, Stride: 1},
+	},
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/js_lexer/js_lexer.go b/source/vendor/github.com/evanw/esbuild/internal/js_lexer/js_lexer.go
new file mode 100644
index 0000000..3776f97
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/js_lexer/js_lexer.go
@@ -0,0 +1,2665 @@
+package js_lexer
+
+// The lexer converts a source file to a stream of tokens. Unlike many
+// compilers, esbuild does not run the lexer to completion before the parser is
+// started. Instead, the lexer is called repeatedly by the parser as the parser
+// parses the file. This is because many tokens are context-sensitive and need
+// high-level information from the parser. Examples are regular expression
+// literals and JSX elements.
+//
+// For efficiency, the text associated with textual tokens is stored in two
+// separate ways depending on the token. Identifiers use UTF-8 encoding which
+// allows them to be slices of the input file without allocating extra memory.
+// Strings use UTF-16 encoding so they can represent unicode surrogates
+// accurately.
+
+import (
+	"fmt"
+	"strconv"
+	"strings"
+	"unicode/utf8"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/config"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type T uint8
+
+// If you add a new token, remember to add it to "tokenToString" too
+const (
+	TEndOfFile T = iota
+	TSyntaxError
+
+	// "#!/usr/bin/env node"
+	THashbang
+
+	// Literals
+	TNoSubstitutionTemplateLiteral // Contents are in lexer.StringLiteral ([]uint16)
+	TNumericLiteral                // Contents are in lexer.Number (float64)
+	TStringLiteral                 // Contents are in lexer.StringLiteral ([]uint16)
+	TBigIntegerLiteral             // Contents are in lexer.Identifier (string)
+
+	// Pseudo-literals
+	TTemplateHead   // Contents are in lexer.StringLiteral ([]uint16)
+	TTemplateMiddle // Contents are in lexer.StringLiteral ([]uint16)
+	TTemplateTail   // Contents are in lexer.StringLiteral ([]uint16)
+
+	// Punctuation
+	TAmpersand
+	TAmpersandAmpersand
+	TAsterisk
+	TAsteriskAsterisk
+	TAt
+	TBar
+	TBarBar
+	TCaret
+	TCloseBrace
+	TCloseBracket
+	TCloseParen
+	TColon
+	TComma
+	TDot
+	TDotDotDot
+	TEqualsEquals
+	TEqualsEqualsEquals
+	TEqualsGreaterThan
+	TExclamation
+	TExclamationEquals
+	TExclamationEqualsEquals
+	TGreaterThan
+	TGreaterThanEquals
+	TGreaterThanGreaterThan
+	TGreaterThanGreaterThanGreaterThan
+	TLessThan
+	TLessThanEquals
+	TLessThanLessThan
+	TMinus
+	TMinusMinus
+	TOpenBrace
+	TOpenBracket
+	TOpenParen
+	TPercent
+	TPlus
+	TPlusPlus
+	TQuestion
+	TQuestionDot
+	TQuestionQuestion
+	TSemicolon
+	TSlash
+	TTilde
+
+	// Assignments (keep in sync with IsAssign() below)
+	TAmpersandAmpersandEquals
+	TAmpersandEquals
+	TAsteriskAsteriskEquals
+	TAsteriskEquals
+	TBarBarEquals
+	TBarEquals
+	TCaretEquals
+	TEquals
+	TGreaterThanGreaterThanEquals
+	TGreaterThanGreaterThanGreaterThanEquals
+	TLessThanLessThanEquals
+	TMinusEquals
+	TPercentEquals
+	TPlusEquals
+	TQuestionQuestionEquals
+	TSlashEquals
+
+	// Class-private fields and methods
+	TPrivateIdentifier
+
+	// Identifiers
+	TIdentifier     // Contents are in lexer.Identifier (string)
+	TEscapedKeyword // A keyword that has been escaped as an identifer
+
+	// Reserved words
+	TBreak
+	TCase
+	TCatch
+	TClass
+	TConst
+	TContinue
+	TDebugger
+	TDefault
+	TDelete
+	TDo
+	TElse
+	TEnum
+	TExport
+	TExtends
+	TFalse
+	TFinally
+	TFor
+	TFunction
+	TIf
+	TImport
+	TIn
+	TInstanceof
+	TNew
+	TNull
+	TReturn
+	TSuper
+	TSwitch
+	TThis
+	TThrow
+	TTrue
+	TTry
+	TTypeof
+	TVar
+	TVoid
+	TWhile
+	TWith
+)
+
+func (t T) IsAssign() bool {
+	return t >= TAmpersandAmpersandEquals && t <= TSlashEquals
+}
+
+var Keywords = map[string]T{
+	// Reserved words
+	"break":      TBreak,
+	"case":       TCase,
+	"catch":      TCatch,
+	"class":      TClass,
+	"const":      TConst,
+	"continue":   TContinue,
+	"debugger":   TDebugger,
+	"default":    TDefault,
+	"delete":     TDelete,
+	"do":         TDo,
+	"else":       TElse,
+	"enum":       TEnum,
+	"export":     TExport,
+	"extends":    TExtends,
+	"false":      TFalse,
+	"finally":    TFinally,
+	"for":        TFor,
+	"function":   TFunction,
+	"if":         TIf,
+	"import":     TImport,
+	"in":         TIn,
+	"instanceof": TInstanceof,
+	"new":        TNew,
+	"null":       TNull,
+	"return":     TReturn,
+	"super":      TSuper,
+	"switch":     TSwitch,
+	"this":       TThis,
+	"throw":      TThrow,
+	"true":       TTrue,
+	"try":        TTry,
+	"typeof":     TTypeof,
+	"var":        TVar,
+	"void":       TVoid,
+	"while":      TWhile,
+	"with":       TWith,
+}
+
+var StrictModeReservedWords = map[string]bool{
+	"implements": true,
+	"interface":  true,
+	"let":        true,
+	"package":    true,
+	"private":    true,
+	"protected":  true,
+	"public":     true,
+	"static":     true,
+	"yield":      true,
+}
+
+// This represents a string that is maybe a substring of the current file's
+// "source.Contents" string. The point of doing this is that if it is a
+// substring (the common case), then we can represent it more efficiently.
+//
+// For compactness and performance, the JS AST represents identifiers as a
+// symbol reference instead of as a string. However, we need to track the
+// string between the first pass and the second pass because the string is only
+// resolved to a symbol in the second pass. To avoid allocating extra memory
+// to store the string, we instead use an index+length slice of the original JS
+// source code. That index is what "Start" represents here. The length is just
+// "len(String)".
+//
+// Set "Start" to invalid (the zero value) if "String" is not a substring of
+// "source.Contents". This is the case for escaped identifiers. For example,
+// the identifier "fo\u006f" would be "MaybeSubstring{String: "foo"}". It's
+// critical that any code changing the "String" also set "Start" to the zero
+// value, which is best done by just overwriting the whole "MaybeSubstring".
+//
+// The substring range used to be recovered automatically from the string but
+// that relied on the Go "unsafe" package which can hypothetically break under
+// certain Go compiler optimization passes, so it has been removed and replaced
+// with this more error-prone approach that doesn't use "unsafe".
+type MaybeSubstring struct {
+	String string
+	Start  ast.Index32
+}
+
+type Lexer struct {
+	LegalCommentsBeforeToken     []logger.Range
+	CommentsBeforeToken          []logger.Range
+	AllComments                  []logger.Range
+	Identifier                   MaybeSubstring
+	log                          logger.Log
+	source                       logger.Source
+	JSXFactoryPragmaComment      logger.Span
+	JSXFragmentPragmaComment     logger.Span
+	JSXRuntimePragmaComment      logger.Span
+	JSXImportSourcePragmaComment logger.Span
+	SourceMappingURL             logger.Span
+	BadArrowInTSXSuggestion      string
+
+	// Escape sequences in string literals are decoded lazily because they are
+	// not interpreted inside tagged templates, and tagged templates can contain
+	// invalid escape sequences. If the decoded array is nil, the encoded value
+	// should be passed to "tryToDecodeEscapeSequences" first.
+	decodedStringLiteralOrNil []uint16
+	encodedStringLiteralText  string
+
+	errorSuffix string
+	tracker     logger.LineColumnTracker
+
+	encodedStringLiteralStart int
+
+	Number                          float64
+	current                         int
+	start                           int
+	end                             int
+	ApproximateNewlineCount         int
+	CouldBeBadArrowInTSX            int
+	BadArrowInTSXRange              logger.Range
+	LegacyOctalLoc                  logger.Loc
+	AwaitKeywordLoc                 logger.Loc
+	FnOrArrowStartLoc               logger.Loc
+	PreviousBackslashQuoteInJSX     logger.Range
+	LegacyHTMLCommentRange          logger.Range
+	codePoint                       rune
+	prevErrorLoc                    logger.Loc
+	json                            JSONFlavor
+	Token                           T
+	ts                              config.TSOptions
+	HasNewlineBefore                bool
+	HasCommentBefore                CommentBefore
+	IsLegacyOctalLiteral            bool
+	PrevTokenWasAwaitKeyword        bool
+	rescanCloseBraceAsTemplateToken bool
+	forGlobalName                   bool
+
+	// The log is disabled during speculative scans that may backtrack
+	IsLogDisabled bool
+}
+
+type CommentBefore uint8
+
+const (
+	PureCommentBefore CommentBefore = 1 << iota
+	KeyCommentBefore
+	NoSideEffectsCommentBefore
+)
+
+type LexerPanic struct{}
+
+func NewLexer(log logger.Log, source logger.Source, ts config.TSOptions) Lexer {
+	lexer := Lexer{
+		log:               log,
+		source:            source,
+		tracker:           logger.MakeLineColumnTracker(&source),
+		prevErrorLoc:      logger.Loc{Start: -1},
+		FnOrArrowStartLoc: logger.Loc{Start: -1},
+		ts:                ts,
+		json:              NotJSON,
+	}
+	lexer.step()
+	lexer.Next()
+	return lexer
+}
+
+func NewLexerGlobalName(log logger.Log, source logger.Source) Lexer {
+	lexer := Lexer{
+		log:               log,
+		source:            source,
+		tracker:           logger.MakeLineColumnTracker(&source),
+		prevErrorLoc:      logger.Loc{Start: -1},
+		FnOrArrowStartLoc: logger.Loc{Start: -1},
+		forGlobalName:     true,
+		json:              NotJSON,
+	}
+	lexer.step()
+	lexer.Next()
+	return lexer
+}
+
+type JSONFlavor uint8
+
+const (
+	// Specification: https://json.org/
+	JSON JSONFlavor = iota
+
+	// TypeScript's JSON superset is not documented but appears to allow:
+	// - Comments: https://github.com/microsoft/TypeScript/issues/4987
+	// - Trailing commas
+	// - Full JS number syntax
+	TSConfigJSON
+
+	// This is used by the JavaScript lexer
+	NotJSON
+)
+
+func NewLexerJSON(log logger.Log, source logger.Source, json JSONFlavor, errorSuffix string) Lexer {
+	lexer := Lexer{
+		log:               log,
+		source:            source,
+		tracker:           logger.MakeLineColumnTracker(&source),
+		prevErrorLoc:      logger.Loc{Start: -1},
+		FnOrArrowStartLoc: logger.Loc{Start: -1},
+		errorSuffix:       errorSuffix,
+		json:              json,
+	}
+	lexer.step()
+	lexer.Next()
+	return lexer
+}
+
+func (lexer *Lexer) Loc() logger.Loc {
+	return logger.Loc{Start: int32(lexer.start)}
+}
+
+func (lexer *Lexer) Range() logger.Range {
+	return logger.Range{Loc: logger.Loc{Start: int32(lexer.start)}, Len: int32(lexer.end - lexer.start)}
+}
+
+func (lexer *Lexer) Raw() string {
+	return lexer.source.Contents[lexer.start:lexer.end]
+}
+
+func (lexer *Lexer) rawIdentifier() MaybeSubstring {
+	return MaybeSubstring{lexer.Raw(), ast.MakeIndex32(uint32(lexer.start))}
+}
+
+func (lexer *Lexer) StringLiteral() []uint16 {
+	if lexer.decodedStringLiteralOrNil == nil {
+		// Lazily decode escape sequences if needed
+		if decoded, ok, end := lexer.tryToDecodeEscapeSequences(lexer.encodedStringLiteralStart, lexer.encodedStringLiteralText, true /* reportErrors */); !ok {
+			lexer.end = end
+			lexer.SyntaxError()
+		} else {
+			lexer.decodedStringLiteralOrNil = decoded
+		}
+	}
+	return lexer.decodedStringLiteralOrNil
+}
+
+func (lexer *Lexer) CookedAndRawTemplateContents() ([]uint16, string) {
+	var raw string
+
+	switch lexer.Token {
+	case TNoSubstitutionTemplateLiteral, TTemplateTail:
+		// "`x`" or "}x`"
+		raw = lexer.source.Contents[lexer.start+1 : lexer.end-1]
+
+	case TTemplateHead, TTemplateMiddle:
+		// "`x${" or "}x${"
+		raw = lexer.source.Contents[lexer.start+1 : lexer.end-2]
+	}
+
+	if strings.IndexByte(raw, '\r') != -1 {
+		// From the specification:
+		//
+		// 11.8.6.1 Static Semantics: TV and TRV
+		//
+		// TV excludes the code units of LineContinuation while TRV includes
+		// them. <CR><LF> and <CR> LineTerminatorSequences are normalized to
+		// <LF> for both TV and TRV. An explicit EscapeSequence is needed to
+		// include a <CR> or <CR><LF> sequence.
+
+		bytes := []byte(raw)
+		end := 0
+		i := 0
+
+		for i < len(bytes) {
+			c := bytes[i]
+			i++
+
+			if c == '\r' {
+				// Convert '\r\n' into '\n'
+				if i < len(bytes) && bytes[i] == '\n' {
+					i++
+				}
+
+				// Convert '\r' into '\n'
+				c = '\n'
+			}
+
+			bytes[end] = c
+			end++
+		}
+
+		raw = string(bytes[:end])
+	}
+
+	// This will return nil on failure, which will become "undefined" for the tag
+	cooked, _, _ := lexer.tryToDecodeEscapeSequences(lexer.start+1, raw, false /* reportErrors */)
+	return cooked, raw
+}
+
+func (lexer *Lexer) IsIdentifierOrKeyword() bool {
+	return lexer.Token >= TIdentifier
+}
+
+func (lexer *Lexer) IsContextualKeyword(text string) bool {
+	return lexer.Token == TIdentifier && lexer.Raw() == text
+}
+
+func (lexer *Lexer) ExpectContextualKeyword(text string) {
+	if !lexer.IsContextualKeyword(text) {
+		lexer.ExpectedString(fmt.Sprintf("%q", text))
+	}
+	lexer.Next()
+}
+
+func (lexer *Lexer) SyntaxError() {
+	loc := logger.Loc{Start: int32(lexer.end)}
+	message := "Unexpected end of file"
+	if lexer.end < len(lexer.source.Contents) {
+		c, _ := utf8.DecodeRuneInString(lexer.source.Contents[lexer.end:])
+		if c < 0x20 {
+			message = fmt.Sprintf("Syntax error \"\\x%02X\"", c)
+		} else if c >= 0x80 {
+			message = fmt.Sprintf("Syntax error \"\\u{%x}\"", c)
+		} else if c != '"' {
+			message = fmt.Sprintf("Syntax error \"%c\"", c)
+		} else {
+			message = "Syntax error '\"'"
+		}
+	}
+	lexer.addRangeError(logger.Range{Loc: loc}, message)
+	panic(LexerPanic{})
+}
+
+func (lexer *Lexer) ExpectedString(text string) {
+	// Provide a friendly error message about "await" without "async"
+	if lexer.PrevTokenWasAwaitKeyword {
+		var notes []logger.MsgData
+		if lexer.FnOrArrowStartLoc.Start != -1 {
+			note := lexer.tracker.MsgData(logger.Range{Loc: lexer.FnOrArrowStartLoc},
+				"Consider adding the \"async\" keyword here:")
+			note.Location.Suggestion = "async"
+			notes = []logger.MsgData{note}
+		}
+		lexer.AddRangeErrorWithNotes(RangeOfIdentifier(lexer.source, lexer.AwaitKeywordLoc),
+			"\"await\" can only be used inside an \"async\" function",
+			notes)
+		panic(LexerPanic{})
+	}
+
+	found := fmt.Sprintf("%q", lexer.Raw())
+	if lexer.start == len(lexer.source.Contents) {
+		found = "end of file"
+	}
+
+	suggestion := ""
+	if strings.HasPrefix(text, "\"") && strings.HasSuffix(text, "\"") {
+		suggestion = text[1 : len(text)-1]
+	}
+
+	lexer.addRangeErrorWithSuggestion(lexer.Range(), fmt.Sprintf("Expected %s%s but found %s", text, lexer.errorSuffix, found), suggestion)
+	panic(LexerPanic{})
+}
+
+func (lexer *Lexer) Expected(token T) {
+	if text, ok := tokenToString[token]; ok {
+		lexer.ExpectedString(text)
+	} else {
+		lexer.Unexpected()
+	}
+}
+
+func (lexer *Lexer) Unexpected() {
+	found := fmt.Sprintf("%q", lexer.Raw())
+	if lexer.start == len(lexer.source.Contents) {
+		found = "end of file"
+	}
+	lexer.addRangeError(lexer.Range(), fmt.Sprintf("Unexpected %s%s", found, lexer.errorSuffix))
+	panic(LexerPanic{})
+}
+
+func (lexer *Lexer) Expect(token T) {
+	if lexer.Token != token {
+		lexer.Expected(token)
+	}
+	lexer.Next()
+}
+
+func (lexer *Lexer) ExpectOrInsertSemicolon() {
+	if lexer.Token == TSemicolon || (!lexer.HasNewlineBefore &&
+		lexer.Token != TCloseBrace && lexer.Token != TEndOfFile) {
+		lexer.Expect(TSemicolon)
+	}
+}
+
+// This parses a single "<" token. If that is the first part of a longer token,
+// this function splits off the first "<" and leaves the remainder of the
+// current token as another, smaller token. For example, "<<=" becomes "<=".
+func (lexer *Lexer) ExpectLessThan(isInsideJSXElement bool) {
+	switch lexer.Token {
+	case TLessThan:
+		if isInsideJSXElement {
+			lexer.NextInsideJSXElement()
+		} else {
+			lexer.Next()
+		}
+
+	case TLessThanEquals:
+		lexer.Token = TEquals
+		lexer.start++
+		lexer.maybeExpandEquals()
+
+	case TLessThanLessThan:
+		lexer.Token = TLessThan
+		lexer.start++
+
+	case TLessThanLessThanEquals:
+		lexer.Token = TLessThanEquals
+		lexer.start++
+
+	default:
+		lexer.Expected(TLessThan)
+	}
+}
+
+// This parses a single ">" token. If that is the first part of a longer token,
+// this function splits off the first ">" and leaves the remainder of the
+// current token as another, smaller token. For example, ">>=" becomes ">=".
+func (lexer *Lexer) ExpectGreaterThan(isInsideJSXElement bool) {
+	switch lexer.Token {
+	case TGreaterThan:
+		if isInsideJSXElement {
+			lexer.NextInsideJSXElement()
+		} else {
+			lexer.Next()
+		}
+
+	case TGreaterThanEquals:
+		lexer.Token = TEquals
+		lexer.start++
+		lexer.maybeExpandEquals()
+
+	case TGreaterThanGreaterThan:
+		lexer.Token = TGreaterThan
+		lexer.start++
+
+	case TGreaterThanGreaterThanEquals:
+		lexer.Token = TGreaterThanEquals
+		lexer.start++
+
+	case TGreaterThanGreaterThanGreaterThan:
+		lexer.Token = TGreaterThanGreaterThan
+		lexer.start++
+
+	case TGreaterThanGreaterThanGreaterThanEquals:
+		lexer.Token = TGreaterThanGreaterThanEquals
+		lexer.start++
+
+	default:
+		lexer.Expected(TGreaterThan)
+	}
+}
+
+func (lexer *Lexer) maybeExpandEquals() {
+	switch lexer.codePoint {
+	case '>':
+		// "=" + ">" = "=>"
+		lexer.Token = TEqualsGreaterThan
+		lexer.step()
+
+	case '=':
+		// "=" + "=" = "=="
+		lexer.Token = TEqualsEquals
+		lexer.step()
+
+		if lexer.Token == '=' {
+			// "=" + "==" = "==="
+			lexer.Token = TEqualsEqualsEquals
+			lexer.step()
+		}
+	}
+}
+
+func RangeOfIdentifier(source logger.Source, loc logger.Loc) logger.Range {
+	text := source.Contents[loc.Start:]
+	if len(text) == 0 {
+		return logger.Range{Loc: loc, Len: 0}
+	}
+
+	i := 0
+	c, _ := utf8.DecodeRuneInString(text[i:])
+
+	// Handle private names
+	if c == '#' {
+		i++
+		c, _ = utf8.DecodeRuneInString(text[i:])
+	}
+
+	if js_ast.IsIdentifierStart(c) || c == '\\' {
+		// Search for the end of the identifier
+		for i < len(text) {
+			c2, width2 := utf8.DecodeRuneInString(text[i:])
+			if c2 == '\\' {
+				i += width2
+
+				// Skip over bracketed unicode escapes such as "\u{10000}"
+				if i+2 < len(text) && text[i] == 'u' && text[i+1] == '{' {
+					i += 2
+					for i < len(text) {
+						if text[i] == '}' {
+							i++
+							break
+						}
+						i++
+					}
+				}
+			} else if !js_ast.IsIdentifierContinue(c2) {
+				return logger.Range{Loc: loc, Len: int32(i)}
+			} else {
+				i += width2
+			}
+		}
+	}
+
+	// When minifying, this identifier may have originally been a string
+	return source.RangeOfString(loc)
+}
+
+type KeyOrValue uint8
+
+const (
+	KeyRange KeyOrValue = iota
+	ValueRange
+	KeyAndValueRange
+)
+
+func RangeOfImportAssertOrWith(source logger.Source, assertOrWith ast.AssertOrWithEntry, which KeyOrValue) logger.Range {
+	if which == KeyRange {
+		return RangeOfIdentifier(source, assertOrWith.KeyLoc)
+	}
+	if which == ValueRange {
+		return source.RangeOfString(assertOrWith.ValueLoc)
+	}
+	loc := RangeOfIdentifier(source, assertOrWith.KeyLoc).Loc
+	return logger.Range{Loc: loc, Len: source.RangeOfString(assertOrWith.ValueLoc).End() - loc.Start}
+}
+
+func (lexer *Lexer) ExpectJSXElementChild(token T) {
+	if lexer.Token != token {
+		lexer.Expected(token)
+	}
+	lexer.NextJSXElementChild()
+}
+
+func (lexer *Lexer) NextJSXElementChild() {
+	lexer.HasNewlineBefore = false
+	originalStart := lexer.end
+
+	for {
+		lexer.start = lexer.end
+		lexer.Token = 0
+
+		switch lexer.codePoint {
+		case -1: // This indicates the end of the file
+			lexer.Token = TEndOfFile
+
+		case '{':
+			lexer.step()
+			lexer.Token = TOpenBrace
+
+		case '<':
+			lexer.step()
+			lexer.Token = TLessThan
+
+		default:
+			needsFixing := false
+
+		stringLiteral:
+			for {
+				switch lexer.codePoint {
+				case -1, '{', '<':
+					// Stop when the string ends
+					break stringLiteral
+
+				case '&', '\r', '\n', '\u2028', '\u2029':
+					// This needs fixing if it has an entity or if it's a multi-line string
+					needsFixing = true
+					lexer.step()
+
+				case '}', '>':
+					// These technically aren't valid JSX: https://facebook.github.io/jsx/
+					//
+					//   JSXTextCharacter :
+					//     * SourceCharacter but not one of {, <, > or }
+					//
+					var replacement string
+					if lexer.codePoint == '}' {
+						replacement = "{'}'}"
+					} else {
+						replacement = "{'>'}"
+					}
+					msg := logger.Msg{
+						Kind: logger.Error,
+						Data: lexer.tracker.MsgData(logger.Range{Loc: logger.Loc{Start: int32(lexer.end)}, Len: 1},
+							fmt.Sprintf("The character \"%c\" is not valid inside a JSX element", lexer.codePoint)),
+					}
+
+					// Attempt to provide a better error message if this looks like an arrow function
+					if lexer.CouldBeBadArrowInTSX > 0 && lexer.codePoint == '>' && lexer.source.Contents[lexer.end-1] == '=' {
+						msg.Notes = []logger.MsgData{lexer.tracker.MsgData(lexer.BadArrowInTSXRange,
+							"TypeScript's TSX syntax interprets arrow functions with a single generic type parameter as an opening JSX element. "+
+								"If you want it to be interpreted as an arrow function instead, you need to add a trailing comma after the type parameter to disambiguate:")}
+						msg.Notes[0].Location.Suggestion = lexer.BadArrowInTSXSuggestion
+					} else {
+						msg.Notes = []logger.MsgData{{Text: fmt.Sprintf("Did you mean to escape it as %q instead?", replacement)}}
+						msg.Data.Location.Suggestion = replacement
+						if !lexer.ts.Parse {
+							// TypeScript treats this as an error but Babel doesn't treat this
+							// as an error yet, so allow this in JS for now. Babel version 8
+							// was supposed to be released in 2021 but was never released. If
+							// it's released in the future, this can be changed to an error too.
+							//
+							// More context:
+							// * TypeScript change: https://github.com/microsoft/TypeScript/issues/36341
+							// * Babel 8 change: https://github.com/babel/babel/issues/11042
+							// * Babel 8 release: https://github.com/babel/babel/issues/10746
+							//
+							msg.Kind = logger.Warning
+						}
+					}
+
+					lexer.log.AddMsg(msg)
+					lexer.step()
+
+				default:
+					// Non-ASCII strings need the slow path
+					if lexer.codePoint >= 0x80 {
+						needsFixing = true
+					}
+					lexer.step()
+				}
+			}
+
+			lexer.Token = TStringLiteral
+			text := lexer.source.Contents[originalStart:lexer.end]
+
+			if needsFixing {
+				// Slow path
+				lexer.decodedStringLiteralOrNil = fixWhitespaceAndDecodeJSXEntities(text)
+			} else {
+				// Fast path
+				n := len(text)
+				copy := make([]uint16, n)
+				for i := 0; i < n; i++ {
+					copy[i] = uint16(text[i])
+				}
+				lexer.decodedStringLiteralOrNil = copy
+			}
+		}
+
+		break
+	}
+}
+
+func (lexer *Lexer) ExpectInsideJSXElement(token T) {
+	if lexer.Token != token {
+		lexer.Expected(token)
+	}
+	lexer.NextInsideJSXElement()
+}
+
+func (lexer *Lexer) NextInsideJSXElement() {
+	lexer.HasNewlineBefore = false
+
+	for {
+		lexer.start = lexer.end
+		lexer.Token = 0
+
+		switch lexer.codePoint {
+		case -1: // This indicates the end of the file
+			lexer.Token = TEndOfFile
+
+		case '\r', '\n', '\u2028', '\u2029':
+			lexer.step()
+			lexer.HasNewlineBefore = true
+			continue
+
+		case '\t', ' ':
+			lexer.step()
+			continue
+
+		case '.':
+			lexer.step()
+			lexer.Token = TDot
+
+		case ':':
+			lexer.step()
+			lexer.Token = TColon
+
+		case '=':
+			lexer.step()
+			lexer.Token = TEquals
+
+		case '{':
+			lexer.step()
+			lexer.Token = TOpenBrace
+
+		case '}':
+			lexer.step()
+			lexer.Token = TCloseBrace
+
+		case '<':
+			lexer.step()
+			lexer.Token = TLessThan
+
+		case '>':
+			lexer.step()
+			lexer.Token = TGreaterThan
+
+		case '/':
+			// '/' or '//' or '/* ... */'
+			lexer.step()
+			switch lexer.codePoint {
+			case '/':
+			singleLineComment:
+				for {
+					lexer.step()
+					switch lexer.codePoint {
+					case '\r', '\n', '\u2028', '\u2029':
+						break singleLineComment
+
+					case -1: // This indicates the end of the file
+						break singleLineComment
+					}
+				}
+				continue
+
+			case '*':
+				lexer.step()
+				startRange := lexer.Range()
+			multiLineComment:
+				for {
+					switch lexer.codePoint {
+					case '*':
+						lexer.step()
+						if lexer.codePoint == '/' {
+							lexer.step()
+							break multiLineComment
+						}
+
+					case '\r', '\n', '\u2028', '\u2029':
+						lexer.step()
+						lexer.HasNewlineBefore = true
+
+					case -1: // This indicates the end of the file
+						lexer.start = lexer.end
+						lexer.AddRangeErrorWithNotes(logger.Range{Loc: lexer.Loc()}, "Expected \"*/\" to terminate multi-line comment",
+							[]logger.MsgData{lexer.tracker.MsgData(startRange, "The multi-line comment starts here:")})
+						panic(LexerPanic{})
+
+					default:
+						lexer.step()
+					}
+				}
+				continue
+
+			default:
+				lexer.Token = TSlash
+			}
+
+		case '\'', '"':
+			var backslash logger.Range
+			quote := lexer.codePoint
+			needsDecode := false
+			lexer.step()
+
+		stringLiteral:
+			for {
+				switch lexer.codePoint {
+				case -1: // This indicates the end of the file
+					lexer.SyntaxError()
+
+				case '&':
+					needsDecode = true
+					lexer.step()
+
+				case '\\':
+					backslash = logger.Range{Loc: logger.Loc{Start: int32(lexer.end)}, Len: 1}
+					lexer.step()
+					continue
+
+				case quote:
+					if backslash.Len > 0 {
+						backslash.Len++
+						lexer.PreviousBackslashQuoteInJSX = backslash
+					}
+					lexer.step()
+					break stringLiteral
+
+				default:
+					// Non-ASCII strings need the slow path
+					if lexer.codePoint >= 0x80 {
+						needsDecode = true
+					}
+					lexer.step()
+				}
+				backslash = logger.Range{}
+			}
+
+			lexer.Token = TStringLiteral
+			text := lexer.source.Contents[lexer.start+1 : lexer.end-1]
+
+			if needsDecode {
+				// Slow path
+				lexer.decodedStringLiteralOrNil = decodeJSXEntities([]uint16{}, text)
+			} else {
+				// Fast path
+				n := len(text)
+				copy := make([]uint16, n)
+				for i := 0; i < n; i++ {
+					copy[i] = uint16(text[i])
+				}
+				lexer.decodedStringLiteralOrNil = copy
+			}
+
+		default:
+			// Check for unusual whitespace characters
+			if js_ast.IsWhitespace(lexer.codePoint) {
+				lexer.step()
+				continue
+			}
+
+			if js_ast.IsIdentifierStart(lexer.codePoint) {
+				lexer.step()
+				for js_ast.IsIdentifierContinue(lexer.codePoint) || lexer.codePoint == '-' {
+					lexer.step()
+				}
+
+				lexer.Identifier = lexer.rawIdentifier()
+				lexer.Token = TIdentifier
+				break
+			}
+
+			lexer.end = lexer.current
+			lexer.Token = TSyntaxError
+		}
+
+		return
+	}
+}
+
+func (lexer *Lexer) Next() {
+	lexer.HasNewlineBefore = lexer.end == 0
+	lexer.HasCommentBefore = 0
+	lexer.PrevTokenWasAwaitKeyword = false
+	lexer.LegalCommentsBeforeToken = lexer.LegalCommentsBeforeToken[:0]
+	lexer.CommentsBeforeToken = lexer.CommentsBeforeToken[:0]
+
+	for {
+		lexer.start = lexer.end
+		lexer.Token = 0
+
+		switch lexer.codePoint {
+		case -1: // This indicates the end of the file
+			lexer.Token = TEndOfFile
+
+		case '#':
+			if lexer.start == 0 && strings.HasPrefix(lexer.source.Contents, "#!") {
+				// "#!/usr/bin/env node"
+				lexer.Token = THashbang
+			hashbang:
+				for {
+					lexer.step()
+					switch lexer.codePoint {
+					case '\r', '\n', '\u2028', '\u2029':
+						break hashbang
+
+					case -1: // This indicates the end of the file
+						break hashbang
+					}
+				}
+				lexer.Identifier = lexer.rawIdentifier()
+			} else {
+				// "#foo"
+				lexer.step()
+				if lexer.codePoint == '\\' {
+					lexer.Identifier, _ = lexer.scanIdentifierWithEscapes(privateIdentifier)
+				} else {
+					if !js_ast.IsIdentifierStart(lexer.codePoint) {
+						lexer.SyntaxError()
+					}
+					lexer.step()
+					for js_ast.IsIdentifierContinue(lexer.codePoint) {
+						lexer.step()
+					}
+					if lexer.codePoint == '\\' {
+						lexer.Identifier, _ = lexer.scanIdentifierWithEscapes(privateIdentifier)
+					} else {
+						lexer.Identifier = lexer.rawIdentifier()
+					}
+				}
+				lexer.Token = TPrivateIdentifier
+			}
+
+		case '\r', '\n', '\u2028', '\u2029':
+			lexer.step()
+			lexer.HasNewlineBefore = true
+			continue
+
+		case '\t', ' ':
+			lexer.step()
+			continue
+
+		case '(':
+			lexer.step()
+			lexer.Token = TOpenParen
+
+		case ')':
+			lexer.step()
+			lexer.Token = TCloseParen
+
+		case '[':
+			lexer.step()
+			lexer.Token = TOpenBracket
+
+		case ']':
+			lexer.step()
+			lexer.Token = TCloseBracket
+
+		case '{':
+			lexer.step()
+			lexer.Token = TOpenBrace
+
+		case '}':
+			lexer.step()
+			lexer.Token = TCloseBrace
+
+		case ',':
+			lexer.step()
+			lexer.Token = TComma
+
+		case ':':
+			lexer.step()
+			lexer.Token = TColon
+
+		case ';':
+			lexer.step()
+			lexer.Token = TSemicolon
+
+		case '@':
+			lexer.step()
+			lexer.Token = TAt
+
+		case '~':
+			lexer.step()
+			lexer.Token = TTilde
+
+		case '?':
+			// '?' or '?.' or '??' or '??='
+			lexer.step()
+			switch lexer.codePoint {
+			case '?':
+				lexer.step()
+				switch lexer.codePoint {
+				case '=':
+					lexer.step()
+					lexer.Token = TQuestionQuestionEquals
+				default:
+					lexer.Token = TQuestionQuestion
+				}
+			case '.':
+				lexer.Token = TQuestion
+				current := lexer.current
+				contents := lexer.source.Contents
+
+				// Lookahead to disambiguate with 'a?.1:b'
+				if current < len(contents) {
+					c := contents[current]
+					if c < '0' || c > '9' {
+						lexer.step()
+						lexer.Token = TQuestionDot
+					}
+				}
+			default:
+				lexer.Token = TQuestion
+			}
+
+		case '%':
+			// '%' or '%='
+			lexer.step()
+			switch lexer.codePoint {
+			case '=':
+				lexer.step()
+				lexer.Token = TPercentEquals
+			default:
+				lexer.Token = TPercent
+			}
+
+		case '&':
+			// '&' or '&=' or '&&' or '&&='
+			lexer.step()
+			switch lexer.codePoint {
+			case '=':
+				lexer.step()
+				lexer.Token = TAmpersandEquals
+			case '&':
+				lexer.step()
+				switch lexer.codePoint {
+				case '=':
+					lexer.step()
+					lexer.Token = TAmpersandAmpersandEquals
+				default:
+					lexer.Token = TAmpersandAmpersand
+				}
+			default:
+				lexer.Token = TAmpersand
+			}
+
+		case '|':
+			// '|' or '|=' or '||' or '||='
+			lexer.step()
+			switch lexer.codePoint {
+			case '=':
+				lexer.step()
+				lexer.Token = TBarEquals
+			case '|':
+				lexer.step()
+				switch lexer.codePoint {
+				case '=':
+					lexer.step()
+					lexer.Token = TBarBarEquals
+				default:
+					lexer.Token = TBarBar
+				}
+			default:
+				lexer.Token = TBar
+			}
+
+		case '^':
+			// '^' or '^='
+			lexer.step()
+			switch lexer.codePoint {
+			case '=':
+				lexer.step()
+				lexer.Token = TCaretEquals
+			default:
+				lexer.Token = TCaret
+			}
+
+		case '+':
+			// '+' or '+=' or '++'
+			lexer.step()
+			switch lexer.codePoint {
+			case '=':
+				lexer.step()
+				lexer.Token = TPlusEquals
+			case '+':
+				lexer.step()
+				lexer.Token = TPlusPlus
+			default:
+				lexer.Token = TPlus
+			}
+
+		case '-':
+			// '-' or '-=' or '--' or '-->'
+			lexer.step()
+			switch lexer.codePoint {
+			case '=':
+				lexer.step()
+				lexer.Token = TMinusEquals
+			case '-':
+				lexer.step()
+
+				// Handle legacy HTML-style comments
+				if lexer.codePoint == '>' && lexer.HasNewlineBefore {
+					lexer.step()
+					lexer.LegacyHTMLCommentRange = lexer.Range()
+					lexer.log.AddID(logger.MsgID_JS_HTMLCommentInJS, logger.Warning, &lexer.tracker, lexer.Range(),
+						"Treating \"-->\" as the start of a legacy HTML single-line comment")
+				singleLineHTMLCloseComment:
+					for {
+						switch lexer.codePoint {
+						case '\r', '\n', '\u2028', '\u2029':
+							break singleLineHTMLCloseComment
+
+						case -1: // This indicates the end of the file
+							break singleLineHTMLCloseComment
+						}
+						lexer.step()
+					}
+					continue
+				}
+
+				lexer.Token = TMinusMinus
+			default:
+				lexer.Token = TMinus
+				if lexer.json == JSON && lexer.codePoint != '.' && (lexer.codePoint < '0' || lexer.codePoint > '9') {
+					lexer.Unexpected()
+				}
+			}
+
+		case '*':
+			// '*' or '*=' or '**' or '**='
+			lexer.step()
+			switch lexer.codePoint {
+			case '=':
+				lexer.step()
+				lexer.Token = TAsteriskEquals
+
+			case '*':
+				lexer.step()
+				switch lexer.codePoint {
+				case '=':
+					lexer.step()
+					lexer.Token = TAsteriskAsteriskEquals
+
+				default:
+					lexer.Token = TAsteriskAsterisk
+				}
+
+			default:
+				lexer.Token = TAsterisk
+			}
+
+		case '/':
+			// '/' or '/=' or '//' or '/* ... */'
+			lexer.step()
+			if lexer.forGlobalName {
+				lexer.Token = TSlash
+				break
+			}
+			switch lexer.codePoint {
+			case '=':
+				lexer.step()
+				lexer.Token = TSlashEquals
+
+			case '/':
+			singleLineComment:
+				for {
+					lexer.step()
+					switch lexer.codePoint {
+					case '\r', '\n', '\u2028', '\u2029':
+						break singleLineComment
+
+					case -1: // This indicates the end of the file
+						break singleLineComment
+					}
+				}
+				if lexer.json == JSON {
+					lexer.addRangeError(lexer.Range(), "JSON does not support comments")
+				}
+				lexer.scanCommentText()
+				continue
+
+			case '*':
+				lexer.step()
+				startRange := lexer.Range()
+			multiLineComment:
+				for {
+					switch lexer.codePoint {
+					case '*':
+						lexer.step()
+						if lexer.codePoint == '/' {
+							lexer.step()
+							break multiLineComment
+						}
+
+					case '\r', '\n', '\u2028', '\u2029':
+						lexer.step()
+						lexer.HasNewlineBefore = true
+
+					case -1: // This indicates the end of the file
+						lexer.start = lexer.end
+						lexer.AddRangeErrorWithNotes(logger.Range{Loc: lexer.Loc()}, "Expected \"*/\" to terminate multi-line comment",
+							[]logger.MsgData{lexer.tracker.MsgData(startRange, "The multi-line comment starts here:")})
+						panic(LexerPanic{})
+
+					default:
+						lexer.step()
+					}
+				}
+				if lexer.json == JSON {
+					lexer.addRangeError(lexer.Range(), "JSON does not support comments")
+				}
+				lexer.scanCommentText()
+				continue
+
+			default:
+				lexer.Token = TSlash
+			}
+
+		case '=':
+			// '=' or '=>' or '==' or '==='
+			lexer.step()
+			switch lexer.codePoint {
+			case '>':
+				lexer.step()
+				lexer.Token = TEqualsGreaterThan
+			case '=':
+				lexer.step()
+				switch lexer.codePoint {
+				case '=':
+					lexer.step()
+					lexer.Token = TEqualsEqualsEquals
+				default:
+					lexer.Token = TEqualsEquals
+				}
+			default:
+				lexer.Token = TEquals
+			}
+
+		case '<':
+			// '<' or '<<' or '<=' or '<<=' or '<!--'
+			lexer.step()
+			switch lexer.codePoint {
+			case '=':
+				lexer.step()
+				lexer.Token = TLessThanEquals
+			case '<':
+				lexer.step()
+				switch lexer.codePoint {
+				case '=':
+					lexer.step()
+					lexer.Token = TLessThanLessThanEquals
+				default:
+					lexer.Token = TLessThanLessThan
+				}
+
+				// Handle legacy HTML-style comments
+			case '!':
+				if strings.HasPrefix(lexer.source.Contents[lexer.start:], "<!--") {
+					lexer.step()
+					lexer.step()
+					lexer.step()
+					lexer.LegacyHTMLCommentRange = lexer.Range()
+					lexer.log.AddID(logger.MsgID_JS_HTMLCommentInJS, logger.Warning, &lexer.tracker, lexer.Range(),
+						"Treating \"<!--\" as the start of a legacy HTML single-line comment")
+				singleLineHTMLOpenComment:
+					for {
+						switch lexer.codePoint {
+						case '\r', '\n', '\u2028', '\u2029':
+							break singleLineHTMLOpenComment
+
+						case -1: // This indicates the end of the file
+							break singleLineHTMLOpenComment
+						}
+						lexer.step()
+					}
+					continue
+				}
+
+				lexer.Token = TLessThan
+
+			default:
+				lexer.Token = TLessThan
+			}
+
+		case '>':
+			// '>' or '>>' or '>>>' or '>=' or '>>=' or '>>>='
+			lexer.step()
+			switch lexer.codePoint {
+			case '=':
+				lexer.step()
+				lexer.Token = TGreaterThanEquals
+			case '>':
+				lexer.step()
+				switch lexer.codePoint {
+				case '=':
+					lexer.step()
+					lexer.Token = TGreaterThanGreaterThanEquals
+				case '>':
+					lexer.step()
+					switch lexer.codePoint {
+					case '=':
+						lexer.step()
+						lexer.Token = TGreaterThanGreaterThanGreaterThanEquals
+					default:
+						lexer.Token = TGreaterThanGreaterThanGreaterThan
+					}
+				default:
+					lexer.Token = TGreaterThanGreaterThan
+				}
+			default:
+				lexer.Token = TGreaterThan
+			}
+
+		case '!':
+			// '!' or '!=' or '!=='
+			lexer.step()
+			switch lexer.codePoint {
+			case '=':
+				lexer.step()
+				switch lexer.codePoint {
+				case '=':
+					lexer.step()
+					lexer.Token = TExclamationEqualsEquals
+				default:
+					lexer.Token = TExclamationEquals
+				}
+			default:
+				lexer.Token = TExclamation
+			}
+
+		case '\'', '"', '`':
+			quote := lexer.codePoint
+			needsSlowPath := false
+			suffixLen := 1
+
+			if quote != '`' {
+				lexer.Token = TStringLiteral
+			} else if lexer.rescanCloseBraceAsTemplateToken {
+				lexer.Token = TTemplateTail
+			} else {
+				lexer.Token = TNoSubstitutionTemplateLiteral
+			}
+			lexer.step()
+
+		stringLiteral:
+			for {
+				switch lexer.codePoint {
+				case '\\':
+					needsSlowPath = true
+					lexer.step()
+
+					// Handle Windows CRLF
+					if lexer.codePoint == '\r' && lexer.json != JSON {
+						lexer.step()
+						if lexer.codePoint == '\n' {
+							lexer.step()
+						}
+						continue
+					}
+
+				case -1: // This indicates the end of the file
+					lexer.addRangeError(logger.Range{Loc: logger.Loc{Start: int32(lexer.end)}}, "Unterminated string literal")
+					panic(LexerPanic{})
+
+				case '\r':
+					if quote != '`' {
+						lexer.addRangeError(logger.Range{Loc: logger.Loc{Start: int32(lexer.end)}}, "Unterminated string literal")
+						panic(LexerPanic{})
+					}
+
+					// Template literals require newline normalization
+					needsSlowPath = true
+
+				case '\n':
+					if quote != '`' {
+						lexer.addRangeError(logger.Range{Loc: logger.Loc{Start: int32(lexer.end)}}, "Unterminated string literal")
+						panic(LexerPanic{})
+					}
+
+				case '$':
+					if quote == '`' {
+						lexer.step()
+						if lexer.codePoint == '{' {
+							suffixLen = 2
+							lexer.step()
+							if lexer.rescanCloseBraceAsTemplateToken {
+								lexer.Token = TTemplateMiddle
+							} else {
+								lexer.Token = TTemplateHead
+							}
+							break stringLiteral
+						}
+						continue stringLiteral
+					}
+
+				case quote:
+					lexer.step()
+					break stringLiteral
+
+				default:
+					// Non-ASCII strings need the slow path
+					if lexer.codePoint >= 0x80 {
+						needsSlowPath = true
+					} else if lexer.json == JSON && lexer.codePoint < 0x20 {
+						lexer.SyntaxError()
+					}
+				}
+				lexer.step()
+			}
+
+			text := lexer.source.Contents[lexer.start+1 : lexer.end-suffixLen]
+
+			if needsSlowPath {
+				// Slow path
+				lexer.decodedStringLiteralOrNil = nil
+				lexer.encodedStringLiteralStart = lexer.start + 1
+				lexer.encodedStringLiteralText = text
+			} else {
+				// Fast path
+				n := len(text)
+				copy := make([]uint16, n)
+				for i := 0; i < n; i++ {
+					copy[i] = uint16(text[i])
+				}
+				lexer.decodedStringLiteralOrNil = copy
+			}
+
+			if quote == '\'' && (lexer.json == JSON || lexer.json == TSConfigJSON) {
+				lexer.addRangeError(lexer.Range(), "JSON strings must use double quotes")
+			}
+
+		// Note: This case is hot in profiles
+		case '_', '$',
+			'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm',
+			'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z',
+			'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M',
+			'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z':
+			// This is a fast path for long ASCII identifiers. Doing this in a loop
+			// first instead of doing "step()" and "js_ast.IsIdentifierContinue()" like we
+			// do after this is noticeably faster in the common case of ASCII-only
+			// text. For example, doing this sped up end-to-end consuming of a large
+			// TypeScript type declaration file from 97ms to 79ms (around 20% faster).
+			contents := lexer.source.Contents
+			n := len(contents)
+			i := lexer.current
+			for i < n {
+				c := contents[i]
+				if (c < 'a' || c > 'z') && (c < 'A' || c > 'Z') && (c < '0' || c > '9') && c != '_' && c != '$' {
+					break
+				}
+				i++
+			}
+			lexer.current = i
+
+			// Now do the slow path for any remaining non-ASCII identifier characters
+			lexer.step()
+			if lexer.codePoint >= 0x80 {
+				for js_ast.IsIdentifierContinue(lexer.codePoint) {
+					lexer.step()
+				}
+			}
+
+			// If there's a slash, then we're in the extra-slow (and extra-rare) case
+			// where the identifier has embedded escapes
+			if lexer.codePoint == '\\' {
+				lexer.Identifier, lexer.Token = lexer.scanIdentifierWithEscapes(normalIdentifier)
+				break
+			}
+
+			// Otherwise (if there was no escape) we can slice the code verbatim
+			lexer.Identifier = lexer.rawIdentifier()
+			lexer.Token = Keywords[lexer.Raw()]
+			if lexer.Token == 0 {
+				lexer.Token = TIdentifier
+			}
+
+		case '\\':
+			lexer.Identifier, lexer.Token = lexer.scanIdentifierWithEscapes(normalIdentifier)
+
+		case '.', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9':
+			lexer.parseNumericLiteralOrDot()
+
+		default:
+			// Check for unusual whitespace characters
+			if js_ast.IsWhitespace(lexer.codePoint) {
+				lexer.step()
+				continue
+			}
+
+			if js_ast.IsIdentifierStart(lexer.codePoint) {
+				lexer.step()
+				for js_ast.IsIdentifierContinue(lexer.codePoint) {
+					lexer.step()
+				}
+				if lexer.codePoint == '\\' {
+					lexer.Identifier, lexer.Token = lexer.scanIdentifierWithEscapes(normalIdentifier)
+				} else {
+					lexer.Token = TIdentifier
+					lexer.Identifier = lexer.rawIdentifier()
+				}
+				break
+			}
+
+			lexer.end = lexer.current
+			lexer.Token = TSyntaxError
+		}
+
+		return
+	}
+}
+
+type identifierKind uint8
+
+const (
+	normalIdentifier identifierKind = iota
+	privateIdentifier
+)
+
+// This is an edge case that doesn't really exist in the wild, so it doesn't
+// need to be as fast as possible.
+func (lexer *Lexer) scanIdentifierWithEscapes(kind identifierKind) (MaybeSubstring, T) {
+	// First pass: scan over the identifier to see how long it is
+	for {
+		// Scan a unicode escape sequence. There is at least one because that's
+		// what caused us to get on this slow path in the first place.
+		if lexer.codePoint == '\\' {
+			lexer.step()
+			if lexer.codePoint != 'u' {
+				lexer.SyntaxError()
+			}
+			lexer.step()
+			if lexer.codePoint == '{' {
+				// Variable-length
+				lexer.step()
+				for lexer.codePoint != '}' {
+					switch lexer.codePoint {
+					case '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
+						'a', 'b', 'c', 'd', 'e', 'f',
+						'A', 'B', 'C', 'D', 'E', 'F':
+						lexer.step()
+					default:
+						lexer.SyntaxError()
+					}
+				}
+				lexer.step()
+			} else {
+				// Fixed-length
+				for j := 0; j < 4; j++ {
+					switch lexer.codePoint {
+					case '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
+						'a', 'b', 'c', 'd', 'e', 'f',
+						'A', 'B', 'C', 'D', 'E', 'F':
+						lexer.step()
+					default:
+						lexer.SyntaxError()
+					}
+				}
+			}
+			continue
+		}
+
+		// Stop when we reach the end of the identifier
+		if !js_ast.IsIdentifierContinue(lexer.codePoint) {
+			break
+		}
+		lexer.step()
+	}
+
+	// Second pass: re-use our existing escape sequence parser
+	decoded, ok, end := lexer.tryToDecodeEscapeSequences(lexer.start, lexer.Raw(), true /* reportErrors */)
+	if !ok {
+		lexer.end = end
+		lexer.SyntaxError()
+	}
+	text := string(helpers.UTF16ToString(decoded))
+
+	// Even though it was escaped, it must still be a valid identifier
+	identifier := text
+	if kind == privateIdentifier {
+		identifier = identifier[1:] // Skip over the "#"
+	}
+	if !js_ast.IsIdentifier(identifier) {
+		lexer.addRangeError(logger.Range{Loc: logger.Loc{Start: int32(lexer.start)}, Len: int32(lexer.end - lexer.start)},
+			fmt.Sprintf("Invalid identifier: %q", text))
+	}
+
+	// Escaped keywords are not allowed to work as actual keywords, but they are
+	// allowed wherever we allow identifiers or keywords. For example:
+	//
+	//   // This is an error (equivalent to "var var;")
+	//   var \u0076\u0061\u0072;
+	//
+	//   // This is an error (equivalent to "var foo;" except for this rule)
+	//   \u0076\u0061\u0072 foo;
+	//
+	//   // This is an fine (equivalent to "foo.var;")
+	//   foo.\u0076\u0061\u0072;
+	//
+	if Keywords[text] != 0 {
+		return MaybeSubstring{String: text}, TEscapedKeyword
+	} else {
+		return MaybeSubstring{String: text}, TIdentifier
+	}
+}
+
+func (lexer *Lexer) parseNumericLiteralOrDot() {
+	// Number or dot
+	first := lexer.codePoint
+	lexer.step()
+
+	// Dot without a digit after it
+	if first == '.' && (lexer.codePoint < '0' || lexer.codePoint > '9') {
+		// "..."
+		if lexer.codePoint == '.' &&
+			lexer.current < len(lexer.source.Contents) &&
+			lexer.source.Contents[lexer.current] == '.' {
+			lexer.step()
+			lexer.step()
+			lexer.Token = TDotDotDot
+			return
+		}
+
+		// "."
+		lexer.Token = TDot
+		return
+	}
+
+	underscoreCount := 0
+	lastUnderscoreEnd := 0
+	hasDotOrExponent := first == '.'
+	isMissingDigitAfterDot := false
+	base := 0.0
+	lexer.IsLegacyOctalLiteral = false
+
+	// Assume this is a number, but potentially change to a bigint later
+	lexer.Token = TNumericLiteral
+
+	// Check for binary, octal, or hexadecimal literal
+	if first == '0' {
+		switch lexer.codePoint {
+		case 'b', 'B':
+			base = 2
+
+		case 'o', 'O':
+			base = 8
+
+		case 'x', 'X':
+			base = 16
+
+		case '0', '1', '2', '3', '4', '5', '6', '7', '_':
+			base = 8
+			lexer.IsLegacyOctalLiteral = true
+
+		case '8', '9':
+			lexer.IsLegacyOctalLiteral = true
+		}
+	}
+
+	if base != 0 {
+		// Integer literal
+		isFirst := true
+		isInvalidLegacyOctalLiteral := false
+		lexer.Number = 0
+		if !lexer.IsLegacyOctalLiteral {
+			lexer.step()
+		}
+
+	integerLiteral:
+		for {
+			switch lexer.codePoint {
+			case '_':
+				// Cannot have multiple underscores in a row
+				if lastUnderscoreEnd > 0 && lexer.end == lastUnderscoreEnd+1 {
+					lexer.SyntaxError()
+				}
+
+				// The first digit must exist
+				if isFirst || lexer.IsLegacyOctalLiteral {
+					lexer.SyntaxError()
+				}
+
+				lastUnderscoreEnd = lexer.end
+				underscoreCount++
+
+			case '0', '1':
+				lexer.Number = lexer.Number*base + float64(lexer.codePoint-'0')
+
+			case '2', '3', '4', '5', '6', '7':
+				if base == 2 {
+					lexer.SyntaxError()
+				}
+				lexer.Number = lexer.Number*base + float64(lexer.codePoint-'0')
+
+			case '8', '9':
+				if lexer.IsLegacyOctalLiteral {
+					isInvalidLegacyOctalLiteral = true
+				} else if base < 10 {
+					lexer.SyntaxError()
+				}
+				lexer.Number = lexer.Number*base + float64(lexer.codePoint-'0')
+
+			case 'A', 'B', 'C', 'D', 'E', 'F':
+				if base != 16 {
+					lexer.SyntaxError()
+				}
+				lexer.Number = lexer.Number*base + float64(lexer.codePoint+10-'A')
+
+			case 'a', 'b', 'c', 'd', 'e', 'f':
+				if base != 16 {
+					lexer.SyntaxError()
+				}
+				lexer.Number = lexer.Number*base + float64(lexer.codePoint+10-'a')
+
+			default:
+				// The first digit must exist
+				if isFirst {
+					lexer.SyntaxError()
+				}
+
+				break integerLiteral
+			}
+
+			lexer.step()
+			isFirst = false
+		}
+
+		isBigIntegerLiteral := lexer.codePoint == 'n' && !hasDotOrExponent
+
+		// Slow path: do we need to re-scan the input as text?
+		if isBigIntegerLiteral || isInvalidLegacyOctalLiteral {
+			text := lexer.rawIdentifier()
+
+			// Can't use a leading zero for bigint literals
+			if isBigIntegerLiteral && lexer.IsLegacyOctalLiteral {
+				lexer.SyntaxError()
+			}
+
+			// Filter out underscores
+			if underscoreCount > 0 {
+				bytes := make([]byte, 0, len(text.String)-underscoreCount)
+				for i := 0; i < len(text.String); i++ {
+					c := text.String[i]
+					if c != '_' {
+						bytes = append(bytes, c)
+					}
+				}
+				text = MaybeSubstring{String: string(bytes)}
+			}
+
+			// Store bigints as text to avoid precision loss
+			if isBigIntegerLiteral {
+				lexer.Identifier = text
+			} else if isInvalidLegacyOctalLiteral {
+				// Legacy octal literals may turn out to be a base 10 literal after all
+				value, _ := strconv.ParseFloat(text.String, 64)
+				lexer.Number = value
+			}
+		}
+	} else {
+		// Floating-point literal
+		isInvalidLegacyOctalLiteral := first == '0' && (lexer.codePoint == '8' || lexer.codePoint == '9')
+
+		// Initial digits
+		for {
+			if lexer.codePoint < '0' || lexer.codePoint > '9' {
+				if lexer.codePoint != '_' {
+					break
+				}
+
+				// Cannot have multiple underscores in a row
+				if lastUnderscoreEnd > 0 && lexer.end == lastUnderscoreEnd+1 {
+					lexer.SyntaxError()
+				}
+
+				// The specification forbids underscores in this case
+				if isInvalidLegacyOctalLiteral {
+					lexer.SyntaxError()
+				}
+
+				lastUnderscoreEnd = lexer.end
+				underscoreCount++
+			}
+			lexer.step()
+		}
+
+		// Fractional digits
+		if first != '.' && lexer.codePoint == '.' {
+			// An underscore must not come last
+			if lastUnderscoreEnd > 0 && lexer.end == lastUnderscoreEnd+1 {
+				lexer.end--
+				lexer.SyntaxError()
+			}
+
+			hasDotOrExponent = true
+			lexer.step()
+			if lexer.codePoint == '_' {
+				lexer.SyntaxError()
+			}
+			isMissingDigitAfterDot = true
+			for {
+				if lexer.codePoint >= '0' && lexer.codePoint <= '9' {
+					isMissingDigitAfterDot = false
+				} else {
+					if lexer.codePoint != '_' {
+						break
+					}
+
+					// Cannot have multiple underscores in a row
+					if lastUnderscoreEnd > 0 && lexer.end == lastUnderscoreEnd+1 {
+						lexer.SyntaxError()
+					}
+
+					lastUnderscoreEnd = lexer.end
+					underscoreCount++
+				}
+				lexer.step()
+			}
+		}
+
+		// Exponent
+		if lexer.codePoint == 'e' || lexer.codePoint == 'E' {
+			// An underscore must not come last
+			if lastUnderscoreEnd > 0 && lexer.end == lastUnderscoreEnd+1 {
+				lexer.end--
+				lexer.SyntaxError()
+			}
+
+			hasDotOrExponent = true
+			lexer.step()
+			if lexer.codePoint == '+' || lexer.codePoint == '-' {
+				lexer.step()
+			}
+			if lexer.codePoint < '0' || lexer.codePoint > '9' {
+				lexer.SyntaxError()
+			}
+			for {
+				if lexer.codePoint < '0' || lexer.codePoint > '9' {
+					if lexer.codePoint != '_' {
+						break
+					}
+
+					// Cannot have multiple underscores in a row
+					if lastUnderscoreEnd > 0 && lexer.end == lastUnderscoreEnd+1 {
+						lexer.SyntaxError()
+					}
+
+					lastUnderscoreEnd = lexer.end
+					underscoreCount++
+				}
+				lexer.step()
+			}
+		}
+
+		// Take a slice of the text to parse
+		text := lexer.rawIdentifier()
+
+		// Filter out underscores
+		if underscoreCount > 0 {
+			bytes := make([]byte, 0, len(text.String)-underscoreCount)
+			for i := 0; i < len(text.String); i++ {
+				c := text.String[i]
+				if c != '_' {
+					bytes = append(bytes, c)
+				}
+			}
+			text = MaybeSubstring{String: string(bytes)}
+		}
+
+		if lexer.codePoint == 'n' && !hasDotOrExponent {
+			// The only bigint literal that can start with 0 is "0n"
+			if len(text.String) > 1 && first == '0' {
+				lexer.SyntaxError()
+			}
+
+			// Store bigints as text to avoid precision loss
+			lexer.Identifier = text
+		} else if !hasDotOrExponent && lexer.end-lexer.start < 10 {
+			// Parse a 32-bit integer (very fast path)
+			var number uint32 = 0
+			for _, c := range text.String {
+				number = number*10 + uint32(c-'0')
+			}
+			lexer.Number = float64(number)
+		} else {
+			// Parse a double-precision floating-point number
+			value, _ := strconv.ParseFloat(text.String, 64)
+			lexer.Number = value
+		}
+	}
+
+	// An underscore must not come last
+	if lastUnderscoreEnd > 0 && lexer.end == lastUnderscoreEnd+1 {
+		lexer.end--
+		lexer.SyntaxError()
+	}
+
+	// Handle bigint literals after the underscore-at-end check above
+	if lexer.codePoint == 'n' && !hasDotOrExponent {
+		lexer.Token = TBigIntegerLiteral
+		lexer.step()
+	}
+
+	// Identifiers can't occur immediately after numbers
+	if js_ast.IsIdentifierStart(lexer.codePoint) {
+		lexer.SyntaxError()
+	}
+
+	// None of these are allowed in JSON
+	if lexer.json == JSON && (first == '.' || base != 0 || underscoreCount > 0 || isMissingDigitAfterDot) {
+		lexer.Unexpected()
+	}
+}
+
+func (lexer *Lexer) ScanRegExp() {
+	validateAndStep := func() {
+		if lexer.codePoint == '\\' {
+			lexer.step()
+		}
+
+		switch lexer.codePoint {
+		case -1, // This indicates the end of the file
+			'\r', '\n', 0x2028, 0x2029: // Newlines aren't allowed in regular expressions
+			lexer.addRangeError(logger.Range{Loc: logger.Loc{Start: int32(lexer.end)}}, "Unterminated regular expression")
+			panic(LexerPanic{})
+
+		default:
+			lexer.step()
+		}
+	}
+
+	for {
+		switch lexer.codePoint {
+		case '/':
+			lexer.step()
+			bits := uint32(0)
+			for js_ast.IsIdentifierContinue(lexer.codePoint) {
+				switch lexer.codePoint {
+				case 'd', 'g', 'i', 'm', 's', 'u', 'v', 'y':
+					bit := uint32(1) << uint32(lexer.codePoint-'a')
+					if (bit & bits) != 0 {
+						// Reject duplicate flags
+						r1 := logger.Range{Loc: logger.Loc{Start: int32(lexer.start)}, Len: 1}
+						r2 := logger.Range{Loc: logger.Loc{Start: int32(lexer.end)}, Len: 1}
+						for r1.Loc.Start < r2.Loc.Start && lexer.source.Contents[r1.Loc.Start] != byte(lexer.codePoint) {
+							r1.Loc.Start++
+						}
+						lexer.log.AddErrorWithNotes(&lexer.tracker, r2,
+							fmt.Sprintf("Duplicate flag \"%c\" in regular expression", lexer.codePoint),
+							[]logger.MsgData{lexer.tracker.MsgData(r1,
+								fmt.Sprintf("The first \"%c\" was here:", lexer.codePoint))})
+					} else {
+						bits |= bit
+					}
+					lexer.step()
+
+				default:
+					lexer.SyntaxError()
+				}
+			}
+			return
+
+		case '[':
+			lexer.step()
+			for lexer.codePoint != ']' {
+				validateAndStep()
+			}
+			lexer.step()
+
+		default:
+			validateAndStep()
+		}
+	}
+}
+
+func decodeJSXEntities(decoded []uint16, text string) []uint16 {
+	i := 0
+
+	for i < len(text) {
+		c, width := utf8.DecodeRuneInString(text[i:])
+		i += width
+
+		if c == '&' {
+			length := strings.IndexByte(text[i:], ';')
+			if length > 0 {
+				entity := text[i : i+length]
+				if entity[0] == '#' {
+					number := entity[1:]
+					base := 10
+					if len(number) > 1 && number[0] == 'x' {
+						number = number[1:]
+						base = 16
+					}
+					if value, err := strconv.ParseInt(number, base, 32); err == nil {
+						c = rune(value)
+						i += length + 1
+					}
+				} else if value, ok := jsxEntity[entity]; ok {
+					c = value
+					i += length + 1
+				}
+			}
+		}
+
+		if c <= 0xFFFF {
+			decoded = append(decoded, uint16(c))
+		} else {
+			c -= 0x10000
+			decoded = append(decoded, uint16(0xD800+((c>>10)&0x3FF)), uint16(0xDC00+(c&0x3FF)))
+		}
+	}
+
+	return decoded
+}
+
+func fixWhitespaceAndDecodeJSXEntities(text string) []uint16 {
+	afterLastNonWhitespace := -1
+	decoded := []uint16{}
+	i := 0
+
+	// Trim whitespace off the end of the first line
+	firstNonWhitespace := 0
+
+	// Split into lines
+	for i < len(text) {
+		c, width := utf8.DecodeRuneInString(text[i:])
+
+		switch c {
+		case '\r', '\n', '\u2028', '\u2029':
+			// Newline
+			if firstNonWhitespace != -1 && afterLastNonWhitespace != -1 {
+				if len(decoded) > 0 {
+					decoded = append(decoded, ' ')
+				}
+
+				// Trim whitespace off the start and end of lines in the middle
+				decoded = decodeJSXEntities(decoded, text[firstNonWhitespace:afterLastNonWhitespace])
+			}
+
+			// Reset for the next line
+			firstNonWhitespace = -1
+
+		case '\t', ' ':
+			// Whitespace
+
+		default:
+			// Check for unusual whitespace characters
+			if !js_ast.IsWhitespace(c) {
+				afterLastNonWhitespace = i + width
+				if firstNonWhitespace == -1 {
+					firstNonWhitespace = i
+				}
+			}
+		}
+
+		i += width
+	}
+
+	if firstNonWhitespace != -1 {
+		if len(decoded) > 0 {
+			decoded = append(decoded, ' ')
+		}
+
+		// Trim whitespace off the start of the last line
+		decoded = decodeJSXEntities(decoded, text[firstNonWhitespace:])
+	}
+
+	return decoded
+}
+
+// If this fails, this returns "nil, false, end" where "end" is the value to
+// store to "lexer.end" before calling "lexer.SyntaxError()" if relevant
+func (lexer *Lexer) tryToDecodeEscapeSequences(start int, text string, reportErrors bool) ([]uint16, bool, int) {
+	decoded := []uint16{}
+	i := 0
+
+	for i < len(text) {
+		c, width := utf8.DecodeRuneInString(text[i:])
+		i += width
+
+		switch c {
+		case '\r':
+			// From the specification:
+			//
+			// 11.8.6.1 Static Semantics: TV and TRV
+			//
+			// TV excludes the code units of LineContinuation while TRV includes
+			// them. <CR><LF> and <CR> LineTerminatorSequences are normalized to
+			// <LF> for both TV and TRV. An explicit EscapeSequence is needed to
+			// include a <CR> or <CR><LF> sequence.
+
+			// Convert '\r\n' into '\n'
+			if i < len(text) && text[i] == '\n' {
+				i++
+			}
+
+			// Convert '\r' into '\n'
+			decoded = append(decoded, '\n')
+			continue
+
+		case '\\':
+			c2, width2 := utf8.DecodeRuneInString(text[i:])
+			i += width2
+
+			switch c2 {
+			case 'b':
+				decoded = append(decoded, '\b')
+				continue
+
+			case 'f':
+				decoded = append(decoded, '\f')
+				continue
+
+			case 'n':
+				decoded = append(decoded, '\n')
+				continue
+
+			case 'r':
+				decoded = append(decoded, '\r')
+				continue
+
+			case 't':
+				decoded = append(decoded, '\t')
+				continue
+
+			case 'v':
+				if lexer.json == JSON {
+					return nil, false, start + i - width2
+				}
+
+				decoded = append(decoded, '\v')
+				continue
+
+			case '0', '1', '2', '3', '4', '5', '6', '7':
+				octalStart := i - 2
+				if lexer.json == JSON {
+					return nil, false, start + i - width2
+				}
+
+				// 1-3 digit octal
+				isBad := false
+				value := c2 - '0'
+				c3, width3 := utf8.DecodeRuneInString(text[i:])
+				switch c3 {
+				case '0', '1', '2', '3', '4', '5', '6', '7':
+					value = value*8 + c3 - '0'
+					i += width3
+					c4, width4 := utf8.DecodeRuneInString(text[i:])
+					switch c4 {
+					case '0', '1', '2', '3', '4', '5', '6', '7':
+						temp := value*8 + c4 - '0'
+						if temp < 256 {
+							value = temp
+							i += width4
+						}
+					case '8', '9':
+						isBad = true
+					}
+				case '8', '9':
+					isBad = true
+				}
+				c = value
+
+				// Forbid the use of octal literals other than "\0"
+				if isBad || text[octalStart:i] != "\\0" {
+					lexer.LegacyOctalLoc = logger.Loc{Start: int32(start + octalStart)}
+				}
+
+			case '8', '9':
+				c = c2
+
+				// Forbid the invalid octal literals "\8" and "\9"
+				lexer.LegacyOctalLoc = logger.Loc{Start: int32(start + i - 2)}
+
+			case 'x':
+				if lexer.json == JSON {
+					return nil, false, start + i - width2
+				}
+
+				// 2-digit hexadecimal
+				value := '\000'
+				for j := 0; j < 2; j++ {
+					c3, width3 := utf8.DecodeRuneInString(text[i:])
+					i += width3
+					switch c3 {
+					case '0', '1', '2', '3', '4', '5', '6', '7', '8', '9':
+						value = value*16 | (c3 - '0')
+					case 'a', 'b', 'c', 'd', 'e', 'f':
+						value = value*16 | (c3 + 10 - 'a')
+					case 'A', 'B', 'C', 'D', 'E', 'F':
+						value = value*16 | (c3 + 10 - 'A')
+					default:
+						return nil, false, start + i - width3
+					}
+				}
+				c = value
+
+			case 'u':
+				// Unicode
+				value := '\000'
+
+				// Check the first character
+				c3, width3 := utf8.DecodeRuneInString(text[i:])
+				i += width3
+
+				if c3 == '{' {
+					if lexer.json == JSON {
+						return nil, false, start + i - width2
+					}
+
+					// Variable-length
+					hexStart := i - width - width2 - width3
+					isFirst := true
+					isOutOfRange := false
+				variableLength:
+					for {
+						c3, width3 = utf8.DecodeRuneInString(text[i:])
+						i += width3
+
+						switch c3 {
+						case '0', '1', '2', '3', '4', '5', '6', '7', '8', '9':
+							value = value*16 | (c3 - '0')
+						case 'a', 'b', 'c', 'd', 'e', 'f':
+							value = value*16 | (c3 + 10 - 'a')
+						case 'A', 'B', 'C', 'D', 'E', 'F':
+							value = value*16 | (c3 + 10 - 'A')
+						case '}':
+							if isFirst {
+								return nil, false, start + i - width3
+							}
+							break variableLength
+						default:
+							return nil, false, start + i - width3
+						}
+
+						if value > utf8.MaxRune {
+							isOutOfRange = true
+						}
+
+						isFirst = false
+					}
+
+					if isOutOfRange && reportErrors {
+						lexer.addRangeError(logger.Range{Loc: logger.Loc{Start: int32(start + hexStart)}, Len: int32(i - hexStart)},
+							"Unicode escape sequence is out of range")
+						panic(LexerPanic{})
+					}
+				} else {
+					// Fixed-length
+					for j := 0; j < 4; j++ {
+						switch c3 {
+						case '0', '1', '2', '3', '4', '5', '6', '7', '8', '9':
+							value = value*16 | (c3 - '0')
+						case 'a', 'b', 'c', 'd', 'e', 'f':
+							value = value*16 | (c3 + 10 - 'a')
+						case 'A', 'B', 'C', 'D', 'E', 'F':
+							value = value*16 | (c3 + 10 - 'A')
+						default:
+							return nil, false, start + i - width3
+						}
+
+						if j < 3 {
+							c3, width3 = utf8.DecodeRuneInString(text[i:])
+							i += width3
+						}
+					}
+				}
+				c = value
+
+			case '\r':
+				if lexer.json == JSON {
+					return nil, false, start + i - width2
+				}
+
+				// Ignore line continuations. A line continuation is not an escaped newline.
+				if i < len(text) && text[i] == '\n' {
+					// Make sure Windows CRLF counts as a single newline
+					i++
+				}
+				continue
+
+			case '\n', '\u2028', '\u2029':
+				if lexer.json == JSON {
+					return nil, false, start + i - width2
+				}
+
+				// Ignore line continuations. A line continuation is not an escaped newline.
+				continue
+
+			default:
+				if lexer.json == JSON {
+					switch c2 {
+					case '"', '\\', '/':
+
+					default:
+						return nil, false, start + i - width2
+					}
+				}
+
+				c = c2
+			}
+		}
+
+		if c <= 0xFFFF {
+			decoded = append(decoded, uint16(c))
+		} else {
+			c -= 0x10000
+			decoded = append(decoded, uint16(0xD800+((c>>10)&0x3FF)), uint16(0xDC00+(c&0x3FF)))
+		}
+	}
+
+	return decoded, true, 0
+}
+
+func (lexer *Lexer) RescanCloseBraceAsTemplateToken() {
+	if lexer.Token != TCloseBrace {
+		lexer.Expected(TCloseBrace)
+	}
+
+	lexer.rescanCloseBraceAsTemplateToken = true
+	lexer.codePoint = '`'
+	lexer.current = lexer.end
+	lexer.end -= 1
+	lexer.Next()
+	lexer.rescanCloseBraceAsTemplateToken = false
+}
+
+func (lexer *Lexer) step() {
+	codePoint, width := utf8.DecodeRuneInString(lexer.source.Contents[lexer.current:])
+
+	// Use -1 to indicate the end of the file
+	if width == 0 {
+		codePoint = -1
+	}
+
+	// Track the approximate number of newlines in the file so we can preallocate
+	// the line offset table in the printer for source maps. The line offset table
+	// is the #1 highest allocation in the heap profile, so this is worth doing.
+	// This count is approximate because it handles "\n" and "\r\n" (the common
+	// cases) but not "\r" or "\u2028" or "\u2029". Getting this wrong is harmless
+	// because it's only a preallocation. The array will just grow if it's too small.
+	if codePoint == '\n' {
+		lexer.ApproximateNewlineCount++
+	}
+
+	lexer.codePoint = codePoint
+	lexer.end = lexer.current
+	lexer.current += width
+}
+
+func (lexer *Lexer) addRangeError(r logger.Range, text string) {
+	// Don't report multiple errors in the same spot
+	if r.Loc == lexer.prevErrorLoc {
+		return
+	}
+	lexer.prevErrorLoc = r.Loc
+
+	if !lexer.IsLogDisabled {
+		lexer.log.AddError(&lexer.tracker, r, text)
+	}
+}
+
+func (lexer *Lexer) addRangeErrorWithSuggestion(r logger.Range, text string, suggestion string) {
+	// Don't report multiple errors in the same spot
+	if r.Loc == lexer.prevErrorLoc {
+		return
+	}
+	lexer.prevErrorLoc = r.Loc
+
+	if !lexer.IsLogDisabled {
+		data := lexer.tracker.MsgData(r, text)
+		data.Location.Suggestion = suggestion
+		lexer.log.AddMsg(logger.Msg{Kind: logger.Error, Data: data})
+	}
+}
+
+func (lexer *Lexer) AddRangeErrorWithNotes(r logger.Range, text string, notes []logger.MsgData) {
+	// Don't report multiple errors in the same spot
+	if r.Loc == lexer.prevErrorLoc {
+		return
+	}
+	lexer.prevErrorLoc = r.Loc
+
+	if !lexer.IsLogDisabled {
+		lexer.log.AddErrorWithNotes(&lexer.tracker, r, text, notes)
+	}
+}
+
+func hasPrefixWithWordBoundary(text string, prefix string) bool {
+	t := len(text)
+	p := len(prefix)
+	if t >= p && text[0:p] == prefix {
+		if t == p {
+			return true
+		}
+		c, _ := utf8.DecodeRuneInString(text[p:])
+		if !js_ast.IsIdentifierContinue(c) {
+			return true
+		}
+	}
+	return false
+}
+
+type pragmaArg uint8
+
+const (
+	pragmaNoSpaceFirst pragmaArg = iota
+	pragmaSkipSpaceFirst
+)
+
+func scanForPragmaArg(kind pragmaArg, start int, pragma string, text string) (logger.Span, bool) {
+	text = text[len(pragma):]
+	start += len(pragma)
+
+	if text == "" {
+		return logger.Span{}, false
+	}
+
+	// One or more whitespace characters
+	c, width := utf8.DecodeRuneInString(text)
+	if kind == pragmaSkipSpaceFirst {
+		if !js_ast.IsWhitespace(c) {
+			return logger.Span{}, false
+		}
+		for js_ast.IsWhitespace(c) {
+			text = text[width:]
+			start += width
+			if text == "" {
+				return logger.Span{}, false
+			}
+			c, width = utf8.DecodeRuneInString(text)
+		}
+	}
+
+	// One or more non-whitespace characters
+	i := 0
+	for !js_ast.IsWhitespace(c) {
+		i += width
+		if i >= len(text) {
+			break
+		}
+		c, width = utf8.DecodeRuneInString(text[i:])
+		if js_ast.IsWhitespace(c) {
+			break
+		}
+	}
+
+	return logger.Span{
+		Text: text[:i],
+		Range: logger.Range{
+			Loc: logger.Loc{Start: int32(start)},
+			Len: int32(i),
+		},
+	}, true
+}
+
+func isUpperASCII(c byte) bool {
+	return c >= 'A' && c <= 'Z'
+}
+
+func isLetterASCII(c byte) bool {
+	return (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z')
+}
+
+func (lexer *Lexer) scanCommentText() {
+	text := lexer.source.Contents[lexer.start:lexer.end]
+	hasLegalAnnotation := len(text) > 2 && text[2] == '!'
+	isMultiLineComment := text[1] == '*'
+	omitFromGeneralCommentPreservation := false
+
+	// Save the original comment text so we can subtract comments from the
+	// character frequency analysis used by symbol minification
+	lexer.AllComments = append(lexer.AllComments, lexer.Range())
+
+	// Omit the trailing "*/" from the checks below
+	endOfCommentText := len(text)
+	if isMultiLineComment {
+		endOfCommentText -= 2
+	}
+
+	for i, n := 0, len(text); i < n; i++ {
+		switch text[i] {
+		case '#':
+			rest := text[i+1 : endOfCommentText]
+			if hasPrefixWithWordBoundary(rest, "__PURE__") {
+				omitFromGeneralCommentPreservation = true
+				lexer.HasCommentBefore |= PureCommentBefore
+			} else if hasPrefixWithWordBoundary(rest, "__KEY__") {
+				omitFromGeneralCommentPreservation = true
+				lexer.HasCommentBefore |= KeyCommentBefore
+			} else if hasPrefixWithWordBoundary(rest, "__NO_SIDE_EFFECTS__") {
+				omitFromGeneralCommentPreservation = true
+				lexer.HasCommentBefore |= NoSideEffectsCommentBefore
+			} else if i == 2 && strings.HasPrefix(rest, " sourceMappingURL=") {
+				if arg, ok := scanForPragmaArg(pragmaNoSpaceFirst, lexer.start+i+1, " sourceMappingURL=", rest); ok {
+					omitFromGeneralCommentPreservation = true
+					lexer.SourceMappingURL = arg
+				}
+			}
+
+		case '@':
+			rest := text[i+1 : endOfCommentText]
+			if hasPrefixWithWordBoundary(rest, "__PURE__") {
+				omitFromGeneralCommentPreservation = true
+				lexer.HasCommentBefore |= PureCommentBefore
+			} else if hasPrefixWithWordBoundary(rest, "__KEY__") {
+				omitFromGeneralCommentPreservation = true
+				lexer.HasCommentBefore |= KeyCommentBefore
+			} else if hasPrefixWithWordBoundary(rest, "__NO_SIDE_EFFECTS__") {
+				omitFromGeneralCommentPreservation = true
+				lexer.HasCommentBefore |= NoSideEffectsCommentBefore
+			} else if hasPrefixWithWordBoundary(rest, "preserve") || hasPrefixWithWordBoundary(rest, "license") {
+				hasLegalAnnotation = true
+			} else if hasPrefixWithWordBoundary(rest, "jsx") {
+				if arg, ok := scanForPragmaArg(pragmaSkipSpaceFirst, lexer.start+i+1, "jsx", rest); ok {
+					lexer.JSXFactoryPragmaComment = arg
+				}
+			} else if hasPrefixWithWordBoundary(rest, "jsxFrag") {
+				if arg, ok := scanForPragmaArg(pragmaSkipSpaceFirst, lexer.start+i+1, "jsxFrag", rest); ok {
+					lexer.JSXFragmentPragmaComment = arg
+				}
+			} else if hasPrefixWithWordBoundary(rest, "jsxRuntime") {
+				if arg, ok := scanForPragmaArg(pragmaSkipSpaceFirst, lexer.start+i+1, "jsxRuntime", rest); ok {
+					lexer.JSXRuntimePragmaComment = arg
+				}
+			} else if hasPrefixWithWordBoundary(rest, "jsxImportSource") {
+				if arg, ok := scanForPragmaArg(pragmaSkipSpaceFirst, lexer.start+i+1, "jsxImportSource", rest); ok {
+					lexer.JSXImportSourcePragmaComment = arg
+				}
+			} else if i == 2 && strings.HasPrefix(rest, " sourceMappingURL=") {
+				if arg, ok := scanForPragmaArg(pragmaNoSpaceFirst, lexer.start+i+1, " sourceMappingURL=", rest); ok {
+					omitFromGeneralCommentPreservation = true
+					lexer.SourceMappingURL = arg
+				}
+			}
+		}
+	}
+
+	if hasLegalAnnotation {
+		lexer.LegalCommentsBeforeToken = append(lexer.LegalCommentsBeforeToken, lexer.Range())
+	}
+
+	if !omitFromGeneralCommentPreservation {
+		lexer.CommentsBeforeToken = append(lexer.CommentsBeforeToken, lexer.Range())
+	}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/js_lexer/tables.go b/source/vendor/github.com/evanw/esbuild/internal/js_lexer/tables.go
new file mode 100644
index 0000000..c96b6b5
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/js_lexer/tables.go
@@ -0,0 +1,382 @@
+package js_lexer
+
+var tokenToString = map[T]string{
+	TEndOfFile:   "end of file",
+	TSyntaxError: "syntax error",
+	THashbang:    "hashbang comment",
+
+	// Literals
+	TNoSubstitutionTemplateLiteral: "template literal",
+	TNumericLiteral:                "number",
+	TStringLiteral:                 "string",
+	TBigIntegerLiteral:             "bigint",
+
+	// Pseudo-literals
+	TTemplateHead:   "template literal",
+	TTemplateMiddle: "template literal",
+	TTemplateTail:   "template literal",
+
+	// Punctuation
+	TAmpersand:                         "\"&\"",
+	TAmpersandAmpersand:                "\"&&\"",
+	TAsterisk:                          "\"*\"",
+	TAsteriskAsterisk:                  "\"**\"",
+	TAt:                                "\"@\"",
+	TBar:                               "\"|\"",
+	TBarBar:                            "\"||\"",
+	TCaret:                             "\"^\"",
+	TCloseBrace:                        "\"}\"",
+	TCloseBracket:                      "\"]\"",
+	TCloseParen:                        "\")\"",
+	TColon:                             "\":\"",
+	TComma:                             "\",\"",
+	TDot:                               "\".\"",
+	TDotDotDot:                         "\"...\"",
+	TEqualsEquals:                      "\"==\"",
+	TEqualsEqualsEquals:                "\"===\"",
+	TEqualsGreaterThan:                 "\"=>\"",
+	TExclamation:                       "\"!\"",
+	TExclamationEquals:                 "\"!=\"",
+	TExclamationEqualsEquals:           "\"!==\"",
+	TGreaterThan:                       "\">\"",
+	TGreaterThanEquals:                 "\">=\"",
+	TGreaterThanGreaterThan:            "\">>\"",
+	TGreaterThanGreaterThanGreaterThan: "\">>>\"",
+	TLessThan:                          "\"<\"",
+	TLessThanEquals:                    "\"<=\"",
+	TLessThanLessThan:                  "\"<<\"",
+	TMinus:                             "\"-\"",
+	TMinusMinus:                        "\"--\"",
+	TOpenBrace:                         "\"{\"",
+	TOpenBracket:                       "\"[\"",
+	TOpenParen:                         "\"(\"",
+	TPercent:                           "\"%\"",
+	TPlus:                              "\"+\"",
+	TPlusPlus:                          "\"++\"",
+	TQuestion:                          "\"?\"",
+	TQuestionDot:                       "\"?.\"",
+	TQuestionQuestion:                  "\"??\"",
+	TSemicolon:                         "\";\"",
+	TSlash:                             "\"/\"",
+	TTilde:                             "\"~\"",
+
+	// Assignments
+	TAmpersandAmpersandEquals:                "\"&&=\"",
+	TAmpersandEquals:                         "\"&=\"",
+	TAsteriskAsteriskEquals:                  "\"**=\"",
+	TAsteriskEquals:                          "\"*=\"",
+	TBarBarEquals:                            "\"||=\"",
+	TBarEquals:                               "\"|=\"",
+	TCaretEquals:                             "\"^=\"",
+	TEquals:                                  "\"=\"",
+	TGreaterThanGreaterThanEquals:            "\">>=\"",
+	TGreaterThanGreaterThanGreaterThanEquals: "\">>>=\"",
+	TLessThanLessThanEquals:                  "\"<<=\"",
+	TMinusEquals:                             "\"-=\"",
+	TPercentEquals:                           "\"%=\"",
+	TPlusEquals:                              "\"+=\"",
+	TQuestionQuestionEquals:                  "\"??=\"",
+	TSlashEquals:                             "\"/=\"",
+
+	// Class-private fields and methods
+	TPrivateIdentifier: "private identifier",
+
+	// Identifiers
+	TIdentifier:     "identifier",
+	TEscapedKeyword: "escaped keyword",
+
+	// Reserved words
+	TBreak:      "\"break\"",
+	TCase:       "\"case\"",
+	TCatch:      "\"catch\"",
+	TClass:      "\"class\"",
+	TConst:      "\"const\"",
+	TContinue:   "\"continue\"",
+	TDebugger:   "\"debugger\"",
+	TDefault:    "\"default\"",
+	TDelete:     "\"delete\"",
+	TDo:         "\"do\"",
+	TElse:       "\"else\"",
+	TEnum:       "\"enum\"",
+	TExport:     "\"export\"",
+	TExtends:    "\"extends\"",
+	TFalse:      "\"false\"",
+	TFinally:    "\"finally\"",
+	TFor:        "\"for\"",
+	TFunction:   "\"function\"",
+	TIf:         "\"if\"",
+	TImport:     "\"import\"",
+	TIn:         "\"in\"",
+	TInstanceof: "\"instanceof\"",
+	TNew:        "\"new\"",
+	TNull:       "\"null\"",
+	TReturn:     "\"return\"",
+	TSuper:      "\"super\"",
+	TSwitch:     "\"switch\"",
+	TThis:       "\"this\"",
+	TThrow:      "\"throw\"",
+	TTrue:       "\"true\"",
+	TTry:        "\"try\"",
+	TTypeof:     "\"typeof\"",
+	TVar:        "\"var\"",
+	TVoid:       "\"void\"",
+	TWhile:      "\"while\"",
+	TWith:       "\"with\"",
+}
+
+// This is from https://github.com/microsoft/TypeScript/blob/master/src/compiler/transformers/jsx.ts
+var jsxEntity = map[string]rune{
+	"quot":     0x0022,
+	"amp":      0x0026,
+	"apos":     0x0027,
+	"lt":       0x003C,
+	"gt":       0x003E,
+	"nbsp":     0x00A0,
+	"iexcl":    0x00A1,
+	"cent":     0x00A2,
+	"pound":    0x00A3,
+	"curren":   0x00A4,
+	"yen":      0x00A5,
+	"brvbar":   0x00A6,
+	"sect":     0x00A7,
+	"uml":      0x00A8,
+	"copy":     0x00A9,
+	"ordf":     0x00AA,
+	"laquo":    0x00AB,
+	"not":      0x00AC,
+	"shy":      0x00AD,
+	"reg":      0x00AE,
+	"macr":     0x00AF,
+	"deg":      0x00B0,
+	"plusmn":   0x00B1,
+	"sup2":     0x00B2,
+	"sup3":     0x00B3,
+	"acute":    0x00B4,
+	"micro":    0x00B5,
+	"para":     0x00B6,
+	"middot":   0x00B7,
+	"cedil":    0x00B8,
+	"sup1":     0x00B9,
+	"ordm":     0x00BA,
+	"raquo":    0x00BB,
+	"frac14":   0x00BC,
+	"frac12":   0x00BD,
+	"frac34":   0x00BE,
+	"iquest":   0x00BF,
+	"Agrave":   0x00C0,
+	"Aacute":   0x00C1,
+	"Acirc":    0x00C2,
+	"Atilde":   0x00C3,
+	"Auml":     0x00C4,
+	"Aring":    0x00C5,
+	"AElig":    0x00C6,
+	"Ccedil":   0x00C7,
+	"Egrave":   0x00C8,
+	"Eacute":   0x00C9,
+	"Ecirc":    0x00CA,
+	"Euml":     0x00CB,
+	"Igrave":   0x00CC,
+	"Iacute":   0x00CD,
+	"Icirc":    0x00CE,
+	"Iuml":     0x00CF,
+	"ETH":      0x00D0,
+	"Ntilde":   0x00D1,
+	"Ograve":   0x00D2,
+	"Oacute":   0x00D3,
+	"Ocirc":    0x00D4,
+	"Otilde":   0x00D5,
+	"Ouml":     0x00D6,
+	"times":    0x00D7,
+	"Oslash":   0x00D8,
+	"Ugrave":   0x00D9,
+	"Uacute":   0x00DA,
+	"Ucirc":    0x00DB,
+	"Uuml":     0x00DC,
+	"Yacute":   0x00DD,
+	"THORN":    0x00DE,
+	"szlig":    0x00DF,
+	"agrave":   0x00E0,
+	"aacute":   0x00E1,
+	"acirc":    0x00E2,
+	"atilde":   0x00E3,
+	"auml":     0x00E4,
+	"aring":    0x00E5,
+	"aelig":    0x00E6,
+	"ccedil":   0x00E7,
+	"egrave":   0x00E8,
+	"eacute":   0x00E9,
+	"ecirc":    0x00EA,
+	"euml":     0x00EB,
+	"igrave":   0x00EC,
+	"iacute":   0x00ED,
+	"icirc":    0x00EE,
+	"iuml":     0x00EF,
+	"eth":      0x00F0,
+	"ntilde":   0x00F1,
+	"ograve":   0x00F2,
+	"oacute":   0x00F3,
+	"ocirc":    0x00F4,
+	"otilde":   0x00F5,
+	"ouml":     0x00F6,
+	"divide":   0x00F7,
+	"oslash":   0x00F8,
+	"ugrave":   0x00F9,
+	"uacute":   0x00FA,
+	"ucirc":    0x00FB,
+	"uuml":     0x00FC,
+	"yacute":   0x00FD,
+	"thorn":    0x00FE,
+	"yuml":     0x00FF,
+	"OElig":    0x0152,
+	"oelig":    0x0153,
+	"Scaron":   0x0160,
+	"scaron":   0x0161,
+	"Yuml":     0x0178,
+	"fnof":     0x0192,
+	"circ":     0x02C6,
+	"tilde":    0x02DC,
+	"Alpha":    0x0391,
+	"Beta":     0x0392,
+	"Gamma":    0x0393,
+	"Delta":    0x0394,
+	"Epsilon":  0x0395,
+	"Zeta":     0x0396,
+	"Eta":      0x0397,
+	"Theta":    0x0398,
+	"Iota":     0x0399,
+	"Kappa":    0x039A,
+	"Lambda":   0x039B,
+	"Mu":       0x039C,
+	"Nu":       0x039D,
+	"Xi":       0x039E,
+	"Omicron":  0x039F,
+	"Pi":       0x03A0,
+	"Rho":      0x03A1,
+	"Sigma":    0x03A3,
+	"Tau":      0x03A4,
+	"Upsilon":  0x03A5,
+	"Phi":      0x03A6,
+	"Chi":      0x03A7,
+	"Psi":      0x03A8,
+	"Omega":    0x03A9,
+	"alpha":    0x03B1,
+	"beta":     0x03B2,
+	"gamma":    0x03B3,
+	"delta":    0x03B4,
+	"epsilon":  0x03B5,
+	"zeta":     0x03B6,
+	"eta":      0x03B7,
+	"theta":    0x03B8,
+	"iota":     0x03B9,
+	"kappa":    0x03BA,
+	"lambda":   0x03BB,
+	"mu":       0x03BC,
+	"nu":       0x03BD,
+	"xi":       0x03BE,
+	"omicron":  0x03BF,
+	"pi":       0x03C0,
+	"rho":      0x03C1,
+	"sigmaf":   0x03C2,
+	"sigma":    0x03C3,
+	"tau":      0x03C4,
+	"upsilon":  0x03C5,
+	"phi":      0x03C6,
+	"chi":      0x03C7,
+	"psi":      0x03C8,
+	"omega":    0x03C9,
+	"thetasym": 0x03D1,
+	"upsih":    0x03D2,
+	"piv":      0x03D6,
+	"ensp":     0x2002,
+	"emsp":     0x2003,
+	"thinsp":   0x2009,
+	"zwnj":     0x200C,
+	"zwj":      0x200D,
+	"lrm":      0x200E,
+	"rlm":      0x200F,
+	"ndash":    0x2013,
+	"mdash":    0x2014,
+	"lsquo":    0x2018,
+	"rsquo":    0x2019,
+	"sbquo":    0x201A,
+	"ldquo":    0x201C,
+	"rdquo":    0x201D,
+	"bdquo":    0x201E,
+	"dagger":   0x2020,
+	"Dagger":   0x2021,
+	"bull":     0x2022,
+	"hellip":   0x2026,
+	"permil":   0x2030,
+	"prime":    0x2032,
+	"Prime":    0x2033,
+	"lsaquo":   0x2039,
+	"rsaquo":   0x203A,
+	"oline":    0x203E,
+	"frasl":    0x2044,
+	"euro":     0x20AC,
+	"image":    0x2111,
+	"weierp":   0x2118,
+	"real":     0x211C,
+	"trade":    0x2122,
+	"alefsym":  0x2135,
+	"larr":     0x2190,
+	"uarr":     0x2191,
+	"rarr":     0x2192,
+	"darr":     0x2193,
+	"harr":     0x2194,
+	"crarr":    0x21B5,
+	"lArr":     0x21D0,
+	"uArr":     0x21D1,
+	"rArr":     0x21D2,
+	"dArr":     0x21D3,
+	"hArr":     0x21D4,
+	"forall":   0x2200,
+	"part":     0x2202,
+	"exist":    0x2203,
+	"empty":    0x2205,
+	"nabla":    0x2207,
+	"isin":     0x2208,
+	"notin":    0x2209,
+	"ni":       0x220B,
+	"prod":     0x220F,
+	"sum":      0x2211,
+	"minus":    0x2212,
+	"lowast":   0x2217,
+	"radic":    0x221A,
+	"prop":     0x221D,
+	"infin":    0x221E,
+	"ang":      0x2220,
+	"and":      0x2227,
+	"or":       0x2228,
+	"cap":      0x2229,
+	"cup":      0x222A,
+	"int":      0x222B,
+	"there4":   0x2234,
+	"sim":      0x223C,
+	"cong":     0x2245,
+	"asymp":    0x2248,
+	"ne":       0x2260,
+	"equiv":    0x2261,
+	"le":       0x2264,
+	"ge":       0x2265,
+	"sub":      0x2282,
+	"sup":      0x2283,
+	"nsub":     0x2284,
+	"sube":     0x2286,
+	"supe":     0x2287,
+	"oplus":    0x2295,
+	"otimes":   0x2297,
+	"perp":     0x22A5,
+	"sdot":     0x22C5,
+	"lceil":    0x2308,
+	"rceil":    0x2309,
+	"lfloor":   0x230A,
+	"rfloor":   0x230B,
+	"lang":     0x2329,
+	"rang":     0x232A,
+	"loz":      0x25CA,
+	"spades":   0x2660,
+	"clubs":    0x2663,
+	"hearts":   0x2665,
+	"diams":    0x2666,
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/js_parser/global_name_parser.go b/source/vendor/github.com/evanw/esbuild/internal/js_parser/global_name_parser.go
new file mode 100644
index 0000000..0064990
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/js_parser/global_name_parser.go
@@ -0,0 +1,49 @@
+package js_parser
+
+import (
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_lexer"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+func ParseGlobalName(log logger.Log, source logger.Source) (result []string, ok bool) {
+	ok = true
+	defer func() {
+		r := recover()
+		if _, isLexerPanic := r.(js_lexer.LexerPanic); isLexerPanic {
+			ok = false
+		} else if r != nil {
+			panic(r)
+		}
+	}()
+
+	lexer := js_lexer.NewLexerGlobalName(log, source)
+
+	// Start off with an identifier
+	result = append(result, lexer.Identifier.String)
+	lexer.Expect(js_lexer.TIdentifier)
+
+	// Follow with dot or index expressions
+	for lexer.Token != js_lexer.TEndOfFile {
+		switch lexer.Token {
+		case js_lexer.TDot:
+			lexer.Next()
+			if !lexer.IsIdentifierOrKeyword() {
+				lexer.Expect(js_lexer.TIdentifier)
+			}
+			result = append(result, lexer.Identifier.String)
+			lexer.Next()
+
+		case js_lexer.TOpenBracket:
+			lexer.Next()
+			result = append(result, helpers.UTF16ToString(lexer.StringLiteral()))
+			lexer.Expect(js_lexer.TStringLiteral)
+			lexer.Expect(js_lexer.TCloseBracket)
+
+		default:
+			lexer.Expect(js_lexer.TDot)
+		}
+	}
+
+	return
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/js_parser/js_parser.go b/source/vendor/github.com/evanw/esbuild/internal/js_parser/js_parser.go
new file mode 100644
index 0000000..45daca9
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/js_parser/js_parser.go
@@ -0,0 +1,18021 @@
+package js_parser
+
+import (
+	"fmt"
+	"math"
+	"regexp"
+	"sort"
+	"strings"
+	"unicode/utf8"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/config"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/js_lexer"
+	"github.com/evanw/esbuild/internal/logger"
+	"github.com/evanw/esbuild/internal/renamer"
+	"github.com/evanw/esbuild/internal/runtime"
+)
+
+// This parser does two passes:
+//
+// 1. Parse the source into an AST, create the scope tree, and declare symbols.
+//
+//  2. Visit each node in the AST, bind identifiers to declared symbols, do
+//     constant folding, substitute compile-time variable definitions, and
+//     lower certain syntactic constructs as appropriate given the language
+//     target.
+//
+// So many things have been put in so few passes because we want to minimize
+// the number of full-tree passes to improve performance. However, we need
+// to have at least two separate passes to handle variable hoisting. See the
+// comment about scopesInOrder below for more information.
+type parser struct {
+	options                    Options
+	log                        logger.Log
+	source                     logger.Source
+	tracker                    logger.LineColumnTracker
+	fnOrArrowDataParse         fnOrArrowDataParse
+	fnOnlyDataVisit            fnOnlyDataVisit
+	allocatedNames             []string
+	currentScope               *js_ast.Scope
+	scopesForCurrentPart       []*js_ast.Scope
+	symbols                    []ast.Symbol
+	astHelpers                 js_ast.HelperContext
+	tsUseCounts                []uint32
+	injectedDefineSymbols      []ast.Ref
+	injectedSymbolSources      map[ast.Ref]injectedSymbolSource
+	injectedDotNames           map[string][]injectedDotName
+	dropLabelsMap              map[string]struct{}
+	exprComments               map[logger.Loc][]string
+	mangledProps               map[string]ast.Ref
+	reservedProps              map[string]bool
+	symbolUses                 map[ast.Ref]js_ast.SymbolUse
+	importSymbolPropertyUses   map[ast.Ref]map[string]js_ast.SymbolUse
+	symbolCallUses             map[ast.Ref]js_ast.SymbolCallUse
+	declaredSymbols            []js_ast.DeclaredSymbol
+	globPatternImports         []globPatternImport
+	runtimeImports             map[string]ast.LocRef
+	duplicateCaseChecker       duplicateCaseChecker
+	unrepresentableIdentifiers map[string]bool
+	legacyOctalLiterals        map[js_ast.E]logger.Range
+	scopesInOrderForEnum       map[logger.Loc][]scopeOrder
+	binaryExprStack            []binaryExprVisitor
+
+	// For strict mode handling
+	hoistedRefForSloppyModeBlockFn map[ast.Ref]ast.Ref
+
+	// For lowering private methods
+	privateGetters map[ast.Ref]ast.Ref
+	privateSetters map[ast.Ref]ast.Ref
+
+	// These are for TypeScript
+	//
+	// We build up enough information about the TypeScript namespace hierarchy to
+	// be able to resolve scope lookups and property accesses for TypeScript enum
+	// and namespace features. Each JavaScript scope object inside a namespace
+	// has a reference to a map of exported namespace members from sibling scopes.
+	//
+	// In addition, there is a map from each relevant symbol reference to the data
+	// associated with that namespace or namespace member: "refToTSNamespaceMemberData".
+	// This gives enough info to be able to resolve queries into the namespace.
+	//
+	// When visiting expressions, namespace metadata is associated with the most
+	// recently visited node. If namespace metadata is present, "tsNamespaceTarget"
+	// will be set to the most recently visited node (as a way to mark that this
+	// node has metadata) and "tsNamespaceMemberData" will be set to the metadata.
+	refToTSNamespaceMemberData map[ast.Ref]js_ast.TSNamespaceMemberData
+	tsNamespaceTarget          js_ast.E
+	tsNamespaceMemberData      js_ast.TSNamespaceMemberData
+	emittedNamespaceVars       map[ast.Ref]bool
+	isExportedInsideNamespace  map[ast.Ref]ast.Ref
+	localTypeNames             map[string]bool
+	tsEnums                    map[ast.Ref]map[string]js_ast.TSEnumValue
+	constValues                map[ast.Ref]js_ast.ConstValue
+	propDerivedCtorValue       js_ast.E
+	propMethodDecoratorScope   *js_ast.Scope
+
+	// This is the reference to the generated function argument for the namespace,
+	// which is different than the reference to the namespace itself:
+	//
+	//   namespace ns {
+	//   }
+	//
+	// The code above is transformed into something like this:
+	//
+	//   var ns1;
+	//   (function(ns2) {
+	//   })(ns1 || (ns1 = {}));
+	//
+	// This variable is "ns2" not "ns1". It is only used during the second
+	// "visit" pass.
+	enclosingNamespaceArgRef *ast.Ref
+
+	// Imports (both ES6 and CommonJS) are tracked at the top level
+	importRecords               []ast.ImportRecord
+	importRecordsForCurrentPart []uint32
+	exportStarImportRecords     []uint32
+
+	// These are for handling ES6 imports and exports
+	importItemsForNamespace map[ast.Ref]namespaceImportItems
+	isImportItem            map[ast.Ref]bool
+	namedImports            map[ast.Ref]js_ast.NamedImport
+	namedExports            map[string]js_ast.NamedExport
+	topLevelSymbolToParts   map[ast.Ref][]uint32
+	importNamespaceCCMap    map[importNamespaceCall]bool
+
+	// The parser does two passes and we need to pass the scope tree information
+	// from the first pass to the second pass. That's done by tracking the calls
+	// to pushScopeForParsePass() and popScope() during the first pass in
+	// scopesInOrder.
+	//
+	// Then, when the second pass calls pushScopeForVisitPass() and popScope(),
+	// we consume entries from scopesInOrder and make sure they are in the same
+	// order. This way the second pass can efficiently use the same scope tree
+	// as the first pass without having to attach the scope tree to the AST.
+	//
+	// We need to split this into two passes because the pass that declares the
+	// symbols must be separate from the pass that binds identifiers to declared
+	// symbols to handle declaring a hoisted "var" symbol in a nested scope and
+	// binding a name to it in a parent or sibling scope.
+	scopesInOrder []scopeOrder
+
+	// These propagate the name from the parent context into an anonymous child
+	// expression. For example:
+	//
+	//   let foo = function() {}
+	//   assert.strictEqual(foo.name, 'foo')
+	//
+	nameToKeep      string
+	nameToKeepIsFor js_ast.E
+
+	// These properties are for the visit pass, which runs after the parse pass.
+	// The visit pass binds identifiers to declared symbols, does constant
+	// folding, substitutes compile-time variable definitions, and lowers certain
+	// syntactic constructs as appropriate.
+	stmtExprValue                        js_ast.E
+	callTarget                           js_ast.E
+	dotOrIndexTarget                     js_ast.E
+	templateTag                          js_ast.E
+	deleteTarget                         js_ast.E
+	loopBody                             js_ast.S
+	suspiciousLogicalOperatorInsideArrow js_ast.E
+	moduleScope                          *js_ast.Scope
+
+	// This is internal-only data used for the implementation of Yarn PnP
+	manifestForYarnPnP     js_ast.Expr
+	stringLocalsForYarnPnP map[ast.Ref]stringLocalForYarnPnP
+
+	// This helps recognize the "await import()" pattern. When this is present,
+	// warnings about non-string import paths will be omitted inside try blocks.
+	awaitTarget js_ast.E
+
+	// This helps recognize the "import().catch()" pattern. We also try to avoid
+	// warning about this just like the "try { await import() }" pattern.
+	thenCatchChain thenCatchChain
+
+	// When bundling, hoisted top-level local variables declared with "var" in
+	// nested scopes are moved up to be declared in the top-level scope instead.
+	// The old "var" statements are turned into regular assignments instead. This
+	// makes it easier to quickly scan the top-level statements for "var" locals
+	// with the guarantee that all will be found.
+	relocatedTopLevelVars []ast.LocRef
+
+	// We need to lower private names such as "#foo" if they are used in a brand
+	// check such as "#foo in x" even if the private name syntax would otherwise
+	// be supported. This is because private names are a newly-added feature.
+	//
+	// However, this parser operates in only two passes for speed. The first pass
+	// parses things and declares variables, and the second pass lowers things and
+	// resolves references to declared variables. So the existence of a "#foo in x"
+	// expression for a specific "#foo" cannot be used to decide to lower "#foo"
+	// because it's too late by that point. There may be another expression such
+	// as "x.#foo" before that point and that must be lowered as well even though
+	// it has already been visited.
+	//
+	// Instead what we do is track just the names of fields used in private brand
+	// checks during the first pass. This tracks the names themselves, not symbol
+	// references. Then, during the second pass when we are about to enter into
+	// a class, we conservatively decide to lower all private names in that class
+	// which are used in a brand check anywhere in the file.
+	lowerAllOfThesePrivateNames map[string]bool
+
+	// Temporary variables used for lowering
+	tempLetsToDeclare         []ast.Ref
+	tempRefsToDeclare         []tempRef
+	topLevelTempRefsToDeclare []tempRef
+
+	lexer js_lexer.Lexer
+
+	// Private field access in a decorator lowers all private fields in that class
+	parseExperimentalDecoratorNesting int
+
+	// Temporary variables used for lowering
+	tempRefCount         int
+	topLevelTempRefCount int
+
+	// We need to scan over the source contents to recover the line and column offsets
+	jsxSourceLoc    int
+	jsxSourceLine   int
+	jsxSourceColumn int
+
+	exportsRef    ast.Ref
+	requireRef    ast.Ref
+	moduleRef     ast.Ref
+	importMetaRef ast.Ref
+	promiseRef    ast.Ref
+	regExpRef     ast.Ref
+	superCtorRef  ast.Ref
+
+	// Imports from "react/jsx-runtime" and "react", respectively.
+	// (Or whatever was specified in the "importSource" option)
+	jsxRuntimeImports map[string]ast.LocRef
+	jsxLegacyImports  map[string]ast.LocRef
+
+	// For lowering private methods
+	weakMapRef ast.Ref
+	weakSetRef ast.Ref
+
+	esmImportStatementKeyword logger.Range
+	esmImportMeta             logger.Range
+	esmExportKeyword          logger.Range
+	enclosingClassKeyword     logger.Range
+	topLevelAwaitKeyword      logger.Range
+	liveTopLevelAwaitKeyword  logger.Range
+
+	latestArrowArgLoc      logger.Loc
+	forbidSuffixAfterAsLoc logger.Loc
+	firstJSXElementLoc     logger.Loc
+
+	fnOrArrowDataVisit fnOrArrowDataVisit
+
+	// ArrowFunction is a special case in the grammar. Although it appears to be
+	// a PrimaryExpression, it's actually an AssignmentExpression. This means if
+	// a AssignmentExpression ends up producing an ArrowFunction then nothing can
+	// come after it other than the comma operator, since the comma operator is
+	// the only thing above AssignmentExpression under the Expression rule:
+	//
+	//   AssignmentExpression:
+	//     ArrowFunction
+	//     ConditionalExpression
+	//     LeftHandSideExpression = AssignmentExpression
+	//     LeftHandSideExpression AssignmentOperator AssignmentExpression
+	//
+	//   Expression:
+	//     AssignmentExpression
+	//     Expression , AssignmentExpression
+	//
+	afterArrowBodyLoc logger.Loc
+
+	// Setting this to true disables warnings about code that is very likely to
+	// be a bug. This is used to ignore issues inside "node_modules" directories.
+	// This has caught real issues in the past. However, it's not esbuild's job
+	// to find bugs in other libraries, and these warnings are problematic for
+	// people using these libraries with esbuild. The only fix is to either
+	// disable all esbuild warnings and not get warnings about your own code, or
+	// to try to get the warning fixed in the affected library. This is
+	// especially annoying if the warning is a false positive as was the case in
+	// https://github.com/firebase/firebase-js-sdk/issues/3814. So these warnings
+	// are now disabled for code inside "node_modules" directories.
+	suppressWarningsAboutWeirdCode bool
+
+	// A file is considered to be an ECMAScript module if it has any of the
+	// features of one (e.g. the "export" keyword), otherwise it's considered
+	// a CommonJS module.
+	//
+	// However, we have a single exception: a file where the only ESM feature
+	// is the "import" keyword is allowed to have CommonJS exports. This feature
+	// is necessary to be able to synchronously import ESM code into CommonJS,
+	// which we need to enable in a few important cases. Some examples are:
+	// our runtime code, injected files (the "inject" feature is ESM-only),
+	// and certain automatically-generated virtual modules from plugins.
+	isFileConsideredToHaveESMExports bool // Use only for export-related stuff
+	isFileConsideredESM              bool // Use for all other stuff
+
+	// Inside a TypeScript namespace, an "export declare" statement can be used
+	// to cause a namespace to be emitted even though it has no other observable
+	// effect. This flag is used to implement this feature.
+	//
+	// Specifically, namespaces should be generated for all of the following
+	// namespaces below except for "f", which should not be generated:
+	//
+	//   namespace a { export declare const a }
+	//   namespace b { export declare let [[b]] }
+	//   namespace c { export declare function c() }
+	//   namespace d { export declare class d {} }
+	//   namespace e { export declare enum e {} }
+	//   namespace f { export declare namespace f {} }
+	//
+	// The TypeScript compiler compiles this into the following code (notice "f"
+	// is missing):
+	//
+	//   var a; (function (a_1) {})(a || (a = {}));
+	//   var b; (function (b_1) {})(b || (b = {}));
+	//   var c; (function (c_1) {})(c || (c = {}));
+	//   var d; (function (d_1) {})(d || (d = {}));
+	//   var e; (function (e_1) {})(e || (e = {}));
+	//
+	// Note that this should not be implemented by declaring symbols for "export
+	// declare" statements because the TypeScript compiler doesn't generate any
+	// code for these statements, so these statements are actually references to
+	// global variables. There is one exception, which is that local variables
+	// *should* be declared as symbols because they are replaced with. This seems
+	// like very arbitrary behavior but it's what the TypeScript compiler does,
+	// so we try to match it.
+	//
+	// Specifically, in the following code below "a" and "b" should be declared
+	// and should be substituted with "ns.a" and "ns.b" but the other symbols
+	// shouldn't. References to the other symbols actually refer to global
+	// variables instead of to symbols that are exported from the namespace.
+	// This is the case as of TypeScript 4.3. I assume this is a TypeScript bug:
+	//
+	//   namespace ns {
+	//     export declare const a
+	//     export declare let [[b]]
+	//     export declare function c()
+	//     export declare class d { }
+	//     export declare enum e { }
+	//     console.log(a, b, c, d, e)
+	//   }
+	//
+	// The TypeScript compiler compiles this into the following code:
+	//
+	//   var ns;
+	//   (function (ns) {
+	//       console.log(ns.a, ns.b, c, d, e);
+	//   })(ns || (ns = {}));
+	//
+	// Relevant issue: https://github.com/evanw/esbuild/issues/1158
+	hasNonLocalExportDeclareInsideNamespace bool
+
+	// When this flag is enabled, we attempt to fold all expressions that
+	// TypeScript would consider to be "constant expressions". This flag is
+	// enabled inside each enum body block since TypeScript requires numeric
+	// constant folding in enum definitions.
+	//
+	// We also enable this flag in certain cases in JavaScript files such as when
+	// parsing "const" declarations at the top of a non-ESM file, but we still
+	// reuse TypeScript's notion of "constant expressions" for our own convenience.
+	//
+	// As of TypeScript 5.0, a "constant expression" is defined as follows:
+	//
+	//   An expression is considered a constant expression if it is
+	//
+	//   * a number or string literal,
+	//   * a unary +, -, or ~ applied to a numeric constant expression,
+	//   * a binary +, -, *, /, %, **, <<, >>, >>>, |, &, ^ applied to two numeric constant expressions,
+	//   * a binary + applied to two constant expressions whereof at least one is a string,
+	//   * a template expression where each substitution expression is a constant expression,
+	//   * a parenthesized constant expression,
+	//   * a dotted name (e.g. x.y.z) that references a const variable with a constant expression initializer and no type annotation,
+	//   * a dotted name that references an enum member with an enum literal type, or
+	//   * a dotted name indexed by a string literal (e.g. x.y["z"]) that references an enum member with an enum literal type.
+	//
+	// More detail: https://github.com/microsoft/TypeScript/pull/50528. Note that
+	// we don't implement certain items in this list. For example, we don't do all
+	// number-to-string conversions since ours might differ from how JavaScript
+	// would do it, which would be a correctness issue.
+	shouldFoldTypeScriptConstantExpressions bool
+
+	allowIn                     bool
+	allowPrivateIdentifiers     bool
+	hasTopLevelReturn           bool
+	latestReturnHadSemicolon    bool
+	messageAboutThisIsUndefined bool
+	isControlFlowDead           bool
+
+	// If this is true, then all top-level statements are wrapped in a try/catch
+	willWrapModuleInTryCatchForUsing bool
+}
+
+type globPatternImport struct {
+	assertOrWith     *ast.ImportAssertOrWith
+	parts            []helpers.GlobPart
+	name             string
+	approximateRange logger.Range
+	ref              ast.Ref
+	kind             ast.ImportKind
+}
+
+type namespaceImportItems struct {
+	entries           map[string]ast.LocRef
+	importRecordIndex uint32
+}
+
+type stringLocalForYarnPnP struct {
+	value []uint16
+	loc   logger.Loc
+}
+
+type injectedSymbolSource struct {
+	source logger.Source
+	loc    logger.Loc
+}
+
+type injectedDotName struct {
+	parts               []string
+	injectedDefineIndex uint32
+}
+
+type importNamespaceCallKind uint8
+
+const (
+	exprKindCall importNamespaceCallKind = iota
+	exprKindNew
+	exprKindJSXTag
+)
+
+type importNamespaceCall struct {
+	ref  ast.Ref
+	kind importNamespaceCallKind
+}
+
+type thenCatchChain struct {
+	nextTarget      js_ast.E
+	catchLoc        logger.Loc
+	hasMultipleArgs bool
+	hasCatch        bool
+}
+
+// This is used as part of an incremental build cache key. Some of these values
+// can potentially change between builds if they are derived from nearby
+// "package.json" or "tsconfig.json" files that were changed since the last
+// build.
+type Options struct {
+	injectedFiles  []config.InjectedFile
+	jsx            config.JSXOptions
+	tsAlwaysStrict *config.TSAlwaysStrict
+	mangleProps    *regexp.Regexp
+	reserveProps   *regexp.Regexp
+	dropLabels     []string
+
+	// This pointer will always be different for each build but the contents
+	// shouldn't ever behave different semantically. We ignore this field for the
+	// equality comparison.
+	defines *config.ProcessedDefines
+
+	// This is an embedded struct. Always access these directly instead of off
+	// the name "optionsThatSupportStructuralEquality". This is only grouped like
+	// this to make the equality comparison easier and safer (and hopefully faster).
+	optionsThatSupportStructuralEquality
+}
+
+type optionsThatSupportStructuralEquality struct {
+	originalTargetEnv                 string
+	moduleTypeData                    js_ast.ModuleTypeData
+	unsupportedJSFeatures             compat.JSFeature
+	unsupportedJSFeatureOverrides     compat.JSFeature
+	unsupportedJSFeatureOverridesMask compat.JSFeature
+
+	// Byte-sized values go here (gathered together here to keep this object compact)
+	ts                     config.TSOptions
+	mode                   config.Mode
+	platform               config.Platform
+	outputFormat           config.Format
+	asciiOnly              bool
+	keepNames              bool
+	minifySyntax           bool
+	minifyIdentifiers      bool
+	minifyWhitespace       bool
+	omitRuntimeForTests    bool
+	omitJSXRuntimeForTests bool
+	ignoreDCEAnnotations   bool
+	treeShaking            bool
+	dropDebugger           bool
+	mangleQuoted           bool
+
+	// This is an internal-only option used for the implementation of Yarn PnP
+	decodeHydrateRuntimeStateYarnPnP bool
+}
+
+func OptionsForYarnPnP() Options {
+	return Options{
+		optionsThatSupportStructuralEquality: optionsThatSupportStructuralEquality{
+			decodeHydrateRuntimeStateYarnPnP: true,
+		},
+	}
+}
+
+func OptionsFromConfig(options *config.Options) Options {
+	return Options{
+		injectedFiles:  options.InjectedFiles,
+		jsx:            options.JSX,
+		defines:        options.Defines,
+		tsAlwaysStrict: options.TSAlwaysStrict,
+		mangleProps:    options.MangleProps,
+		reserveProps:   options.ReserveProps,
+		dropLabels:     options.DropLabels,
+
+		optionsThatSupportStructuralEquality: optionsThatSupportStructuralEquality{
+			unsupportedJSFeatures:             options.UnsupportedJSFeatures,
+			unsupportedJSFeatureOverrides:     options.UnsupportedJSFeatureOverrides,
+			unsupportedJSFeatureOverridesMask: options.UnsupportedJSFeatureOverridesMask,
+			originalTargetEnv:                 options.OriginalTargetEnv,
+			ts:                                options.TS,
+			mode:                              options.Mode,
+			platform:                          options.Platform,
+			outputFormat:                      options.OutputFormat,
+			moduleTypeData:                    options.ModuleTypeData,
+			asciiOnly:                         options.ASCIIOnly,
+			keepNames:                         options.KeepNames,
+			minifySyntax:                      options.MinifySyntax,
+			minifyIdentifiers:                 options.MinifyIdentifiers,
+			minifyWhitespace:                  options.MinifyWhitespace,
+			omitRuntimeForTests:               options.OmitRuntimeForTests,
+			omitJSXRuntimeForTests:            options.OmitJSXRuntimeForTests,
+			ignoreDCEAnnotations:              options.IgnoreDCEAnnotations,
+			treeShaking:                       options.TreeShaking,
+			dropDebugger:                      options.DropDebugger,
+			mangleQuoted:                      options.MangleQuoted,
+		},
+	}
+}
+
+func (a *Options) Equal(b *Options) bool {
+	// Compare "optionsThatSupportStructuralEquality"
+	if a.optionsThatSupportStructuralEquality != b.optionsThatSupportStructuralEquality {
+		return false
+	}
+
+	// Compare "tsAlwaysStrict"
+	if (a.tsAlwaysStrict == nil && b.tsAlwaysStrict != nil) || (a.tsAlwaysStrict != nil && b.tsAlwaysStrict == nil) ||
+		(a.tsAlwaysStrict != nil && b.tsAlwaysStrict != nil && *a.tsAlwaysStrict != *b.tsAlwaysStrict) {
+		return false
+	}
+
+	// Compare "mangleProps" and "reserveProps"
+	if !isSameRegexp(a.mangleProps, b.mangleProps) || !isSameRegexp(a.reserveProps, b.reserveProps) {
+		return false
+	}
+
+	// Compare "dropLabels"
+	if !helpers.StringArraysEqual(a.dropLabels, b.dropLabels) {
+		return false
+	}
+
+	// Compare "injectedFiles"
+	if len(a.injectedFiles) != len(b.injectedFiles) {
+		return false
+	}
+	for i, x := range a.injectedFiles {
+		y := b.injectedFiles[i]
+		if x.Source != y.Source || x.DefineName != y.DefineName || len(x.Exports) != len(y.Exports) {
+			return false
+		}
+		for j := range x.Exports {
+			if x.Exports[j] != y.Exports[j] {
+				return false
+			}
+		}
+	}
+
+	// Compare "jsx"
+	if a.jsx.Parse != b.jsx.Parse || !jsxExprsEqual(a.jsx.Factory, b.jsx.Factory) || !jsxExprsEqual(a.jsx.Fragment, b.jsx.Fragment) {
+		return false
+	}
+
+	// Do a cheap assert that the defines object hasn't changed
+	if (a.defines != nil || b.defines != nil) && (a.defines == nil || b.defines == nil ||
+		len(a.defines.IdentifierDefines) != len(b.defines.IdentifierDefines) ||
+		len(a.defines.DotDefines) != len(b.defines.DotDefines)) {
+		panic("Internal error")
+	}
+
+	return true
+}
+
+func isSameRegexp(a *regexp.Regexp, b *regexp.Regexp) bool {
+	if a == nil {
+		return b == nil
+	} else {
+		return b != nil && a.String() == b.String()
+	}
+}
+
+func jsxExprsEqual(a config.DefineExpr, b config.DefineExpr) bool {
+	if !helpers.StringArraysEqual(a.Parts, b.Parts) {
+		return false
+	}
+
+	if a.Constant != nil {
+		if b.Constant == nil || !js_ast.ValuesLookTheSame(a.Constant, b.Constant) {
+			return false
+		}
+	} else if b.Constant != nil {
+		return false
+	}
+
+	return true
+}
+
+type tempRef struct {
+	valueOrNil js_ast.Expr
+	ref        ast.Ref
+}
+
+const (
+	locModuleScope = -1
+)
+
+type scopeOrder struct {
+	scope *js_ast.Scope
+	loc   logger.Loc
+}
+
+type awaitOrYield uint8
+
+const (
+	// The keyword is used as an identifier, not a special expression
+	allowIdent awaitOrYield = iota
+
+	// Declaring the identifier is forbidden, and the keyword is used as a special expression
+	allowExpr
+
+	// Declaring the identifier is forbidden, and using the identifier is also forbidden
+	forbidAll
+)
+
+// This is function-specific information used during parsing. It is saved and
+// restored on the call stack around code that parses nested functions and
+// arrow expressions.
+type fnOrArrowDataParse struct {
+	arrowArgErrors      *deferredArrowArgErrors
+	decoratorScope      *js_ast.Scope
+	asyncRange          logger.Range
+	needsAsyncLoc       logger.Loc
+	await               awaitOrYield
+	yield               awaitOrYield
+	allowSuperCall      bool
+	allowSuperProperty  bool
+	isTopLevel          bool
+	isConstructor       bool
+	isTypeScriptDeclare bool
+	isThisDisallowed    bool
+	isReturnDisallowed  bool
+
+	// In TypeScript, forward declarations of functions have no bodies
+	allowMissingBodyForTypeScript bool
+}
+
+// This is function-specific information used during visiting. It is saved and
+// restored on the call stack around code that parses nested functions and
+// arrow expressions.
+type fnOrArrowDataVisit struct {
+	// This is used to silence unresolvable imports due to "require" calls inside
+	// a try/catch statement. The assumption is that the try/catch statement is
+	// there to handle the case where the reference to "require" crashes.
+	tryBodyCount int32
+	tryCatchLoc  logger.Loc
+
+	isArrow                        bool
+	isAsync                        bool
+	isGenerator                    bool
+	isInsideLoop                   bool
+	isInsideSwitch                 bool
+	isDerivedClassCtor             bool
+	isOutsideFnOrArrow             bool
+	shouldLowerSuperPropertyAccess bool
+}
+
+// This is function-specific information used during visiting. It is saved and
+// restored on the call stack around code that parses nested functions (but not
+// nested arrow functions).
+type fnOnlyDataVisit struct {
+	// This is a reference to the magic "arguments" variable that exists inside
+	// functions in JavaScript. It will be non-nil inside functions and nil
+	// otherwise.
+	argumentsRef *ast.Ref
+
+	// Arrow functions don't capture the value of "this" and "arguments". Instead,
+	// the values are inherited from the surrounding context. If arrow functions
+	// are turned into regular functions due to lowering, we will need to generate
+	// local variables to capture these values so they are preserved correctly.
+	thisCaptureRef      *ast.Ref
+	argumentsCaptureRef *ast.Ref
+
+	// If true, we're inside a static class context where "this" expressions
+	// should be replaced with the class name.
+	shouldReplaceThisWithInnerClassNameRef bool
+
+	// This is true if "this" is equal to the class name. It's true if we're in a
+	// static class field initializer, a static class method, or a static class
+	// block.
+	isInStaticClassContext bool
+
+	// This is a reference to the enclosing class name if there is one. It's used
+	// to implement "this" and "super" references. A name is automatically generated
+	// if one is missing so this will always be present inside a class body.
+	innerClassNameRef *ast.Ref
+
+	// If we're inside an async arrow function and async functions are not
+	// supported, then we will have to convert that arrow function to a generator
+	// function. That means references to "arguments" inside the arrow function
+	// will have to reference a captured variable instead of the real variable.
+	isInsideAsyncArrowFn bool
+
+	// If false, disallow "new.target" expressions. We disallow all "new.target"
+	// expressions at the top-level of the file (i.e. not inside a function or
+	// a class field). Technically since CommonJS files are wrapped in a function
+	// you can use "new.target" in node as an alias for "undefined" but we don't
+	// support that.
+	isNewTargetAllowed bool
+
+	// If false, the value for "this" is the top-level module scope "this" value.
+	// That means it's "undefined" for ECMAScript modules and "exports" for
+	// CommonJS modules. We track this information so that we can substitute the
+	// correct value for these top-level "this" references at compile time instead
+	// of passing the "this" expression through to the output and leaving the
+	// interpretation up to the run-time behavior of the generated code.
+	//
+	// If true, the value for "this" is nested inside something (either a function
+	// or a class declaration). That means the top-level module scope "this" value
+	// has been shadowed and is now inaccessible.
+	isThisNested bool
+
+	// Do not warn about "this" being undefined for code that the TypeScript
+	// compiler generates that looks like this:
+	//
+	//   var __rest = (this && this.__rest) || function (s, e) {
+	//     ...
+	//   };
+	//
+	silenceMessageAboutThisBeingUndefined bool
+}
+
+const bloomFilterSize = 251
+
+type duplicateCaseValue struct {
+	value js_ast.Expr
+	hash  uint32
+}
+
+type duplicateCaseChecker struct {
+	cases       []duplicateCaseValue
+	bloomFilter [(bloomFilterSize + 7) / 8]byte
+}
+
+func (dc *duplicateCaseChecker) reset() {
+	// Preserve capacity
+	dc.cases = dc.cases[:0]
+
+	// This should be optimized by the compiler. See this for more information:
+	// https://github.com/golang/go/issues/5373
+	bytes := dc.bloomFilter
+	for i := range bytes {
+		bytes[i] = 0
+	}
+}
+
+func (dc *duplicateCaseChecker) check(p *parser, expr js_ast.Expr) {
+	if hash, ok := duplicateCaseHash(expr); ok {
+		bucket := hash % bloomFilterSize
+		entry := &dc.bloomFilter[bucket/8]
+		mask := byte(1) << (bucket % 8)
+
+		// Check for collisions
+		if (*entry & mask) != 0 {
+			for _, c := range dc.cases {
+				if c.hash == hash {
+					if equals, couldBeIncorrect := duplicateCaseEquals(c.value, expr); equals {
+						var laterRange logger.Range
+						var earlierRange logger.Range
+						if _, ok := expr.Data.(*js_ast.EString); ok {
+							laterRange = p.source.RangeOfString(expr.Loc)
+						} else {
+							laterRange = p.source.RangeOfOperatorBefore(expr.Loc, "case")
+						}
+						if _, ok := c.value.Data.(*js_ast.EString); ok {
+							earlierRange = p.source.RangeOfString(c.value.Loc)
+						} else {
+							earlierRange = p.source.RangeOfOperatorBefore(c.value.Loc, "case")
+						}
+						text := "This case clause will never be evaluated because it duplicates an earlier case clause"
+						if couldBeIncorrect {
+							text = "This case clause may never be evaluated because it likely duplicates an earlier case clause"
+						}
+						kind := logger.Warning
+						if p.suppressWarningsAboutWeirdCode {
+							kind = logger.Debug
+						}
+						p.log.AddIDWithNotes(logger.MsgID_JS_DuplicateCase, kind, &p.tracker, laterRange, text,
+							[]logger.MsgData{p.tracker.MsgData(earlierRange, "The earlier case clause is here:")})
+					}
+					return
+				}
+			}
+		}
+
+		*entry |= mask
+		dc.cases = append(dc.cases, duplicateCaseValue{hash: hash, value: expr})
+	}
+}
+
+func duplicateCaseHash(expr js_ast.Expr) (uint32, bool) {
+	switch e := expr.Data.(type) {
+	case *js_ast.EInlinedEnum:
+		return duplicateCaseHash(e.Value)
+
+	case *js_ast.ENull:
+		return 0, true
+
+	case *js_ast.EUndefined:
+		return 1, true
+
+	case *js_ast.EBoolean:
+		if e.Value {
+			return helpers.HashCombine(2, 1), true
+		}
+		return helpers.HashCombine(2, 0), true
+
+	case *js_ast.ENumber:
+		bits := math.Float64bits(e.Value)
+		return helpers.HashCombine(helpers.HashCombine(3, uint32(bits)), uint32(bits>>32)), true
+
+	case *js_ast.EString:
+		hash := uint32(4)
+		for _, c := range e.Value {
+			hash = helpers.HashCombine(hash, uint32(c))
+		}
+		return hash, true
+
+	case *js_ast.EBigInt:
+		hash := uint32(5)
+		for _, c := range e.Value {
+			hash = helpers.HashCombine(hash, uint32(c))
+		}
+		return hash, true
+
+	case *js_ast.EIdentifier:
+		return helpers.HashCombine(6, e.Ref.InnerIndex), true
+
+	case *js_ast.EDot:
+		if target, ok := duplicateCaseHash(e.Target); ok {
+			return helpers.HashCombineString(helpers.HashCombine(7, target), e.Name), true
+		}
+
+	case *js_ast.EIndex:
+		if target, ok := duplicateCaseHash(e.Target); ok {
+			if index, ok := duplicateCaseHash(e.Index); ok {
+				return helpers.HashCombine(helpers.HashCombine(8, target), index), true
+			}
+		}
+	}
+
+	return 0, false
+}
+
+func duplicateCaseEquals(left js_ast.Expr, right js_ast.Expr) (equals bool, couldBeIncorrect bool) {
+	if b, ok := right.Data.(*js_ast.EInlinedEnum); ok {
+		return duplicateCaseEquals(left, b.Value)
+	}
+
+	switch a := left.Data.(type) {
+	case *js_ast.EInlinedEnum:
+		return duplicateCaseEquals(a.Value, right)
+
+	case *js_ast.ENull:
+		_, ok := right.Data.(*js_ast.ENull)
+		return ok, false
+
+	case *js_ast.EUndefined:
+		_, ok := right.Data.(*js_ast.EUndefined)
+		return ok, false
+
+	case *js_ast.EBoolean:
+		b, ok := right.Data.(*js_ast.EBoolean)
+		return ok && a.Value == b.Value, false
+
+	case *js_ast.ENumber:
+		b, ok := right.Data.(*js_ast.ENumber)
+		return ok && a.Value == b.Value, false
+
+	case *js_ast.EString:
+		b, ok := right.Data.(*js_ast.EString)
+		return ok && helpers.UTF16EqualsUTF16(a.Value, b.Value), false
+
+	case *js_ast.EBigInt:
+		if b, ok := right.Data.(*js_ast.EBigInt); ok {
+			equal, ok := js_ast.CheckEqualityBigInt(a.Value, b.Value)
+			return ok && equal, false
+		}
+
+	case *js_ast.EIdentifier:
+		b, ok := right.Data.(*js_ast.EIdentifier)
+		return ok && a.Ref == b.Ref, false
+
+	case *js_ast.EDot:
+		if b, ok := right.Data.(*js_ast.EDot); ok && a.OptionalChain == b.OptionalChain && a.Name == b.Name {
+			equals, _ := duplicateCaseEquals(a.Target, b.Target)
+			return equals, true
+		}
+
+	case *js_ast.EIndex:
+		if b, ok := right.Data.(*js_ast.EIndex); ok && a.OptionalChain == b.OptionalChain {
+			if equals, _ := duplicateCaseEquals(a.Index, b.Index); equals {
+				equals, _ := duplicateCaseEquals(a.Target, b.Target)
+				return equals, true
+			}
+		}
+	}
+
+	return false, false
+}
+
+type duplicatePropertiesIn uint8
+
+const (
+	duplicatePropertiesInObject duplicatePropertiesIn = iota
+	duplicatePropertiesInClass
+)
+
+func (p *parser) warnAboutDuplicateProperties(properties []js_ast.Property, in duplicatePropertiesIn) {
+	if len(properties) < 2 {
+		return
+	}
+
+	type keyKind uint8
+	type existingKey struct {
+		loc  logger.Loc
+		kind keyKind
+	}
+	const (
+		keyMissing keyKind = iota
+		keyNormal
+		keyGet
+		keySet
+		keyGetAndSet
+	)
+	instanceKeys := make(map[string]existingKey)
+	staticKeys := make(map[string]existingKey)
+
+	for _, property := range properties {
+		if property.Kind != js_ast.PropertySpread {
+			if str, ok := property.Key.Data.(*js_ast.EString); ok {
+				var keys map[string]existingKey
+				if property.Flags.Has(js_ast.PropertyIsStatic) {
+					keys = staticKeys
+				} else {
+					keys = instanceKeys
+				}
+				key := helpers.UTF16ToString(str.Value)
+				prevKey := keys[key]
+				nextKey := existingKey{kind: keyNormal, loc: property.Key.Loc}
+
+				if property.Kind == js_ast.PropertyGetter {
+					nextKey.kind = keyGet
+				} else if property.Kind == js_ast.PropertySetter {
+					nextKey.kind = keySet
+				}
+
+				if prevKey.kind != keyMissing && (in != duplicatePropertiesInObject || key != "__proto__") && (in != duplicatePropertiesInClass || key != "constructor") {
+					if (prevKey.kind == keyGet && nextKey.kind == keySet) || (prevKey.kind == keySet && nextKey.kind == keyGet) {
+						nextKey.kind = keyGetAndSet
+					} else {
+						var id logger.MsgID
+						var what string
+						var where string
+						switch in {
+						case duplicatePropertiesInObject:
+							id = logger.MsgID_JS_DuplicateObjectKey
+							what = "key"
+							where = "object literal"
+						case duplicatePropertiesInClass:
+							id = logger.MsgID_JS_DuplicateClassMember
+							what = "member"
+							where = "class body"
+						}
+						r := js_lexer.RangeOfIdentifier(p.source, property.Key.Loc)
+						p.log.AddIDWithNotes(id, logger.Warning, &p.tracker, r,
+							fmt.Sprintf("Duplicate %s %q in %s", what, key, where),
+							[]logger.MsgData{p.tracker.MsgData(js_lexer.RangeOfIdentifier(p.source, prevKey.loc),
+								fmt.Sprintf("The original %s %q is here:", what, key))})
+					}
+				}
+
+				keys[key] = nextKey
+			}
+		}
+	}
+}
+
+func isJumpStatement(data js_ast.S) bool {
+	switch data.(type) {
+	case *js_ast.SBreak, *js_ast.SContinue, *js_ast.SReturn, *js_ast.SThrow:
+		return true
+	}
+
+	return false
+}
+
+func jumpStmtsLookTheSame(left js_ast.S, right js_ast.S) bool {
+	switch a := left.(type) {
+	case *js_ast.SBreak:
+		b, ok := right.(*js_ast.SBreak)
+		return ok && (a.Label == nil) == (b.Label == nil) && (a.Label == nil || a.Label.Ref == b.Label.Ref)
+
+	case *js_ast.SContinue:
+		b, ok := right.(*js_ast.SContinue)
+		return ok && (a.Label == nil) == (b.Label == nil) && (a.Label == nil || a.Label.Ref == b.Label.Ref)
+
+	case *js_ast.SReturn:
+		b, ok := right.(*js_ast.SReturn)
+		return ok && (a.ValueOrNil.Data == nil) == (b.ValueOrNil.Data == nil) &&
+			(a.ValueOrNil.Data == nil || js_ast.ValuesLookTheSame(a.ValueOrNil.Data, b.ValueOrNil.Data))
+
+	case *js_ast.SThrow:
+		b, ok := right.(*js_ast.SThrow)
+		return ok && js_ast.ValuesLookTheSame(a.Value.Data, b.Value.Data)
+	}
+
+	return false
+}
+
+func (p *parser) selectLocalKind(kind js_ast.LocalKind) js_ast.LocalKind {
+	// Use "var" instead of "let" and "const" if the variable declaration may
+	// need to be separated from the initializer. This allows us to safely move
+	// this declaration into a nested scope.
+	if p.currentScope.Parent == nil && (kind == js_ast.LocalLet || kind == js_ast.LocalConst) &&
+		(p.options.mode == config.ModeBundle || p.willWrapModuleInTryCatchForUsing) {
+		return js_ast.LocalVar
+	}
+
+	// Optimization: use "let" instead of "const" because it's shorter. This is
+	// only done when bundling because assigning to "const" is only an error when
+	// bundling.
+	if p.options.mode == config.ModeBundle && kind == js_ast.LocalConst && p.options.minifySyntax {
+		return js_ast.LocalLet
+	}
+
+	return kind
+}
+
+func (p *parser) pushScopeForParsePass(kind js_ast.ScopeKind, loc logger.Loc) int {
+	parent := p.currentScope
+	scope := &js_ast.Scope{
+		Kind:    kind,
+		Parent:  parent,
+		Members: make(map[string]js_ast.ScopeMember),
+		Label:   ast.LocRef{Ref: ast.InvalidRef},
+	}
+	if parent != nil {
+		parent.Children = append(parent.Children, scope)
+		scope.StrictMode = parent.StrictMode
+		scope.UseStrictLoc = parent.UseStrictLoc
+	}
+	p.currentScope = scope
+
+	// Enforce that scope locations are strictly increasing to help catch bugs
+	// where the pushed scopes are mismatched between the first and second passes
+	if len(p.scopesInOrder) > 0 {
+		prevStart := p.scopesInOrder[len(p.scopesInOrder)-1].loc.Start
+		if prevStart >= loc.Start {
+			panic(fmt.Sprintf("Scope location %d must be greater than %d", loc.Start, prevStart))
+		}
+	}
+
+	// Copy down function arguments into the function body scope. That way we get
+	// errors if a statement in the function body tries to re-declare any of the
+	// arguments.
+	if kind == js_ast.ScopeFunctionBody {
+		if scope.Parent.Kind != js_ast.ScopeFunctionArgs {
+			panic("Internal error")
+		}
+		for name, member := range scope.Parent.Members {
+			// Don't copy down the optional function expression name. Re-declaring
+			// the name of a function expression is allowed.
+			kind := p.symbols[member.Ref.InnerIndex].Kind
+			if kind != ast.SymbolHoistedFunction {
+				scope.Members[name] = member
+			}
+		}
+	}
+
+	// Remember the length in case we call popAndDiscardScope() later
+	scopeIndex := len(p.scopesInOrder)
+	p.scopesInOrder = append(p.scopesInOrder, scopeOrder{loc: loc, scope: scope})
+	return scopeIndex
+}
+
+func (p *parser) popScope() {
+	// We cannot rename anything inside a scope containing a direct eval() call
+	if p.currentScope.ContainsDirectEval {
+		for _, member := range p.currentScope.Members {
+			// Using direct eval when bundling is not a good idea in general because
+			// esbuild must assume that it can potentially reach anything in any of
+			// the containing scopes. We try to make it work but this isn't possible
+			// in some cases.
+			//
+			// For example, symbols imported using an ESM import are a live binding
+			// to the underlying symbol in another file. This is emulated during
+			// scope hoisting by erasing the ESM import and just referencing the
+			// underlying symbol in the flattened bundle directly. However, that
+			// symbol may have a different name which could break uses of direct
+			// eval:
+			//
+			//   // Before bundling
+			//   import { foo as bar } from './foo.js'
+			//   console.log(eval('bar'))
+			//
+			//   // After bundling
+			//   let foo = 123 // The contents of "foo.js"
+			//   console.log(eval('bar'))
+			//
+			// There really isn't any way to fix this. You can't just rename "foo" to
+			// "bar" in the example above because there may be a third bundled file
+			// that also contains direct eval and imports the same symbol with a
+			// different conflicting import alias. And there is no way to store a
+			// live binding to the underlying symbol in a variable with the import's
+			// name so that direct eval can access it:
+			//
+			//   // After bundling
+			//   let foo = 123 // The contents of "foo.js"
+			//   const bar = /* cannot express a live binding to "foo" here */
+			//   console.log(eval('bar'))
+			//
+			// Technically a "with" statement could potentially make this work (with
+			// a big hit to performance), but they are deprecated and are unavailable
+			// in strict mode. This is a non-starter since all ESM code is strict mode.
+			//
+			// So while we still try to obey the requirement that all symbol names are
+			// pinned when direct eval is present, we make an exception for top-level
+			// symbols in an ESM file when bundling is enabled. We make no guarantee
+			// that "eval" will be able to reach these symbols and we allow them to be
+			// renamed or removed by tree shaking.
+			if p.options.mode == config.ModeBundle && p.currentScope.Parent == nil && p.isFileConsideredESM {
+				continue
+			}
+
+			p.symbols[member.Ref.InnerIndex].Flags |= ast.MustNotBeRenamed
+		}
+	}
+
+	p.currentScope = p.currentScope.Parent
+}
+
+func (p *parser) popAndDiscardScope(scopeIndex int) {
+	// Unwind any newly-added scopes in reverse order
+	for i := len(p.scopesInOrder) - 1; i >= scopeIndex; i-- {
+		scope := p.scopesInOrder[i].scope
+		parent := scope.Parent
+		last := len(parent.Children) - 1
+		if parent.Children[last] != scope {
+			panic("Internal error")
+		}
+		parent.Children = parent.Children[:last]
+	}
+
+	// Move up to the parent scope
+	p.currentScope = p.currentScope.Parent
+
+	// Truncate the scope order where we started to pretend we never saw this scope
+	p.scopesInOrder = p.scopesInOrder[:scopeIndex]
+}
+
+func (p *parser) popAndFlattenScope(scopeIndex int) {
+	// Move up to the parent scope
+	toFlatten := p.currentScope
+	parent := toFlatten.Parent
+	p.currentScope = parent
+
+	// Erase this scope from the order. This will shift over the indices of all
+	// the scopes that were created after us. However, we shouldn't have to
+	// worry about other code with outstanding scope indices for these scopes.
+	// These scopes were all created in between this scope's push and pop
+	// operations, so they should all be child scopes and should all be popped
+	// by the time we get here.
+	copy(p.scopesInOrder[scopeIndex:], p.scopesInOrder[scopeIndex+1:])
+	p.scopesInOrder = p.scopesInOrder[:len(p.scopesInOrder)-1]
+
+	// Remove the last child from the parent scope
+	last := len(parent.Children) - 1
+	if parent.Children[last] != toFlatten {
+		panic("Internal error")
+	}
+	parent.Children = parent.Children[:last]
+
+	// Reparent our child scopes into our parent
+	for _, scope := range toFlatten.Children {
+		scope.Parent = parent
+		parent.Children = append(parent.Children, scope)
+	}
+}
+
+// Undo all scopes pushed and popped after this scope index. This assumes that
+// the scope stack is at the same level now as it was at the given scope index.
+func (p *parser) discardScopesUpTo(scopeIndex int) {
+	// Remove any direct children from their parent
+	children := p.currentScope.Children
+	for _, child := range p.scopesInOrder[scopeIndex:] {
+		if child.scope.Parent == p.currentScope {
+			for i := len(children) - 1; i >= 0; i-- {
+				if children[i] == child.scope {
+					children = append(children[:i], children[i+1:]...)
+					break
+				}
+			}
+		}
+	}
+	p.currentScope.Children = children
+
+	// Truncate the scope order where we started to pretend we never saw this scope
+	p.scopesInOrder = p.scopesInOrder[:scopeIndex]
+}
+
+func (p *parser) newSymbol(kind ast.SymbolKind, name string) ast.Ref {
+	ref := ast.Ref{SourceIndex: p.source.Index, InnerIndex: uint32(len(p.symbols))}
+	p.symbols = append(p.symbols, ast.Symbol{
+		Kind:         kind,
+		OriginalName: name,
+		Link:         ast.InvalidRef,
+	})
+	if p.options.ts.Parse {
+		p.tsUseCounts = append(p.tsUseCounts, 0)
+	}
+	return ref
+}
+
+// This is similar to "ast.MergeSymbols" but it works with this parser's
+// one-level symbol map instead of the linker's two-level symbol map. It also
+// doesn't handle cycles since they shouldn't come up due to the way this
+// function is used.
+func (p *parser) mergeSymbols(old ast.Ref, new ast.Ref) ast.Ref {
+	if old == new {
+		return new
+	}
+
+	oldSymbol := &p.symbols[old.InnerIndex]
+	if oldSymbol.Link != ast.InvalidRef {
+		oldSymbol.Link = p.mergeSymbols(oldSymbol.Link, new)
+		return oldSymbol.Link
+	}
+
+	newSymbol := &p.symbols[new.InnerIndex]
+	if newSymbol.Link != ast.InvalidRef {
+		newSymbol.Link = p.mergeSymbols(old, newSymbol.Link)
+		return newSymbol.Link
+	}
+
+	oldSymbol.Link = new
+	newSymbol.MergeContentsWith(oldSymbol)
+	return new
+}
+
+type mergeResult int
+
+const (
+	mergeForbidden = iota
+	mergeReplaceWithNew
+	mergeOverwriteWithNew
+	mergeKeepExisting
+	mergeBecomePrivateGetSetPair
+	mergeBecomePrivateStaticGetSetPair
+)
+
+func (p *parser) canMergeSymbols(scope *js_ast.Scope, existing ast.SymbolKind, new ast.SymbolKind) mergeResult {
+	if existing == ast.SymbolUnbound {
+		return mergeReplaceWithNew
+	}
+
+	// In TypeScript, imports are allowed to silently collide with symbols within
+	// the module. Presumably this is because the imports may be type-only:
+	//
+	//   import {Foo} from 'bar'
+	//   class Foo {}
+	//
+	if p.options.ts.Parse && existing == ast.SymbolImport {
+		return mergeReplaceWithNew
+	}
+
+	// "enum Foo {} enum Foo {}"
+	if new == ast.SymbolTSEnum && existing == ast.SymbolTSEnum {
+		return mergeKeepExisting
+	}
+
+	// "namespace Foo { ... } enum Foo {}"
+	if new == ast.SymbolTSEnum && existing == ast.SymbolTSNamespace {
+		return mergeReplaceWithNew
+	}
+
+	// "namespace Foo { ... } namespace Foo { ... }"
+	// "function Foo() {} namespace Foo { ... }"
+	// "enum Foo {} namespace Foo { ... }"
+	if new == ast.SymbolTSNamespace {
+		switch existing {
+		case ast.SymbolTSNamespace, ast.SymbolHoistedFunction, ast.SymbolGeneratorOrAsyncFunction, ast.SymbolTSEnum, ast.SymbolClass:
+			return mergeKeepExisting
+		}
+	}
+
+	// "var foo; var foo;"
+	// "var foo; function foo() {}"
+	// "function foo() {} var foo;"
+	// "function *foo() {} function *foo() {}" but not "{ function *foo() {} function *foo() {} }"
+	if new.IsHoistedOrFunction() && existing.IsHoistedOrFunction() &&
+		(scope.Kind == js_ast.ScopeEntry ||
+			scope.Kind == js_ast.ScopeFunctionBody ||
+			scope.Kind == js_ast.ScopeFunctionArgs ||
+			(new == existing && new.IsHoisted())) {
+		return mergeReplaceWithNew
+	}
+
+	// "get #foo() {} set #foo() {}"
+	// "set #foo() {} get #foo() {}"
+	if (existing == ast.SymbolPrivateGet && new == ast.SymbolPrivateSet) ||
+		(existing == ast.SymbolPrivateSet && new == ast.SymbolPrivateGet) {
+		return mergeBecomePrivateGetSetPair
+	}
+	if (existing == ast.SymbolPrivateStaticGet && new == ast.SymbolPrivateStaticSet) ||
+		(existing == ast.SymbolPrivateStaticSet && new == ast.SymbolPrivateStaticGet) {
+		return mergeBecomePrivateStaticGetSetPair
+	}
+
+	// "try {} catch (e) { var e }"
+	if existing == ast.SymbolCatchIdentifier && new == ast.SymbolHoisted {
+		return mergeReplaceWithNew
+	}
+
+	// "function() { var arguments }"
+	if existing == ast.SymbolArguments && new == ast.SymbolHoisted {
+		return mergeKeepExisting
+	}
+
+	// "function() { let arguments }"
+	if existing == ast.SymbolArguments && new != ast.SymbolHoisted {
+		return mergeOverwriteWithNew
+	}
+
+	return mergeForbidden
+}
+
+func (p *parser) addSymbolAlreadyDeclaredError(name string, newLoc logger.Loc, oldLoc logger.Loc) {
+	p.log.AddErrorWithNotes(&p.tracker,
+		js_lexer.RangeOfIdentifier(p.source, newLoc),
+		fmt.Sprintf("The symbol %q has already been declared", name),
+
+		[]logger.MsgData{p.tracker.MsgData(
+			js_lexer.RangeOfIdentifier(p.source, oldLoc),
+			fmt.Sprintf("The symbol %q was originally declared here:", name),
+		)},
+	)
+}
+
+func (p *parser) declareSymbol(kind ast.SymbolKind, loc logger.Loc, name string) ast.Ref {
+	p.checkForUnrepresentableIdentifier(loc, name)
+
+	// Allocate a new symbol
+	ref := p.newSymbol(kind, name)
+
+	// Check for a collision in the declaring scope
+	if existing, ok := p.currentScope.Members[name]; ok {
+		symbol := &p.symbols[existing.Ref.InnerIndex]
+
+		switch p.canMergeSymbols(p.currentScope, symbol.Kind, kind) {
+		case mergeForbidden:
+			p.addSymbolAlreadyDeclaredError(name, loc, existing.Loc)
+			return existing.Ref
+
+		case mergeKeepExisting:
+			ref = existing.Ref
+
+		case mergeReplaceWithNew:
+			symbol.Link = ref
+			p.currentScope.Replaced = append(p.currentScope.Replaced, existing)
+
+			// If these are both functions, remove the overwritten declaration
+			if p.options.minifySyntax && kind.IsFunction() && symbol.Kind.IsFunction() {
+				symbol.Flags |= ast.RemoveOverwrittenFunctionDeclaration
+			}
+
+		case mergeBecomePrivateGetSetPair:
+			ref = existing.Ref
+			symbol.Kind = ast.SymbolPrivateGetSetPair
+
+		case mergeBecomePrivateStaticGetSetPair:
+			ref = existing.Ref
+			symbol.Kind = ast.SymbolPrivateStaticGetSetPair
+
+		case mergeOverwriteWithNew:
+		}
+	}
+
+	// Overwrite this name in the declaring scope
+	p.currentScope.Members[name] = js_ast.ScopeMember{Ref: ref, Loc: loc}
+	return ref
+
+}
+
+// This type is just so we can use Go's native sort function
+type scopeMemberArray []js_ast.ScopeMember
+
+func (a scopeMemberArray) Len() int          { return len(a) }
+func (a scopeMemberArray) Swap(i int, j int) { a[i], a[j] = a[j], a[i] }
+
+func (a scopeMemberArray) Less(i int, j int) bool {
+	ai := a[i].Ref
+	bj := a[j].Ref
+	return ai.InnerIndex < bj.InnerIndex || (ai.InnerIndex == bj.InnerIndex && ai.SourceIndex < bj.SourceIndex)
+}
+
+func (p *parser) hoistSymbols(scope *js_ast.Scope) {
+	// Duplicate function declarations are forbidden in nested blocks in strict
+	// mode. Separately, they are also forbidden at the top-level of modules.
+	// This check needs to be delayed until now instead of being done when the
+	// functions are declared because we potentially need to scan the whole file
+	// to know if the file is considered to be in strict mode (or is considered
+	// to be a module). We might only encounter an "export {}" clause at the end
+	// of the file.
+	if (scope.StrictMode != js_ast.SloppyMode && scope.Kind == js_ast.ScopeBlock) || (scope.Parent == nil && p.isFileConsideredESM) {
+		for _, replaced := range scope.Replaced {
+			symbol := &p.symbols[replaced.Ref.InnerIndex]
+			if symbol.Kind.IsFunction() {
+				if member, ok := scope.Members[symbol.OriginalName]; ok && p.symbols[member.Ref.InnerIndex].Kind.IsFunction() {
+					var notes []logger.MsgData
+					if scope.Parent == nil && p.isFileConsideredESM {
+						_, notes = p.whyESModule()
+						notes[0].Text = fmt.Sprintf("Duplicate top-level function declarations are not allowed in an ECMAScript module. %s", notes[0].Text)
+					} else {
+						var where string
+						where, notes = p.whyStrictMode(scope)
+						notes[0].Text = fmt.Sprintf("Duplicate function declarations are not allowed in nested blocks %s. %s", where, notes[0].Text)
+					}
+
+					p.log.AddErrorWithNotes(&p.tracker,
+						js_lexer.RangeOfIdentifier(p.source, member.Loc),
+						fmt.Sprintf("The symbol %q has already been declared", symbol.OriginalName),
+
+						append([]logger.MsgData{p.tracker.MsgData(
+							js_lexer.RangeOfIdentifier(p.source, replaced.Loc),
+							fmt.Sprintf("The symbol %q was originally declared here:", symbol.OriginalName),
+						)}, notes...),
+					)
+				}
+			}
+		}
+	}
+
+	if !scope.Kind.StopsHoisting() {
+		// We create new symbols in the loop below, so the iteration order of the
+		// loop must be deterministic to avoid generating different minified names
+		sortedMembers := make(scopeMemberArray, 0, len(scope.Members))
+		for _, member := range scope.Members {
+			sortedMembers = append(sortedMembers, member)
+		}
+		sort.Sort(sortedMembers)
+
+	nextMember:
+		for _, member := range sortedMembers {
+			symbol := &p.symbols[member.Ref.InnerIndex]
+
+			// Handle non-hoisted collisions between catch bindings and the catch body.
+			// This implements "B.3.4 VariableStatements in Catch Blocks" from Annex B
+			// of the ECMAScript standard version 6+ (except for the hoisted case, which
+			// is handled later on below):
+			//
+			// * It is a Syntax Error if any element of the BoundNames of CatchParameter
+			//   also occurs in the LexicallyDeclaredNames of Block.
+			//
+			// * It is a Syntax Error if any element of the BoundNames of CatchParameter
+			//   also occurs in the VarDeclaredNames of Block unless CatchParameter is
+			//   CatchParameter : BindingIdentifier .
+			//
+			if scope.Parent.Kind == js_ast.ScopeCatchBinding && symbol.Kind != ast.SymbolHoisted {
+				if existingMember, ok := scope.Parent.Members[symbol.OriginalName]; ok {
+					p.addSymbolAlreadyDeclaredError(symbol.OriginalName, member.Loc, existingMember.Loc)
+					continue
+				}
+			}
+
+			if !symbol.Kind.IsHoisted() {
+				continue
+			}
+
+			// Implement "Block-Level Function Declarations Web Legacy Compatibility
+			// Semantics" from Annex B of the ECMAScript standard version 6+
+			isSloppyModeBlockLevelFnStmt := false
+			originalMemberRef := member.Ref
+			if symbol.Kind == ast.SymbolHoistedFunction {
+				// Block-level function declarations behave like "let" in strict mode
+				if scope.StrictMode != js_ast.SloppyMode {
+					continue
+				}
+
+				// In sloppy mode, block level functions behave like "let" except with
+				// an assignment to "var", sort of. This code:
+				//
+				//   if (x) {
+				//     f();
+				//     function f() {}
+				//   }
+				//   f();
+				//
+				// behaves like this code:
+				//
+				//   if (x) {
+				//     let f2 = function() {}
+				//     var f = f2;
+				//     f2();
+				//   }
+				//   f();
+				//
+				hoistedRef := p.newSymbol(ast.SymbolHoisted, symbol.OriginalName)
+				scope.Generated = append(scope.Generated, hoistedRef)
+				if p.hoistedRefForSloppyModeBlockFn == nil {
+					p.hoistedRefForSloppyModeBlockFn = make(map[ast.Ref]ast.Ref)
+				}
+				p.hoistedRefForSloppyModeBlockFn[member.Ref] = hoistedRef
+				symbol = &p.symbols[hoistedRef.InnerIndex]
+				member.Ref = hoistedRef
+				isSloppyModeBlockLevelFnStmt = true
+			}
+
+			// Check for collisions that would prevent to hoisting "var" symbols up to the enclosing function scope
+			s := scope.Parent
+			for {
+				// Variable declarations hoisted past a "with" statement may actually end
+				// up overwriting a property on the target of the "with" statement instead
+				// of initializing the variable. We must not rename them or we risk
+				// causing a behavior change.
+				//
+				//   var obj = { foo: 1 }
+				//   with (obj) { var foo = 2 }
+				//   assert(foo === undefined)
+				//   assert(obj.foo === 2)
+				//
+				if s.Kind == js_ast.ScopeWith {
+					symbol.Flags |= ast.MustNotBeRenamed
+				}
+
+				if existingMember, ok := s.Members[symbol.OriginalName]; ok {
+					existingSymbol := &p.symbols[existingMember.Ref.InnerIndex]
+
+					// We can hoist the symbol from the child scope into the symbol in
+					// this scope if:
+					//
+					//   - The symbol is unbound (i.e. a global variable access)
+					//   - The symbol is also another hoisted variable
+					//   - The symbol is a function of any kind and we're in a function or module scope
+					//
+					// Is this unbound (i.e. a global access) or also hoisted?
+					if existingSymbol.Kind == ast.SymbolUnbound || existingSymbol.Kind == ast.SymbolHoisted ||
+						(existingSymbol.Kind.IsFunction() && (s.Kind == js_ast.ScopeEntry || s.Kind == js_ast.ScopeFunctionBody)) {
+						// Silently merge this symbol into the existing symbol
+						symbol.Link = existingMember.Ref
+						s.Members[symbol.OriginalName] = existingMember
+						continue nextMember
+					}
+
+					// Otherwise if this isn't a catch identifier or "arguments", it's a collision
+					if existingSymbol.Kind != ast.SymbolCatchIdentifier && existingSymbol.Kind != ast.SymbolArguments {
+						// An identifier binding from a catch statement and a function
+						// declaration can both silently shadow another hoisted symbol
+						if symbol.Kind != ast.SymbolCatchIdentifier && symbol.Kind != ast.SymbolHoistedFunction {
+							if !isSloppyModeBlockLevelFnStmt {
+								p.addSymbolAlreadyDeclaredError(symbol.OriginalName, member.Loc, existingMember.Loc)
+							} else if s == scope.Parent {
+								// Never mind about this, turns out it's not needed after all
+								delete(p.hoistedRefForSloppyModeBlockFn, originalMemberRef)
+							}
+						}
+						continue nextMember
+					}
+
+					// If this is a catch identifier, silently merge the existing symbol
+					// into this symbol but continue hoisting past this catch scope
+					existingSymbol.Link = member.Ref
+					s.Members[symbol.OriginalName] = member
+				}
+
+				if s.Kind.StopsHoisting() {
+					// Declare the member in the scope that stopped the hoisting
+					s.Members[symbol.OriginalName] = member
+					break
+				}
+				s = s.Parent
+			}
+		}
+	}
+
+	for _, child := range scope.Children {
+		p.hoistSymbols(child)
+	}
+}
+
+func (p *parser) declareBinding(kind ast.SymbolKind, binding js_ast.Binding, opts parseStmtOpts) {
+	js_ast.ForEachIdentifierBinding(binding, func(loc logger.Loc, b *js_ast.BIdentifier) {
+		if !opts.isTypeScriptDeclare || (opts.isNamespaceScope && opts.isExport) {
+			b.Ref = p.declareSymbol(kind, loc, p.loadNameFromRef(b.Ref))
+		}
+	})
+}
+
+func (p *parser) recordUsage(ref ast.Ref) {
+	// The use count stored in the symbol is used for generating symbol names
+	// during minification. These counts shouldn't include references inside dead
+	// code regions since those will be culled.
+	if !p.isControlFlowDead {
+		p.symbols[ref.InnerIndex].UseCountEstimate++
+		use := p.symbolUses[ref]
+		use.CountEstimate++
+		p.symbolUses[ref] = use
+	}
+
+	// The correctness of TypeScript-to-JavaScript conversion relies on accurate
+	// symbol use counts for the whole file, including dead code regions. This is
+	// tracked separately in a parser-only data structure.
+	if p.options.ts.Parse {
+		p.tsUseCounts[ref.InnerIndex]++
+	}
+}
+
+func (p *parser) ignoreUsage(ref ast.Ref) {
+	// Roll back the use count increment in recordUsage()
+	if !p.isControlFlowDead {
+		p.symbols[ref.InnerIndex].UseCountEstimate--
+		use := p.symbolUses[ref]
+		use.CountEstimate--
+		if use.CountEstimate == 0 {
+			delete(p.symbolUses, ref)
+		} else {
+			p.symbolUses[ref] = use
+		}
+	}
+
+	// Don't roll back the "tsUseCounts" increment. This must be counted even if
+	// the value is ignored because that's what the TypeScript compiler does.
+}
+
+func (p *parser) ignoreUsageOfIdentifierInDotChain(expr js_ast.Expr) {
+	for {
+		switch e := expr.Data.(type) {
+		case *js_ast.EIdentifier:
+			p.ignoreUsage(e.Ref)
+
+		case *js_ast.EDot:
+			expr = e.Target
+			continue
+
+		case *js_ast.EIndex:
+			if _, ok := e.Index.Data.(*js_ast.EString); ok {
+				expr = e.Target
+				continue
+			}
+		}
+
+		return
+	}
+}
+
+func (p *parser) importFromRuntime(loc logger.Loc, name string) js_ast.Expr {
+	it, ok := p.runtimeImports[name]
+	if !ok {
+		it.Loc = loc
+		it.Ref = p.newSymbol(ast.SymbolOther, name)
+		p.moduleScope.Generated = append(p.moduleScope.Generated, it.Ref)
+		p.runtimeImports[name] = it
+	}
+	p.recordUsage(it.Ref)
+	return js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: it.Ref}}
+}
+
+func (p *parser) callRuntime(loc logger.Loc, name string, args []js_ast.Expr) js_ast.Expr {
+	return js_ast.Expr{Loc: loc, Data: &js_ast.ECall{
+		Target: p.importFromRuntime(loc, name),
+		Args:   args,
+	}}
+}
+
+type JSXImport uint8
+
+const (
+	JSXImportJSX JSXImport = iota
+	JSXImportJSXS
+	JSXImportFragment
+	JSXImportCreateElement
+)
+
+func (p *parser) importJSXSymbol(loc logger.Loc, jsx JSXImport) js_ast.Expr {
+	var symbols map[string]ast.LocRef
+	var name string
+
+	switch jsx {
+	case JSXImportJSX:
+		symbols = p.jsxRuntimeImports
+		if p.options.jsx.Development {
+			name = "jsxDEV"
+		} else {
+			name = "jsx"
+		}
+
+	case JSXImportJSXS:
+		symbols = p.jsxRuntimeImports
+		if p.options.jsx.Development {
+			name = "jsxDEV"
+		} else {
+			name = "jsxs"
+		}
+
+	case JSXImportFragment:
+		symbols = p.jsxRuntimeImports
+		name = "Fragment"
+
+	case JSXImportCreateElement:
+		symbols = p.jsxLegacyImports
+		name = "createElement"
+	}
+
+	it, ok := symbols[name]
+	if !ok {
+		it.Loc = loc
+		it.Ref = p.newSymbol(ast.SymbolOther, name)
+		p.moduleScope.Generated = append(p.moduleScope.Generated, it.Ref)
+		p.isImportItem[it.Ref] = true
+		symbols[name] = it
+	}
+
+	p.recordUsage(it.Ref)
+	return p.handleIdentifier(loc, &js_ast.EIdentifier{Ref: it.Ref}, identifierOpts{
+		wasOriginallyIdentifier: true,
+	})
+}
+
+func (p *parser) valueToSubstituteForRequire(loc logger.Loc) js_ast.Expr {
+	if p.source.Index != runtime.SourceIndex &&
+		config.ShouldCallRuntimeRequire(p.options.mode, p.options.outputFormat) {
+		return p.importFromRuntime(loc, "__require")
+	}
+
+	p.recordUsage(p.requireRef)
+	return js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: p.requireRef}}
+}
+
+func (p *parser) makePromiseRef() ast.Ref {
+	if p.promiseRef == ast.InvalidRef {
+		p.promiseRef = p.newSymbol(ast.SymbolUnbound, "Promise")
+	}
+	return p.promiseRef
+}
+
+func (p *parser) makeRegExpRef() ast.Ref {
+	if p.regExpRef == ast.InvalidRef {
+		p.regExpRef = p.newSymbol(ast.SymbolUnbound, "RegExp")
+		p.moduleScope.Generated = append(p.moduleScope.Generated, p.regExpRef)
+	}
+	return p.regExpRef
+}
+
+// The name is temporarily stored in the ref until the scope traversal pass
+// happens, at which point a symbol will be generated and the ref will point
+// to the symbol instead.
+//
+// The scope traversal pass will reconstruct the name using one of two methods.
+// In the common case, the name is a slice of the file itself. In that case we
+// can just store the slice and not need to allocate any extra memory. In the
+// rare case, the name is an externally-allocated string. In that case we store
+// an index to the string and use that index during the scope traversal pass.
+func (p *parser) storeNameInRef(name js_lexer.MaybeSubstring) ast.Ref {
+	// Is the data in "name" a subset of the data in "p.source.Contents"?
+	if name.Start.IsValid() {
+		// The name is a slice of the file contents, so we can just reference it by
+		// length and don't have to allocate anything. This is the common case.
+		//
+		// It's stored as a negative value so we'll crash if we try to use it. That
+		// way we'll catch cases where we've forgotten to call loadNameFromRef().
+		// The length is the negative part because we know it's non-zero.
+		return ast.Ref{SourceIndex: -uint32(len(name.String)), InnerIndex: uint32(name.Start.GetIndex())}
+	} else {
+		// The name is some memory allocated elsewhere. This is either an inline
+		// string constant in the parser or an identifier with escape sequences
+		// in the source code, which is very unusual. Stash it away for later.
+		// This uses allocations but it should hopefully be very uncommon.
+		ref := ast.Ref{SourceIndex: 0x80000000, InnerIndex: uint32(len(p.allocatedNames))}
+		p.allocatedNames = append(p.allocatedNames, name.String)
+		return ref
+	}
+}
+
+// This is the inverse of storeNameInRef() above
+func (p *parser) loadNameFromRef(ref ast.Ref) string {
+	if ref.SourceIndex == 0x80000000 {
+		return p.allocatedNames[ref.InnerIndex]
+	} else {
+		if (ref.SourceIndex & 0x80000000) == 0 {
+			panic("Internal error: invalid symbol reference")
+		}
+		return p.source.Contents[ref.InnerIndex : int32(ref.InnerIndex)-int32(ref.SourceIndex)]
+	}
+}
+
+// Due to ES6 destructuring patterns, there are many cases where it's
+// impossible to distinguish between an array or object literal and a
+// destructuring assignment until we hit the "=" operator later on.
+// This object defers errors about being in one state or the other
+// until we discover which state we're in.
+type deferredErrors struct {
+	// These are errors for expressions
+	invalidExprDefaultValue  logger.Range
+	invalidExprAfterQuestion logger.Range
+	arraySpreadFeature       logger.Range
+
+	// These errors are for arrow functions
+	invalidParens []logger.Range
+}
+
+func (from *deferredErrors) mergeInto(to *deferredErrors) {
+	if from.invalidExprDefaultValue.Len > 0 {
+		to.invalidExprDefaultValue = from.invalidExprDefaultValue
+	}
+	if from.invalidExprAfterQuestion.Len > 0 {
+		to.invalidExprAfterQuestion = from.invalidExprAfterQuestion
+	}
+	if from.arraySpreadFeature.Len > 0 {
+		to.arraySpreadFeature = from.arraySpreadFeature
+	}
+	if len(from.invalidParens) > 0 {
+		if len(to.invalidParens) > 0 {
+			to.invalidParens = append(to.invalidParens, from.invalidParens...)
+		} else {
+			to.invalidParens = from.invalidParens
+		}
+	}
+}
+
+func (p *parser) logExprErrors(errors *deferredErrors) {
+	if errors.invalidExprDefaultValue.Len > 0 {
+		p.log.AddError(&p.tracker, errors.invalidExprDefaultValue, "Unexpected \"=\"")
+	}
+
+	if errors.invalidExprAfterQuestion.Len > 0 {
+		r := errors.invalidExprAfterQuestion
+		p.log.AddError(&p.tracker, r, fmt.Sprintf("Unexpected %q", p.source.Contents[r.Loc.Start:r.Loc.Start+r.Len]))
+	}
+
+	if errors.arraySpreadFeature.Len > 0 {
+		p.markSyntaxFeature(compat.ArraySpread, errors.arraySpreadFeature)
+	}
+}
+
+func (p *parser) logDeferredArrowArgErrors(errors *deferredErrors) {
+	for _, paren := range errors.invalidParens {
+		p.log.AddError(&p.tracker, paren, "Invalid binding pattern")
+	}
+}
+
+func (p *parser) logNullishCoalescingErrorPrecedenceError(op string) {
+	prevOp := "??"
+	if p.lexer.Token == js_lexer.TQuestionQuestion {
+		op, prevOp = prevOp, op
+	}
+	// p.log.AddError(&p.tracker, p.lexer.Range(), fmt.Sprintf("The %q operator requires parentheses"))
+	p.log.AddErrorWithNotes(&p.tracker, p.lexer.Range(), fmt.Sprintf("Cannot use %q with %q without parentheses", op, prevOp),
+		[]logger.MsgData{{Text: fmt.Sprintf("Expressions of the form \"x %s y %s z\" are not allowed in JavaScript. "+
+			"You must disambiguate between \"(x %s y) %s z\" and \"x %s (y %s z)\" by adding parentheses.", prevOp, op, prevOp, op, prevOp, op)}})
+}
+
+func defineValueCanBeUsedInAssignTarget(data js_ast.E) bool {
+	switch data.(type) {
+	case *js_ast.EIdentifier, *js_ast.EDot:
+		return true
+	}
+
+	// Substituting a constant into an assignment target (e.g. "x = 1" becomes
+	// "0 = 1") will cause a syntax error, so we avoid doing this. The caller
+	// will log a warning instead.
+	return false
+}
+
+func (p *parser) logAssignToDefine(r logger.Range, name string, expr js_ast.Expr) {
+	// If this is a compound expression, pretty-print it for the error message.
+	// We don't use a literal slice of the source text in case it contains
+	// problematic things (e.g. spans multiple lines, has embedded comments).
+	if expr.Data != nil {
+		var parts []string
+		for {
+			if id, ok := expr.Data.(*js_ast.EIdentifier); ok {
+				parts = append(parts, p.loadNameFromRef(id.Ref))
+				break
+			} else if dot, ok := expr.Data.(*js_ast.EDot); ok {
+				parts = append(parts, dot.Name)
+				parts = append(parts, ".")
+				expr = dot.Target
+			} else if index, ok := expr.Data.(*js_ast.EIndex); ok {
+				if str, ok := index.Index.Data.(*js_ast.EString); ok {
+					parts = append(parts, "]")
+					parts = append(parts, string(helpers.QuoteSingle(helpers.UTF16ToString(str.Value), false)))
+					parts = append(parts, "[")
+					expr = index.Target
+				} else {
+					return
+				}
+			} else {
+				return
+			}
+		}
+		for i, j := 0, len(parts)-1; i < j; i, j = i+1, j-1 {
+			parts[i], parts[j] = parts[j], parts[i]
+		}
+		name = strings.Join(parts, "")
+	}
+
+	kind := logger.Warning
+	if p.suppressWarningsAboutWeirdCode {
+		kind = logger.Debug
+	}
+
+	p.log.AddIDWithNotes(logger.MsgID_JS_AssignToDefine, kind, &p.tracker, r,
+		fmt.Sprintf("Suspicious assignment to defined constant %q", name),
+		[]logger.MsgData{{Text: fmt.Sprintf(
+			"The expression %q has been configured to be replaced with a constant using the \"define\" feature. "+
+				"If this expression is supposed to be a compile-time constant, then it doesn't make sense to assign to it here. "+
+				"Or if this expression is supposed to change at run-time, this \"define\" substitution should be removed.", name)}})
+}
+
+// The "await" and "yield" expressions are never allowed in argument lists but
+// may or may not be allowed otherwise depending on the details of the enclosing
+// function or module. This needs to be handled when parsing an arrow function
+// argument list because we don't know if these expressions are not allowed until
+// we reach the "=>" token (or discover the absence of one).
+//
+// Specifically, for await:
+//
+//	// This is ok
+//	async function foo() { (x = await y) }
+//
+//	// This is an error
+//	async function foo() { (x = await y) => {} }
+//
+// And for yield:
+//
+//	// This is ok
+//	function* foo() { (x = yield y) }
+//
+//	// This is an error
+//	function* foo() { (x = yield y) => {} }
+type deferredArrowArgErrors struct {
+	invalidExprAwait logger.Range
+	invalidExprYield logger.Range
+}
+
+func (p *parser) logArrowArgErrors(errors *deferredArrowArgErrors) {
+	if errors.invalidExprAwait.Len > 0 {
+		p.log.AddError(&p.tracker, errors.invalidExprAwait, "Cannot use an \"await\" expression here:")
+	}
+
+	if errors.invalidExprYield.Len > 0 {
+		p.log.AddError(&p.tracker, errors.invalidExprYield, "Cannot use a \"yield\" expression here:")
+	}
+}
+
+func (p *parser) keyNameForError(key js_ast.Expr) string {
+	switch k := key.Data.(type) {
+	case *js_ast.EString:
+		return fmt.Sprintf("%q", helpers.UTF16ToString(k.Value))
+	case *js_ast.EPrivateIdentifier:
+		return fmt.Sprintf("%q", p.loadNameFromRef(k.Ref))
+	}
+	return "property"
+}
+
+func (p *parser) checkForLegacyOctalLiteral(e js_ast.E) {
+	if p.lexer.IsLegacyOctalLiteral {
+		if p.legacyOctalLiterals == nil {
+			p.legacyOctalLiterals = make(map[js_ast.E]logger.Range)
+		}
+		p.legacyOctalLiterals[e] = p.lexer.Range()
+	}
+}
+
+func (p *parser) notesForAssertTypeJSON(record *ast.ImportRecord, alias string) []logger.MsgData {
+	return []logger.MsgData{p.tracker.MsgData(
+		js_lexer.RangeOfImportAssertOrWith(p.source, *ast.FindAssertOrWithEntry(record.AssertOrWith.Entries, "type"), js_lexer.KeyAndValueRange),
+		"The JSON import assertion is here:"),
+		{Text: fmt.Sprintf("You can either keep the import assertion and only use the \"default\" import, "+
+			"or you can remove the import assertion and use the %q import.", alias)}}
+}
+
+// This assumes the caller has already checked for TStringLiteral or TNoSubstitutionTemplateLiteral
+func (p *parser) parseStringLiteral() js_ast.Expr {
+	var legacyOctalLoc logger.Loc
+	loc := p.lexer.Loc()
+	text := p.lexer.StringLiteral()
+
+	// Enable using a "/* @__KEY__ */" comment to turn a string into a key
+	hasPropertyKeyComment := (p.lexer.HasCommentBefore & js_lexer.KeyCommentBefore) != 0
+	if hasPropertyKeyComment {
+		if name := helpers.UTF16ToString(text); p.isMangledProp(name) {
+			value := js_ast.Expr{Loc: loc, Data: &js_ast.ENameOfSymbol{
+				Ref:                   p.storeNameInRef(js_lexer.MaybeSubstring{String: name}),
+				HasPropertyKeyComment: true,
+			}}
+			p.lexer.Next()
+			return value
+		}
+	}
+
+	if p.lexer.LegacyOctalLoc.Start > loc.Start {
+		legacyOctalLoc = p.lexer.LegacyOctalLoc
+	}
+	value := js_ast.Expr{Loc: loc, Data: &js_ast.EString{
+		Value:                 text,
+		LegacyOctalLoc:        legacyOctalLoc,
+		PreferTemplate:        p.lexer.Token == js_lexer.TNoSubstitutionTemplateLiteral,
+		HasPropertyKeyComment: hasPropertyKeyComment,
+	}}
+	p.lexer.Next()
+	return value
+}
+
+type propertyOpts struct {
+	decorators       []js_ast.Decorator
+	decoratorScope   *js_ast.Scope
+	decoratorContext decoratorContextFlags
+
+	asyncRange     logger.Range
+	generatorRange logger.Range
+	tsDeclareRange logger.Range
+	classKeyword   logger.Range
+	isAsync        bool
+	isGenerator    bool
+
+	// Class-related options
+	isStatic        bool
+	isTSAbstract    bool
+	isClass         bool
+	classHasExtends bool
+}
+
+func (p *parser) parseProperty(startLoc logger.Loc, kind js_ast.PropertyKind, opts propertyOpts, errors *deferredErrors) (js_ast.Property, bool) {
+	var flags js_ast.PropertyFlags
+	var key js_ast.Expr
+	var closeBracketLoc logger.Loc
+	keyRange := p.lexer.Range()
+
+	switch p.lexer.Token {
+	case js_lexer.TNumericLiteral:
+		key = js_ast.Expr{Loc: p.lexer.Loc(), Data: &js_ast.ENumber{Value: p.lexer.Number}}
+		p.checkForLegacyOctalLiteral(key.Data)
+		p.lexer.Next()
+
+	case js_lexer.TStringLiteral:
+		key = p.parseStringLiteral()
+		if !p.options.minifySyntax {
+			flags |= js_ast.PropertyPreferQuotedKey
+		}
+
+	case js_lexer.TBigIntegerLiteral:
+		key = js_ast.Expr{Loc: p.lexer.Loc(), Data: &js_ast.EBigInt{Value: p.lexer.Identifier.String}}
+		p.markSyntaxFeature(compat.Bigint, p.lexer.Range())
+		p.lexer.Next()
+
+	case js_lexer.TPrivateIdentifier:
+		if p.options.ts.Parse && p.options.ts.Config.ExperimentalDecorators == config.True && len(opts.decorators) > 0 {
+			p.log.AddError(&p.tracker, p.lexer.Range(), "TypeScript experimental decorators cannot be used on private identifiers")
+		} else if !opts.isClass {
+			p.lexer.Expected(js_lexer.TIdentifier)
+		} else if opts.tsDeclareRange.Len != 0 {
+			p.log.AddError(&p.tracker, opts.tsDeclareRange, "\"declare\" cannot be used with a private identifier")
+		}
+		name := p.lexer.Identifier
+		key = js_ast.Expr{Loc: p.lexer.Loc(), Data: &js_ast.EPrivateIdentifier{Ref: p.storeNameInRef(name)}}
+		p.reportPrivateNameUsage(name.String)
+		p.lexer.Next()
+
+	case js_lexer.TOpenBracket:
+		flags |= js_ast.PropertyIsComputed
+		p.markSyntaxFeature(compat.ObjectExtensions, p.lexer.Range())
+		p.lexer.Next()
+		wasIdentifier := p.lexer.Token == js_lexer.TIdentifier
+		expr := p.parseExpr(js_ast.LComma)
+
+		// Handle index signatures
+		if p.options.ts.Parse && p.lexer.Token == js_lexer.TColon && wasIdentifier && opts.isClass {
+			if _, ok := expr.Data.(*js_ast.EIdentifier); ok {
+				if opts.tsDeclareRange.Len != 0 {
+					p.log.AddError(&p.tracker, opts.tsDeclareRange, "\"declare\" cannot be used with an index signature")
+				}
+
+				// "[key: string]: any;"
+				p.lexer.Next()
+				p.skipTypeScriptType(js_ast.LLowest)
+				p.lexer.Expect(js_lexer.TCloseBracket)
+				p.lexer.Expect(js_lexer.TColon)
+				p.skipTypeScriptType(js_ast.LLowest)
+				p.lexer.ExpectOrInsertSemicolon()
+
+				// Skip this property entirely
+				return js_ast.Property{}, false
+			}
+		}
+
+		closeBracketLoc = p.saveExprCommentsHere()
+		p.lexer.Expect(js_lexer.TCloseBracket)
+		key = expr
+
+	case js_lexer.TAsterisk:
+		if kind != js_ast.PropertyField && (kind != js_ast.PropertyMethod || opts.isGenerator) {
+			p.lexer.Unexpected()
+		}
+		opts.isGenerator = true
+		opts.generatorRange = p.lexer.Range()
+		p.lexer.Next()
+		return p.parseProperty(startLoc, js_ast.PropertyMethod, opts, errors)
+
+	default:
+		name := p.lexer.Identifier
+		raw := p.lexer.Raw()
+		nameRange := p.lexer.Range()
+		if !p.lexer.IsIdentifierOrKeyword() {
+			p.lexer.Expect(js_lexer.TIdentifier)
+		}
+		p.lexer.Next()
+
+		// Support contextual keywords
+		if kind == js_ast.PropertyField {
+			// Does the following token look like a key?
+			couldBeModifierKeyword := p.lexer.IsIdentifierOrKeyword()
+			if !couldBeModifierKeyword {
+				switch p.lexer.Token {
+				case js_lexer.TOpenBracket, js_lexer.TNumericLiteral, js_lexer.TStringLiteral,
+					js_lexer.TAsterisk, js_lexer.TPrivateIdentifier:
+					couldBeModifierKeyword = true
+				}
+			}
+
+			// If so, check for a modifier keyword
+			if couldBeModifierKeyword {
+				switch name.String {
+				case "get":
+					if !opts.isAsync && raw == name.String {
+						p.markSyntaxFeature(compat.ObjectAccessors, nameRange)
+						return p.parseProperty(startLoc, js_ast.PropertyGetter, opts, nil)
+					}
+
+				case "set":
+					if !opts.isAsync && raw == name.String {
+						p.markSyntaxFeature(compat.ObjectAccessors, nameRange)
+						return p.parseProperty(startLoc, js_ast.PropertySetter, opts, nil)
+					}
+
+				case "accessor":
+					if !p.lexer.HasNewlineBefore && !opts.isAsync && opts.isClass && raw == name.String {
+						return p.parseProperty(startLoc, js_ast.PropertyAutoAccessor, opts, nil)
+					}
+
+				case "async":
+					if !p.lexer.HasNewlineBefore && !opts.isAsync && raw == name.String {
+						opts.isAsync = true
+						opts.asyncRange = nameRange
+						return p.parseProperty(startLoc, js_ast.PropertyMethod, opts, nil)
+					}
+
+				case "static":
+					if !opts.isStatic && !opts.isAsync && opts.isClass && raw == name.String {
+						opts.isStatic = true
+						return p.parseProperty(startLoc, kind, opts, nil)
+					}
+
+				case "declare":
+					if !p.lexer.HasNewlineBefore && opts.isClass && p.options.ts.Parse && opts.tsDeclareRange.Len == 0 && raw == name.String {
+						opts.tsDeclareRange = nameRange
+						scopeIndex := len(p.scopesInOrder)
+
+						if prop, ok := p.parseProperty(startLoc, kind, opts, nil); ok &&
+							prop.Kind == js_ast.PropertyField && prop.ValueOrNil.Data == nil &&
+							(p.options.ts.Config.ExperimentalDecorators == config.True && len(opts.decorators) > 0) {
+							// If this is a well-formed class field with the "declare" keyword,
+							// only keep the declaration to preserve its side-effects when
+							// there are TypeScript experimental decorators present:
+							//
+							//   class Foo {
+							//     // Remove this
+							//     declare [(console.log('side effect 1'), 'foo')]
+							//
+							//     // Keep this
+							//     @decorator(console.log('side effect 2')) declare bar
+							//   }
+							//
+							// This behavior is surprisingly somehow valid with TypeScript
+							// experimental decorators, which was possibly by accident.
+							// TypeScript does not allow this with JavaScript decorators.
+							//
+							// References:
+							//
+							//   https://github.com/evanw/esbuild/issues/1675
+							//   https://github.com/microsoft/TypeScript/issues/46345
+							//
+							prop.Kind = js_ast.PropertyDeclareOrAbstract
+							return prop, true
+						}
+
+						p.discardScopesUpTo(scopeIndex)
+						return js_ast.Property{}, false
+					}
+
+				case "abstract":
+					if !p.lexer.HasNewlineBefore && opts.isClass && p.options.ts.Parse && !opts.isTSAbstract && raw == name.String {
+						opts.isTSAbstract = true
+						scopeIndex := len(p.scopesInOrder)
+
+						if prop, ok := p.parseProperty(startLoc, kind, opts, nil); ok &&
+							prop.Kind == js_ast.PropertyField && prop.ValueOrNil.Data == nil &&
+							(p.options.ts.Config.ExperimentalDecorators == config.True && len(opts.decorators) > 0) {
+							// If this is a well-formed class field with the "abstract" keyword,
+							// only keep the declaration to preserve its side-effects when
+							// there are TypeScript experimental decorators present:
+							//
+							//   abstract class Foo {
+							//     // Remove this
+							//     abstract [(console.log('side effect 1'), 'foo')]
+							//
+							//     // Keep this
+							//     @decorator(console.log('side effect 2')) abstract bar
+							//   }
+							//
+							// This behavior is valid with TypeScript experimental decorators.
+							// TypeScript does not allow this with JavaScript decorators.
+							//
+							// References:
+							//
+							//   https://github.com/evanw/esbuild/issues/3684
+							//
+							prop.Kind = js_ast.PropertyDeclareOrAbstract
+							return prop, true
+						}
+
+						p.discardScopesUpTo(scopeIndex)
+						return js_ast.Property{}, false
+					}
+
+				case "private", "protected", "public", "readonly", "override":
+					// Skip over TypeScript keywords
+					if opts.isClass && p.options.ts.Parse && raw == name.String {
+						return p.parseProperty(startLoc, kind, opts, nil)
+					}
+				}
+			} else if p.lexer.Token == js_lexer.TOpenBrace && name.String == "static" && len(opts.decorators) == 0 {
+				loc := p.lexer.Loc()
+				p.lexer.Next()
+
+				oldFnOrArrowDataParse := p.fnOrArrowDataParse
+				p.fnOrArrowDataParse = fnOrArrowDataParse{
+					isReturnDisallowed: true,
+					allowSuperProperty: true,
+					await:              forbidAll,
+				}
+
+				p.pushScopeForParsePass(js_ast.ScopeClassStaticInit, loc)
+				stmts := p.parseStmtsUpTo(js_lexer.TCloseBrace, parseStmtOpts{})
+				p.popScope()
+
+				p.fnOrArrowDataParse = oldFnOrArrowDataParse
+
+				closeBraceLoc := p.lexer.Loc()
+				p.lexer.Expect(js_lexer.TCloseBrace)
+				return js_ast.Property{
+					Kind: js_ast.PropertyClassStaticBlock,
+					Loc:  startLoc,
+					ClassStaticBlock: &js_ast.ClassStaticBlock{
+						Loc:   loc,
+						Block: js_ast.SBlock{Stmts: stmts, CloseBraceLoc: closeBraceLoc},
+					},
+				}, true
+			}
+		}
+
+		if p.isMangledProp(name.String) {
+			key = js_ast.Expr{Loc: nameRange.Loc, Data: &js_ast.ENameOfSymbol{Ref: p.storeNameInRef(name)}}
+		} else {
+			key = js_ast.Expr{Loc: nameRange.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16(name.String)}}
+		}
+
+		// Parse a shorthand property
+		if !opts.isClass && kind == js_ast.PropertyField && p.lexer.Token != js_lexer.TColon &&
+			p.lexer.Token != js_lexer.TOpenParen && p.lexer.Token != js_lexer.TLessThan &&
+			js_lexer.Keywords[name.String] == js_lexer.T(0) {
+
+			// Forbid invalid identifiers
+			if (p.fnOrArrowDataParse.await != allowIdent && name.String == "await") ||
+				(p.fnOrArrowDataParse.yield != allowIdent && name.String == "yield") {
+				p.log.AddError(&p.tracker, nameRange, fmt.Sprintf("Cannot use %q as an identifier here:", name.String))
+			}
+
+			ref := p.storeNameInRef(name)
+			value := js_ast.Expr{Loc: key.Loc, Data: &js_ast.EIdentifier{Ref: ref}}
+
+			// Destructuring patterns have an optional default value
+			var initializerOrNil js_ast.Expr
+			if errors != nil && p.lexer.Token == js_lexer.TEquals {
+				errors.invalidExprDefaultValue = p.lexer.Range()
+				p.lexer.Next()
+				initializerOrNil = p.parseExpr(js_ast.LComma)
+			}
+
+			return js_ast.Property{
+				Kind:             kind,
+				Loc:              startLoc,
+				Key:              key,
+				ValueOrNil:       value,
+				InitializerOrNil: initializerOrNil,
+				Flags:            js_ast.PropertyWasShorthand,
+			}, true
+		}
+	}
+
+	hasTypeParameters := false
+	hasDefiniteAssignmentAssertionOperator := false
+
+	if p.options.ts.Parse {
+		if opts.isClass {
+			if p.lexer.Token == js_lexer.TQuestion {
+				// "class X { foo?: number }"
+				// "class X { foo?(): number }"
+				p.lexer.Next()
+			} else if p.lexer.Token == js_lexer.TExclamation && !p.lexer.HasNewlineBefore &&
+				(kind == js_ast.PropertyField || kind == js_ast.PropertyAutoAccessor) {
+				// "class X { foo!: number }"
+				p.lexer.Next()
+				hasDefiniteAssignmentAssertionOperator = true
+			}
+		}
+
+		// "class X { foo?<T>(): T }"
+		// "const x = { foo<T>(): T {} }"
+		if !hasDefiniteAssignmentAssertionOperator && kind != js_ast.PropertyAutoAccessor {
+			hasTypeParameters = p.skipTypeScriptTypeParameters(allowConstModifier) != didNotSkipAnything
+		}
+	}
+
+	// Parse a class field with an optional initial value
+	if kind == js_ast.PropertyAutoAccessor || (opts.isClass && kind == js_ast.PropertyField &&
+		!hasTypeParameters && (p.lexer.Token != js_lexer.TOpenParen || hasDefiniteAssignmentAssertionOperator)) {
+		var initializerOrNil js_ast.Expr
+
+		// Forbid the names "constructor" and "prototype" in some cases
+		if !flags.Has(js_ast.PropertyIsComputed) {
+			if str, ok := key.Data.(*js_ast.EString); ok && (helpers.UTF16EqualsString(str.Value, "constructor") ||
+				(opts.isStatic && helpers.UTF16EqualsString(str.Value, "prototype"))) {
+				p.log.AddError(&p.tracker, keyRange, fmt.Sprintf("Invalid field name %q", helpers.UTF16ToString(str.Value)))
+			}
+		}
+
+		// Skip over types
+		if p.options.ts.Parse && p.lexer.Token == js_lexer.TColon {
+			p.lexer.Next()
+			p.skipTypeScriptType(js_ast.LLowest)
+		}
+
+		if p.lexer.Token == js_lexer.TEquals {
+			p.lexer.Next()
+
+			// "this" and "super" property access is allowed in field initializers
+			oldIsThisDisallowed := p.fnOrArrowDataParse.isThisDisallowed
+			oldAllowSuperProperty := p.fnOrArrowDataParse.allowSuperProperty
+			p.fnOrArrowDataParse.isThisDisallowed = false
+			p.fnOrArrowDataParse.allowSuperProperty = true
+
+			initializerOrNil = p.parseExpr(js_ast.LComma)
+
+			p.fnOrArrowDataParse.isThisDisallowed = oldIsThisDisallowed
+			p.fnOrArrowDataParse.allowSuperProperty = oldAllowSuperProperty
+		}
+
+		// Special-case private identifiers
+		if private, ok := key.Data.(*js_ast.EPrivateIdentifier); ok {
+			name := p.loadNameFromRef(private.Ref)
+			if name == "#constructor" {
+				p.log.AddError(&p.tracker, keyRange, fmt.Sprintf("Invalid field name %q", name))
+			}
+			var declare ast.SymbolKind
+			if kind == js_ast.PropertyAutoAccessor {
+				if opts.isStatic {
+					declare = ast.SymbolPrivateStaticGetSetPair
+				} else {
+					declare = ast.SymbolPrivateGetSetPair
+				}
+				private.Ref = p.declareSymbol(declare, key.Loc, name)
+				p.privateGetters[private.Ref] = p.newSymbol(ast.SymbolOther, name[1:]+"_get")
+				p.privateSetters[private.Ref] = p.newSymbol(ast.SymbolOther, name[1:]+"_set")
+			} else {
+				if opts.isStatic {
+					declare = ast.SymbolPrivateStaticField
+				} else {
+					declare = ast.SymbolPrivateField
+				}
+				private.Ref = p.declareSymbol(declare, key.Loc, name)
+			}
+		}
+
+		p.lexer.ExpectOrInsertSemicolon()
+		if opts.isStatic {
+			flags |= js_ast.PropertyIsStatic
+		}
+		return js_ast.Property{
+			Decorators:       opts.decorators,
+			Loc:              startLoc,
+			Kind:             kind,
+			Flags:            flags,
+			Key:              key,
+			InitializerOrNil: initializerOrNil,
+			CloseBracketLoc:  closeBracketLoc,
+		}, true
+	}
+
+	// Parse a method expression
+	if p.lexer.Token == js_lexer.TOpenParen || kind.IsMethodDefinition() || opts.isClass {
+		hasError := false
+
+		if !hasError && opts.tsDeclareRange.Len != 0 {
+			what := "method"
+			if kind == js_ast.PropertyGetter {
+				what = "getter"
+			} else if kind == js_ast.PropertySetter {
+				what = "setter"
+			}
+			p.log.AddError(&p.tracker, opts.tsDeclareRange, "\"declare\" cannot be used with a "+what)
+			hasError = true
+		}
+
+		if opts.isAsync && p.markAsyncFn(opts.asyncRange, opts.isGenerator) {
+			hasError = true
+		}
+
+		if !hasError && opts.isGenerator && p.markSyntaxFeature(compat.Generator, opts.generatorRange) {
+			hasError = true
+		}
+
+		if !hasError && p.lexer.Token == js_lexer.TOpenParen && kind != js_ast.PropertyGetter && kind != js_ast.PropertySetter && p.markSyntaxFeature(compat.ObjectExtensions, p.lexer.Range()) {
+			hasError = true
+		}
+
+		loc := p.lexer.Loc()
+		scopeIndex := p.pushScopeForParsePass(js_ast.ScopeFunctionArgs, loc)
+		isConstructor := false
+
+		// Forbid the names "constructor" and "prototype" in some cases
+		if opts.isClass && !flags.Has(js_ast.PropertyIsComputed) {
+			if str, ok := key.Data.(*js_ast.EString); ok {
+				if !opts.isStatic && helpers.UTF16EqualsString(str.Value, "constructor") {
+					switch {
+					case kind == js_ast.PropertyGetter:
+						p.log.AddError(&p.tracker, keyRange, "Class constructor cannot be a getter")
+					case kind == js_ast.PropertySetter:
+						p.log.AddError(&p.tracker, keyRange, "Class constructor cannot be a setter")
+					case opts.isAsync:
+						p.log.AddError(&p.tracker, keyRange, "Class constructor cannot be an async function")
+					case opts.isGenerator:
+						p.log.AddError(&p.tracker, keyRange, "Class constructor cannot be a generator")
+					default:
+						isConstructor = true
+					}
+				} else if opts.isStatic && helpers.UTF16EqualsString(str.Value, "prototype") {
+					p.log.AddError(&p.tracker, keyRange, "Invalid static method name \"prototype\"")
+				}
+			}
+		}
+
+		await := allowIdent
+		yield := allowIdent
+		if opts.isAsync {
+			await = allowExpr
+		}
+		if opts.isGenerator {
+			yield = allowExpr
+		}
+
+		fn, hadBody := p.parseFn(nil, opts.classKeyword, opts.decoratorContext, fnOrArrowDataParse{
+			needsAsyncLoc:      key.Loc,
+			asyncRange:         opts.asyncRange,
+			await:              await,
+			yield:              yield,
+			allowSuperCall:     opts.classHasExtends && isConstructor,
+			allowSuperProperty: true,
+			decoratorScope:     opts.decoratorScope,
+			isConstructor:      isConstructor,
+
+			// Only allow omitting the body if we're parsing TypeScript class
+			allowMissingBodyForTypeScript: p.options.ts.Parse && opts.isClass,
+		})
+
+		// "class Foo { foo(): void; foo(): void {} }"
+		if !hadBody {
+			// Skip this property entirely
+			p.popAndDiscardScope(scopeIndex)
+			return js_ast.Property{}, false
+		}
+
+		p.popScope()
+		fn.IsUniqueFormalParameters = true
+		value := js_ast.Expr{Loc: loc, Data: &js_ast.EFunction{Fn: fn}}
+
+		// Enforce argument rules for accessors
+		switch kind {
+		case js_ast.PropertyGetter:
+			if len(fn.Args) > 0 {
+				r := js_lexer.RangeOfIdentifier(p.source, fn.Args[0].Binding.Loc)
+				p.log.AddError(&p.tracker, r, fmt.Sprintf("Getter %s must have zero arguments", p.keyNameForError(key)))
+			}
+
+		case js_ast.PropertySetter:
+			if len(fn.Args) != 1 {
+				r := js_lexer.RangeOfIdentifier(p.source, key.Loc)
+				if len(fn.Args) > 1 {
+					r = js_lexer.RangeOfIdentifier(p.source, fn.Args[1].Binding.Loc)
+				}
+				p.log.AddError(&p.tracker, r, fmt.Sprintf("Setter %s must have exactly one argument", p.keyNameForError(key)))
+			}
+
+		default:
+			kind = js_ast.PropertyMethod
+		}
+
+		// Special-case private identifiers
+		if private, ok := key.Data.(*js_ast.EPrivateIdentifier); ok {
+			var declare ast.SymbolKind
+			var suffix string
+			switch kind {
+			case js_ast.PropertyGetter:
+				if opts.isStatic {
+					declare = ast.SymbolPrivateStaticGet
+				} else {
+					declare = ast.SymbolPrivateGet
+				}
+				suffix = "_get"
+			case js_ast.PropertySetter:
+				if opts.isStatic {
+					declare = ast.SymbolPrivateStaticSet
+				} else {
+					declare = ast.SymbolPrivateSet
+				}
+				suffix = "_set"
+			default:
+				if opts.isStatic {
+					declare = ast.SymbolPrivateStaticMethod
+				} else {
+					declare = ast.SymbolPrivateMethod
+				}
+				suffix = "_fn"
+			}
+			name := p.loadNameFromRef(private.Ref)
+			if name == "#constructor" {
+				p.log.AddError(&p.tracker, keyRange, fmt.Sprintf("Invalid method name %q", name))
+			}
+			private.Ref = p.declareSymbol(declare, key.Loc, name)
+			methodRef := p.newSymbol(ast.SymbolOther, name[1:]+suffix)
+			if kind == js_ast.PropertySetter {
+				p.privateSetters[private.Ref] = methodRef
+			} else {
+				p.privateGetters[private.Ref] = methodRef
+			}
+		}
+
+		if opts.isStatic {
+			flags |= js_ast.PropertyIsStatic
+		}
+		return js_ast.Property{
+			Decorators:      opts.decorators,
+			Loc:             startLoc,
+			Kind:            kind,
+			Flags:           flags,
+			Key:             key,
+			ValueOrNil:      value,
+			CloseBracketLoc: closeBracketLoc,
+		}, true
+	}
+
+	// Parse an object key/value pair
+	p.lexer.Expect(js_lexer.TColon)
+	value := p.parseExprOrBindings(js_ast.LComma, errors)
+	return js_ast.Property{
+		Loc:             startLoc,
+		Kind:            kind,
+		Flags:           flags,
+		Key:             key,
+		ValueOrNil:      value,
+		CloseBracketLoc: closeBracketLoc,
+	}, true
+}
+
+func (p *parser) parsePropertyBinding() js_ast.PropertyBinding {
+	var key js_ast.Expr
+	var closeBracketLoc logger.Loc
+	isComputed := false
+	preferQuotedKey := false
+	loc := p.lexer.Loc()
+
+	switch p.lexer.Token {
+	case js_lexer.TDotDotDot:
+		p.lexer.Next()
+		value := js_ast.Binding{Loc: p.saveExprCommentsHere(), Data: &js_ast.BIdentifier{Ref: p.storeNameInRef(p.lexer.Identifier)}}
+		p.lexer.Expect(js_lexer.TIdentifier)
+		return js_ast.PropertyBinding{
+			Loc:      loc,
+			IsSpread: true,
+			Value:    value,
+		}
+
+	case js_lexer.TNumericLiteral:
+		key = js_ast.Expr{Loc: p.lexer.Loc(), Data: &js_ast.ENumber{Value: p.lexer.Number}}
+		p.checkForLegacyOctalLiteral(key.Data)
+		p.lexer.Next()
+
+	case js_lexer.TStringLiteral:
+		key = p.parseStringLiteral()
+		preferQuotedKey = !p.options.minifySyntax
+
+	case js_lexer.TBigIntegerLiteral:
+		key = js_ast.Expr{Loc: p.lexer.Loc(), Data: &js_ast.EBigInt{Value: p.lexer.Identifier.String}}
+		p.markSyntaxFeature(compat.Bigint, p.lexer.Range())
+		p.lexer.Next()
+
+	case js_lexer.TOpenBracket:
+		isComputed = true
+		p.lexer.Next()
+		key = p.parseExpr(js_ast.LComma)
+		closeBracketLoc = p.saveExprCommentsHere()
+		p.lexer.Expect(js_lexer.TCloseBracket)
+
+	default:
+		name := p.lexer.Identifier
+		nameRange := p.lexer.Range()
+		if !p.lexer.IsIdentifierOrKeyword() {
+			p.lexer.Expect(js_lexer.TIdentifier)
+		}
+		p.lexer.Next()
+		if p.isMangledProp(name.String) {
+			key = js_ast.Expr{Loc: nameRange.Loc, Data: &js_ast.ENameOfSymbol{Ref: p.storeNameInRef(name)}}
+		} else {
+			key = js_ast.Expr{Loc: nameRange.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16(name.String)}}
+		}
+
+		if p.lexer.Token != js_lexer.TColon && p.lexer.Token != js_lexer.TOpenParen {
+			// Forbid invalid identifiers
+			if (p.fnOrArrowDataParse.await != allowIdent && name.String == "await") ||
+				(p.fnOrArrowDataParse.yield != allowIdent && name.String == "yield") {
+				p.log.AddError(&p.tracker, nameRange, fmt.Sprintf("Cannot use %q as an identifier here:", name.String))
+			}
+
+			ref := p.storeNameInRef(name)
+			value := js_ast.Binding{Loc: nameRange.Loc, Data: &js_ast.BIdentifier{Ref: ref}}
+
+			var defaultValueOrNil js_ast.Expr
+			if p.lexer.Token == js_lexer.TEquals {
+				p.lexer.Next()
+				defaultValueOrNil = p.parseExpr(js_ast.LComma)
+			}
+
+			return js_ast.PropertyBinding{
+				Loc:               loc,
+				Key:               key,
+				Value:             value,
+				DefaultValueOrNil: defaultValueOrNil,
+			}
+		}
+	}
+
+	p.lexer.Expect(js_lexer.TColon)
+	value := p.parseBinding(parseBindingOpts{})
+
+	var defaultValueOrNil js_ast.Expr
+	if p.lexer.Token == js_lexer.TEquals {
+		p.lexer.Next()
+		defaultValueOrNil = p.parseExpr(js_ast.LComma)
+	}
+
+	return js_ast.PropertyBinding{
+		Loc:               loc,
+		IsComputed:        isComputed,
+		PreferQuotedKey:   preferQuotedKey,
+		Key:               key,
+		Value:             value,
+		DefaultValueOrNil: defaultValueOrNil,
+		CloseBracketLoc:   closeBracketLoc,
+	}
+}
+
+// These properties have special semantics in JavaScript. They must not be
+// mangled or we could potentially fail to parse valid JavaScript syntax or
+// generate invalid JavaScript syntax as output.
+//
+// This list is only intended to contain properties specific to the JavaScript
+// language itself to avoid syntax errors in the generated output. It's not
+// intended to contain properties for JavaScript APIs. Those must be provided
+// by the user.
+var permanentReservedProps = map[string]bool{
+	"__proto__":   true,
+	"constructor": true,
+	"prototype":   true,
+}
+
+func (p *parser) isMangledProp(name string) bool {
+	if p.options.mangleProps == nil {
+		return false
+	}
+	if p.options.mangleProps.MatchString(name) && !permanentReservedProps[name] && (p.options.reserveProps == nil || !p.options.reserveProps.MatchString(name)) {
+		return true
+	}
+	reservedProps := p.reservedProps
+	if reservedProps == nil {
+		reservedProps = make(map[string]bool)
+		p.reservedProps = reservedProps
+	}
+	reservedProps[name] = true
+	return false
+}
+
+func (p *parser) symbolForMangledProp(name string) ast.Ref {
+	mangledProps := p.mangledProps
+	if mangledProps == nil {
+		mangledProps = make(map[string]ast.Ref)
+		p.mangledProps = mangledProps
+	}
+	ref, ok := mangledProps[name]
+	if !ok {
+		ref = p.newSymbol(ast.SymbolMangledProp, name)
+		mangledProps[name] = ref
+	}
+	if !p.isControlFlowDead {
+		p.symbols[ref.InnerIndex].UseCountEstimate++
+	}
+	return ref
+}
+
+type wasOriginallyDotOrIndex uint8
+
+const (
+	wasOriginallyDot wasOriginallyDotOrIndex = iota
+	wasOriginallyIndex
+)
+
+func (p *parser) dotOrMangledPropParse(
+	target js_ast.Expr,
+	name js_lexer.MaybeSubstring,
+	nameLoc logger.Loc,
+	optionalChain js_ast.OptionalChain,
+	original wasOriginallyDotOrIndex,
+) js_ast.E {
+	if (original != wasOriginallyIndex || p.options.mangleQuoted) && p.isMangledProp(name.String) {
+		return &js_ast.EIndex{
+			Target:        target,
+			Index:         js_ast.Expr{Loc: nameLoc, Data: &js_ast.ENameOfSymbol{Ref: p.storeNameInRef(name)}},
+			OptionalChain: optionalChain,
+		}
+	}
+
+	return &js_ast.EDot{
+		Target:        target,
+		Name:          name.String,
+		NameLoc:       nameLoc,
+		OptionalChain: optionalChain,
+	}
+}
+
+func (p *parser) dotOrMangledPropVisit(target js_ast.Expr, name string, nameLoc logger.Loc) js_ast.E {
+	if p.isMangledProp(name) {
+		return &js_ast.EIndex{
+			Target: target,
+			Index:  js_ast.Expr{Loc: nameLoc, Data: &js_ast.ENameOfSymbol{Ref: p.symbolForMangledProp(name)}},
+		}
+	}
+
+	return &js_ast.EDot{
+		Target:  target,
+		Name:    name,
+		NameLoc: nameLoc,
+	}
+}
+
+func (p *parser) parseArrowBody(args []js_ast.Arg, data fnOrArrowDataParse) *js_ast.EArrow {
+	arrowLoc := p.lexer.Loc()
+
+	// Newlines are not allowed before "=>"
+	if p.lexer.HasNewlineBefore {
+		p.log.AddError(&p.tracker, p.lexer.Range(), "Unexpected newline before \"=>\"")
+		panic(js_lexer.LexerPanic{})
+	}
+
+	p.lexer.Expect(js_lexer.TEqualsGreaterThan)
+
+	for _, arg := range args {
+		p.declareBinding(ast.SymbolHoisted, arg.Binding, parseStmtOpts{})
+	}
+
+	// The ability to use "this" and "super" is inherited by arrow functions
+	data.isThisDisallowed = p.fnOrArrowDataParse.isThisDisallowed
+	data.allowSuperCall = p.fnOrArrowDataParse.allowSuperCall
+	data.allowSuperProperty = p.fnOrArrowDataParse.allowSuperProperty
+
+	if p.lexer.Token == js_lexer.TOpenBrace {
+		body := p.parseFnBody(data)
+		p.afterArrowBodyLoc = p.lexer.Loc()
+		return &js_ast.EArrow{Args: args, Body: body}
+	}
+
+	p.pushScopeForParsePass(js_ast.ScopeFunctionBody, arrowLoc)
+	defer p.popScope()
+
+	oldFnOrArrowData := p.fnOrArrowDataParse
+	p.fnOrArrowDataParse = data
+	expr := p.parseExpr(js_ast.LComma)
+	p.fnOrArrowDataParse = oldFnOrArrowData
+	return &js_ast.EArrow{
+		Args:       args,
+		PreferExpr: true,
+		Body:       js_ast.FnBody{Loc: arrowLoc, Block: js_ast.SBlock{Stmts: []js_ast.Stmt{{Loc: expr.Loc, Data: &js_ast.SReturn{ValueOrNil: expr}}}}},
+	}
+}
+
+func (p *parser) checkForArrowAfterTheCurrentToken() bool {
+	oldLexer := p.lexer
+	p.lexer.IsLogDisabled = true
+
+	// Implement backtracking by restoring the lexer's memory to its original state
+	defer func() {
+		r := recover()
+		if _, isLexerPanic := r.(js_lexer.LexerPanic); isLexerPanic {
+			p.lexer = oldLexer
+		} else if r != nil {
+			panic(r)
+		}
+	}()
+
+	p.lexer.Next()
+	isArrowAfterThisToken := p.lexer.Token == js_lexer.TEqualsGreaterThan
+
+	p.lexer = oldLexer
+	return isArrowAfterThisToken
+}
+
+// This parses an expression. This assumes we've already parsed the "async"
+// keyword and are currently looking at the following token.
+func (p *parser) parseAsyncPrefixExpr(asyncRange logger.Range, level js_ast.L, flags exprFlag) js_ast.Expr {
+	// "async function() {}"
+	if !p.lexer.HasNewlineBefore && p.lexer.Token == js_lexer.TFunction {
+		return p.parseFnExpr(asyncRange.Loc, true /* isAsync */, asyncRange)
+	}
+
+	// Check the precedence level to avoid parsing an arrow function in
+	// "new async () => {}". This also avoids parsing "new async()" as
+	// "new (async())()" instead.
+	if !p.lexer.HasNewlineBefore && level < js_ast.LMember {
+		switch p.lexer.Token {
+		// "async => {}"
+		case js_lexer.TEqualsGreaterThan:
+			if level <= js_ast.LAssign {
+				arg := js_ast.Arg{Binding: js_ast.Binding{Loc: asyncRange.Loc, Data: &js_ast.BIdentifier{
+					Ref: p.storeNameInRef(js_lexer.MaybeSubstring{String: "async"})}}}
+
+				p.pushScopeForParsePass(js_ast.ScopeFunctionArgs, asyncRange.Loc)
+				defer p.popScope()
+
+				return js_ast.Expr{Loc: asyncRange.Loc, Data: p.parseArrowBody([]js_ast.Arg{arg}, fnOrArrowDataParse{
+					needsAsyncLoc: asyncRange.Loc,
+				})}
+			}
+
+		// "async x => {}"
+		case js_lexer.TIdentifier:
+			if level <= js_ast.LAssign {
+				// See https://github.com/tc39/ecma262/issues/2034 for details
+				isArrowFn := true
+				if (flags&exprFlagForLoopInit) != 0 && p.lexer.Identifier.String == "of" {
+					// "for (async of" is only an arrow function if the next token is "=>"
+					isArrowFn = p.checkForArrowAfterTheCurrentToken()
+
+					// Do not allow "for (async of []) ;" but do allow "for await (async of []) ;"
+					if !isArrowFn && (flags&exprFlagForAwaitLoopInit) == 0 && p.lexer.Raw() == "of" {
+						r := logger.Range{Loc: asyncRange.Loc, Len: p.lexer.Range().End() - asyncRange.Loc.Start}
+						p.log.AddError(&p.tracker, r, "For loop initializers cannot start with \"async of\"")
+						panic(js_lexer.LexerPanic{})
+					}
+				}
+
+				if isArrowFn {
+					p.markAsyncFn(asyncRange, false)
+					ref := p.storeNameInRef(p.lexer.Identifier)
+					arg := js_ast.Arg{Binding: js_ast.Binding{Loc: p.lexer.Loc(), Data: &js_ast.BIdentifier{Ref: ref}}}
+					p.lexer.Next()
+
+					p.pushScopeForParsePass(js_ast.ScopeFunctionArgs, asyncRange.Loc)
+					defer p.popScope()
+
+					arrow := p.parseArrowBody([]js_ast.Arg{arg}, fnOrArrowDataParse{
+						needsAsyncLoc: arg.Binding.Loc,
+						await:         allowExpr,
+					})
+					arrow.IsAsync = true
+					return js_ast.Expr{Loc: asyncRange.Loc, Data: arrow}
+				}
+			}
+
+		// "async()"
+		// "async () => {}"
+		case js_lexer.TOpenParen:
+			p.lexer.Next()
+			return p.parseParenExpr(asyncRange.Loc, level, parenExprOpts{asyncRange: asyncRange})
+
+		// "async<T>()"
+		// "async <T>() => {}"
+		case js_lexer.TLessThan:
+			if p.options.ts.Parse && (!p.options.jsx.Parse || p.isTSArrowFnJSX()) {
+				if result := p.trySkipTypeScriptTypeParametersThenOpenParenWithBacktracking(); result != didNotSkipAnything {
+					p.lexer.Next()
+					return p.parseParenExpr(asyncRange.Loc, level, parenExprOpts{
+						asyncRange:   asyncRange,
+						forceArrowFn: result == definitelyTypeParameters,
+					})
+				}
+			}
+		}
+	}
+
+	// "async"
+	// "async + 1"
+	return js_ast.Expr{Loc: asyncRange.Loc, Data: &js_ast.EIdentifier{
+		Ref: p.storeNameInRef(js_lexer.MaybeSubstring{String: "async"})}}
+}
+
+func (p *parser) parseFnExpr(loc logger.Loc, isAsync bool, asyncRange logger.Range) js_ast.Expr {
+	p.lexer.Next()
+	isGenerator := p.lexer.Token == js_lexer.TAsterisk
+	hasError := false
+	if isAsync {
+		hasError = p.markAsyncFn(asyncRange, isGenerator)
+	}
+	if isGenerator {
+		if !hasError {
+			p.markSyntaxFeature(compat.Generator, p.lexer.Range())
+		}
+		p.lexer.Next()
+	}
+	var name *ast.LocRef
+
+	p.pushScopeForParsePass(js_ast.ScopeFunctionArgs, loc)
+	defer p.popScope()
+
+	// The name is optional
+	if p.lexer.Token == js_lexer.TIdentifier {
+		// Don't declare the name "arguments" since it's shadowed and inaccessible
+		name = &ast.LocRef{Loc: p.lexer.Loc()}
+		if text := p.lexer.Identifier.String; text != "arguments" {
+			name.Ref = p.declareSymbol(ast.SymbolHoistedFunction, name.Loc, text)
+		} else {
+			name.Ref = p.newSymbol(ast.SymbolHoistedFunction, text)
+		}
+		p.lexer.Next()
+	}
+
+	// Even anonymous functions can have TypeScript type parameters
+	if p.options.ts.Parse {
+		p.skipTypeScriptTypeParameters(allowConstModifier)
+	}
+
+	await := allowIdent
+	yield := allowIdent
+	if isAsync {
+		await = allowExpr
+	}
+	if isGenerator {
+		yield = allowExpr
+	}
+
+	fn, _ := p.parseFn(name, logger.Range{}, 0, fnOrArrowDataParse{
+		needsAsyncLoc: loc,
+		asyncRange:    asyncRange,
+		await:         await,
+		yield:         yield,
+	})
+	p.validateFunctionName(fn, fnExpr)
+	return js_ast.Expr{Loc: loc, Data: &js_ast.EFunction{Fn: fn}}
+}
+
+type parenExprOpts struct {
+	asyncRange   logger.Range
+	forceArrowFn bool
+}
+
+// This assumes that the open parenthesis has already been parsed by the caller
+func (p *parser) parseParenExpr(loc logger.Loc, level js_ast.L, opts parenExprOpts) js_ast.Expr {
+	items := []js_ast.Expr{}
+	errors := deferredErrors{}
+	arrowArgErrors := deferredArrowArgErrors{}
+	spreadRange := logger.Range{}
+	typeColonRange := logger.Range{}
+	commaAfterSpread := logger.Loc{}
+	isAsync := opts.asyncRange.Len > 0
+
+	// Push a scope assuming this is an arrow function. It may not be, in which
+	// case we'll need to roll this change back. This has to be done ahead of
+	// parsing the arguments instead of later on when we hit the "=>" token and
+	// we know it's an arrow function because the arguments may have default
+	// values that introduce new scopes and declare new symbols. If this is an
+	// arrow function, then those new scopes will need to be parented under the
+	// scope of the arrow function itself.
+	scopeIndex := p.pushScopeForParsePass(js_ast.ScopeFunctionArgs, loc)
+
+	// Allow "in" inside parentheses
+	oldAllowIn := p.allowIn
+	p.allowIn = true
+
+	// Forbid "await" and "yield", but only for arrow functions
+	oldFnOrArrowData := p.fnOrArrowDataParse
+	p.fnOrArrowDataParse.arrowArgErrors = &arrowArgErrors
+
+	// Scan over the comma-separated arguments or expressions
+	for p.lexer.Token != js_lexer.TCloseParen {
+		itemLoc := p.lexer.Loc()
+		isSpread := p.lexer.Token == js_lexer.TDotDotDot
+
+		if isSpread {
+			spreadRange = p.lexer.Range()
+			p.markSyntaxFeature(compat.RestArgument, spreadRange)
+			p.lexer.Next()
+		}
+
+		// We don't know yet whether these are arguments or expressions, so parse
+		// a superset of the expression syntax. Errors about things that are valid
+		// in one but not in the other are deferred.
+		p.latestArrowArgLoc = p.lexer.Loc()
+		item := p.parseExprOrBindings(js_ast.LComma, &errors)
+
+		if isSpread {
+			item = js_ast.Expr{Loc: itemLoc, Data: &js_ast.ESpread{Value: item}}
+		}
+
+		// Skip over types
+		if p.options.ts.Parse && p.lexer.Token == js_lexer.TColon {
+			typeColonRange = p.lexer.Range()
+			p.lexer.Next()
+			p.skipTypeScriptType(js_ast.LLowest)
+		}
+
+		// There may be a "=" after the type (but not after an "as" cast)
+		if p.options.ts.Parse && p.lexer.Token == js_lexer.TEquals && p.lexer.Loc() != p.forbidSuffixAfterAsLoc {
+			p.lexer.Next()
+			item = js_ast.Assign(item, p.parseExpr(js_ast.LComma))
+		}
+
+		items = append(items, item)
+		if p.lexer.Token != js_lexer.TComma {
+			break
+		}
+
+		// Spread arguments must come last. If there's a spread argument followed
+		// by a comma, throw an error if we use these expressions as bindings.
+		if isSpread {
+			commaAfterSpread = p.lexer.Loc()
+		}
+
+		// Eat the comma token
+		p.lexer.Next()
+	}
+
+	// The parenthetical construct must end with a close parenthesis
+	p.lexer.Expect(js_lexer.TCloseParen)
+
+	// Restore "in" operator status before we parse the arrow function body
+	p.allowIn = oldAllowIn
+
+	// Also restore "await" and "yield" expression errors
+	p.fnOrArrowDataParse = oldFnOrArrowData
+
+	// Are these arguments to an arrow function?
+	if p.lexer.Token == js_lexer.TEqualsGreaterThan || opts.forceArrowFn || (p.options.ts.Parse && p.lexer.Token == js_lexer.TColon) {
+		// Arrow functions are not allowed inside certain expressions
+		if level > js_ast.LAssign {
+			p.lexer.Unexpected()
+		}
+
+		var invalidLog invalidLog
+		args := []js_ast.Arg{}
+
+		if isAsync {
+			p.markAsyncFn(opts.asyncRange, false)
+		}
+
+		// First, try converting the expressions to bindings
+		for _, item := range items {
+			isSpread := false
+			if spread, ok := item.Data.(*js_ast.ESpread); ok {
+				item = spread.Value
+				isSpread = true
+			}
+			binding, initializerOrNil, log := p.convertExprToBindingAndInitializer(item, invalidLog, isSpread)
+			invalidLog = log
+			args = append(args, js_ast.Arg{Binding: binding, DefaultOrNil: initializerOrNil})
+		}
+
+		// Avoid parsing TypeScript code like "a ? (1 + 2) : (3 + 4)" as an arrow
+		// function. The ":" after the ")" may be a return type annotation, so we
+		// attempt to convert the expressions to bindings first before deciding
+		// whether this is an arrow function, and only pick an arrow function if
+		// there were no conversion errors.
+		if p.lexer.Token == js_lexer.TEqualsGreaterThan || (len(invalidLog.invalidTokens) == 0 &&
+			p.trySkipTypeScriptArrowReturnTypeWithBacktracking()) || opts.forceArrowFn {
+			if commaAfterSpread.Start != 0 {
+				p.log.AddError(&p.tracker, logger.Range{Loc: commaAfterSpread, Len: 1}, "Unexpected \",\" after rest pattern")
+			}
+			p.logArrowArgErrors(&arrowArgErrors)
+			p.logDeferredArrowArgErrors(&errors)
+
+			// Now that we've decided we're an arrow function, report binding pattern
+			// conversion errors
+			if len(invalidLog.invalidTokens) > 0 {
+				for _, token := range invalidLog.invalidTokens {
+					p.log.AddError(&p.tracker, token, "Invalid binding pattern")
+				}
+				panic(js_lexer.LexerPanic{})
+			}
+
+			// Also report syntax features used in bindings
+			for _, entry := range invalidLog.syntaxFeatures {
+				p.markSyntaxFeature(entry.feature, entry.token)
+			}
+
+			await := allowIdent
+			if isAsync {
+				await = allowExpr
+			}
+
+			arrow := p.parseArrowBody(args, fnOrArrowDataParse{
+				needsAsyncLoc: loc,
+				await:         await,
+			})
+			arrow.IsAsync = isAsync
+			arrow.HasRestArg = spreadRange.Len > 0
+			p.popScope()
+			return js_ast.Expr{Loc: loc, Data: arrow}
+		}
+	}
+
+	// If we get here, it's not an arrow function so undo the pushing of the
+	// scope we did earlier. This needs to flatten any child scopes into the
+	// parent scope as if the scope was never pushed in the first place.
+	p.popAndFlattenScope(scopeIndex)
+
+	// If this isn't an arrow function, then types aren't allowed
+	if typeColonRange.Len > 0 {
+		p.log.AddError(&p.tracker, typeColonRange, "Unexpected \":\"")
+		panic(js_lexer.LexerPanic{})
+	}
+
+	// Are these arguments for a call to a function named "async"?
+	if isAsync {
+		p.logExprErrors(&errors)
+		async := js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{
+			Ref: p.storeNameInRef(js_lexer.MaybeSubstring{String: "async"})}}
+		return js_ast.Expr{Loc: loc, Data: &js_ast.ECall{
+			Target: async,
+			Args:   items,
+		}}
+	}
+
+	// Is this a chain of expressions and comma operators?
+	if len(items) > 0 {
+		p.logExprErrors(&errors)
+		if spreadRange.Len > 0 {
+			p.log.AddError(&p.tracker, spreadRange, "Unexpected \"...\"")
+			panic(js_lexer.LexerPanic{})
+		}
+		value := js_ast.JoinAllWithComma(items)
+		p.markExprAsParenthesized(value, loc, isAsync)
+		return value
+	}
+
+	// Indicate that we expected an arrow function
+	p.lexer.Expected(js_lexer.TEqualsGreaterThan)
+	return js_ast.Expr{}
+}
+
+type invalidLog struct {
+	invalidTokens  []logger.Range
+	syntaxFeatures []syntaxFeature
+}
+
+type syntaxFeature struct {
+	feature compat.JSFeature
+	token   logger.Range
+}
+
+func (p *parser) convertExprToBindingAndInitializer(
+	expr js_ast.Expr, invalidLog invalidLog, isSpread bool,
+) (js_ast.Binding, js_ast.Expr, invalidLog) {
+	var initializerOrNil js_ast.Expr
+	if assign, ok := expr.Data.(*js_ast.EBinary); ok && assign.Op == js_ast.BinOpAssign {
+		initializerOrNil = assign.Right
+		expr = assign.Left
+	}
+	binding, invalidLog := p.convertExprToBinding(expr, invalidLog)
+	if initializerOrNil.Data != nil {
+		equalsRange := p.source.RangeOfOperatorBefore(initializerOrNil.Loc, "=")
+		if isSpread {
+			p.log.AddError(&p.tracker, equalsRange, "A rest argument cannot have a default initializer")
+		} else {
+			invalidLog.syntaxFeatures = append(invalidLog.syntaxFeatures, syntaxFeature{
+				feature: compat.DefaultArgument,
+				token:   equalsRange,
+			})
+		}
+	}
+	return binding, initializerOrNil, invalidLog
+}
+
+// Note: do not write to "p.log" in this function. Any errors due to conversion
+// from expression to binding should be written to "invalidLog" instead. That
+// way we can potentially keep this as an expression if it turns out it's not
+// needed as a binding after all.
+func (p *parser) convertExprToBinding(expr js_ast.Expr, invalidLog invalidLog) (js_ast.Binding, invalidLog) {
+	switch e := expr.Data.(type) {
+	case *js_ast.EMissing:
+		return js_ast.Binding{Loc: expr.Loc, Data: js_ast.BMissingShared}, invalidLog
+
+	case *js_ast.EIdentifier:
+		return js_ast.Binding{Loc: expr.Loc, Data: &js_ast.BIdentifier{Ref: e.Ref}}, invalidLog
+
+	case *js_ast.EArray:
+		if e.CommaAfterSpread.Start != 0 {
+			invalidLog.invalidTokens = append(invalidLog.invalidTokens, logger.Range{Loc: e.CommaAfterSpread, Len: 1})
+		}
+		invalidLog.syntaxFeatures = append(invalidLog.syntaxFeatures,
+			syntaxFeature{feature: compat.Destructuring, token: p.source.RangeOfOperatorAfter(expr.Loc, "[")})
+		items := []js_ast.ArrayBinding{}
+		isSpread := false
+		for _, item := range e.Items {
+			if i, ok := item.Data.(*js_ast.ESpread); ok {
+				isSpread = true
+				item = i.Value
+				if _, ok := item.Data.(*js_ast.EIdentifier); !ok {
+					p.markSyntaxFeature(compat.NestedRestBinding, p.source.RangeOfOperatorAfter(item.Loc, "["))
+				}
+			}
+			binding, initializerOrNil, log := p.convertExprToBindingAndInitializer(item, invalidLog, isSpread)
+			invalidLog = log
+			items = append(items, js_ast.ArrayBinding{
+				Binding:           binding,
+				DefaultValueOrNil: initializerOrNil,
+				Loc:               item.Loc,
+			})
+		}
+		return js_ast.Binding{Loc: expr.Loc, Data: &js_ast.BArray{
+			Items:           items,
+			HasSpread:       isSpread,
+			IsSingleLine:    e.IsSingleLine,
+			CloseBracketLoc: e.CloseBracketLoc,
+		}}, invalidLog
+
+	case *js_ast.EObject:
+		if e.CommaAfterSpread.Start != 0 {
+			invalidLog.invalidTokens = append(invalidLog.invalidTokens, logger.Range{Loc: e.CommaAfterSpread, Len: 1})
+		}
+		invalidLog.syntaxFeatures = append(invalidLog.syntaxFeatures,
+			syntaxFeature{feature: compat.Destructuring, token: p.source.RangeOfOperatorAfter(expr.Loc, "{")})
+		properties := []js_ast.PropertyBinding{}
+		for _, property := range e.Properties {
+			if property.Kind.IsMethodDefinition() {
+				invalidLog.invalidTokens = append(invalidLog.invalidTokens, js_lexer.RangeOfIdentifier(p.source, property.Key.Loc))
+				continue
+			}
+			binding, initializerOrNil, log := p.convertExprToBindingAndInitializer(property.ValueOrNil, invalidLog, false)
+			invalidLog = log
+			if initializerOrNil.Data == nil {
+				initializerOrNil = property.InitializerOrNil
+			}
+			properties = append(properties, js_ast.PropertyBinding{
+				Loc:               property.Loc,
+				IsSpread:          property.Kind == js_ast.PropertySpread,
+				IsComputed:        property.Flags.Has(js_ast.PropertyIsComputed),
+				Key:               property.Key,
+				Value:             binding,
+				DefaultValueOrNil: initializerOrNil,
+			})
+		}
+		return js_ast.Binding{Loc: expr.Loc, Data: &js_ast.BObject{
+			Properties:    properties,
+			IsSingleLine:  e.IsSingleLine,
+			CloseBraceLoc: e.CloseBraceLoc,
+		}}, invalidLog
+
+	default:
+		invalidLog.invalidTokens = append(invalidLog.invalidTokens, logger.Range{Loc: expr.Loc})
+		return js_ast.Binding{}, invalidLog
+	}
+}
+
+func (p *parser) saveExprCommentsHere() logger.Loc {
+	loc := p.lexer.Loc()
+	if p.exprComments != nil && len(p.lexer.CommentsBeforeToken) > 0 {
+		comments := make([]string, len(p.lexer.CommentsBeforeToken))
+		for i, comment := range p.lexer.CommentsBeforeToken {
+			comments[i] = p.source.CommentTextWithoutIndent(comment)
+		}
+		p.exprComments[loc] = comments
+		p.lexer.CommentsBeforeToken = p.lexer.CommentsBeforeToken[0:]
+	}
+	return loc
+}
+
+type exprFlag uint8
+
+const (
+	exprFlagDecorator exprFlag = 1 << iota
+	exprFlagForLoopInit
+	exprFlagForAwaitLoopInit
+)
+
+func (p *parser) parsePrefix(level js_ast.L, errors *deferredErrors, flags exprFlag) js_ast.Expr {
+	loc := p.saveExprCommentsHere()
+
+	switch p.lexer.Token {
+	case js_lexer.TSuper:
+		superRange := p.lexer.Range()
+		p.lexer.Next()
+
+		switch p.lexer.Token {
+		case js_lexer.TOpenParen:
+			if level < js_ast.LCall && p.fnOrArrowDataParse.allowSuperCall {
+				return js_ast.Expr{Loc: loc, Data: js_ast.ESuperShared}
+			}
+
+		case js_lexer.TDot, js_lexer.TOpenBracket:
+			if p.fnOrArrowDataParse.allowSuperProperty {
+				return js_ast.Expr{Loc: loc, Data: js_ast.ESuperShared}
+			}
+		}
+
+		p.log.AddError(&p.tracker, superRange, "Unexpected \"super\"")
+		return js_ast.Expr{Loc: loc, Data: js_ast.ESuperShared}
+
+	case js_lexer.TOpenParen:
+		if errors != nil {
+			errors.invalidParens = append(errors.invalidParens, p.lexer.Range())
+		}
+
+		p.lexer.Next()
+
+		// Arrow functions aren't allowed in the middle of expressions
+		if level > js_ast.LAssign {
+			// Allow "in" inside parentheses
+			oldAllowIn := p.allowIn
+			p.allowIn = true
+
+			value := p.parseExpr(js_ast.LLowest)
+			p.markExprAsParenthesized(value, loc, false)
+			p.lexer.Expect(js_lexer.TCloseParen)
+
+			p.allowIn = oldAllowIn
+			return value
+		}
+
+		value := p.parseParenExpr(loc, level, parenExprOpts{})
+		return value
+
+	case js_lexer.TFalse:
+		p.lexer.Next()
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EBoolean{Value: false}}
+
+	case js_lexer.TTrue:
+		p.lexer.Next()
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EBoolean{Value: true}}
+
+	case js_lexer.TNull:
+		p.lexer.Next()
+		return js_ast.Expr{Loc: loc, Data: js_ast.ENullShared}
+
+	case js_lexer.TThis:
+		if p.fnOrArrowDataParse.isThisDisallowed {
+			p.log.AddError(&p.tracker, p.lexer.Range(), "Cannot use \"this\" here:")
+		}
+		p.lexer.Next()
+		return js_ast.Expr{Loc: loc, Data: js_ast.EThisShared}
+
+	case js_lexer.TPrivateIdentifier:
+		if !p.allowPrivateIdentifiers || !p.allowIn || level >= js_ast.LCompare {
+			p.lexer.Unexpected()
+		}
+
+		name := p.lexer.Identifier
+		p.lexer.Next()
+
+		// Check for "#foo in bar"
+		if p.lexer.Token != js_lexer.TIn {
+			p.lexer.Expected(js_lexer.TIn)
+		}
+
+		// Make sure to lower all matching private names
+		if p.options.unsupportedJSFeatures.Has(compat.ClassPrivateBrandCheck) {
+			if p.lowerAllOfThesePrivateNames == nil {
+				p.lowerAllOfThesePrivateNames = make(map[string]bool)
+			}
+			p.lowerAllOfThesePrivateNames[name.String] = true
+		}
+
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EPrivateIdentifier{Ref: p.storeNameInRef(name)}}
+
+	case js_lexer.TIdentifier:
+		name := p.lexer.Identifier
+		nameRange := p.lexer.Range()
+		raw := p.lexer.Raw()
+		p.lexer.Next()
+
+		// Handle async and await expressions
+		switch name.String {
+		case "async":
+			if raw == "async" {
+				return p.parseAsyncPrefixExpr(nameRange, level, flags)
+			}
+
+		case "await":
+			switch p.fnOrArrowDataParse.await {
+			case forbidAll:
+				p.log.AddError(&p.tracker, nameRange, "The keyword \"await\" cannot be used here:")
+
+			case allowExpr:
+				if raw != "await" {
+					p.log.AddError(&p.tracker, nameRange, "The keyword \"await\" cannot be escaped")
+				} else {
+					if p.fnOrArrowDataParse.isTopLevel {
+						p.topLevelAwaitKeyword = nameRange
+					}
+					if p.fnOrArrowDataParse.arrowArgErrors != nil {
+						p.fnOrArrowDataParse.arrowArgErrors.invalidExprAwait = nameRange
+					}
+					value := p.parseExpr(js_ast.LPrefix)
+					if p.lexer.Token == js_lexer.TAsteriskAsterisk {
+						p.lexer.Unexpected()
+					}
+					return js_ast.Expr{Loc: loc, Data: &js_ast.EAwait{Value: value}}
+				}
+
+			case allowIdent:
+				p.lexer.PrevTokenWasAwaitKeyword = true
+				p.lexer.AwaitKeywordLoc = loc
+				p.lexer.FnOrArrowStartLoc = p.fnOrArrowDataParse.needsAsyncLoc
+			}
+
+		case "yield":
+			switch p.fnOrArrowDataParse.yield {
+			case forbidAll:
+				p.log.AddError(&p.tracker, nameRange, "The keyword \"yield\" cannot be used here:")
+
+			case allowExpr:
+				if raw != "yield" {
+					p.log.AddError(&p.tracker, nameRange, "The keyword \"yield\" cannot be escaped")
+				} else {
+					if level > js_ast.LAssign {
+						p.log.AddError(&p.tracker, nameRange, "Cannot use a \"yield\" expression here without parentheses:")
+					}
+					if p.fnOrArrowDataParse.arrowArgErrors != nil {
+						p.fnOrArrowDataParse.arrowArgErrors.invalidExprYield = nameRange
+					}
+					return p.parseYieldExpr(loc)
+				}
+
+			case allowIdent:
+				if !p.lexer.HasNewlineBefore {
+					// Try to gracefully recover if "yield" is used in the wrong place
+					switch p.lexer.Token {
+					case js_lexer.TNull, js_lexer.TIdentifier, js_lexer.TFalse, js_lexer.TTrue,
+						js_lexer.TNumericLiteral, js_lexer.TBigIntegerLiteral, js_lexer.TStringLiteral:
+						p.log.AddError(&p.tracker, nameRange, "Cannot use \"yield\" outside a generator function")
+						return p.parseYieldExpr(loc)
+					}
+				}
+			}
+		}
+
+		// Handle the start of an arrow expression
+		if p.lexer.Token == js_lexer.TEqualsGreaterThan && level <= js_ast.LAssign {
+			ref := p.storeNameInRef(name)
+			arg := js_ast.Arg{Binding: js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: ref}}}
+
+			p.pushScopeForParsePass(js_ast.ScopeFunctionArgs, loc)
+			defer p.popScope()
+
+			return js_ast.Expr{Loc: loc, Data: p.parseArrowBody([]js_ast.Arg{arg}, fnOrArrowDataParse{
+				needsAsyncLoc: loc,
+			})}
+		}
+
+		ref := p.storeNameInRef(name)
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: ref}}
+
+	case js_lexer.TStringLiteral, js_lexer.TNoSubstitutionTemplateLiteral:
+		return p.parseStringLiteral()
+
+	case js_lexer.TTemplateHead:
+		var legacyOctalLoc logger.Loc
+		headLoc := p.lexer.Loc()
+		head := p.lexer.StringLiteral()
+		if p.lexer.LegacyOctalLoc.Start > loc.Start {
+			legacyOctalLoc = p.lexer.LegacyOctalLoc
+		}
+		parts, tailLegacyOctalLoc := p.parseTemplateParts(false /* includeRaw */)
+		if tailLegacyOctalLoc.Start > 0 {
+			legacyOctalLoc = tailLegacyOctalLoc
+		}
+		return js_ast.Expr{Loc: loc, Data: &js_ast.ETemplate{
+			HeadLoc:        headLoc,
+			HeadCooked:     head,
+			Parts:          parts,
+			LegacyOctalLoc: legacyOctalLoc,
+		}}
+
+	case js_lexer.TNumericLiteral:
+		value := js_ast.Expr{Loc: loc, Data: &js_ast.ENumber{Value: p.lexer.Number}}
+		p.checkForLegacyOctalLiteral(value.Data)
+		p.lexer.Next()
+		return value
+
+	case js_lexer.TBigIntegerLiteral:
+		value := p.lexer.Identifier
+		p.markSyntaxFeature(compat.Bigint, p.lexer.Range())
+		p.lexer.Next()
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EBigInt{Value: value.String}}
+
+	case js_lexer.TSlash, js_lexer.TSlashEquals:
+		p.lexer.ScanRegExp()
+		value := p.lexer.Raw()
+		p.lexer.Next()
+		return js_ast.Expr{Loc: loc, Data: &js_ast.ERegExp{Value: value}}
+
+	case js_lexer.TVoid:
+		p.lexer.Next()
+		value := p.parseExpr(js_ast.LPrefix)
+		if p.lexer.Token == js_lexer.TAsteriskAsterisk {
+			p.lexer.Unexpected()
+		}
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EUnary{Op: js_ast.UnOpVoid, Value: value}}
+
+	case js_lexer.TTypeof:
+		p.lexer.Next()
+		value := p.parseExpr(js_ast.LPrefix)
+		if p.lexer.Token == js_lexer.TAsteriskAsterisk {
+			p.lexer.Unexpected()
+		}
+		_, valueIsIdentifier := value.Data.(*js_ast.EIdentifier)
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EUnary{
+			Op:                            js_ast.UnOpTypeof,
+			Value:                         value,
+			WasOriginallyTypeofIdentifier: valueIsIdentifier,
+		}}
+
+	case js_lexer.TDelete:
+		p.lexer.Next()
+		value := p.parseExpr(js_ast.LPrefix)
+		if p.lexer.Token == js_lexer.TAsteriskAsterisk {
+			p.lexer.Unexpected()
+		}
+		if index, ok := value.Data.(*js_ast.EIndex); ok {
+			if private, ok := index.Index.Data.(*js_ast.EPrivateIdentifier); ok {
+				name := p.loadNameFromRef(private.Ref)
+				r := logger.Range{Loc: index.Index.Loc, Len: int32(len(name))}
+				p.log.AddError(&p.tracker, r, fmt.Sprintf("Deleting the private name %q is forbidden", name))
+			}
+		}
+		_, valueIsIdentifier := value.Data.(*js_ast.EIdentifier)
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EUnary{
+			Op:    js_ast.UnOpDelete,
+			Value: value,
+			WasOriginallyDeleteOfIdentifierOrPropertyAccess: valueIsIdentifier || js_ast.IsPropertyAccess(value),
+		}}
+
+	case js_lexer.TPlus:
+		p.lexer.Next()
+		value := p.parseExpr(js_ast.LPrefix)
+		if p.lexer.Token == js_lexer.TAsteriskAsterisk {
+			p.lexer.Unexpected()
+		}
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EUnary{Op: js_ast.UnOpPos, Value: value}}
+
+	case js_lexer.TMinus:
+		p.lexer.Next()
+		value := p.parseExpr(js_ast.LPrefix)
+		if p.lexer.Token == js_lexer.TAsteriskAsterisk {
+			p.lexer.Unexpected()
+		}
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EUnary{Op: js_ast.UnOpNeg, Value: value}}
+
+	case js_lexer.TTilde:
+		p.lexer.Next()
+		value := p.parseExpr(js_ast.LPrefix)
+		if p.lexer.Token == js_lexer.TAsteriskAsterisk {
+			p.lexer.Unexpected()
+		}
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EUnary{Op: js_ast.UnOpCpl, Value: value}}
+
+	case js_lexer.TExclamation:
+		p.lexer.Next()
+		value := p.parseExpr(js_ast.LPrefix)
+		if p.lexer.Token == js_lexer.TAsteriskAsterisk {
+			p.lexer.Unexpected()
+		}
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EUnary{Op: js_ast.UnOpNot, Value: value}}
+
+	case js_lexer.TMinusMinus:
+		p.lexer.Next()
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EUnary{Op: js_ast.UnOpPreDec, Value: p.parseExpr(js_ast.LPrefix)}}
+
+	case js_lexer.TPlusPlus:
+		p.lexer.Next()
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EUnary{Op: js_ast.UnOpPreInc, Value: p.parseExpr(js_ast.LPrefix)}}
+
+	case js_lexer.TFunction:
+		return p.parseFnExpr(loc, false /* isAsync */, logger.Range{})
+
+	case js_lexer.TClass:
+		return p.parseClassExpr(nil)
+
+	case js_lexer.TAt:
+		// Parse decorators before class expressions
+		decorators := p.parseDecorators(p.currentScope, logger.Range{}, decoratorBeforeClassExpr)
+		return p.parseClassExpr(decorators)
+
+	case js_lexer.TNew:
+		p.lexer.Next()
+
+		// Special-case the weird "new.target" expression here
+		if p.lexer.Token == js_lexer.TDot {
+			p.lexer.Next()
+			if p.lexer.Token != js_lexer.TIdentifier || p.lexer.Raw() != "target" {
+				p.lexer.Unexpected()
+			}
+			r := logger.Range{Loc: loc, Len: p.lexer.Range().End() - loc.Start}
+			p.markSyntaxFeature(compat.NewTarget, r)
+			p.lexer.Next()
+			return js_ast.Expr{Loc: loc, Data: &js_ast.ENewTarget{Range: r}}
+		}
+
+		target := p.parseExprWithFlags(js_ast.LMember, flags)
+		args := []js_ast.Expr{}
+		var closeParenLoc logger.Loc
+		var isMultiLine bool
+
+		if p.lexer.Token == js_lexer.TOpenParen {
+			args, closeParenLoc, isMultiLine = p.parseCallArgs()
+		}
+
+		return js_ast.Expr{Loc: loc, Data: &js_ast.ENew{
+			Target:        target,
+			Args:          args,
+			CloseParenLoc: closeParenLoc,
+			IsMultiLine:   isMultiLine,
+		}}
+
+	case js_lexer.TOpenBracket:
+		p.lexer.Next()
+		isSingleLine := !p.lexer.HasNewlineBefore
+		items := []js_ast.Expr{}
+		selfErrors := deferredErrors{}
+		commaAfterSpread := logger.Loc{}
+
+		// Allow "in" inside arrays
+		oldAllowIn := p.allowIn
+		p.allowIn = true
+
+		for p.lexer.Token != js_lexer.TCloseBracket {
+			switch p.lexer.Token {
+			case js_lexer.TComma:
+				items = append(items, js_ast.Expr{Loc: p.lexer.Loc(), Data: js_ast.EMissingShared})
+
+			case js_lexer.TDotDotDot:
+				if errors != nil {
+					errors.arraySpreadFeature = p.lexer.Range()
+				} else {
+					p.markSyntaxFeature(compat.ArraySpread, p.lexer.Range())
+				}
+				dotsLoc := p.saveExprCommentsHere()
+				p.lexer.Next()
+				item := p.parseExprOrBindings(js_ast.LComma, &selfErrors)
+				items = append(items, js_ast.Expr{Loc: dotsLoc, Data: &js_ast.ESpread{Value: item}})
+
+				// Commas are not allowed here when destructuring
+				if p.lexer.Token == js_lexer.TComma {
+					commaAfterSpread = p.lexer.Loc()
+				}
+
+			default:
+				item := p.parseExprOrBindings(js_ast.LComma, &selfErrors)
+				items = append(items, item)
+			}
+
+			if p.lexer.Token != js_lexer.TComma {
+				break
+			}
+			if p.lexer.HasNewlineBefore {
+				isSingleLine = false
+			}
+			p.lexer.Next()
+			if p.lexer.HasNewlineBefore {
+				isSingleLine = false
+			}
+		}
+
+		if p.lexer.HasNewlineBefore {
+			isSingleLine = false
+		}
+		closeBracketLoc := p.saveExprCommentsHere()
+		p.lexer.Expect(js_lexer.TCloseBracket)
+		p.allowIn = oldAllowIn
+
+		if p.willNeedBindingPattern() {
+			// Is this a binding pattern?
+		} else if errors == nil {
+			// Is this an expression?
+			p.logExprErrors(&selfErrors)
+		} else {
+			// In this case, we can't distinguish between the two yet
+			selfErrors.mergeInto(errors)
+		}
+
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EArray{
+			Items:            items,
+			CommaAfterSpread: commaAfterSpread,
+			IsSingleLine:     isSingleLine,
+			CloseBracketLoc:  closeBracketLoc,
+		}}
+
+	case js_lexer.TOpenBrace:
+		p.lexer.Next()
+		isSingleLine := !p.lexer.HasNewlineBefore
+		properties := []js_ast.Property{}
+		selfErrors := deferredErrors{}
+		commaAfterSpread := logger.Loc{}
+
+		// Allow "in" inside object literals
+		oldAllowIn := p.allowIn
+		p.allowIn = true
+
+		for p.lexer.Token != js_lexer.TCloseBrace {
+			if p.lexer.Token == js_lexer.TDotDotDot {
+				dotLoc := p.saveExprCommentsHere()
+				p.lexer.Next()
+				value := p.parseExprOrBindings(js_ast.LComma, &selfErrors)
+				properties = append(properties, js_ast.Property{
+					Kind:       js_ast.PropertySpread,
+					Loc:        dotLoc,
+					ValueOrNil: value,
+				})
+
+				// Commas are not allowed here when destructuring
+				if p.lexer.Token == js_lexer.TComma {
+					commaAfterSpread = p.lexer.Loc()
+				}
+			} else {
+				// This property may turn out to be a type in TypeScript, which should be ignored
+				if property, ok := p.parseProperty(p.saveExprCommentsHere(), js_ast.PropertyField, propertyOpts{}, &selfErrors); ok {
+					properties = append(properties, property)
+				}
+			}
+
+			if p.lexer.Token != js_lexer.TComma {
+				break
+			}
+			if p.lexer.HasNewlineBefore {
+				isSingleLine = false
+			}
+			p.lexer.Next()
+			if p.lexer.HasNewlineBefore {
+				isSingleLine = false
+			}
+		}
+
+		if p.lexer.HasNewlineBefore {
+			isSingleLine = false
+		}
+		closeBraceLoc := p.saveExprCommentsHere()
+		p.lexer.Expect(js_lexer.TCloseBrace)
+		p.allowIn = oldAllowIn
+
+		if p.willNeedBindingPattern() {
+			// Is this a binding pattern?
+		} else if errors == nil {
+			// Is this an expression?
+			p.logExprErrors(&selfErrors)
+		} else {
+			// In this case, we can't distinguish between the two yet
+			selfErrors.mergeInto(errors)
+		}
+
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EObject{
+			Properties:       properties,
+			CommaAfterSpread: commaAfterSpread,
+			IsSingleLine:     isSingleLine,
+			CloseBraceLoc:    closeBraceLoc,
+		}}
+
+	case js_lexer.TLessThan:
+		// This is a very complicated and highly ambiguous area of TypeScript
+		// syntax. Many similar-looking things are overloaded.
+		//
+		// TS:
+		//
+		//   A type cast:
+		//     <A>(x)
+		//     <[]>(x)
+		//     <A[]>(x)
+		//     <const>(x)
+		//
+		//   An arrow function with type parameters:
+		//     <A>(x) => {}
+		//     <A, B>(x) => {}
+		//     <A = B>(x) => {}
+		//     <A extends B>(x) => {}
+		//     <const A>(x) => {}
+		//     <const A extends B>(x) => {}
+		//
+		//   A syntax error:
+		//     <>() => {}
+		//
+		// TSX:
+		//
+		//   A JSX element:
+		//     <>() => {}</>
+		//     <A>(x) => {}</A>
+		//     <A extends/>
+		//     <A extends>(x) => {}</A>
+		//     <A extends={false}>(x) => {}</A>
+		//     <const A extends/>
+		//     <const A extends>(x) => {}</const>
+		//
+		//   An arrow function with type parameters:
+		//     <A,>(x) => {}
+		//     <A, B>(x) => {}
+		//     <A = B>(x) => {}
+		//     <A extends B>(x) => {}
+		//     <const>(x)</const>
+		//     <const A extends B>(x) => {}
+		//
+		//   A syntax error:
+		//     <[]>(x)
+		//     <A[]>(x)
+		//     <>() => {}
+		//     <A>(x) => {}
+
+		if p.options.ts.Parse && p.options.jsx.Parse && p.isTSArrowFnJSX() {
+			p.skipTypeScriptTypeParameters(allowConstModifier)
+			p.lexer.Expect(js_lexer.TOpenParen)
+			return p.parseParenExpr(loc, level, parenExprOpts{forceArrowFn: true})
+		}
+
+		// Print a friendly error message when parsing JSX as JavaScript
+		if !p.options.jsx.Parse && !p.options.ts.Parse {
+			var how string
+			switch logger.API {
+			case logger.CLIAPI:
+				how = " You can use \"--loader:.js=jsx\" to do that."
+			case logger.JSAPI:
+				how = " You can use \"loader: { '.js': 'jsx' }\" to do that."
+			case logger.GoAPI:
+				how = " You can use 'Loader: map[string]api.Loader{\".js\": api.LoaderJSX}' to do that."
+			}
+			p.log.AddErrorWithNotes(&p.tracker, p.lexer.Range(), "The JSX syntax extension is not currently enabled", []logger.MsgData{{
+				Text: "The esbuild loader for this file is currently set to \"js\" but it must be set to \"jsx\" to be able to parse JSX syntax." + how}})
+			p.options.jsx.Parse = true
+		}
+
+		if p.options.jsx.Parse {
+			// Use NextInsideJSXElement() instead of Next() so we parse "<<" as "<"
+			p.lexer.NextInsideJSXElement()
+			element := p.parseJSXElement(loc)
+
+			// The call to parseJSXElement() above doesn't consume the last
+			// TGreaterThan because the caller knows what Next() function to call.
+			// Use Next() instead of NextInsideJSXElement() here since the next
+			// token is an expression.
+			p.lexer.Next()
+			return element
+		}
+
+		if p.options.ts.Parse {
+			// This is either an old-style type cast or a generic lambda function
+
+			// TypeScript 4.5 introduced the ".mts" and ".cts" extensions that forbid
+			// the use of an expression starting with "<" that would be ambiguous
+			// when the file is in JSX mode.
+			if p.options.ts.NoAmbiguousLessThan && !p.isTSArrowFnJSX() {
+				p.log.AddError(&p.tracker, p.lexer.Range(),
+					"This syntax is not allowed in files with the \".mts\" or \".cts\" extension")
+			}
+
+			// "<T>(x)"
+			// "<T>(x) => {}"
+			if result := p.trySkipTypeScriptTypeParametersThenOpenParenWithBacktracking(); result != didNotSkipAnything {
+				p.lexer.Expect(js_lexer.TOpenParen)
+				return p.parseParenExpr(loc, level, parenExprOpts{
+					forceArrowFn: result == definitelyTypeParameters,
+				})
+			}
+
+			// "<T>x"
+			p.lexer.Next()
+			p.skipTypeScriptType(js_ast.LLowest)
+			p.lexer.ExpectGreaterThan(false /* isInsideJSXElement */)
+			value := p.parsePrefix(level, errors, flags)
+			return value
+		}
+
+		p.lexer.Unexpected()
+		return js_ast.Expr{}
+
+	case js_lexer.TImport:
+		p.lexer.Next()
+		return p.parseImportExpr(loc, level)
+
+	default:
+		p.lexer.Unexpected()
+		return js_ast.Expr{}
+	}
+}
+
+func (p *parser) parseYieldExpr(loc logger.Loc) js_ast.Expr {
+	// Parse a yield-from expression, which yields from an iterator
+	isStar := p.lexer.Token == js_lexer.TAsterisk
+	if isStar && !p.lexer.HasNewlineBefore {
+		p.lexer.Next()
+	}
+
+	var valueOrNil js_ast.Expr
+
+	// The yield expression only has a value in certain cases
+	if isStar {
+		valueOrNil = p.parseExpr(js_ast.LYield)
+	} else {
+		switch p.lexer.Token {
+		case js_lexer.TCloseBrace, js_lexer.TCloseBracket, js_lexer.TCloseParen,
+			js_lexer.TColon, js_lexer.TComma, js_lexer.TSemicolon:
+
+		default:
+			if !p.lexer.HasNewlineBefore {
+				valueOrNil = p.parseExpr(js_ast.LYield)
+			}
+		}
+	}
+
+	return js_ast.Expr{Loc: loc, Data: &js_ast.EYield{ValueOrNil: valueOrNil, IsStar: isStar}}
+}
+
+func (p *parser) willNeedBindingPattern() bool {
+	switch p.lexer.Token {
+	case js_lexer.TEquals:
+		// "[a] = b;"
+		return true
+
+	case js_lexer.TIn:
+		// "for ([a] in b) {}"
+		return !p.allowIn
+
+	case js_lexer.TIdentifier:
+		// "for ([a] of b) {}"
+		return !p.allowIn && p.lexer.IsContextualKeyword("of")
+
+	default:
+		return false
+	}
+}
+
+// Note: The caller has already parsed the "import" keyword
+func (p *parser) parseImportExpr(loc logger.Loc, level js_ast.L) js_ast.Expr {
+	// Parse an "import.meta" expression
+	if p.lexer.Token == js_lexer.TDot {
+		p.lexer.Next()
+		if !p.lexer.IsContextualKeyword("meta") {
+			p.lexer.ExpectedString("\"meta\"")
+		}
+		p.esmImportMeta = logger.Range{Loc: loc, Len: p.lexer.Range().End() - loc.Start}
+		p.lexer.Next()
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EImportMeta{RangeLen: p.esmImportMeta.Len}}
+	}
+
+	if level > js_ast.LCall {
+		r := js_lexer.RangeOfIdentifier(p.source, loc)
+		p.log.AddError(&p.tracker, r, "Cannot use an \"import\" expression here without parentheses:")
+	}
+
+	// Allow "in" inside call arguments
+	oldAllowIn := p.allowIn
+	p.allowIn = true
+
+	p.lexer.Expect(js_lexer.TOpenParen)
+
+	value := p.parseExpr(js_ast.LComma)
+	var optionsOrNil js_ast.Expr
+
+	if p.lexer.Token == js_lexer.TComma {
+		// "import('./foo.json', )"
+		p.lexer.Next()
+
+		if p.lexer.Token != js_lexer.TCloseParen {
+			// "import('./foo.json', { assert: { type: 'json' } })"
+			optionsOrNil = p.parseExpr(js_ast.LComma)
+
+			if p.lexer.Token == js_lexer.TComma {
+				// "import('./foo.json', { assert: { type: 'json' } }, )"
+				p.lexer.Next()
+			}
+		}
+	}
+
+	closeParenLoc := p.saveExprCommentsHere()
+	p.lexer.Expect(js_lexer.TCloseParen)
+
+	p.allowIn = oldAllowIn
+	return js_ast.Expr{Loc: loc, Data: &js_ast.EImportCall{
+		Expr:          value,
+		OptionsOrNil:  optionsOrNil,
+		CloseParenLoc: closeParenLoc,
+	}}
+}
+
+func (p *parser) parseExprOrBindings(level js_ast.L, errors *deferredErrors) js_ast.Expr {
+	return p.parseExprCommon(level, errors, 0)
+}
+
+func (p *parser) parseExpr(level js_ast.L) js_ast.Expr {
+	return p.parseExprCommon(level, nil, 0)
+}
+
+func (p *parser) parseExprWithFlags(level js_ast.L, flags exprFlag) js_ast.Expr {
+	return p.parseExprCommon(level, nil, flags)
+}
+
+func (p *parser) parseExprCommon(level js_ast.L, errors *deferredErrors, flags exprFlag) js_ast.Expr {
+	lexerCommentFlags := p.lexer.HasCommentBefore
+	expr := p.parsePrefix(level, errors, flags)
+
+	if (lexerCommentFlags&(js_lexer.PureCommentBefore|js_lexer.NoSideEffectsCommentBefore)) != 0 && !p.options.ignoreDCEAnnotations {
+		if (lexerCommentFlags & js_lexer.NoSideEffectsCommentBefore) != 0 {
+			switch e := expr.Data.(type) {
+			case *js_ast.EArrow:
+				e.HasNoSideEffectsComment = true
+			case *js_ast.EFunction:
+				e.Fn.HasNoSideEffectsComment = true
+			}
+		}
+
+		// There is no formal spec for "__PURE__" comments but from reverse-
+		// engineering, it looks like they apply to the next CallExpression or
+		// NewExpression. So in "/* @__PURE__ */ a().b() + c()" the comment applies
+		// to the expression "a().b()".
+		if (lexerCommentFlags&js_lexer.PureCommentBefore) != 0 && level < js_ast.LCall {
+			expr = p.parseSuffix(expr, js_ast.LCall-1, errors, flags)
+			switch e := expr.Data.(type) {
+			case *js_ast.ECall:
+				e.CanBeUnwrappedIfUnused = true
+			case *js_ast.ENew:
+				e.CanBeUnwrappedIfUnused = true
+			}
+		}
+	}
+
+	return p.parseSuffix(expr, level, errors, flags)
+}
+
+func (p *parser) parseSuffix(left js_ast.Expr, level js_ast.L, errors *deferredErrors, flags exprFlag) js_ast.Expr {
+	optionalChain := js_ast.OptionalChainNone
+
+	for {
+		if p.lexer.Loc() == p.afterArrowBodyLoc {
+			for {
+				switch p.lexer.Token {
+				case js_lexer.TComma:
+					if level >= js_ast.LComma {
+						return left
+					}
+					p.lexer.Next()
+					left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpComma, Left: left, Right: p.parseExpr(js_ast.LComma)}}
+
+				default:
+					return left
+				}
+			}
+		}
+
+		// Stop now if this token is forbidden to follow a TypeScript "as" cast
+		if p.lexer.Loc() == p.forbidSuffixAfterAsLoc {
+			return left
+		}
+
+		// Reset the optional chain flag by default. That way we won't accidentally
+		// treat "c.d" as OptionalChainContinue in "a?.b + c.d".
+		oldOptionalChain := optionalChain
+		optionalChain = js_ast.OptionalChainNone
+
+		switch p.lexer.Token {
+		case js_lexer.TDot:
+			p.lexer.Next()
+
+			if p.lexer.Token == js_lexer.TPrivateIdentifier && p.allowPrivateIdentifiers {
+				// "a.#b"
+				// "a?.b.#c"
+				if _, ok := left.Data.(*js_ast.ESuper); ok {
+					p.lexer.Expected(js_lexer.TIdentifier)
+				}
+				name := p.lexer.Identifier
+				nameLoc := p.lexer.Loc()
+				p.reportPrivateNameUsage(name.String)
+				p.lexer.Next()
+				ref := p.storeNameInRef(name)
+				left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EIndex{
+					Target:        left,
+					Index:         js_ast.Expr{Loc: nameLoc, Data: &js_ast.EPrivateIdentifier{Ref: ref}},
+					OptionalChain: oldOptionalChain,
+				}}
+			} else {
+				// "a.b"
+				// "a?.b.c"
+				if !p.lexer.IsIdentifierOrKeyword() {
+					p.lexer.Expect(js_lexer.TIdentifier)
+				}
+				name := p.lexer.Identifier
+				nameLoc := p.lexer.Loc()
+				p.lexer.Next()
+				left = js_ast.Expr{Loc: left.Loc, Data: p.dotOrMangledPropParse(left, name, nameLoc, oldOptionalChain, wasOriginallyDot)}
+			}
+
+			optionalChain = oldOptionalChain
+
+		case js_lexer.TQuestionDot:
+			p.lexer.Next()
+			optionalStart := js_ast.OptionalChainStart
+
+			// Remove unnecessary optional chains
+			if p.options.minifySyntax {
+				if isNullOrUndefined, _, ok := js_ast.ToNullOrUndefinedWithSideEffects(left.Data); ok && !isNullOrUndefined {
+					optionalStart = js_ast.OptionalChainNone
+				}
+			}
+
+			switch p.lexer.Token {
+			case js_lexer.TOpenBracket:
+				// "a?.[b]"
+				p.lexer.Next()
+
+				// Allow "in" inside the brackets
+				oldAllowIn := p.allowIn
+				p.allowIn = true
+
+				index := p.parseExpr(js_ast.LLowest)
+
+				p.allowIn = oldAllowIn
+
+				closeBracketLoc := p.saveExprCommentsHere()
+				p.lexer.Expect(js_lexer.TCloseBracket)
+				left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EIndex{
+					Target:          left,
+					Index:           index,
+					OptionalChain:   optionalStart,
+					CloseBracketLoc: closeBracketLoc,
+				}}
+
+			case js_lexer.TOpenParen:
+				// "a?.()"
+				if level >= js_ast.LCall {
+					return left
+				}
+				kind := js_ast.NormalCall
+				if js_ast.IsPropertyAccess(left) {
+					kind = js_ast.TargetWasOriginallyPropertyAccess
+				}
+				args, closeParenLoc, isMultiLine := p.parseCallArgs()
+				left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.ECall{
+					Target:        left,
+					Args:          args,
+					CloseParenLoc: closeParenLoc,
+					OptionalChain: optionalStart,
+					IsMultiLine:   isMultiLine,
+					Kind:          kind,
+				}}
+
+			case js_lexer.TLessThan, js_lexer.TLessThanLessThan:
+				// "a?.<T>()"
+				// "a?.<<T>() => T>()"
+				if !p.options.ts.Parse {
+					p.lexer.Expected(js_lexer.TIdentifier)
+				}
+				p.skipTypeScriptTypeArguments(skipTypeScriptTypeArgumentsOpts{})
+				if p.lexer.Token != js_lexer.TOpenParen {
+					p.lexer.Expected(js_lexer.TOpenParen)
+				}
+				if level >= js_ast.LCall {
+					return left
+				}
+				kind := js_ast.NormalCall
+				if js_ast.IsPropertyAccess(left) {
+					kind = js_ast.TargetWasOriginallyPropertyAccess
+				}
+				args, closeParenLoc, isMultiLine := p.parseCallArgs()
+				left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.ECall{
+					Target:        left,
+					Args:          args,
+					CloseParenLoc: closeParenLoc,
+					OptionalChain: optionalStart,
+					IsMultiLine:   isMultiLine,
+					Kind:          kind,
+				}}
+
+			default:
+				if p.lexer.Token == js_lexer.TPrivateIdentifier && p.allowPrivateIdentifiers {
+					// "a?.#b"
+					name := p.lexer.Identifier
+					nameLoc := p.lexer.Loc()
+					p.reportPrivateNameUsage(name.String)
+					p.lexer.Next()
+					ref := p.storeNameInRef(name)
+					left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EIndex{
+						Target:        left,
+						Index:         js_ast.Expr{Loc: nameLoc, Data: &js_ast.EPrivateIdentifier{Ref: ref}},
+						OptionalChain: optionalStart,
+					}}
+				} else {
+					// "a?.b"
+					if !p.lexer.IsIdentifierOrKeyword() {
+						p.lexer.Expect(js_lexer.TIdentifier)
+					}
+					name := p.lexer.Identifier
+					nameLoc := p.lexer.Loc()
+					p.lexer.Next()
+					left = js_ast.Expr{Loc: left.Loc, Data: p.dotOrMangledPropParse(left, name, nameLoc, optionalStart, wasOriginallyDot)}
+				}
+			}
+
+			// Only continue if we have started
+			if optionalStart == js_ast.OptionalChainStart {
+				optionalChain = js_ast.OptionalChainContinue
+			}
+
+		case js_lexer.TNoSubstitutionTemplateLiteral:
+			if oldOptionalChain != js_ast.OptionalChainNone {
+				p.log.AddError(&p.tracker, p.lexer.Range(), "Template literals cannot have an optional chain as a tag")
+			}
+			headLoc := p.lexer.Loc()
+			headCooked, headRaw := p.lexer.CookedAndRawTemplateContents()
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.ETemplate{
+				TagOrNil:                       left,
+				HeadLoc:                        headLoc,
+				HeadCooked:                     headCooked,
+				HeadRaw:                        headRaw,
+				TagWasOriginallyPropertyAccess: js_ast.IsPropertyAccess(left),
+			}}
+
+		case js_lexer.TTemplateHead:
+			if oldOptionalChain != js_ast.OptionalChainNone {
+				p.log.AddError(&p.tracker, p.lexer.Range(), "Template literals cannot have an optional chain as a tag")
+			}
+			headLoc := p.lexer.Loc()
+			headCooked, headRaw := p.lexer.CookedAndRawTemplateContents()
+			parts, _ := p.parseTemplateParts(true /* includeRaw */)
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.ETemplate{
+				TagOrNil:                       left,
+				HeadLoc:                        headLoc,
+				HeadCooked:                     headCooked,
+				HeadRaw:                        headRaw,
+				Parts:                          parts,
+				TagWasOriginallyPropertyAccess: js_ast.IsPropertyAccess(left),
+			}}
+
+		case js_lexer.TOpenBracket:
+			// When parsing a decorator, ignore EIndex expressions since they may be
+			// part of a computed property:
+			//
+			//   class Foo {
+			//     @foo ['computed']() {}
+			//   }
+			//
+			// This matches the behavior of the TypeScript compiler.
+			if (flags & exprFlagDecorator) != 0 {
+				return left
+			}
+
+			p.lexer.Next()
+
+			// Allow "in" inside the brackets
+			oldAllowIn := p.allowIn
+			p.allowIn = true
+
+			index := p.parseExpr(js_ast.LLowest)
+
+			p.allowIn = oldAllowIn
+
+			closeBracketLoc := p.saveExprCommentsHere()
+			p.lexer.Expect(js_lexer.TCloseBracket)
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EIndex{
+				Target:          left,
+				Index:           index,
+				OptionalChain:   oldOptionalChain,
+				CloseBracketLoc: closeBracketLoc,
+			}}
+			optionalChain = oldOptionalChain
+
+		case js_lexer.TOpenParen:
+			if level >= js_ast.LCall {
+				return left
+			}
+			kind := js_ast.NormalCall
+			if js_ast.IsPropertyAccess(left) {
+				kind = js_ast.TargetWasOriginallyPropertyAccess
+			}
+			args, closeParenLoc, isMultiLine := p.parseCallArgs()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.ECall{
+				Target:        left,
+				Args:          args,
+				CloseParenLoc: closeParenLoc,
+				OptionalChain: oldOptionalChain,
+				IsMultiLine:   isMultiLine,
+				Kind:          kind,
+			}}
+			optionalChain = oldOptionalChain
+
+		case js_lexer.TQuestion:
+			if level >= js_ast.LConditional {
+				return left
+			}
+			p.lexer.Next()
+
+			// Stop now if we're parsing one of these:
+			// "(a?) => {}"
+			// "(a?: b) => {}"
+			// "(a?, b?) => {}"
+			if p.options.ts.Parse && left.Loc == p.latestArrowArgLoc && (p.lexer.Token == js_lexer.TColon ||
+				p.lexer.Token == js_lexer.TCloseParen || p.lexer.Token == js_lexer.TComma) {
+				if errors == nil {
+					p.lexer.Unexpected()
+				}
+				errors.invalidExprAfterQuestion = p.lexer.Range()
+				return left
+			}
+
+			// Allow "in" in between "?" and ":"
+			oldAllowIn := p.allowIn
+			p.allowIn = true
+
+			yes := p.parseExpr(js_ast.LComma)
+
+			p.allowIn = oldAllowIn
+
+			p.lexer.Expect(js_lexer.TColon)
+			no := p.parseExpr(js_ast.LComma)
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EIf{Test: left, Yes: yes, No: no}}
+
+		case js_lexer.TExclamation:
+			// Skip over TypeScript non-null assertions
+			if p.lexer.HasNewlineBefore {
+				return left
+			}
+			if !p.options.ts.Parse {
+				p.lexer.Unexpected()
+			}
+			p.lexer.Next()
+			optionalChain = oldOptionalChain
+
+		case js_lexer.TMinusMinus:
+			if p.lexer.HasNewlineBefore || level >= js_ast.LPostfix {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EUnary{Op: js_ast.UnOpPostDec, Value: left}}
+
+		case js_lexer.TPlusPlus:
+			if p.lexer.HasNewlineBefore || level >= js_ast.LPostfix {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EUnary{Op: js_ast.UnOpPostInc, Value: left}}
+
+		case js_lexer.TComma:
+			if level >= js_ast.LComma {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpComma, Left: left, Right: p.parseExpr(js_ast.LComma)}}
+
+		case js_lexer.TPlus:
+			if level >= js_ast.LAdd {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpAdd, Left: left, Right: p.parseExpr(js_ast.LAdd)}}
+
+		case js_lexer.TPlusEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpAddAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TMinus:
+			if level >= js_ast.LAdd {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpSub, Left: left, Right: p.parseExpr(js_ast.LAdd)}}
+
+		case js_lexer.TMinusEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpSubAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TAsterisk:
+			if level >= js_ast.LMultiply {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpMul, Left: left, Right: p.parseExpr(js_ast.LMultiply)}}
+
+		case js_lexer.TAsteriskAsterisk:
+			if level >= js_ast.LExponentiation {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpPow, Left: left, Right: p.parseExpr(js_ast.LExponentiation - 1)}}
+
+		case js_lexer.TAsteriskAsteriskEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpPowAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TAsteriskEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpMulAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TPercent:
+			if level >= js_ast.LMultiply {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpRem, Left: left, Right: p.parseExpr(js_ast.LMultiply)}}
+
+		case js_lexer.TPercentEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpRemAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TSlash:
+			if level >= js_ast.LMultiply {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpDiv, Left: left, Right: p.parseExpr(js_ast.LMultiply)}}
+
+		case js_lexer.TSlashEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpDivAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TEqualsEquals:
+			if level >= js_ast.LEquals {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpLooseEq, Left: left, Right: p.parseExpr(js_ast.LEquals)}}
+
+		case js_lexer.TExclamationEquals:
+			if level >= js_ast.LEquals {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpLooseNe, Left: left, Right: p.parseExpr(js_ast.LEquals)}}
+
+		case js_lexer.TEqualsEqualsEquals:
+			if level >= js_ast.LEquals {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpStrictEq, Left: left, Right: p.parseExpr(js_ast.LEquals)}}
+
+		case js_lexer.TExclamationEqualsEquals:
+			if level >= js_ast.LEquals {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpStrictNe, Left: left, Right: p.parseExpr(js_ast.LEquals)}}
+
+		case js_lexer.TLessThan:
+			// TypeScript allows type arguments to be specified with angle brackets
+			// inside an expression. Unlike in other languages, this unfortunately
+			// appears to require backtracking to parse.
+			if p.options.ts.Parse && p.trySkipTypeArgumentsInExpressionWithBacktracking() {
+				optionalChain = oldOptionalChain
+				continue
+			}
+
+			if level >= js_ast.LCompare {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpLt, Left: left, Right: p.parseExpr(js_ast.LCompare)}}
+
+		case js_lexer.TLessThanEquals:
+			if level >= js_ast.LCompare {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpLe, Left: left, Right: p.parseExpr(js_ast.LCompare)}}
+
+		case js_lexer.TGreaterThan:
+			if level >= js_ast.LCompare {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpGt, Left: left, Right: p.parseExpr(js_ast.LCompare)}}
+
+		case js_lexer.TGreaterThanEquals:
+			if level >= js_ast.LCompare {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpGe, Left: left, Right: p.parseExpr(js_ast.LCompare)}}
+
+		case js_lexer.TLessThanLessThan:
+			// TypeScript allows type arguments to be specified with angle brackets
+			// inside an expression. Unlike in other languages, this unfortunately
+			// appears to require backtracking to parse.
+			if p.options.ts.Parse && p.trySkipTypeArgumentsInExpressionWithBacktracking() {
+				optionalChain = oldOptionalChain
+				continue
+			}
+
+			if level >= js_ast.LShift {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpShl, Left: left, Right: p.parseExpr(js_ast.LShift)}}
+
+		case js_lexer.TLessThanLessThanEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpShlAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TGreaterThanGreaterThan:
+			if level >= js_ast.LShift {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpShr, Left: left, Right: p.parseExpr(js_ast.LShift)}}
+
+		case js_lexer.TGreaterThanGreaterThanEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpShrAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TGreaterThanGreaterThanGreaterThan:
+			if level >= js_ast.LShift {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpUShr, Left: left, Right: p.parseExpr(js_ast.LShift)}}
+
+		case js_lexer.TGreaterThanGreaterThanGreaterThanEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpUShrAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TQuestionQuestion:
+			if level >= js_ast.LNullishCoalescing {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpNullishCoalescing, Left: left, Right: p.parseExpr(js_ast.LNullishCoalescing)}}
+
+		case js_lexer.TQuestionQuestionEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpNullishCoalescingAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TBarBar:
+			if level >= js_ast.LLogicalOr {
+				return left
+			}
+
+			// Prevent "||" inside "??" from the right
+			if level == js_ast.LNullishCoalescing {
+				p.logNullishCoalescingErrorPrecedenceError("||")
+			}
+
+			p.lexer.Next()
+			right := p.parseExpr(js_ast.LLogicalOr)
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpLogicalOr, Left: left, Right: right}}
+
+			// Prevent "||" inside "??" from the left
+			if level < js_ast.LNullishCoalescing {
+				left = p.parseSuffix(left, js_ast.LNullishCoalescing+1, nil, flags)
+				if p.lexer.Token == js_lexer.TQuestionQuestion {
+					p.logNullishCoalescingErrorPrecedenceError("||")
+				}
+			}
+
+		case js_lexer.TBarBarEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpLogicalOrAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TAmpersandAmpersand:
+			if level >= js_ast.LLogicalAnd {
+				return left
+			}
+
+			// Prevent "&&" inside "??" from the right
+			if level == js_ast.LNullishCoalescing {
+				p.logNullishCoalescingErrorPrecedenceError("&&")
+			}
+
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpLogicalAnd, Left: left, Right: p.parseExpr(js_ast.LLogicalAnd)}}
+
+			// Prevent "&&" inside "??" from the left
+			if level < js_ast.LNullishCoalescing {
+				left = p.parseSuffix(left, js_ast.LNullishCoalescing+1, nil, flags)
+				if p.lexer.Token == js_lexer.TQuestionQuestion {
+					p.logNullishCoalescingErrorPrecedenceError("&&")
+				}
+			}
+
+		case js_lexer.TAmpersandAmpersandEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpLogicalAndAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TBar:
+			if level >= js_ast.LBitwiseOr {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpBitwiseOr, Left: left, Right: p.parseExpr(js_ast.LBitwiseOr)}}
+
+		case js_lexer.TBarEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpBitwiseOrAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TAmpersand:
+			if level >= js_ast.LBitwiseAnd {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpBitwiseAnd, Left: left, Right: p.parseExpr(js_ast.LBitwiseAnd)}}
+
+		case js_lexer.TAmpersandEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpBitwiseAndAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TCaret:
+			if level >= js_ast.LBitwiseXor {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpBitwiseXor, Left: left, Right: p.parseExpr(js_ast.LBitwiseXor)}}
+
+		case js_lexer.TCaretEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpBitwiseXorAssign, Left: left, Right: p.parseExpr(js_ast.LAssign - 1)}}
+
+		case js_lexer.TEquals:
+			if level >= js_ast.LAssign {
+				return left
+			}
+			p.lexer.Next()
+			left = js_ast.Assign(left, p.parseExpr(js_ast.LAssign-1))
+
+		case js_lexer.TIn:
+			if level >= js_ast.LCompare || !p.allowIn {
+				return left
+			}
+
+			// Warn about "!a in b" instead of "!(a in b)"
+			kind := logger.Warning
+			if p.suppressWarningsAboutWeirdCode {
+				kind = logger.Debug
+			}
+			if e, ok := left.Data.(*js_ast.EUnary); ok && e.Op == js_ast.UnOpNot {
+				r := logger.Range{Loc: left.Loc, Len: p.source.LocBeforeWhitespace(p.lexer.Loc()).Start - left.Loc.Start}
+				data := p.tracker.MsgData(r, "Suspicious use of the \"!\" operator inside the \"in\" operator")
+				data.Location.Suggestion = fmt.Sprintf("(%s)", p.source.TextForRange(r))
+				p.log.AddMsgID(logger.MsgID_JS_SuspiciousBooleanNot, logger.Msg{
+					Kind: kind,
+					Data: data,
+					Notes: []logger.MsgData{{Text: "The code \"!x in y\" is parsed as \"(!x) in y\". " +
+						"You need to insert parentheses to get \"!(x in y)\" instead."}},
+				})
+			}
+
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpIn, Left: left, Right: p.parseExpr(js_ast.LCompare)}}
+
+		case js_lexer.TInstanceof:
+			if level >= js_ast.LCompare {
+				return left
+			}
+
+			// Warn about "!a instanceof b" instead of "!(a instanceof b)". Here's an
+			// example of code with this problem: https://github.com/mrdoob/three.js/pull/11182.
+			kind := logger.Warning
+			if p.suppressWarningsAboutWeirdCode {
+				kind = logger.Debug
+			}
+			if e, ok := left.Data.(*js_ast.EUnary); ok && e.Op == js_ast.UnOpNot {
+				r := logger.Range{Loc: left.Loc, Len: p.source.LocBeforeWhitespace(p.lexer.Loc()).Start - left.Loc.Start}
+				data := p.tracker.MsgData(r, "Suspicious use of the \"!\" operator inside the \"instanceof\" operator")
+				data.Location.Suggestion = fmt.Sprintf("(%s)", p.source.TextForRange(r))
+				p.log.AddMsgID(logger.MsgID_JS_SuspiciousBooleanNot, logger.Msg{
+					Kind: kind,
+					Data: data,
+					Notes: []logger.MsgData{{Text: "The code \"!x instanceof y\" is parsed as \"(!x) instanceof y\". " +
+						"You need to insert parentheses to get \"!(x instanceof y)\" instead."}},
+				})
+			}
+
+			p.lexer.Next()
+			left = js_ast.Expr{Loc: left.Loc, Data: &js_ast.EBinary{Op: js_ast.BinOpInstanceof, Left: left, Right: p.parseExpr(js_ast.LCompare)}}
+
+		default:
+			// Handle the TypeScript "as"/"satisfies" operator
+			if p.options.ts.Parse && level < js_ast.LCompare && !p.lexer.HasNewlineBefore && (p.lexer.IsContextualKeyword("as") || p.lexer.IsContextualKeyword("satisfies")) {
+				p.lexer.Next()
+				p.skipTypeScriptType(js_ast.LLowest)
+
+				// These tokens are not allowed to follow a cast expression. This isn't
+				// an outright error because it may be on a new line, in which case it's
+				// the start of a new expression when it's after a cast:
+				//
+				//   x = y as z
+				//   (something);
+				//
+				switch p.lexer.Token {
+				case js_lexer.TPlusPlus, js_lexer.TMinusMinus, js_lexer.TNoSubstitutionTemplateLiteral,
+					js_lexer.TTemplateHead, js_lexer.TOpenParen, js_lexer.TOpenBracket, js_lexer.TQuestionDot:
+					p.forbidSuffixAfterAsLoc = p.lexer.Loc()
+					return left
+				}
+				if p.lexer.Token.IsAssign() {
+					p.forbidSuffixAfterAsLoc = p.lexer.Loc()
+					return left
+				}
+				continue
+			}
+
+			return left
+		}
+	}
+}
+
+func (p *parser) parseExprOrLetOrUsingStmt(opts parseStmtOpts) (js_ast.Expr, js_ast.Stmt, []js_ast.Decl) {
+	couldBeLet := false
+	couldBeUsing := false
+	couldBeAwaitUsing := false
+	tokenRange := p.lexer.Range()
+
+	if p.lexer.Token == js_lexer.TIdentifier {
+		raw := p.lexer.Raw()
+		couldBeLet = raw == "let"
+		couldBeUsing = raw == "using"
+		couldBeAwaitUsing = raw == "await" && p.fnOrArrowDataParse.await == allowExpr
+	}
+
+	if !couldBeLet && !couldBeUsing && !couldBeAwaitUsing {
+		var flags exprFlag
+		if opts.isForLoopInit {
+			flags |= exprFlagForLoopInit
+		}
+		if opts.isForAwaitLoopInit {
+			flags |= exprFlagForAwaitLoopInit
+		}
+		return p.parseExprCommon(js_ast.LLowest, nil, flags), js_ast.Stmt{}, nil
+	}
+
+	name := p.lexer.Identifier
+	p.lexer.Next()
+
+	if couldBeLet {
+		isLet := opts.isExport
+		switch p.lexer.Token {
+		case js_lexer.TIdentifier, js_lexer.TOpenBracket, js_lexer.TOpenBrace:
+			if opts.lexicalDecl == lexicalDeclAllowAll || !p.lexer.HasNewlineBefore || p.lexer.Token == js_lexer.TOpenBracket {
+				isLet = true
+			}
+		}
+		if isLet {
+			// Handle a "let" declaration
+			if opts.lexicalDecl != lexicalDeclAllowAll {
+				p.forbidLexicalDecl(tokenRange.Loc)
+			}
+			p.markSyntaxFeature(compat.ConstAndLet, tokenRange)
+			decls := p.parseAndDeclareDecls(ast.SymbolOther, opts)
+			return js_ast.Expr{}, js_ast.Stmt{Loc: tokenRange.Loc, Data: &js_ast.SLocal{
+				Kind:     js_ast.LocalLet,
+				Decls:    decls,
+				IsExport: opts.isExport,
+			}}, decls
+		}
+	} else if couldBeUsing && p.lexer.Token == js_lexer.TIdentifier && !p.lexer.HasNewlineBefore && (!opts.isForLoopInit || p.lexer.Raw() != "of") {
+		// Handle a "using" declaration
+		if opts.lexicalDecl != lexicalDeclAllowAll {
+			p.forbidLexicalDecl(tokenRange.Loc)
+		}
+		opts.isUsingStmt = true
+		decls := p.parseAndDeclareDecls(ast.SymbolConst, opts)
+		if !opts.isForLoopInit {
+			p.requireInitializers(js_ast.LocalUsing, decls)
+		}
+		return js_ast.Expr{}, js_ast.Stmt{Loc: tokenRange.Loc, Data: &js_ast.SLocal{
+			Kind:     js_ast.LocalUsing,
+			Decls:    decls,
+			IsExport: opts.isExport,
+		}}, decls
+	} else if couldBeAwaitUsing {
+		// Handle an "await using" declaration
+		if p.fnOrArrowDataParse.isTopLevel {
+			p.topLevelAwaitKeyword = tokenRange
+		}
+		var value js_ast.Expr
+		if p.lexer.Token == js_lexer.TIdentifier && p.lexer.Raw() == "using" {
+			usingLoc := p.saveExprCommentsHere()
+			usingRange := p.lexer.Range()
+			p.lexer.Next()
+			if p.lexer.Token == js_lexer.TIdentifier && !p.lexer.HasNewlineBefore {
+				// It's an "await using" declaration if we get here
+				if opts.lexicalDecl != lexicalDeclAllowAll {
+					p.forbidLexicalDecl(usingRange.Loc)
+				}
+				opts.isUsingStmt = true
+				decls := p.parseAndDeclareDecls(ast.SymbolConst, opts)
+				if !opts.isForLoopInit {
+					p.requireInitializers(js_ast.LocalAwaitUsing, decls)
+				}
+				return js_ast.Expr{}, js_ast.Stmt{Loc: tokenRange.Loc, Data: &js_ast.SLocal{
+					Kind:     js_ast.LocalAwaitUsing,
+					Decls:    decls,
+					IsExport: opts.isExport,
+				}}, decls
+			}
+			value = js_ast.Expr{Loc: usingLoc, Data: &js_ast.EIdentifier{Ref: p.storeNameInRef(js_lexer.MaybeSubstring{String: "using"})}}
+		} else {
+			value = p.parseExpr(js_ast.LPrefix)
+		}
+		if p.lexer.Token == js_lexer.TAsteriskAsterisk {
+			p.lexer.Unexpected()
+		}
+		value = p.parseSuffix(value, js_ast.LPrefix, nil, 0)
+		expr := js_ast.Expr{Loc: tokenRange.Loc, Data: &js_ast.EAwait{Value: value}}
+		return p.parseSuffix(expr, js_ast.LLowest, nil, 0), js_ast.Stmt{}, nil
+	}
+
+	// Parse the remainder of this expression that starts with an identifier
+	expr := js_ast.Expr{Loc: tokenRange.Loc, Data: &js_ast.EIdentifier{Ref: p.storeNameInRef(name)}}
+	return p.parseSuffix(expr, js_ast.LLowest, nil, 0), js_ast.Stmt{}, nil
+}
+
+func (p *parser) parseCallArgs() (args []js_ast.Expr, closeParenLoc logger.Loc, isMultiLine bool) {
+	// Allow "in" inside call arguments
+	oldAllowIn := p.allowIn
+	p.allowIn = true
+
+	p.lexer.Expect(js_lexer.TOpenParen)
+
+	for p.lexer.Token != js_lexer.TCloseParen {
+		if p.lexer.HasNewlineBefore {
+			isMultiLine = true
+		}
+		loc := p.lexer.Loc()
+		isSpread := p.lexer.Token == js_lexer.TDotDotDot
+		if isSpread {
+			p.markSyntaxFeature(compat.RestArgument, p.lexer.Range())
+			p.lexer.Next()
+		}
+		arg := p.parseExpr(js_ast.LComma)
+		if isSpread {
+			arg = js_ast.Expr{Loc: loc, Data: &js_ast.ESpread{Value: arg}}
+		}
+		args = append(args, arg)
+		if p.lexer.Token != js_lexer.TComma {
+			break
+		}
+		if p.lexer.HasNewlineBefore {
+			isMultiLine = true
+		}
+		p.lexer.Next()
+	}
+
+	if p.lexer.HasNewlineBefore {
+		isMultiLine = true
+	}
+	closeParenLoc = p.saveExprCommentsHere()
+	p.lexer.Expect(js_lexer.TCloseParen)
+	p.allowIn = oldAllowIn
+	return
+}
+
+func (p *parser) parseJSXNamespacedName() (logger.Range, js_lexer.MaybeSubstring) {
+	nameRange := p.lexer.Range()
+	name := p.lexer.Identifier
+	p.lexer.ExpectInsideJSXElement(js_lexer.TIdentifier)
+
+	// Parse JSX namespaces. These are not supported by React or TypeScript
+	// but someone using JSX syntax in more obscure ways may find a use for
+	// them. A namespaced name is just always turned into a string so you
+	// can't use this feature to reference JavaScript identifiers.
+	if p.lexer.Token == js_lexer.TColon {
+		// Parse the colon
+		nameRange.Len = p.lexer.Range().End() - nameRange.Loc.Start
+		ns := name.String + ":"
+		p.lexer.NextInsideJSXElement()
+
+		// Parse the second identifier
+		if p.lexer.Token == js_lexer.TIdentifier {
+			nameRange.Len = p.lexer.Range().End() - nameRange.Loc.Start
+			ns += p.lexer.Identifier.String
+			p.lexer.NextInsideJSXElement()
+		} else {
+			p.log.AddError(&p.tracker, logger.Range{Loc: logger.Loc{Start: nameRange.End()}},
+				fmt.Sprintf("Expected identifier after %q in namespaced JSX name", ns))
+			panic(js_lexer.LexerPanic{})
+		}
+		return nameRange, js_lexer.MaybeSubstring{String: ns}
+	}
+
+	return nameRange, name
+}
+
+func tagOrFragmentHelpText(tag string) string {
+	if tag == "" {
+		return "fragment tag"
+	}
+	return fmt.Sprintf("%q tag", tag)
+}
+
+func (p *parser) parseJSXTag() (logger.Range, string, js_ast.Expr) {
+	loc := p.lexer.Loc()
+
+	// A missing tag is a fragment
+	if p.lexer.Token == js_lexer.TGreaterThan {
+		return logger.Range{Loc: loc, Len: 0}, "", js_ast.Expr{}
+	}
+
+	// The tag is an identifier
+	tagRange, tagName := p.parseJSXNamespacedName()
+
+	// Certain identifiers are strings
+	if strings.ContainsAny(tagName.String, "-:") || (p.lexer.Token != js_lexer.TDot && tagName.String[0] >= 'a' && tagName.String[0] <= 'z') {
+		return tagRange, tagName.String, js_ast.Expr{Loc: loc, Data: &js_ast.EString{Value: helpers.StringToUTF16(tagName.String)}}
+	}
+
+	// Otherwise, this is an identifier
+	tag := js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: p.storeNameInRef(tagName)}}
+
+	// Parse a member expression chain
+	chain := tagName.String
+	for p.lexer.Token == js_lexer.TDot {
+		p.lexer.NextInsideJSXElement()
+		memberRange := p.lexer.Range()
+		member := p.lexer.Identifier
+		p.lexer.ExpectInsideJSXElement(js_lexer.TIdentifier)
+
+		// Dashes are not allowed in member expression chains
+		index := strings.IndexByte(member.String, '-')
+		if index >= 0 {
+			p.log.AddError(&p.tracker, logger.Range{Loc: logger.Loc{Start: memberRange.Loc.Start + int32(index)}},
+				"Unexpected \"-\"")
+			panic(js_lexer.LexerPanic{})
+		}
+
+		chain += "." + member.String
+		tag = js_ast.Expr{Loc: loc, Data: p.dotOrMangledPropParse(tag, member, memberRange.Loc, js_ast.OptionalChainNone, wasOriginallyDot)}
+		tagRange.Len = memberRange.Loc.Start + memberRange.Len - tagRange.Loc.Start
+	}
+
+	return tagRange, chain, tag
+}
+
+func (p *parser) parseJSXElement(loc logger.Loc) js_ast.Expr {
+	// Keep track of the location of the first JSX element for error messages
+	if p.firstJSXElementLoc.Start == -1 {
+		p.firstJSXElementLoc = loc
+	}
+
+	// Parse the tag
+	startRange, startText, startTagOrNil := p.parseJSXTag()
+
+	// The tag may have TypeScript type arguments: "<Foo<T>/>"
+	if p.options.ts.Parse {
+		// Pass a flag to the type argument skipper because we need to call
+		// js_lexer.NextInsideJSXElement() after we hit the closing ">". The next
+		// token after the ">" might be an attribute name with a dash in it
+		// like this: "<Foo<T> data-disabled/>"
+		p.skipTypeScriptTypeArguments(skipTypeScriptTypeArgumentsOpts{isInsideJSXElement: true})
+	}
+
+	// Parse attributes
+	var previousStringWithBackslashLoc logger.Loc
+	properties := []js_ast.Property{}
+	isSingleLine := true
+	if startTagOrNil.Data != nil {
+	parseAttributes:
+		for {
+			if p.lexer.HasNewlineBefore {
+				isSingleLine = false
+			}
+
+			switch p.lexer.Token {
+			case js_lexer.TIdentifier:
+				// Parse the key
+				keyRange, keyName := p.parseJSXNamespacedName()
+				var key js_ast.Expr
+				if p.isMangledProp(keyName.String) && !strings.ContainsRune(keyName.String, ':') {
+					key = js_ast.Expr{Loc: keyRange.Loc, Data: &js_ast.ENameOfSymbol{Ref: p.storeNameInRef(keyName)}}
+				} else {
+					key = js_ast.Expr{Loc: keyRange.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16(keyName.String)}}
+				}
+
+				// Parse the value
+				var value js_ast.Expr
+				var flags js_ast.PropertyFlags
+				if p.lexer.Token != js_lexer.TEquals {
+					// Implicitly true value
+					flags |= js_ast.PropertyWasShorthand
+					value = js_ast.Expr{Loc: logger.Loc{Start: keyRange.Loc.Start + keyRange.Len}, Data: &js_ast.EBoolean{Value: true}}
+				} else {
+					// Use NextInsideJSXElement() not Next() so we can parse a JSX-style string literal
+					p.lexer.NextInsideJSXElement()
+					if p.lexer.Token == js_lexer.TStringLiteral {
+						stringLoc := p.lexer.Loc()
+						if p.lexer.PreviousBackslashQuoteInJSX.Loc.Start > stringLoc.Start {
+							previousStringWithBackslashLoc = stringLoc
+						}
+						if p.options.jsx.Preserve {
+							value = js_ast.Expr{Loc: stringLoc, Data: &js_ast.EJSXText{Raw: p.lexer.Raw()}}
+						} else {
+							value = js_ast.Expr{Loc: stringLoc, Data: &js_ast.EString{Value: p.lexer.StringLiteral()}}
+						}
+						p.lexer.NextInsideJSXElement()
+					} else if p.lexer.Token == js_lexer.TLessThan {
+						// This may be removed in the future: https://github.com/facebook/jsx/issues/53
+						loc := p.lexer.Loc()
+						p.lexer.NextInsideJSXElement()
+						flags |= js_ast.PropertyWasShorthand
+						value = p.parseJSXElement(loc)
+
+						// The call to parseJSXElement() above doesn't consume the last
+						// TGreaterThan because the caller knows what Next() function to call.
+						// Use NextJSXElementChild() here since the next token is inside a JSX
+						// element.
+						p.lexer.NextInsideJSXElement()
+					} else {
+						// Use Expect() not ExpectInsideJSXElement() so we can parse expression tokens
+						p.lexer.Expect(js_lexer.TOpenBrace)
+						value = p.parseExpr(js_ast.LLowest)
+						p.lexer.ExpectInsideJSXElement(js_lexer.TCloseBrace)
+					}
+				}
+
+				// Add a property
+				properties = append(properties, js_ast.Property{
+					Loc:        keyRange.Loc,
+					Key:        key,
+					ValueOrNil: value,
+					Flags:      flags,
+				})
+
+			case js_lexer.TOpenBrace:
+				// Use Next() not ExpectInsideJSXElement() so we can parse "..."
+				p.lexer.Next()
+				dotLoc := p.saveExprCommentsHere()
+				p.lexer.Expect(js_lexer.TDotDotDot)
+				value := p.parseExpr(js_ast.LComma)
+				properties = append(properties, js_ast.Property{
+					Kind:       js_ast.PropertySpread,
+					Loc:        dotLoc,
+					ValueOrNil: value,
+				})
+
+				// Use NextInsideJSXElement() not Next() so we can parse ">>" as ">"
+				p.lexer.NextInsideJSXElement()
+
+			default:
+				break parseAttributes
+			}
+		}
+
+		// Check for and warn about duplicate attributes
+		if len(properties) > 1 && !p.suppressWarningsAboutWeirdCode {
+			keys := make(map[string]logger.Loc)
+			for _, property := range properties {
+				if property.Kind != js_ast.PropertySpread {
+					if str, ok := property.Key.Data.(*js_ast.EString); ok {
+						key := helpers.UTF16ToString(str.Value)
+						if prevLoc, ok := keys[key]; ok {
+							r := js_lexer.RangeOfIdentifier(p.source, property.Key.Loc)
+							p.log.AddIDWithNotes(logger.MsgID_JS_DuplicateObjectKey, logger.Warning, &p.tracker, r,
+								fmt.Sprintf("Duplicate %q attribute in JSX element", key),
+								[]logger.MsgData{p.tracker.MsgData(js_lexer.RangeOfIdentifier(p.source, prevLoc),
+									fmt.Sprintf("The original %q attribute is here:", key))})
+						}
+						keys[key] = property.Key.Loc
+					}
+				}
+			}
+		}
+	}
+
+	// People sometimes try to use the output of "JSON.stringify()" as a JSX
+	// attribute when automatically-generating JSX code. Doing so is incorrect
+	// because JSX strings work like XML instead of like JS (since JSX is XML-in-
+	// JS). Specifically, using a backslash before a quote does not cause it to
+	// be escaped:
+	//
+	//   JSX ends the "content" attribute here and sets "content" to 'some so-called \\'
+	//                                          v
+	//         <Button content="some so-called \"button text\"" />
+	//                                                      ^
+	//       There is no "=" after the JSX attribute "text", so we expect a ">"
+	//
+	// This code special-cases this error to provide a less obscure error message.
+	if p.lexer.Token == js_lexer.TSyntaxError && p.lexer.Raw() == "\\" && previousStringWithBackslashLoc.Start > 0 {
+		msg := logger.Msg{Kind: logger.Error, Data: p.tracker.MsgData(p.lexer.Range(),
+			"Unexpected backslash in JSX element")}
+
+		// Option 1: Suggest using an XML escape
+		jsEscape := p.source.TextForRange(p.lexer.PreviousBackslashQuoteInJSX)
+		xmlEscape := ""
+		if jsEscape == "\\\"" {
+			xmlEscape = "&quot;"
+		} else if jsEscape == "\\'" {
+			xmlEscape = "&apos;"
+		}
+		if xmlEscape != "" {
+			data := p.tracker.MsgData(p.lexer.PreviousBackslashQuoteInJSX,
+				"Quoted JSX attributes use XML-style escapes instead of JavaScript-style escapes:")
+			data.Location.Suggestion = xmlEscape
+			msg.Notes = append(msg.Notes, data)
+		}
+
+		// Option 2: Suggest using a JavaScript string
+		if stringRange := p.source.RangeOfString(previousStringWithBackslashLoc); stringRange.Len > 0 {
+			data := p.tracker.MsgData(stringRange,
+				"Consider using a JavaScript string inside {...} instead of a quoted JSX attribute:")
+			data.Location.Suggestion = fmt.Sprintf("{%s}", p.source.TextForRange(stringRange))
+			msg.Notes = append(msg.Notes, data)
+		}
+
+		p.log.AddMsg(msg)
+		panic(js_lexer.LexerPanic{})
+	}
+
+	// A slash here is a self-closing element
+	if p.lexer.Token == js_lexer.TSlash {
+		// Use NextInsideJSXElement() not Next() so we can parse ">>" as ">"
+		closeLoc := p.lexer.Loc()
+		p.lexer.NextInsideJSXElement()
+		if p.lexer.Token != js_lexer.TGreaterThan {
+			p.lexer.Expected(js_lexer.TGreaterThan)
+		}
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EJSXElement{
+			TagOrNil:        startTagOrNil,
+			Properties:      properties,
+			CloseLoc:        closeLoc,
+			IsTagSingleLine: isSingleLine,
+		}}
+	}
+
+	// Attempt to provide a better error message for people incorrectly trying to
+	// use arrow functions in TSX (which doesn't work because they are JSX elements)
+	if p.options.ts.Parse && len(properties) == 0 && startText != "" && p.lexer.Token == js_lexer.TGreaterThan &&
+		strings.HasPrefix(p.source.Contents[p.lexer.Loc().Start:], ">(") {
+		badArrowInTSXRange := p.lexer.BadArrowInTSXRange
+		badArrowInTSXSuggestion := p.lexer.BadArrowInTSXSuggestion
+
+		p.lexer.CouldBeBadArrowInTSX++
+		p.lexer.BadArrowInTSXRange = logger.Range{Loc: loc, Len: p.lexer.Range().End() - loc.Start}
+		p.lexer.BadArrowInTSXSuggestion = fmt.Sprintf("<%s,>", startText)
+
+		defer func() {
+			p.lexer.CouldBeBadArrowInTSX--
+			p.lexer.BadArrowInTSXRange = badArrowInTSXRange
+			p.lexer.BadArrowInTSXSuggestion = badArrowInTSXSuggestion
+		}()
+	}
+
+	// Use ExpectJSXElementChild() so we parse child strings
+	p.lexer.ExpectJSXElementChild(js_lexer.TGreaterThan)
+
+	// Parse the children of this element
+	nullableChildren := []js_ast.Expr{}
+	for {
+		switch p.lexer.Token {
+		case js_lexer.TStringLiteral:
+			if p.options.jsx.Preserve {
+				nullableChildren = append(nullableChildren, js_ast.Expr{Loc: p.lexer.Loc(), Data: &js_ast.EJSXText{Raw: p.lexer.Raw()}})
+			} else if str := p.lexer.StringLiteral(); len(str) > 0 {
+				nullableChildren = append(nullableChildren, js_ast.Expr{Loc: p.lexer.Loc(), Data: &js_ast.EString{Value: str}})
+			} else {
+				// Skip this token if it turned out to be empty after trimming
+			}
+			p.lexer.NextJSXElementChild()
+
+		case js_lexer.TOpenBrace:
+			// Use Next() instead of NextJSXElementChild() here since the next token is an expression
+			p.lexer.Next()
+
+			// The expression is optional, and may be absent
+			if p.lexer.Token == js_lexer.TCloseBrace {
+				// Save comments even for absent expressions
+				nullableChildren = append(nullableChildren, js_ast.Expr{Loc: p.saveExprCommentsHere(), Data: nil})
+			} else {
+				if p.lexer.Token == js_lexer.TDotDotDot {
+					// TypeScript preserves "..." before JSX child expressions here.
+					// Babel gives the error "Spread children are not supported in React"
+					// instead, so it should be safe to support this TypeScript-specific
+					// behavior. Note that TypeScript's behavior changed in TypeScript 4.5.
+					// Before that, the "..." was omitted instead of being preserved.
+					itemLoc := p.lexer.Loc()
+					p.markSyntaxFeature(compat.RestArgument, p.lexer.Range())
+					p.lexer.Next()
+					nullableChildren = append(nullableChildren, js_ast.Expr{Loc: itemLoc, Data: &js_ast.ESpread{Value: p.parseExpr(js_ast.LLowest)}})
+				} else {
+					nullableChildren = append(nullableChildren, p.parseExpr(js_ast.LLowest))
+				}
+			}
+
+			// Use ExpectJSXElementChild() so we parse child strings
+			p.lexer.ExpectJSXElementChild(js_lexer.TCloseBrace)
+
+		case js_lexer.TLessThan:
+			lessThanLoc := p.lexer.Loc()
+			p.lexer.NextInsideJSXElement()
+
+			if p.lexer.Token != js_lexer.TSlash {
+				// This is a child element
+				nullableChildren = append(nullableChildren, p.parseJSXElement(lessThanLoc))
+
+				// The call to parseJSXElement() above doesn't consume the last
+				// TGreaterThan because the caller knows what Next() function to call.
+				// Use NextJSXElementChild() here since the next token is an element
+				// child.
+				p.lexer.NextJSXElementChild()
+				continue
+			}
+
+			// This is the closing element
+			p.lexer.NextInsideJSXElement()
+			endRange, endText, _ := p.parseJSXTag()
+			if startText != endText {
+				startTag := tagOrFragmentHelpText(startText)
+				endTag := tagOrFragmentHelpText(endText)
+				msg := logger.Msg{
+					Kind:  logger.Error,
+					Data:  p.tracker.MsgData(endRange, fmt.Sprintf("Unexpected closing %s does not match opening %s", endTag, startTag)),
+					Notes: []logger.MsgData{p.tracker.MsgData(startRange, fmt.Sprintf("The opening %s is here:", startTag))},
+				}
+				msg.Data.Location.Suggestion = startText
+				p.log.AddMsg(msg)
+			}
+			if p.lexer.Token != js_lexer.TGreaterThan {
+				p.lexer.Expected(js_lexer.TGreaterThan)
+			}
+
+			return js_ast.Expr{Loc: loc, Data: &js_ast.EJSXElement{
+				TagOrNil:         startTagOrNil,
+				Properties:       properties,
+				NullableChildren: nullableChildren,
+				CloseLoc:         lessThanLoc,
+				IsTagSingleLine:  isSingleLine,
+			}}
+
+		case js_lexer.TEndOfFile:
+			startTag := tagOrFragmentHelpText(startText)
+			msg := logger.Msg{
+				Kind:  logger.Error,
+				Data:  p.tracker.MsgData(p.lexer.Range(), fmt.Sprintf("Unexpected end of file before a closing %s", startTag)),
+				Notes: []logger.MsgData{p.tracker.MsgData(startRange, fmt.Sprintf("The opening %s is here:", startTag))},
+			}
+			msg.Data.Location.Suggestion = fmt.Sprintf("</%s>", startText)
+			p.log.AddMsg(msg)
+			panic(js_lexer.LexerPanic{})
+
+		default:
+			p.lexer.Unexpected()
+		}
+	}
+}
+
+func (p *parser) parseTemplateParts(includeRaw bool) (parts []js_ast.TemplatePart, legacyOctalLoc logger.Loc) {
+	// Allow "in" inside template literals
+	oldAllowIn := p.allowIn
+	p.allowIn = true
+
+	for {
+		p.lexer.Next()
+		value := p.parseExpr(js_ast.LLowest)
+		tailLoc := p.lexer.Loc()
+		p.lexer.RescanCloseBraceAsTemplateToken()
+		if includeRaw {
+			tailCooked, tailRaw := p.lexer.CookedAndRawTemplateContents()
+			parts = append(parts, js_ast.TemplatePart{
+				Value:      value,
+				TailLoc:    tailLoc,
+				TailCooked: tailCooked,
+				TailRaw:    tailRaw,
+			})
+		} else {
+			parts = append(parts, js_ast.TemplatePart{
+				Value:      value,
+				TailLoc:    tailLoc,
+				TailCooked: p.lexer.StringLiteral(),
+			})
+			if p.lexer.LegacyOctalLoc.Start > tailLoc.Start {
+				legacyOctalLoc = p.lexer.LegacyOctalLoc
+			}
+		}
+		if p.lexer.Token == js_lexer.TTemplateTail {
+			p.lexer.Next()
+			break
+		}
+	}
+
+	p.allowIn = oldAllowIn
+
+	return parts, legacyOctalLoc
+}
+
+func (p *parser) parseAndDeclareDecls(kind ast.SymbolKind, opts parseStmtOpts) []js_ast.Decl {
+	decls := []js_ast.Decl{}
+
+	for {
+		// Forbid "let let" and "const let" but not "var let"
+		if (kind == ast.SymbolOther || kind == ast.SymbolConst) && p.lexer.IsContextualKeyword("let") {
+			p.log.AddError(&p.tracker, p.lexer.Range(), "Cannot use \"let\" as an identifier here:")
+		}
+
+		var valueOrNil js_ast.Expr
+		local := p.parseBinding(parseBindingOpts{isUsingStmt: opts.isUsingStmt})
+		p.declareBinding(kind, local, opts)
+
+		// Skip over types
+		if p.options.ts.Parse {
+			// "let foo!"
+			isDefiniteAssignmentAssertion := p.lexer.Token == js_lexer.TExclamation && !p.lexer.HasNewlineBefore
+			if isDefiniteAssignmentAssertion {
+				p.lexer.Next()
+			}
+
+			// "let foo: number"
+			if isDefiniteAssignmentAssertion || p.lexer.Token == js_lexer.TColon {
+				p.lexer.Expect(js_lexer.TColon)
+				p.skipTypeScriptType(js_ast.LLowest)
+			}
+		}
+
+		if p.lexer.Token == js_lexer.TEquals {
+			p.lexer.Next()
+			valueOrNil = p.parseExpr(js_ast.LComma)
+
+			// Rollup (the tool that invented the "@__NO_SIDE_EFFECTS__" comment) only
+			// applies this to the first declaration, and only when it's a "const".
+			// For more info see: https://github.com/rollup/rollup/pull/5024/files
+			if !p.options.ignoreDCEAnnotations && kind == ast.SymbolConst {
+				switch e := valueOrNil.Data.(type) {
+				case *js_ast.EArrow:
+					if opts.hasNoSideEffectsComment {
+						e.HasNoSideEffectsComment = true
+					}
+					if e.HasNoSideEffectsComment && !opts.isTypeScriptDeclare {
+						if b, ok := local.Data.(*js_ast.BIdentifier); ok {
+							p.symbols[b.Ref.InnerIndex].Flags |= ast.CallCanBeUnwrappedIfUnused
+						}
+					}
+
+				case *js_ast.EFunction:
+					if opts.hasNoSideEffectsComment {
+						e.Fn.HasNoSideEffectsComment = true
+					}
+					if e.Fn.HasNoSideEffectsComment && !opts.isTypeScriptDeclare {
+						if b, ok := local.Data.(*js_ast.BIdentifier); ok {
+							p.symbols[b.Ref.InnerIndex].Flags |= ast.CallCanBeUnwrappedIfUnused
+						}
+					}
+				}
+
+				// Only apply this to the first declaration
+				opts.hasNoSideEffectsComment = false
+			}
+		}
+
+		decls = append(decls, js_ast.Decl{Binding: local, ValueOrNil: valueOrNil})
+
+		if p.lexer.Token != js_lexer.TComma {
+			break
+		}
+		p.lexer.Next()
+	}
+
+	return decls
+}
+
+func (p *parser) requireInitializers(kind js_ast.LocalKind, decls []js_ast.Decl) {
+	for _, d := range decls {
+		if d.ValueOrNil.Data == nil {
+			what := "constant"
+			if kind == js_ast.LocalUsing {
+				what = "declaration"
+			}
+			if id, ok := d.Binding.Data.(*js_ast.BIdentifier); ok {
+				r := js_lexer.RangeOfIdentifier(p.source, d.Binding.Loc)
+				p.log.AddError(&p.tracker, r,
+					fmt.Sprintf("The %s %q must be initialized", what, p.symbols[id.Ref.InnerIndex].OriginalName))
+			} else {
+				p.log.AddError(&p.tracker, logger.Range{Loc: d.Binding.Loc},
+					fmt.Sprintf("This %s must be initialized", what))
+			}
+		}
+	}
+}
+
+func (p *parser) forbidInitializers(decls []js_ast.Decl, loopType string, isVar bool) {
+	if len(decls) > 1 {
+		p.log.AddError(&p.tracker, logger.Range{Loc: decls[0].Binding.Loc},
+			fmt.Sprintf("for-%s loops must have a single declaration", loopType))
+	} else if len(decls) == 1 && decls[0].ValueOrNil.Data != nil {
+		if isVar {
+			if _, ok := decls[0].Binding.Data.(*js_ast.BIdentifier); ok {
+				// This is a weird special case. Initializers are allowed in "var"
+				// statements with identifier bindings.
+				return
+			}
+		}
+		p.log.AddError(&p.tracker, logger.Range{Loc: decls[0].ValueOrNil.Loc},
+			fmt.Sprintf("for-%s loop variables cannot have an initializer", loopType))
+	}
+}
+
+func (p *parser) parseClauseAlias(kind string) js_lexer.MaybeSubstring {
+	loc := p.lexer.Loc()
+
+	// The alias may now be a string (see https://github.com/tc39/ecma262/pull/2154)
+	if p.lexer.Token == js_lexer.TStringLiteral {
+		r := p.source.RangeOfString(loc)
+		alias, problem, ok := helpers.UTF16ToStringWithValidation(p.lexer.StringLiteral())
+		if !ok {
+			p.log.AddError(&p.tracker, r,
+				fmt.Sprintf("This %s alias is invalid because it contains the unpaired Unicode surrogate U+%X", kind, problem))
+		}
+		return js_lexer.MaybeSubstring{String: alias}
+	}
+
+	// The alias may be a keyword
+	if !p.lexer.IsIdentifierOrKeyword() {
+		p.lexer.Expect(js_lexer.TIdentifier)
+	}
+
+	alias := p.lexer.Identifier
+	p.checkForUnrepresentableIdentifier(loc, alias.String)
+	return alias
+}
+
+func (p *parser) parseImportClause() ([]js_ast.ClauseItem, bool) {
+	items := []js_ast.ClauseItem{}
+	p.lexer.Expect(js_lexer.TOpenBrace)
+	isSingleLine := !p.lexer.HasNewlineBefore
+
+	for p.lexer.Token != js_lexer.TCloseBrace {
+		isIdentifier := p.lexer.Token == js_lexer.TIdentifier
+		aliasLoc := p.lexer.Loc()
+		alias := p.parseClauseAlias("import")
+		name := ast.LocRef{Loc: aliasLoc, Ref: p.storeNameInRef(alias)}
+		originalName := alias
+		p.lexer.Next()
+
+		// "import { type xx } from 'mod'"
+		// "import { type xx as yy } from 'mod'"
+		// "import { type 'xx' as yy } from 'mod'"
+		// "import { type as } from 'mod'"
+		// "import { type as as } from 'mod'"
+		// "import { type as as as } from 'mod'"
+		if p.options.ts.Parse && alias.String == "type" && p.lexer.Token != js_lexer.TComma && p.lexer.Token != js_lexer.TCloseBrace {
+			if p.lexer.IsContextualKeyword("as") {
+				p.lexer.Next()
+				if p.lexer.IsContextualKeyword("as") {
+					originalName = p.lexer.Identifier
+					name = ast.LocRef{Loc: p.lexer.Loc(), Ref: p.storeNameInRef(originalName)}
+					p.lexer.Next()
+
+					if p.lexer.Token == js_lexer.TIdentifier {
+						// "import { type as as as } from 'mod'"
+						// "import { type as as foo } from 'mod'"
+						p.lexer.Next()
+					} else {
+						// "import { type as as } from 'mod'"
+						items = append(items, js_ast.ClauseItem{
+							Alias:        alias.String,
+							AliasLoc:     aliasLoc,
+							Name:         name,
+							OriginalName: originalName.String,
+						})
+					}
+				} else if p.lexer.Token == js_lexer.TIdentifier {
+					// "import { type as xxx } from 'mod'"
+					originalName = p.lexer.Identifier
+					name = ast.LocRef{Loc: p.lexer.Loc(), Ref: p.storeNameInRef(originalName)}
+					p.lexer.Expect(js_lexer.TIdentifier)
+
+					// Reject forbidden names
+					if isEvalOrArguments(originalName.String) {
+						r := js_lexer.RangeOfIdentifier(p.source, name.Loc)
+						p.log.AddError(&p.tracker, r, fmt.Sprintf("Cannot use %q as an identifier here:", originalName.String))
+					}
+
+					items = append(items, js_ast.ClauseItem{
+						Alias:        alias.String,
+						AliasLoc:     aliasLoc,
+						Name:         name,
+						OriginalName: originalName.String,
+					})
+				}
+			} else {
+				isIdentifier := p.lexer.Token == js_lexer.TIdentifier
+
+				// "import { type xx } from 'mod'"
+				// "import { type xx as yy } from 'mod'"
+				// "import { type if as yy } from 'mod'"
+				// "import { type 'xx' as yy } from 'mod'"
+				p.parseClauseAlias("import")
+				p.lexer.Next()
+
+				if p.lexer.IsContextualKeyword("as") {
+					p.lexer.Next()
+					p.lexer.Expect(js_lexer.TIdentifier)
+				} else if !isIdentifier {
+					// An import where the name is a keyword must have an alias
+					p.lexer.ExpectedString("\"as\"")
+				}
+			}
+		} else {
+			if p.lexer.IsContextualKeyword("as") {
+				p.lexer.Next()
+				originalName = p.lexer.Identifier
+				name = ast.LocRef{Loc: p.lexer.Loc(), Ref: p.storeNameInRef(originalName)}
+				p.lexer.Expect(js_lexer.TIdentifier)
+			} else if !isIdentifier {
+				// An import where the name is a keyword must have an alias
+				p.lexer.ExpectedString("\"as\"")
+			}
+
+			// Reject forbidden names
+			if isEvalOrArguments(originalName.String) {
+				r := js_lexer.RangeOfIdentifier(p.source, name.Loc)
+				p.log.AddError(&p.tracker, r, fmt.Sprintf("Cannot use %q as an identifier here:", originalName.String))
+			}
+
+			items = append(items, js_ast.ClauseItem{
+				Alias:        alias.String,
+				AliasLoc:     aliasLoc,
+				Name:         name,
+				OriginalName: originalName.String,
+			})
+		}
+
+		if p.lexer.Token != js_lexer.TComma {
+			break
+		}
+		if p.lexer.HasNewlineBefore {
+			isSingleLine = false
+		}
+		p.lexer.Next()
+		if p.lexer.HasNewlineBefore {
+			isSingleLine = false
+		}
+	}
+
+	if p.lexer.HasNewlineBefore {
+		isSingleLine = false
+	}
+	p.lexer.Expect(js_lexer.TCloseBrace)
+	return items, isSingleLine
+}
+
+func (p *parser) parseExportClause() ([]js_ast.ClauseItem, bool) {
+	items := []js_ast.ClauseItem{}
+	firstNonIdentifierLoc := logger.Loc{}
+	p.lexer.Expect(js_lexer.TOpenBrace)
+	isSingleLine := !p.lexer.HasNewlineBefore
+
+	for p.lexer.Token != js_lexer.TCloseBrace {
+		alias := p.parseClauseAlias("export")
+		aliasLoc := p.lexer.Loc()
+		name := ast.LocRef{Loc: aliasLoc, Ref: p.storeNameInRef(alias)}
+		originalName := alias
+
+		// The name can actually be a keyword if we're really an "export from"
+		// statement. However, we won't know until later. Allow keywords as
+		// identifiers for now and throw an error later if there's no "from".
+		//
+		//   // This is fine
+		//   export { default } from 'path'
+		//
+		//   // This is a syntax error
+		//   export { default }
+		//
+		if p.lexer.Token != js_lexer.TIdentifier && firstNonIdentifierLoc.Start == 0 {
+			firstNonIdentifierLoc = p.lexer.Loc()
+		}
+		p.lexer.Next()
+
+		if p.options.ts.Parse && alias.String == "type" && p.lexer.Token != js_lexer.TComma && p.lexer.Token != js_lexer.TCloseBrace {
+			if p.lexer.IsContextualKeyword("as") {
+				p.lexer.Next()
+				if p.lexer.IsContextualKeyword("as") {
+					alias = p.parseClauseAlias("export")
+					aliasLoc = p.lexer.Loc()
+					p.lexer.Next()
+
+					if p.lexer.Token != js_lexer.TComma && p.lexer.Token != js_lexer.TCloseBrace {
+						// "export { type as as as }"
+						// "export { type as as foo }"
+						// "export { type as as 'foo' }"
+						p.parseClauseAlias("export")
+						p.lexer.Next()
+					} else {
+						// "export { type as as }"
+						items = append(items, js_ast.ClauseItem{
+							Alias:        alias.String,
+							AliasLoc:     aliasLoc,
+							Name:         name,
+							OriginalName: originalName.String,
+						})
+					}
+				} else if p.lexer.Token != js_lexer.TComma && p.lexer.Token != js_lexer.TCloseBrace {
+					// "export { type as xxx }"
+					// "export { type as 'xxx' }"
+					alias = p.parseClauseAlias("export")
+					aliasLoc = p.lexer.Loc()
+					p.lexer.Next()
+
+					items = append(items, js_ast.ClauseItem{
+						Alias:        alias.String,
+						AliasLoc:     aliasLoc,
+						Name:         name,
+						OriginalName: originalName.String,
+					})
+				}
+			} else {
+				// The name can actually be a keyword if we're really an "export from"
+				// statement. However, we won't know until later. Allow keywords as
+				// identifiers for now and throw an error later if there's no "from".
+				//
+				//   // This is fine
+				//   export { type default } from 'path'
+				//
+				//   // This is a syntax error
+				//   export { type default }
+				//
+				if p.lexer.Token != js_lexer.TIdentifier && firstNonIdentifierLoc.Start == 0 {
+					firstNonIdentifierLoc = p.lexer.Loc()
+				}
+
+				// "export { type xx }"
+				// "export { type xx as yy }"
+				// "export { type xx as if }"
+				// "export { type default } from 'path'"
+				// "export { type default as if } from 'path'"
+				// "export { type xx as 'yy' }"
+				// "export { type 'xx' } from 'mod'"
+				p.parseClauseAlias("export")
+				p.lexer.Next()
+
+				if p.lexer.IsContextualKeyword("as") {
+					p.lexer.Next()
+					p.parseClauseAlias("export")
+					p.lexer.Next()
+				}
+			}
+		} else {
+			if p.lexer.IsContextualKeyword("as") {
+				p.lexer.Next()
+				alias = p.parseClauseAlias("export")
+				aliasLoc = p.lexer.Loc()
+				p.lexer.Next()
+			}
+
+			items = append(items, js_ast.ClauseItem{
+				Alias:        alias.String,
+				AliasLoc:     aliasLoc,
+				Name:         name,
+				OriginalName: originalName.String,
+			})
+		}
+
+		if p.lexer.Token != js_lexer.TComma {
+			break
+		}
+		if p.lexer.HasNewlineBefore {
+			isSingleLine = false
+		}
+		p.lexer.Next()
+		if p.lexer.HasNewlineBefore {
+			isSingleLine = false
+		}
+	}
+
+	if p.lexer.HasNewlineBefore {
+		isSingleLine = false
+	}
+	p.lexer.Expect(js_lexer.TCloseBrace)
+
+	// Throw an error here if we found a keyword earlier and this isn't an
+	// "export from" statement after all
+	if firstNonIdentifierLoc.Start != 0 && !p.lexer.IsContextualKeyword("from") {
+		r := js_lexer.RangeOfIdentifier(p.source, firstNonIdentifierLoc)
+		p.log.AddError(&p.tracker, r, fmt.Sprintf("Expected identifier but found %q", p.source.TextForRange(r)))
+		panic(js_lexer.LexerPanic{})
+	}
+
+	return items, isSingleLine
+}
+
+type parseBindingOpts struct {
+	isUsingStmt bool
+}
+
+func (p *parser) parseBinding(opts parseBindingOpts) js_ast.Binding {
+	loc := p.lexer.Loc()
+
+	switch p.lexer.Token {
+	case js_lexer.TIdentifier:
+		name := p.lexer.Identifier
+
+		// Forbid invalid identifiers
+		if (p.fnOrArrowDataParse.await != allowIdent && name.String == "await") ||
+			(p.fnOrArrowDataParse.yield != allowIdent && name.String == "yield") {
+			p.log.AddError(&p.tracker, p.lexer.Range(), fmt.Sprintf("Cannot use %q as an identifier here:", name.String))
+		}
+
+		ref := p.storeNameInRef(name)
+		p.lexer.Next()
+		return js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: ref}}
+
+	case js_lexer.TOpenBracket:
+		if opts.isUsingStmt {
+			break
+		}
+		p.markSyntaxFeature(compat.Destructuring, p.lexer.Range())
+		p.lexer.Next()
+		isSingleLine := !p.lexer.HasNewlineBefore
+		items := []js_ast.ArrayBinding{}
+		hasSpread := false
+
+		// "in" expressions are allowed
+		oldAllowIn := p.allowIn
+		p.allowIn = true
+
+		for p.lexer.Token != js_lexer.TCloseBracket {
+			itemLoc := p.saveExprCommentsHere()
+
+			if p.lexer.Token == js_lexer.TComma {
+				binding := js_ast.Binding{Loc: itemLoc, Data: js_ast.BMissingShared}
+				items = append(items, js_ast.ArrayBinding{
+					Binding: binding,
+					Loc:     itemLoc,
+				})
+			} else {
+				if p.lexer.Token == js_lexer.TDotDotDot {
+					p.lexer.Next()
+					hasSpread = true
+
+					// This was a bug in the ES2015 spec that was fixed in ES2016
+					if p.lexer.Token != js_lexer.TIdentifier {
+						p.markSyntaxFeature(compat.NestedRestBinding, p.lexer.Range())
+					}
+				}
+
+				p.saveExprCommentsHere()
+				binding := p.parseBinding(parseBindingOpts{})
+
+				var defaultValueOrNil js_ast.Expr
+				if !hasSpread && p.lexer.Token == js_lexer.TEquals {
+					p.lexer.Next()
+					defaultValueOrNil = p.parseExpr(js_ast.LComma)
+				}
+
+				items = append(items, js_ast.ArrayBinding{
+					Binding:           binding,
+					DefaultValueOrNil: defaultValueOrNil,
+					Loc:               itemLoc,
+				})
+
+				// Commas after spread elements are not allowed
+				if hasSpread && p.lexer.Token == js_lexer.TComma {
+					p.log.AddError(&p.tracker, p.lexer.Range(), "Unexpected \",\" after rest pattern")
+					panic(js_lexer.LexerPanic{})
+				}
+			}
+
+			if p.lexer.Token != js_lexer.TComma {
+				break
+			}
+			if p.lexer.HasNewlineBefore {
+				isSingleLine = false
+			}
+			p.lexer.Next()
+			if p.lexer.HasNewlineBefore {
+				isSingleLine = false
+			}
+		}
+
+		p.allowIn = oldAllowIn
+
+		if p.lexer.HasNewlineBefore {
+			isSingleLine = false
+		}
+		closeBracketLoc := p.saveExprCommentsHere()
+		p.lexer.Expect(js_lexer.TCloseBracket)
+		return js_ast.Binding{Loc: loc, Data: &js_ast.BArray{
+			Items:           items,
+			HasSpread:       hasSpread,
+			IsSingleLine:    isSingleLine,
+			CloseBracketLoc: closeBracketLoc,
+		}}
+
+	case js_lexer.TOpenBrace:
+		if opts.isUsingStmt {
+			break
+		}
+		p.markSyntaxFeature(compat.Destructuring, p.lexer.Range())
+		p.lexer.Next()
+		isSingleLine := !p.lexer.HasNewlineBefore
+		properties := []js_ast.PropertyBinding{}
+
+		// "in" expressions are allowed
+		oldAllowIn := p.allowIn
+		p.allowIn = true
+
+		for p.lexer.Token != js_lexer.TCloseBrace {
+			p.saveExprCommentsHere()
+			property := p.parsePropertyBinding()
+			properties = append(properties, property)
+
+			// Commas after spread elements are not allowed
+			if property.IsSpread && p.lexer.Token == js_lexer.TComma {
+				p.log.AddError(&p.tracker, p.lexer.Range(), "Unexpected \",\" after rest pattern")
+				panic(js_lexer.LexerPanic{})
+			}
+
+			if p.lexer.Token != js_lexer.TComma {
+				break
+			}
+			if p.lexer.HasNewlineBefore {
+				isSingleLine = false
+			}
+			p.lexer.Next()
+			if p.lexer.HasNewlineBefore {
+				isSingleLine = false
+			}
+		}
+
+		p.allowIn = oldAllowIn
+
+		if p.lexer.HasNewlineBefore {
+			isSingleLine = false
+		}
+		closeBraceLoc := p.saveExprCommentsHere()
+		p.lexer.Expect(js_lexer.TCloseBrace)
+		return js_ast.Binding{Loc: loc, Data: &js_ast.BObject{
+			Properties:    properties,
+			IsSingleLine:  isSingleLine,
+			CloseBraceLoc: closeBraceLoc,
+		}}
+	}
+
+	p.lexer.Expect(js_lexer.TIdentifier)
+	return js_ast.Binding{}
+}
+
+func (p *parser) parseFn(
+	name *ast.LocRef,
+	classKeyword logger.Range,
+	decoratorContext decoratorContextFlags,
+	data fnOrArrowDataParse,
+) (fn js_ast.Fn, hadBody bool) {
+	fn.Name = name
+	fn.HasRestArg = false
+	fn.IsAsync = data.await == allowExpr
+	fn.IsGenerator = data.yield == allowExpr
+	fn.ArgumentsRef = ast.InvalidRef
+	fn.OpenParenLoc = p.lexer.Loc()
+	p.lexer.Expect(js_lexer.TOpenParen)
+
+	// Await and yield are not allowed in function arguments
+	oldFnOrArrowData := p.fnOrArrowDataParse
+	if data.await == allowExpr {
+		p.fnOrArrowDataParse.await = forbidAll
+	} else {
+		p.fnOrArrowDataParse.await = allowIdent
+	}
+	if data.yield == allowExpr {
+		p.fnOrArrowDataParse.yield = forbidAll
+	} else {
+		p.fnOrArrowDataParse.yield = allowIdent
+	}
+
+	// Don't suggest inserting "async" before anything if "await" is found
+	p.fnOrArrowDataParse.needsAsyncLoc.Start = -1
+
+	// If "super" is allowed in the body, it's allowed in the arguments
+	p.fnOrArrowDataParse.allowSuperCall = data.allowSuperCall
+	p.fnOrArrowDataParse.allowSuperProperty = data.allowSuperProperty
+
+	for p.lexer.Token != js_lexer.TCloseParen {
+		// Skip over "this" type annotations
+		if p.options.ts.Parse && p.lexer.Token == js_lexer.TThis {
+			p.lexer.Next()
+			if p.lexer.Token == js_lexer.TColon {
+				p.lexer.Next()
+				p.skipTypeScriptType(js_ast.LLowest)
+			}
+			if p.lexer.Token != js_lexer.TComma {
+				break
+			}
+			p.lexer.Next()
+			continue
+		}
+
+		var decorators []js_ast.Decorator
+		if data.decoratorScope != nil {
+			oldAwait := p.fnOrArrowDataParse.await
+			oldNeedsAsyncLoc := p.fnOrArrowDataParse.needsAsyncLoc
+
+			// While TypeScript parameter decorators are expressions, they are not
+			// evaluated where they exist in the code. They are moved to after the
+			// class declaration and evaluated there instead. Specifically this
+			// TypeScript code:
+			//
+			//   class Foo {
+			//     foo(@bar() baz) {}
+			//   }
+			//
+			// becomes this JavaScript code:
+			//
+			//   class Foo {
+			//     foo(baz) {}
+			//   }
+			//   __decorate([
+			//     __param(0, bar())
+			//   ], Foo.prototype, "foo", null);
+			//
+			// One consequence of this is that whether "await" is allowed or not
+			// depends on whether the class declaration itself is inside an "async"
+			// function or not. The TypeScript compiler allows code that does this:
+			//
+			//   async function fn(foo) {
+			//     class Foo {
+			//       foo(@bar(await foo) baz) {}
+			//     }
+			//     return Foo
+			//   }
+			//
+			// because that becomes the following valid JavaScript:
+			//
+			//   async function fn(foo) {
+			//     class Foo {
+			//       foo(baz) {}
+			//     }
+			//     __decorate([
+			//       __param(0, bar(await foo))
+			//     ], Foo.prototype, "foo", null);
+			//     return Foo;
+			//   }
+			//
+			if oldFnOrArrowData.await == allowExpr {
+				p.fnOrArrowDataParse.await = allowExpr
+			} else {
+				p.fnOrArrowDataParse.needsAsyncLoc = oldFnOrArrowData.needsAsyncLoc
+			}
+
+			decorators = p.parseDecorators(data.decoratorScope, classKeyword, decoratorContext|decoratorInFnArgs)
+
+			p.fnOrArrowDataParse.await = oldAwait
+			p.fnOrArrowDataParse.needsAsyncLoc = oldNeedsAsyncLoc
+		}
+
+		if !fn.HasRestArg && p.lexer.Token == js_lexer.TDotDotDot {
+			p.markSyntaxFeature(compat.RestArgument, p.lexer.Range())
+			p.lexer.Next()
+			fn.HasRestArg = true
+		}
+
+		isTypeScriptCtorField := false
+		isIdentifier := p.lexer.Token == js_lexer.TIdentifier
+		text := p.lexer.Identifier.String
+		arg := p.parseBinding(parseBindingOpts{})
+
+		if p.options.ts.Parse {
+			// Skip over TypeScript accessibility modifiers, which turn this argument
+			// into a class field when used inside a class constructor. This is known
+			// as a "parameter property" in TypeScript.
+			if isIdentifier && data.isConstructor {
+				for p.lexer.Token == js_lexer.TIdentifier || p.lexer.Token == js_lexer.TOpenBrace || p.lexer.Token == js_lexer.TOpenBracket {
+					if text != "public" && text != "private" && text != "protected" && text != "readonly" && text != "override" {
+						break
+					}
+					isTypeScriptCtorField = true
+
+					// TypeScript requires an identifier binding
+					if p.lexer.Token != js_lexer.TIdentifier {
+						p.lexer.Expect(js_lexer.TIdentifier)
+					}
+					text = p.lexer.Identifier.String
+
+					// Re-parse the binding (the current binding is the TypeScript keyword)
+					arg = p.parseBinding(parseBindingOpts{})
+				}
+			}
+
+			// "function foo(a?) {}"
+			if p.lexer.Token == js_lexer.TQuestion {
+				p.lexer.Next()
+			}
+
+			// "function foo(a: any) {}"
+			if p.lexer.Token == js_lexer.TColon {
+				p.lexer.Next()
+				p.skipTypeScriptType(js_ast.LLowest)
+			}
+		}
+
+		p.declareBinding(ast.SymbolHoisted, arg, parseStmtOpts{})
+
+		var defaultValueOrNil js_ast.Expr
+		if !fn.HasRestArg && p.lexer.Token == js_lexer.TEquals {
+			p.markSyntaxFeature(compat.DefaultArgument, p.lexer.Range())
+			p.lexer.Next()
+			defaultValueOrNil = p.parseExpr(js_ast.LComma)
+		}
+
+		fn.Args = append(fn.Args, js_ast.Arg{
+			Decorators:   decorators,
+			Binding:      arg,
+			DefaultOrNil: defaultValueOrNil,
+
+			// We need to track this because it affects code generation
+			IsTypeScriptCtorField: isTypeScriptCtorField,
+		})
+
+		if p.lexer.Token != js_lexer.TComma {
+			break
+		}
+		if fn.HasRestArg {
+			// JavaScript does not allow a comma after a rest argument
+			if data.isTypeScriptDeclare {
+				// TypeScript does allow a comma after a rest argument in a "declare" context
+				p.lexer.Next()
+			} else {
+				p.lexer.Expect(js_lexer.TCloseParen)
+			}
+			break
+		}
+		p.lexer.Next()
+	}
+
+	// Reserve the special name "arguments" in this scope. This ensures that it
+	// shadows any variable called "arguments" in any parent scopes. But only do
+	// this if it wasn't already declared above because arguments are allowed to
+	// be called "arguments", in which case the real "arguments" is inaccessible.
+	if _, ok := p.currentScope.Members["arguments"]; !ok {
+		fn.ArgumentsRef = p.declareSymbol(ast.SymbolArguments, fn.OpenParenLoc, "arguments")
+		p.symbols[fn.ArgumentsRef.InnerIndex].Flags |= ast.MustNotBeRenamed
+	}
+
+	p.lexer.Expect(js_lexer.TCloseParen)
+	p.fnOrArrowDataParse = oldFnOrArrowData
+
+	// "function foo(): any {}"
+	if p.options.ts.Parse && p.lexer.Token == js_lexer.TColon {
+		p.lexer.Next()
+		p.skipTypeScriptReturnType()
+	}
+
+	// "function foo(): any;"
+	if data.allowMissingBodyForTypeScript && p.lexer.Token != js_lexer.TOpenBrace {
+		p.lexer.ExpectOrInsertSemicolon()
+		return
+	}
+
+	fn.Body = p.parseFnBody(data)
+	hadBody = true
+	return
+}
+
+type fnKind uint8
+
+const (
+	fnStmt fnKind = iota
+	fnExpr
+)
+
+func (p *parser) validateFunctionName(fn js_ast.Fn, kind fnKind) {
+	// Prevent the function name from being the same as a function-specific keyword
+	if fn.Name != nil {
+		if fn.IsAsync && p.symbols[fn.Name.Ref.InnerIndex].OriginalName == "await" {
+			p.log.AddError(&p.tracker, js_lexer.RangeOfIdentifier(p.source, fn.Name.Loc),
+				"An async function cannot be named \"await\"")
+		} else if fn.IsGenerator && p.symbols[fn.Name.Ref.InnerIndex].OriginalName == "yield" && kind == fnExpr {
+			p.log.AddError(&p.tracker, js_lexer.RangeOfIdentifier(p.source, fn.Name.Loc),
+				"A generator function expression cannot be named \"yield\"")
+		}
+	}
+}
+
+func (p *parser) validateDeclaredSymbolName(loc logger.Loc, name string) {
+	if js_lexer.StrictModeReservedWords[name] {
+		p.markStrictModeFeature(reservedWord, js_lexer.RangeOfIdentifier(p.source, loc), name)
+	} else if isEvalOrArguments(name) {
+		p.markStrictModeFeature(evalOrArguments, js_lexer.RangeOfIdentifier(p.source, loc), name)
+	}
+}
+
+func (p *parser) parseClassStmt(loc logger.Loc, opts parseStmtOpts) js_ast.Stmt {
+	var name *ast.LocRef
+	classKeyword := p.lexer.Range()
+	if p.lexer.Token == js_lexer.TClass {
+		p.markSyntaxFeature(compat.Class, classKeyword)
+		p.lexer.Next()
+	} else {
+		p.lexer.Expected(js_lexer.TClass)
+	}
+
+	if !opts.isNameOptional || (p.lexer.Token == js_lexer.TIdentifier && (!p.options.ts.Parse || p.lexer.Identifier.String != "implements")) {
+		nameLoc := p.lexer.Loc()
+		nameText := p.lexer.Identifier.String
+		p.lexer.Expect(js_lexer.TIdentifier)
+		if p.fnOrArrowDataParse.await != allowIdent && nameText == "await" {
+			p.log.AddError(&p.tracker, js_lexer.RangeOfIdentifier(p.source, nameLoc), "Cannot use \"await\" as an identifier here:")
+		}
+		name = &ast.LocRef{Loc: nameLoc, Ref: ast.InvalidRef}
+		if !opts.isTypeScriptDeclare {
+			name.Ref = p.declareSymbol(ast.SymbolClass, nameLoc, nameText)
+		}
+	}
+
+	// Even anonymous classes can have TypeScript type parameters
+	if p.options.ts.Parse {
+		p.skipTypeScriptTypeParameters(allowInOutVarianceAnnotations | allowConstModifier)
+	}
+
+	classOpts := parseClassOpts{
+		isTypeScriptDeclare: opts.isTypeScriptDeclare,
+	}
+	if opts.deferredDecorators != nil {
+		classOpts.decorators = opts.deferredDecorators.decorators
+	}
+	scopeIndex := p.pushScopeForParsePass(js_ast.ScopeClassName, loc)
+	class := p.parseClass(classKeyword, name, classOpts)
+
+	if opts.isTypeScriptDeclare {
+		p.popAndDiscardScope(scopeIndex)
+
+		if opts.isNamespaceScope && opts.isExport {
+			p.hasNonLocalExportDeclareInsideNamespace = true
+		}
+
+		// Remember that this was a "declare class" so we can allow decorators on it
+		return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptSharedWasDeclareClass}
+	}
+
+	p.popScope()
+	return js_ast.Stmt{Loc: loc, Data: &js_ast.SClass{Class: class, IsExport: opts.isExport}}
+}
+
+func (p *parser) parseClassExpr(decorators []js_ast.Decorator) js_ast.Expr {
+	classKeyword := p.lexer.Range()
+	p.markSyntaxFeature(compat.Class, classKeyword)
+	p.lexer.Expect(js_lexer.TClass)
+	var name *ast.LocRef
+
+	opts := parseClassOpts{
+		decorators:       decorators,
+		decoratorContext: decoratorInClassExpr,
+	}
+	p.pushScopeForParsePass(js_ast.ScopeClassName, classKeyword.Loc)
+
+	// Parse an optional class name
+	if p.lexer.Token == js_lexer.TIdentifier {
+		if nameText := p.lexer.Identifier.String; !p.options.ts.Parse || nameText != "implements" {
+			if p.fnOrArrowDataParse.await != allowIdent && nameText == "await" {
+				p.log.AddError(&p.tracker, p.lexer.Range(), "Cannot use \"await\" as an identifier here:")
+			}
+			name = &ast.LocRef{Loc: p.lexer.Loc(), Ref: p.newSymbol(ast.SymbolOther, nameText)}
+			p.lexer.Next()
+		}
+	}
+
+	// Even anonymous classes can have TypeScript type parameters
+	if p.options.ts.Parse {
+		p.skipTypeScriptTypeParameters(allowInOutVarianceAnnotations | allowConstModifier)
+	}
+
+	class := p.parseClass(classKeyword, name, opts)
+
+	p.popScope()
+	return js_ast.Expr{Loc: classKeyword.Loc, Data: &js_ast.EClass{Class: class}}
+}
+
+type parseClassOpts struct {
+	decorators          []js_ast.Decorator
+	decoratorContext    decoratorContextFlags
+	isTypeScriptDeclare bool
+}
+
+// By the time we call this, the identifier and type parameters have already
+// been parsed. We need to start parsing from the "extends" clause.
+func (p *parser) parseClass(classKeyword logger.Range, name *ast.LocRef, classOpts parseClassOpts) js_ast.Class {
+	var extendsOrNil js_ast.Expr
+
+	if p.lexer.Token == js_lexer.TExtends {
+		p.lexer.Next()
+		extendsOrNil = p.parseExpr(js_ast.LNew)
+
+		// TypeScript's type argument parser inside expressions backtracks if the
+		// first token after the end of the type parameter list is "{", so the
+		// parsed expression above will have backtracked if there are any type
+		// arguments. This means we have to re-parse for any type arguments here.
+		// This seems kind of wasteful to me but it's what the official compiler
+		// does and it probably doesn't have that high of a performance overhead
+		// because "extends" clauses aren't that frequent, so it should be ok.
+		if p.options.ts.Parse {
+			p.skipTypeScriptTypeArguments(skipTypeScriptTypeArgumentsOpts{})
+		}
+	}
+
+	if p.options.ts.Parse && p.lexer.IsContextualKeyword("implements") {
+		p.lexer.Next()
+		for {
+			p.skipTypeScriptType(js_ast.LLowest)
+			if p.lexer.Token != js_lexer.TComma {
+				break
+			}
+			p.lexer.Next()
+		}
+	}
+
+	bodyLoc := p.lexer.Loc()
+	p.lexer.Expect(js_lexer.TOpenBrace)
+	properties := []js_ast.Property{}
+	hasPropertyDecorator := false
+
+	// Allow "in" and private fields inside class bodies
+	oldAllowIn := p.allowIn
+	oldAllowPrivateIdentifiers := p.allowPrivateIdentifiers
+	p.allowIn = true
+	p.allowPrivateIdentifiers = true
+
+	// A scope is needed for private identifiers
+	scopeIndex := p.pushScopeForParsePass(js_ast.ScopeClassBody, bodyLoc)
+
+	opts := propertyOpts{
+		isClass:          true,
+		decoratorScope:   p.currentScope,
+		decoratorContext: classOpts.decoratorContext,
+		classHasExtends:  extendsOrNil.Data != nil,
+		classKeyword:     classKeyword,
+	}
+	hasConstructor := false
+
+	for p.lexer.Token != js_lexer.TCloseBrace {
+		if p.lexer.Token == js_lexer.TSemicolon {
+			p.lexer.Next()
+			continue
+		}
+
+		// Parse decorators for this property
+		firstDecoratorLoc := p.lexer.Loc()
+		scopeIndex := len(p.scopesInOrder)
+		opts.decorators = p.parseDecorators(p.currentScope, classKeyword, opts.decoratorContext)
+		if len(opts.decorators) > 0 {
+			hasPropertyDecorator = true
+		}
+
+		// This property may turn out to be a type in TypeScript, which should be ignored
+		if property, ok := p.parseProperty(p.saveExprCommentsHere(), js_ast.PropertyField, opts, nil); ok {
+			properties = append(properties, property)
+
+			// Forbid decorators on class constructors
+			if key, ok := property.Key.Data.(*js_ast.EString); ok && helpers.UTF16EqualsString(key.Value, "constructor") {
+				if len(opts.decorators) > 0 {
+					p.log.AddError(&p.tracker, logger.Range{Loc: firstDecoratorLoc},
+						"Decorators are not allowed on class constructors")
+				}
+				if property.Kind.IsMethodDefinition() && !property.Flags.Has(js_ast.PropertyIsStatic) && !property.Flags.Has(js_ast.PropertyIsComputed) {
+					if hasConstructor {
+						p.log.AddError(&p.tracker, js_lexer.RangeOfIdentifier(p.source, property.Key.Loc),
+							"Classes cannot contain more than one constructor")
+					}
+					hasConstructor = true
+				}
+			}
+		} else if !classOpts.isTypeScriptDeclare && len(opts.decorators) > 0 {
+			p.log.AddError(&p.tracker, logger.Range{Loc: firstDecoratorLoc, Len: 1}, "Decorators are not valid here")
+			p.discardScopesUpTo(scopeIndex)
+		}
+	}
+
+	// Discard the private identifier scope inside a TypeScript "declare class"
+	if classOpts.isTypeScriptDeclare {
+		p.popAndDiscardScope(scopeIndex)
+	} else {
+		p.popScope()
+	}
+
+	p.allowIn = oldAllowIn
+	p.allowPrivateIdentifiers = oldAllowPrivateIdentifiers
+
+	closeBraceLoc := p.saveExprCommentsHere()
+	p.lexer.Expect(js_lexer.TCloseBrace)
+
+	// TypeScript has legacy behavior that uses assignment semantics instead of
+	// define semantics for class fields when "useDefineForClassFields" is enabled
+	// (in which case TypeScript behaves differently than JavaScript, which is
+	// arguably "wrong").
+	//
+	// This legacy behavior exists because TypeScript added class fields to
+	// TypeScript before they were added to JavaScript. They decided to go with
+	// assignment semantics for whatever reason. Later on TC39 decided to go with
+	// define semantics for class fields instead. This behaves differently if the
+	// base class has a setter with the same name.
+	//
+	// The value of "useDefineForClassFields" defaults to false when it's not
+	// specified and the target is earlier than "ES2022" since the class field
+	// language feature was added in ES2022. However, TypeScript's "target"
+	// setting currently defaults to "ES3" which unfortunately means that the
+	// "useDefineForClassFields" setting defaults to false (i.e. to "wrong").
+	//
+	// We default "useDefineForClassFields" to true (i.e. to "correct") instead.
+	// This is partially because our target defaults to "esnext", and partially
+	// because this is a legacy behavior that no one should be using anymore.
+	// Users that want the wrong behavior can either set "useDefineForClassFields"
+	// to false in "tsconfig.json" explicitly, or set TypeScript's "target" to
+	// "ES2021" or earlier in their in "tsconfig.json" file.
+	useDefineForClassFields := !p.options.ts.Parse || p.options.ts.Config.UseDefineForClassFields == config.True ||
+		(p.options.ts.Config.UseDefineForClassFields == config.Unspecified && p.options.ts.Config.Target != config.TSTargetBelowES2022)
+
+	return js_ast.Class{
+		ClassKeyword:  classKeyword,
+		Decorators:    classOpts.decorators,
+		Name:          name,
+		ExtendsOrNil:  extendsOrNil,
+		BodyLoc:       bodyLoc,
+		Properties:    properties,
+		CloseBraceLoc: closeBraceLoc,
+
+		// Always lower standard decorators if they are present and TypeScript's
+		// "useDefineForClassFields" setting is false even if the configured target
+		// environment supports decorators. This setting changes the behavior of
+		// class fields, and so we must lower decorators so they behave correctly.
+		ShouldLowerStandardDecorators: (len(classOpts.decorators) > 0 || hasPropertyDecorator) &&
+			((!p.options.ts.Parse && p.options.unsupportedJSFeatures.Has(compat.Decorators)) ||
+				(p.options.ts.Parse && p.options.ts.Config.ExperimentalDecorators != config.True &&
+					(p.options.unsupportedJSFeatures.Has(compat.Decorators) || !useDefineForClassFields))),
+
+		UseDefineForClassFields: useDefineForClassFields,
+	}
+}
+
+func (p *parser) parseLabelName() *ast.LocRef {
+	if p.lexer.Token != js_lexer.TIdentifier || p.lexer.HasNewlineBefore {
+		return nil
+	}
+
+	name := ast.LocRef{Loc: p.lexer.Loc(), Ref: p.storeNameInRef(p.lexer.Identifier)}
+	p.lexer.Next()
+	return &name
+}
+
+func (p *parser) parsePath() (logger.Range, string, *ast.ImportAssertOrWith, ast.ImportRecordFlags) {
+	var flags ast.ImportRecordFlags
+	pathRange := p.lexer.Range()
+	pathText := helpers.UTF16ToString(p.lexer.StringLiteral())
+	if p.lexer.Token == js_lexer.TNoSubstitutionTemplateLiteral {
+		p.lexer.Next()
+	} else {
+		p.lexer.Expect(js_lexer.TStringLiteral)
+	}
+
+	// See https://github.com/tc39/proposal-import-attributes for more info
+	var assertOrWith *ast.ImportAssertOrWith
+	if p.lexer.Token == js_lexer.TWith || (!p.lexer.HasNewlineBefore && p.lexer.IsContextualKeyword("assert")) {
+		// "import './foo.json' assert { type: 'json' }"
+		// "import './foo.json' with { type: 'json' }"
+		var entries []ast.AssertOrWithEntry
+		duplicates := make(map[string]logger.Range)
+		keyword := ast.WithKeyword
+		if p.lexer.Token != js_lexer.TWith {
+			keyword = ast.AssertKeyword
+		}
+		keywordLoc := p.saveExprCommentsHere()
+		p.lexer.Next()
+		openBraceLoc := p.saveExprCommentsHere()
+		p.lexer.Expect(js_lexer.TOpenBrace)
+
+		for p.lexer.Token != js_lexer.TCloseBrace {
+			// Parse the key
+			keyLoc := p.saveExprCommentsHere()
+			preferQuotedKey := false
+			var key []uint16
+			var keyText string
+			if p.lexer.IsIdentifierOrKeyword() {
+				keyText = p.lexer.Identifier.String
+				key = helpers.StringToUTF16(keyText)
+			} else if p.lexer.Token == js_lexer.TStringLiteral {
+				key = p.lexer.StringLiteral()
+				keyText = helpers.UTF16ToString(key)
+				preferQuotedKey = !p.options.minifySyntax
+			} else {
+				p.lexer.Expect(js_lexer.TIdentifier)
+			}
+			if prevRange, ok := duplicates[keyText]; ok {
+				what := "attribute"
+				if keyword == ast.AssertKeyword {
+					what = "assertion"
+				}
+				p.log.AddErrorWithNotes(&p.tracker, p.lexer.Range(), fmt.Sprintf("Duplicate import %s %q", what, keyText),
+					[]logger.MsgData{p.tracker.MsgData(prevRange, fmt.Sprintf("The first %q was here:", keyText))})
+			}
+			duplicates[keyText] = p.lexer.Range()
+			p.lexer.Next()
+			p.lexer.Expect(js_lexer.TColon)
+
+			// Parse the value
+			valueLoc := p.saveExprCommentsHere()
+			value := p.lexer.StringLiteral()
+			p.lexer.Expect(js_lexer.TStringLiteral)
+
+			entries = append(entries, ast.AssertOrWithEntry{
+				Key:             key,
+				KeyLoc:          keyLoc,
+				Value:           value,
+				ValueLoc:        valueLoc,
+				PreferQuotedKey: preferQuotedKey,
+			})
+
+			// Using "assert: { type: 'json' }" triggers special behavior
+			if keyword == ast.AssertKeyword && helpers.UTF16EqualsString(key, "type") && helpers.UTF16EqualsString(value, "json") {
+				flags |= ast.AssertTypeJSON
+			}
+
+			if p.lexer.Token != js_lexer.TComma {
+				break
+			}
+			p.lexer.Next()
+		}
+
+		closeBraceLoc := p.saveExprCommentsHere()
+		p.lexer.Expect(js_lexer.TCloseBrace)
+		if keyword == ast.AssertKeyword {
+			p.maybeWarnAboutAssertKeyword(keywordLoc)
+		}
+		assertOrWith = &ast.ImportAssertOrWith{
+			Entries:            entries,
+			Keyword:            keyword,
+			KeywordLoc:         keywordLoc,
+			InnerOpenBraceLoc:  openBraceLoc,
+			InnerCloseBraceLoc: closeBraceLoc,
+		}
+	}
+
+	return pathRange, pathText, assertOrWith, flags
+}
+
+// Let people know if they probably should be using "with" instead of "assert"
+func (p *parser) maybeWarnAboutAssertKeyword(loc logger.Loc) {
+	if p.options.unsupportedJSFeatures.Has(compat.ImportAssertions) && !p.options.unsupportedJSFeatures.Has(compat.ImportAttributes) {
+		where := config.PrettyPrintTargetEnvironment(p.options.originalTargetEnv, p.options.unsupportedJSFeatureOverridesMask)
+		msg := logger.Msg{
+			Kind:  logger.Warning,
+			Data:  p.tracker.MsgData(js_lexer.RangeOfIdentifier(p.source, loc), "The \"assert\" keyword is not supported in "+where),
+			Notes: []logger.MsgData{{Text: "Did you mean to use \"with\" instead of \"assert\"?"}},
+		}
+		msg.Data.Location.Suggestion = "with"
+		p.log.AddMsgID(logger.MsgID_JS_AssertToWith, msg)
+	}
+}
+
+// This assumes the "function" token has already been parsed
+func (p *parser) parseFnStmt(loc logger.Loc, opts parseStmtOpts, isAsync bool, asyncRange logger.Range) js_ast.Stmt {
+	isGenerator := p.lexer.Token == js_lexer.TAsterisk
+	hasError := false
+	if isAsync {
+		hasError = p.markAsyncFn(asyncRange, isGenerator)
+	}
+	if isGenerator {
+		if !hasError {
+			p.markSyntaxFeature(compat.Generator, p.lexer.Range())
+		}
+		p.lexer.Next()
+	}
+
+	switch opts.lexicalDecl {
+	case lexicalDeclForbid:
+		p.forbidLexicalDecl(loc)
+
+	// Allow certain function statements in certain single-statement contexts
+	case lexicalDeclAllowFnInsideIf, lexicalDeclAllowFnInsideLabel:
+		if opts.isTypeScriptDeclare || isGenerator || isAsync {
+			p.forbidLexicalDecl(loc)
+		}
+	}
+
+	var name *ast.LocRef
+	var nameText string
+
+	// The name is optional for "export default function() {}" pseudo-statements
+	if !opts.isNameOptional || p.lexer.Token == js_lexer.TIdentifier {
+		nameLoc := p.lexer.Loc()
+		nameText = p.lexer.Identifier.String
+		if !isAsync && p.fnOrArrowDataParse.await != allowIdent && nameText == "await" {
+			p.log.AddError(&p.tracker, js_lexer.RangeOfIdentifier(p.source, nameLoc), "Cannot use \"await\" as an identifier here:")
+		}
+		p.lexer.Expect(js_lexer.TIdentifier)
+		name = &ast.LocRef{Loc: nameLoc, Ref: ast.InvalidRef}
+	}
+
+	// Even anonymous functions can have TypeScript type parameters
+	if p.options.ts.Parse {
+		p.skipTypeScriptTypeParameters(allowConstModifier)
+	}
+
+	// Introduce a fake block scope for function declarations inside if statements
+	var ifStmtScopeIndex int
+	hasIfScope := opts.lexicalDecl == lexicalDeclAllowFnInsideIf
+	if hasIfScope {
+		ifStmtScopeIndex = p.pushScopeForParsePass(js_ast.ScopeBlock, loc)
+	}
+
+	scopeIndex := p.pushScopeForParsePass(js_ast.ScopeFunctionArgs, p.lexer.Loc())
+
+	await := allowIdent
+	yield := allowIdent
+	if isAsync {
+		await = allowExpr
+	}
+	if isGenerator {
+		yield = allowExpr
+	}
+
+	fn, hadBody := p.parseFn(name, logger.Range{}, 0, fnOrArrowDataParse{
+		needsAsyncLoc:       loc,
+		asyncRange:          asyncRange,
+		await:               await,
+		yield:               yield,
+		isTypeScriptDeclare: opts.isTypeScriptDeclare,
+
+		// Only allow omitting the body if we're parsing TypeScript
+		allowMissingBodyForTypeScript: p.options.ts.Parse,
+	})
+
+	// Don't output anything if it's just a forward declaration of a function
+	if opts.isTypeScriptDeclare || !hadBody {
+		p.popAndDiscardScope(scopeIndex)
+
+		// Balance the fake block scope introduced above
+		if hasIfScope {
+			p.popAndDiscardScope(ifStmtScopeIndex)
+		}
+
+		if opts.isTypeScriptDeclare && opts.isNamespaceScope && opts.isExport {
+			p.hasNonLocalExportDeclareInsideNamespace = true
+		}
+
+		return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptShared}
+	}
+
+	p.popScope()
+
+	// Only declare the function after we know if it had a body or not. Otherwise
+	// TypeScript code such as this will double-declare the symbol:
+	//
+	//     function foo(): void;
+	//     function foo(): void {}
+	//
+	if name != nil {
+		kind := ast.SymbolHoistedFunction
+		if isGenerator || isAsync {
+			kind = ast.SymbolGeneratorOrAsyncFunction
+		}
+		name.Ref = p.declareSymbol(kind, name.Loc, nameText)
+	}
+
+	// Balance the fake block scope introduced above
+	if hasIfScope {
+		p.popScope()
+	}
+
+	fn.HasIfScope = hasIfScope
+	p.validateFunctionName(fn, fnStmt)
+	if opts.hasNoSideEffectsComment && !p.options.ignoreDCEAnnotations {
+		fn.HasNoSideEffectsComment = true
+		if name != nil && !opts.isTypeScriptDeclare {
+			p.symbols[name.Ref.InnerIndex].Flags |= ast.CallCanBeUnwrappedIfUnused
+		}
+	}
+	return js_ast.Stmt{Loc: loc, Data: &js_ast.SFunction{Fn: fn, IsExport: opts.isExport}}
+}
+
+type deferredDecorators struct {
+	decorators []js_ast.Decorator
+}
+
+type decoratorContextFlags uint8
+
+const (
+	decoratorBeforeClassExpr = 1 << iota
+	decoratorInClassExpr
+	decoratorInFnArgs
+)
+
+func (p *parser) parseDecorators(decoratorScope *js_ast.Scope, classKeyword logger.Range, context decoratorContextFlags) (decorators []js_ast.Decorator) {
+	if p.lexer.Token == js_lexer.TAt {
+		if p.options.ts.Parse {
+			if p.options.ts.Config.ExperimentalDecorators == config.True {
+				if (context & decoratorInClassExpr) != 0 {
+					p.lexer.AddRangeErrorWithNotes(p.lexer.Range(), "TypeScript experimental decorators can only be used with class declarations",
+						[]logger.MsgData{p.tracker.MsgData(classKeyword, "This is a class expression, not a class declaration:")})
+				} else if (context & decoratorBeforeClassExpr) != 0 {
+					p.log.AddError(&p.tracker, p.lexer.Range(), "TypeScript experimental decorators cannot be used in expression position")
+				}
+			} else {
+				if (context&decoratorInFnArgs) != 0 && p.options.ts.Config.ExperimentalDecorators != config.True {
+					p.log.AddErrorWithNotes(&p.tracker, p.lexer.Range(), "Parameter decorators only work when experimental decorators are enabled", []logger.MsgData{{
+						Text: "You can enable experimental decorators by adding \"experimentalDecorators\": true to your \"tsconfig.json\" file.",
+					}})
+				}
+			}
+		} else {
+			if (context & decoratorInFnArgs) != 0 {
+				p.log.AddError(&p.tracker, p.lexer.Range(), "Parameter decorators are not allowed in JavaScript")
+			}
+		}
+	}
+
+	// TypeScript decorators cause us to temporarily revert to the scope that
+	// encloses the class declaration, since that's where the generated code
+	// for TypeScript decorators will be inserted.
+	oldScope := p.currentScope
+	p.currentScope = decoratorScope
+
+	for p.lexer.Token == js_lexer.TAt {
+		atLoc := p.lexer.Loc()
+		p.lexer.Next()
+
+		var value js_ast.Expr
+		if p.options.ts.Parse && p.options.ts.Config.ExperimentalDecorators == config.True {
+			// TypeScript's experimental decorator syntax is more permissive than
+			// JavaScript. Parse a new/call expression with "exprFlagDecorator" so
+			// we ignore EIndex expressions, since they may be part of a computed
+			// property:
+			//
+			//   class Foo {
+			//     @foo ['computed']() {}
+			//   }
+			//
+			// This matches the behavior of the TypeScript compiler.
+			p.parseExperimentalDecoratorNesting++
+			value = p.parseExprWithFlags(js_ast.LNew, exprFlagDecorator)
+			p.parseExperimentalDecoratorNesting--
+		} else {
+			// JavaScript's decorator syntax is more restrictive. Parse it using a
+			// special parser that doesn't allow normal expressions (e.g. "?.").
+			value = p.parseDecorator()
+		}
+		decorators = append(decorators, js_ast.Decorator{
+			Value:            value,
+			AtLoc:            atLoc,
+			OmitNewlineAfter: !p.lexer.HasNewlineBefore,
+		})
+	}
+
+	// Avoid "popScope" because this decorator scope is not hierarchical
+	p.currentScope = oldScope
+	return decorators
+}
+
+func (p *parser) parseDecorator() js_ast.Expr {
+	if p.lexer.Token == js_lexer.TOpenParen {
+		p.lexer.Next()
+		value := p.parseExpr(js_ast.LLowest)
+		p.lexer.Expect(js_lexer.TCloseParen)
+		return value
+	}
+
+	name := p.lexer.Identifier
+	nameRange := p.lexer.Range()
+	p.lexer.Expect(js_lexer.TIdentifier)
+
+	// Forbid invalid identifiers
+	if (p.fnOrArrowDataParse.await != allowIdent && name.String == "await") ||
+		(p.fnOrArrowDataParse.yield != allowIdent && name.String == "yield") {
+		p.log.AddError(&p.tracker, nameRange, fmt.Sprintf("Cannot use %q as an identifier here:", name.String))
+	}
+
+	memberExpr := js_ast.Expr{Loc: nameRange.Loc, Data: &js_ast.EIdentifier{Ref: p.storeNameInRef(name)}}
+
+	// Custom error reporting for error recovery
+	var syntaxError logger.MsgData
+	wrapRange := nameRange
+
+loop:
+	for {
+		switch p.lexer.Token {
+		case js_lexer.TExclamation:
+			// Skip over TypeScript non-null assertions
+			if p.lexer.HasNewlineBefore {
+				break loop
+			}
+			if !p.options.ts.Parse {
+				p.lexer.Unexpected()
+			}
+			wrapRange.Len = p.lexer.Range().End() - wrapRange.Loc.Start
+			p.lexer.Next()
+
+		case js_lexer.TDot, js_lexer.TQuestionDot:
+			// The grammar for "DecoratorMemberExpression" currently forbids "?."
+			if p.lexer.Token == js_lexer.TQuestionDot && syntaxError.Location == nil {
+				syntaxError = p.tracker.MsgData(p.lexer.Range(), "JavaScript decorator syntax does not allow \"?.\" here")
+			}
+
+			p.lexer.Next()
+			wrapRange.Len = p.lexer.Range().End() - wrapRange.Loc.Start
+
+			if p.lexer.Token == js_lexer.TPrivateIdentifier {
+				name := p.lexer.Identifier
+				memberExpr.Data = &js_ast.EIndex{
+					Target: memberExpr,
+					Index:  js_ast.Expr{Loc: p.lexer.Loc(), Data: &js_ast.EPrivateIdentifier{Ref: p.storeNameInRef(name)}},
+				}
+				p.reportPrivateNameUsage(name.String)
+				p.lexer.Next()
+			} else {
+				memberExpr.Data = &js_ast.EDot{
+					Target:  memberExpr,
+					Name:    p.lexer.Identifier.String,
+					NameLoc: p.lexer.Loc(),
+				}
+				p.lexer.Expect(js_lexer.TIdentifier)
+			}
+
+		case js_lexer.TOpenParen:
+			args, closeParenLoc, isMultiLine := p.parseCallArgs()
+			memberExpr.Data = &js_ast.ECall{
+				Target:        memberExpr,
+				Args:          args,
+				CloseParenLoc: closeParenLoc,
+				IsMultiLine:   isMultiLine,
+				Kind:          js_ast.TargetWasOriginallyPropertyAccess,
+			}
+			wrapRange.Len = closeParenLoc.Start + 1 - wrapRange.Loc.Start
+
+			// The grammar for "DecoratorCallExpression" currently forbids anything after it
+			if p.lexer.Token == js_lexer.TDot {
+				if syntaxError.Location == nil {
+					syntaxError = p.tracker.MsgData(p.lexer.Range(), "JavaScript decorator syntax does not allow \".\" after a call expression")
+				}
+				continue
+			}
+			break loop
+
+		default:
+			// "@x<y>"
+			// "@x.y<z>"
+			if !p.skipTypeScriptTypeArguments(skipTypeScriptTypeArgumentsOpts{}) {
+				break loop
+			}
+		}
+	}
+
+	// Suggest that non-decorator expressions be wrapped in parentheses
+	if syntaxError.Location != nil {
+		var notes []logger.MsgData
+		if text := p.source.TextForRange(wrapRange); !strings.ContainsRune(text, '\n') {
+			note := p.tracker.MsgData(wrapRange, "Wrap this decorator in parentheses to allow arbitrary expressions:")
+			note.Location.Suggestion = fmt.Sprintf("(%s)", text)
+			notes = []logger.MsgData{note}
+		}
+		p.log.AddMsg(logger.Msg{
+			Kind:  logger.Error,
+			Data:  syntaxError,
+			Notes: notes,
+		})
+	}
+
+	return memberExpr
+}
+
+type lexicalDecl uint8
+
+const (
+	lexicalDeclForbid lexicalDecl = iota
+	lexicalDeclAllowAll
+	lexicalDeclAllowFnInsideIf
+	lexicalDeclAllowFnInsideLabel
+)
+
+type parseStmtOpts struct {
+	deferredDecorators      *deferredDecorators
+	lexicalDecl             lexicalDecl
+	isModuleScope           bool
+	isNamespaceScope        bool
+	isExport                bool
+	isExportDefault         bool
+	isNameOptional          bool // For "export default" pseudo-statements
+	isTypeScriptDeclare     bool
+	isForLoopInit           bool
+	isForAwaitLoopInit      bool
+	allowDirectivePrologue  bool
+	hasNoSideEffectsComment bool
+	isUsingStmt             bool
+}
+
+func (p *parser) parseStmt(opts parseStmtOpts) js_ast.Stmt {
+	loc := p.lexer.Loc()
+
+	if (p.lexer.HasCommentBefore & js_lexer.NoSideEffectsCommentBefore) != 0 {
+		opts.hasNoSideEffectsComment = true
+	}
+
+	// Do not attach any leading comments to the next expression
+	p.lexer.CommentsBeforeToken = p.lexer.CommentsBeforeToken[:0]
+
+	switch p.lexer.Token {
+	case js_lexer.TSemicolon:
+		p.lexer.Next()
+		return js_ast.Stmt{Loc: loc, Data: js_ast.SEmptyShared}
+
+	case js_lexer.TExport:
+		previousExportKeyword := p.esmExportKeyword
+		if opts.isModuleScope {
+			p.esmExportKeyword = p.lexer.Range()
+		} else if !opts.isNamespaceScope {
+			p.lexer.Unexpected()
+		}
+		p.lexer.Next()
+
+		switch p.lexer.Token {
+		case js_lexer.TClass, js_lexer.TConst, js_lexer.TFunction, js_lexer.TVar, js_lexer.TAt:
+			opts.isExport = true
+			return p.parseStmt(opts)
+
+		case js_lexer.TImport:
+			// "export import foo = bar"
+			if p.options.ts.Parse && (opts.isModuleScope || opts.isNamespaceScope) {
+				opts.isExport = true
+				return p.parseStmt(opts)
+			}
+
+			p.lexer.Unexpected()
+			return js_ast.Stmt{}
+
+		case js_lexer.TEnum:
+			if !p.options.ts.Parse {
+				p.lexer.Unexpected()
+			}
+			opts.isExport = true
+			return p.parseStmt(opts)
+
+		case js_lexer.TIdentifier:
+			if p.lexer.IsContextualKeyword("let") {
+				opts.isExport = true
+				return p.parseStmt(opts)
+			}
+
+			if p.lexer.IsContextualKeyword("as") {
+				// "export as namespace ns;"
+				p.lexer.Next()
+				p.lexer.ExpectContextualKeyword("namespace")
+				p.lexer.Expect(js_lexer.TIdentifier)
+				p.lexer.ExpectOrInsertSemicolon()
+				return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptShared}
+			}
+
+			if p.lexer.IsContextualKeyword("async") {
+				// "export async function foo() {}"
+				asyncRange := p.lexer.Range()
+				p.lexer.Next()
+				if p.lexer.HasNewlineBefore {
+					p.log.AddError(&p.tracker, logger.Range{Loc: logger.Loc{Start: asyncRange.End()}},
+						"Unexpected newline after \"async\"")
+					panic(js_lexer.LexerPanic{})
+				}
+				p.lexer.Expect(js_lexer.TFunction)
+				opts.isExport = true
+				return p.parseFnStmt(loc, opts, true /* isAsync */, asyncRange)
+			}
+
+			if p.options.ts.Parse {
+				switch p.lexer.Identifier.String {
+				case "type":
+					// "export type foo = ..."
+					typeRange := p.lexer.Range()
+					p.lexer.Next()
+					if p.lexer.HasNewlineBefore && p.lexer.Token != js_lexer.TOpenBrace && p.lexer.Token != js_lexer.TAsterisk {
+						p.log.AddError(&p.tracker, logger.Range{Loc: logger.Loc{Start: typeRange.End()}},
+							"Unexpected newline after \"type\"")
+						panic(js_lexer.LexerPanic{})
+					}
+					p.skipTypeScriptTypeStmt(parseStmtOpts{isModuleScope: opts.isModuleScope, isExport: true})
+					return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptShared}
+
+				case "namespace", "abstract", "module", "interface":
+					// "export namespace Foo {}"
+					// "export abstract class Foo {}"
+					// "export module Foo {}"
+					// "export interface Foo {}"
+					opts.isExport = true
+					return p.parseStmt(opts)
+
+				case "declare":
+					// "export declare class Foo {}"
+					opts.isExport = true
+					opts.lexicalDecl = lexicalDeclAllowAll
+					opts.isTypeScriptDeclare = true
+					return p.parseStmt(opts)
+				}
+			}
+
+			p.lexer.Unexpected()
+			return js_ast.Stmt{}
+
+		case js_lexer.TDefault:
+			if !opts.isModuleScope && (!opts.isNamespaceScope || !opts.isTypeScriptDeclare) {
+				p.lexer.Unexpected()
+			}
+
+			defaultLoc := p.lexer.Loc()
+			p.lexer.Next()
+
+			// Also pick up comments after the "default" keyword
+			if (p.lexer.HasCommentBefore & js_lexer.NoSideEffectsCommentBefore) != 0 {
+				opts.hasNoSideEffectsComment = true
+			}
+
+			// The default name is lazily generated only if no other name is present
+			createDefaultName := func() ast.LocRef {
+				// This must be named "default" for when "--keep-names" is active
+				defaultName := ast.LocRef{Loc: defaultLoc, Ref: p.newSymbol(ast.SymbolOther, "default")}
+				p.currentScope.Generated = append(p.currentScope.Generated, defaultName.Ref)
+				return defaultName
+			}
+
+			// "export default async function() {}"
+			// "export default async function foo() {}"
+			if p.lexer.IsContextualKeyword("async") {
+				asyncRange := p.lexer.Range()
+				p.lexer.Next()
+
+				if p.lexer.Token == js_lexer.TFunction && !p.lexer.HasNewlineBefore {
+					p.lexer.Next()
+					stmt := p.parseFnStmt(loc, parseStmtOpts{
+						isNameOptional:          true,
+						lexicalDecl:             lexicalDeclAllowAll,
+						hasNoSideEffectsComment: opts.hasNoSideEffectsComment,
+					}, true /* isAsync */, asyncRange)
+					if _, ok := stmt.Data.(*js_ast.STypeScript); ok {
+						return stmt // This was just a type annotation
+					}
+
+					// Use the statement name if present, since it's a better name
+					var defaultName ast.LocRef
+					if s, ok := stmt.Data.(*js_ast.SFunction); ok && s.Fn.Name != nil {
+						defaultName = ast.LocRef{Loc: defaultLoc, Ref: s.Fn.Name.Ref}
+					} else {
+						defaultName = createDefaultName()
+					}
+
+					return js_ast.Stmt{Loc: loc, Data: &js_ast.SExportDefault{DefaultName: defaultName, Value: stmt}}
+				}
+
+				defaultName := createDefaultName()
+				expr := p.parseSuffix(p.parseAsyncPrefixExpr(asyncRange, js_ast.LComma, 0), js_ast.LComma, nil, 0)
+				p.lexer.ExpectOrInsertSemicolon()
+				return js_ast.Stmt{Loc: loc, Data: &js_ast.SExportDefault{
+					DefaultName: defaultName, Value: js_ast.Stmt{Loc: loc, Data: &js_ast.SExpr{Value: expr}}}}
+			}
+
+			// "export default class {}"
+			// "export default class Foo {}"
+			// "export default @x class {}"
+			// "export default @x class Foo {}"
+			// "export default function() {}"
+			// "export default function foo() {}"
+			// "export default interface Foo {}"
+			// "export default interface + 1"
+			if p.lexer.Token == js_lexer.TFunction || p.lexer.Token == js_lexer.TClass || p.lexer.Token == js_lexer.TAt ||
+				(p.options.ts.Parse && p.lexer.IsContextualKeyword("interface")) {
+				stmt := p.parseStmt(parseStmtOpts{
+					deferredDecorators:      opts.deferredDecorators,
+					isNameOptional:          true,
+					isExportDefault:         true,
+					lexicalDecl:             lexicalDeclAllowAll,
+					hasNoSideEffectsComment: opts.hasNoSideEffectsComment,
+				})
+
+				// Use the statement name if present, since it's a better name
+				var defaultName ast.LocRef
+				switch s := stmt.Data.(type) {
+				case *js_ast.STypeScript, *js_ast.SExpr:
+					return stmt // Handle the "interface" case above
+				case *js_ast.SFunction:
+					if s.Fn.Name != nil {
+						defaultName = ast.LocRef{Loc: defaultLoc, Ref: s.Fn.Name.Ref}
+					} else {
+						defaultName = createDefaultName()
+					}
+				case *js_ast.SClass:
+					if s.Class.Name != nil {
+						defaultName = ast.LocRef{Loc: defaultLoc, Ref: s.Class.Name.Ref}
+					} else {
+						defaultName = createDefaultName()
+					}
+				default:
+					panic("Internal error")
+				}
+				return js_ast.Stmt{Loc: loc, Data: &js_ast.SExportDefault{DefaultName: defaultName, Value: stmt}}
+			}
+
+			isIdentifier := p.lexer.Token == js_lexer.TIdentifier
+			name := p.lexer.Identifier.String
+			expr := p.parseExpr(js_ast.LComma)
+
+			// "export default abstract class {}"
+			// "export default abstract class Foo {}"
+			if p.options.ts.Parse && isIdentifier && name == "abstract" && !p.lexer.HasNewlineBefore {
+				if _, ok := expr.Data.(*js_ast.EIdentifier); ok && p.lexer.Token == js_lexer.TClass {
+					stmt := p.parseClassStmt(loc, parseStmtOpts{
+						deferredDecorators: opts.deferredDecorators,
+						isNameOptional:     true,
+					})
+
+					// Use the statement name if present, since it's a better name
+					var defaultName ast.LocRef
+					if s, ok := stmt.Data.(*js_ast.SClass); ok && s.Class.Name != nil {
+						defaultName = ast.LocRef{Loc: defaultLoc, Ref: s.Class.Name.Ref}
+					} else {
+						defaultName = createDefaultName()
+					}
+
+					return js_ast.Stmt{Loc: loc, Data: &js_ast.SExportDefault{DefaultName: defaultName, Value: stmt}}
+				}
+			}
+
+			p.lexer.ExpectOrInsertSemicolon()
+			defaultName := createDefaultName()
+			return js_ast.Stmt{Loc: loc, Data: &js_ast.SExportDefault{
+				DefaultName: defaultName, Value: js_ast.Stmt{Loc: loc, Data: &js_ast.SExpr{Value: expr}}}}
+
+		case js_lexer.TAsterisk:
+			if !opts.isModuleScope && (!opts.isNamespaceScope || !opts.isTypeScriptDeclare) {
+				p.lexer.Unexpected()
+			}
+
+			p.lexer.Next()
+			var namespaceRef ast.Ref
+			var alias *js_ast.ExportStarAlias
+			var pathRange logger.Range
+			var pathText string
+			var assertOrWith *ast.ImportAssertOrWith
+			var flags ast.ImportRecordFlags
+
+			if p.lexer.IsContextualKeyword("as") {
+				// "export * as ns from 'path'"
+				p.lexer.Next()
+				name := p.parseClauseAlias("export")
+				namespaceRef = p.storeNameInRef(name)
+				alias = &js_ast.ExportStarAlias{Loc: p.lexer.Loc(), OriginalName: name.String}
+				p.lexer.Next()
+				p.lexer.ExpectContextualKeyword("from")
+				pathRange, pathText, assertOrWith, flags = p.parsePath()
+			} else {
+				// "export * from 'path'"
+				p.lexer.ExpectContextualKeyword("from")
+				pathRange, pathText, assertOrWith, flags = p.parsePath()
+				name := js_ast.GenerateNonUniqueNameFromPath(pathText) + "_star"
+				namespaceRef = p.storeNameInRef(js_lexer.MaybeSubstring{String: name})
+			}
+			importRecordIndex := p.addImportRecord(ast.ImportStmt, pathRange, pathText, assertOrWith, flags)
+
+			// Export-star statements anywhere in the file disable top-level const
+			// local prefix because import cycles can be used to trigger TDZ
+			p.currentScope.IsAfterConstLocalPrefix = true
+
+			p.lexer.ExpectOrInsertSemicolon()
+			return js_ast.Stmt{Loc: loc, Data: &js_ast.SExportStar{
+				NamespaceRef:      namespaceRef,
+				Alias:             alias,
+				ImportRecordIndex: importRecordIndex,
+			}}
+
+		case js_lexer.TOpenBrace:
+			if !opts.isModuleScope && (!opts.isNamespaceScope || !opts.isTypeScriptDeclare) {
+				p.lexer.Unexpected()
+			}
+
+			items, isSingleLine := p.parseExportClause()
+			if p.lexer.IsContextualKeyword("from") {
+				// "export {} from 'path'"
+				p.lexer.Next()
+				pathLoc, pathText, assertOrWith, flags := p.parsePath()
+				importRecordIndex := p.addImportRecord(ast.ImportStmt, pathLoc, pathText, assertOrWith, flags)
+				name := "import_" + js_ast.GenerateNonUniqueNameFromPath(pathText)
+				namespaceRef := p.storeNameInRef(js_lexer.MaybeSubstring{String: name})
+
+				// Export clause statements anywhere in the file disable top-level const
+				// local prefix because import cycles can be used to trigger TDZ
+				p.currentScope.IsAfterConstLocalPrefix = true
+
+				p.lexer.ExpectOrInsertSemicolon()
+				return js_ast.Stmt{Loc: loc, Data: &js_ast.SExportFrom{
+					Items:             items,
+					NamespaceRef:      namespaceRef,
+					ImportRecordIndex: importRecordIndex,
+					IsSingleLine:      isSingleLine,
+				}}
+			}
+
+			p.lexer.ExpectOrInsertSemicolon()
+			return js_ast.Stmt{Loc: loc, Data: &js_ast.SExportClause{Items: items, IsSingleLine: isSingleLine}}
+
+		case js_lexer.TEquals:
+			// "export = value;"
+			p.esmExportKeyword = previousExportKeyword // This wasn't an ESM export statement after all
+			if p.options.ts.Parse {
+				p.lexer.Next()
+				value := p.parseExpr(js_ast.LLowest)
+				p.lexer.ExpectOrInsertSemicolon()
+				return js_ast.Stmt{Loc: loc, Data: &js_ast.SExportEquals{Value: value}}
+			}
+			p.lexer.Unexpected()
+			return js_ast.Stmt{}
+
+		default:
+			p.lexer.Unexpected()
+			return js_ast.Stmt{}
+		}
+
+	case js_lexer.TFunction:
+		p.lexer.Next()
+		return p.parseFnStmt(loc, opts, false /* isAsync */, logger.Range{})
+
+	case js_lexer.TEnum:
+		if !p.options.ts.Parse {
+			p.lexer.Unexpected()
+		}
+		return p.parseTypeScriptEnumStmt(loc, opts)
+
+	case js_lexer.TAt:
+		// Parse decorators before class statements, which are potentially exported
+		scopeIndex := len(p.scopesInOrder)
+		decorators := p.parseDecorators(p.currentScope, logger.Range{}, 0)
+
+		// "@x export @y class Foo {}"
+		if opts.deferredDecorators != nil {
+			p.log.AddError(&p.tracker, logger.Range{Loc: loc, Len: 1}, "Decorators are not valid here")
+			p.discardScopesUpTo(scopeIndex)
+			return p.parseStmt(opts)
+		}
+
+		// If this turns out to be a "declare class" statement, we need to undo the
+		// scopes that were potentially pushed while parsing the decorator arguments.
+		// That can look like any one of the following:
+		//
+		//   "@decorator declare class Foo {}"
+		//   "@decorator declare abstract class Foo {}"
+		//   "@decorator export declare class Foo {}"
+		//   "@decorator export declare abstract class Foo {}"
+		//
+		opts.deferredDecorators = &deferredDecorators{
+			decorators: decorators,
+		}
+
+		stmt := p.parseStmt(opts)
+
+		// Check for valid decorator targets
+		switch s := stmt.Data.(type) {
+		case *js_ast.SClass:
+			return stmt
+
+		case *js_ast.SExportDefault:
+			switch s.Value.Data.(type) {
+			case *js_ast.SClass:
+				return stmt
+			}
+
+		case *js_ast.STypeScript:
+			if s.WasDeclareClass {
+				// If this is a type declaration, discard any scopes that were pushed
+				// while parsing decorators. Unlike with the class statements above,
+				// these scopes won't end up being visited during the upcoming visit
+				// pass because type declarations aren't visited at all.
+				p.discardScopesUpTo(scopeIndex)
+				return stmt
+			}
+		}
+
+		// Forbid decorators on anything other than a class statement
+		p.log.AddError(&p.tracker, logger.Range{Loc: loc, Len: 1}, "Decorators are not valid here")
+		stmt.Data = js_ast.STypeScriptShared
+		p.discardScopesUpTo(scopeIndex)
+		return stmt
+
+	case js_lexer.TClass:
+		if opts.lexicalDecl != lexicalDeclAllowAll {
+			p.forbidLexicalDecl(loc)
+		}
+		return p.parseClassStmt(loc, opts)
+
+	case js_lexer.TVar:
+		p.lexer.Next()
+		decls := p.parseAndDeclareDecls(ast.SymbolHoisted, opts)
+		p.lexer.ExpectOrInsertSemicolon()
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.SLocal{
+			Kind:     js_ast.LocalVar,
+			Decls:    decls,
+			IsExport: opts.isExport,
+		}}
+
+	case js_lexer.TConst:
+		if opts.lexicalDecl != lexicalDeclAllowAll {
+			p.forbidLexicalDecl(loc)
+		}
+		p.markSyntaxFeature(compat.ConstAndLet, p.lexer.Range())
+		p.lexer.Next()
+
+		if p.options.ts.Parse && p.lexer.Token == js_lexer.TEnum {
+			return p.parseTypeScriptEnumStmt(loc, opts)
+		}
+
+		decls := p.parseAndDeclareDecls(ast.SymbolConst, opts)
+		p.lexer.ExpectOrInsertSemicolon()
+		if !opts.isTypeScriptDeclare {
+			p.requireInitializers(js_ast.LocalConst, decls)
+		}
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.SLocal{
+			Kind:     js_ast.LocalConst,
+			Decls:    decls,
+			IsExport: opts.isExport,
+		}}
+
+	case js_lexer.TIf:
+		p.lexer.Next()
+		p.lexer.Expect(js_lexer.TOpenParen)
+		test := p.parseExpr(js_ast.LLowest)
+		p.lexer.Expect(js_lexer.TCloseParen)
+		isSingleLineYes := !p.lexer.HasNewlineBefore && p.lexer.Token != js_lexer.TOpenBrace
+		yes := p.parseStmt(parseStmtOpts{lexicalDecl: lexicalDeclAllowFnInsideIf})
+		var noOrNil js_ast.Stmt
+		var isSingleLineNo bool
+		if p.lexer.Token == js_lexer.TElse {
+			p.lexer.Next()
+			isSingleLineNo = !p.lexer.HasNewlineBefore && p.lexer.Token != js_lexer.TOpenBrace
+			noOrNil = p.parseStmt(parseStmtOpts{lexicalDecl: lexicalDeclAllowFnInsideIf})
+		}
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.SIf{Test: test, Yes: yes, NoOrNil: noOrNil, IsSingleLineYes: isSingleLineYes, IsSingleLineNo: isSingleLineNo}}
+
+	case js_lexer.TDo:
+		p.lexer.Next()
+		body := p.parseStmt(parseStmtOpts{})
+		p.lexer.Expect(js_lexer.TWhile)
+		p.lexer.Expect(js_lexer.TOpenParen)
+		test := p.parseExpr(js_ast.LLowest)
+		p.lexer.Expect(js_lexer.TCloseParen)
+
+		// This is a weird corner case where automatic semicolon insertion applies
+		// even without a newline present
+		if p.lexer.Token == js_lexer.TSemicolon {
+			p.lexer.Next()
+		}
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.SDoWhile{Body: body, Test: test}}
+
+	case js_lexer.TWhile:
+		p.lexer.Next()
+		p.lexer.Expect(js_lexer.TOpenParen)
+		test := p.parseExpr(js_ast.LLowest)
+		p.lexer.Expect(js_lexer.TCloseParen)
+		isSingleLineBody := !p.lexer.HasNewlineBefore && p.lexer.Token != js_lexer.TOpenBrace
+		body := p.parseStmt(parseStmtOpts{})
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.SWhile{Test: test, Body: body, IsSingleLineBody: isSingleLineBody}}
+
+	case js_lexer.TWith:
+		p.lexer.Next()
+		p.lexer.Expect(js_lexer.TOpenParen)
+		test := p.parseExpr(js_ast.LLowest)
+		bodyLoc := p.lexer.Loc()
+		p.lexer.Expect(js_lexer.TCloseParen)
+
+		// Push a scope so we make sure to prevent any bare identifiers referenced
+		// within the body from being renamed. Renaming them might change the
+		// semantics of the code.
+		p.pushScopeForParsePass(js_ast.ScopeWith, bodyLoc)
+		isSingleLineBody := !p.lexer.HasNewlineBefore && p.lexer.Token != js_lexer.TOpenBrace
+		body := p.parseStmt(parseStmtOpts{})
+		p.popScope()
+
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.SWith{Value: test, BodyLoc: bodyLoc, Body: body, IsSingleLineBody: isSingleLineBody}}
+
+	case js_lexer.TSwitch:
+		p.lexer.Next()
+		p.lexer.Expect(js_lexer.TOpenParen)
+		test := p.parseExpr(js_ast.LLowest)
+		p.lexer.Expect(js_lexer.TCloseParen)
+
+		bodyLoc := p.lexer.Loc()
+		p.pushScopeForParsePass(js_ast.ScopeBlock, bodyLoc)
+		defer p.popScope()
+
+		p.lexer.Expect(js_lexer.TOpenBrace)
+		cases := []js_ast.Case{}
+		foundDefault := false
+
+		for p.lexer.Token != js_lexer.TCloseBrace {
+			var value js_ast.Expr
+			body := []js_ast.Stmt{}
+			caseLoc := p.saveExprCommentsHere()
+
+			if p.lexer.Token == js_lexer.TDefault {
+				if foundDefault {
+					p.log.AddError(&p.tracker, p.lexer.Range(), "Multiple default clauses are not allowed")
+					panic(js_lexer.LexerPanic{})
+				}
+				foundDefault = true
+				p.lexer.Next()
+				p.lexer.Expect(js_lexer.TColon)
+			} else {
+				p.lexer.Expect(js_lexer.TCase)
+				value = p.parseExpr(js_ast.LLowest)
+				p.lexer.Expect(js_lexer.TColon)
+			}
+
+		caseBody:
+			for {
+				switch p.lexer.Token {
+				case js_lexer.TCloseBrace, js_lexer.TCase, js_lexer.TDefault:
+					break caseBody
+
+				default:
+					body = append(body, p.parseStmt(parseStmtOpts{lexicalDecl: lexicalDeclAllowAll}))
+				}
+			}
+
+			cases = append(cases, js_ast.Case{ValueOrNil: value, Body: body, Loc: caseLoc})
+		}
+
+		closeBraceLoc := p.lexer.Loc()
+		p.lexer.Expect(js_lexer.TCloseBrace)
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.SSwitch{
+			Test:          test,
+			Cases:         cases,
+			BodyLoc:       bodyLoc,
+			CloseBraceLoc: closeBraceLoc,
+		}}
+
+	case js_lexer.TTry:
+		p.lexer.Next()
+		blockLoc := p.lexer.Loc()
+		p.lexer.Expect(js_lexer.TOpenBrace)
+		p.pushScopeForParsePass(js_ast.ScopeBlock, loc)
+		body := p.parseStmtsUpTo(js_lexer.TCloseBrace, parseStmtOpts{})
+		p.popScope()
+		closeBraceLoc := p.lexer.Loc()
+		p.lexer.Next()
+
+		var catch *js_ast.Catch = nil
+		var finally *js_ast.Finally = nil
+
+		if p.lexer.Token == js_lexer.TCatch {
+			catchLoc := p.lexer.Loc()
+			p.pushScopeForParsePass(js_ast.ScopeCatchBinding, catchLoc)
+			p.lexer.Next()
+			var bindingOrNil js_ast.Binding
+
+			// The catch binding is optional, and can be omitted
+			if p.lexer.Token == js_lexer.TOpenBrace {
+				if p.options.unsupportedJSFeatures.Has(compat.OptionalCatchBinding) {
+					// Generate a new symbol for the catch binding for older browsers
+					ref := p.newSymbol(ast.SymbolOther, "e")
+					p.currentScope.Generated = append(p.currentScope.Generated, ref)
+					bindingOrNil = js_ast.Binding{Loc: p.lexer.Loc(), Data: &js_ast.BIdentifier{Ref: ref}}
+				}
+			} else {
+				p.lexer.Expect(js_lexer.TOpenParen)
+				bindingOrNil = p.parseBinding(parseBindingOpts{})
+
+				// Skip over types
+				if p.options.ts.Parse && p.lexer.Token == js_lexer.TColon {
+					p.lexer.Expect(js_lexer.TColon)
+					p.skipTypeScriptType(js_ast.LLowest)
+				}
+
+				p.lexer.Expect(js_lexer.TCloseParen)
+
+				// Bare identifiers are a special case
+				kind := ast.SymbolOther
+				if _, ok := bindingOrNil.Data.(*js_ast.BIdentifier); ok {
+					kind = ast.SymbolCatchIdentifier
+				}
+				p.declareBinding(kind, bindingOrNil, parseStmtOpts{})
+			}
+
+			blockLoc := p.lexer.Loc()
+			p.lexer.Expect(js_lexer.TOpenBrace)
+
+			p.pushScopeForParsePass(js_ast.ScopeBlock, blockLoc)
+			stmts := p.parseStmtsUpTo(js_lexer.TCloseBrace, parseStmtOpts{})
+			p.popScope()
+
+			closeBraceLoc := p.lexer.Loc()
+			p.lexer.Next()
+			catch = &js_ast.Catch{Loc: catchLoc, BindingOrNil: bindingOrNil, BlockLoc: blockLoc, Block: js_ast.SBlock{Stmts: stmts, CloseBraceLoc: closeBraceLoc}}
+			p.popScope()
+		}
+
+		if p.lexer.Token == js_lexer.TFinally || catch == nil {
+			finallyLoc := p.lexer.Loc()
+			p.pushScopeForParsePass(js_ast.ScopeBlock, finallyLoc)
+			p.lexer.Expect(js_lexer.TFinally)
+			p.lexer.Expect(js_lexer.TOpenBrace)
+			stmts := p.parseStmtsUpTo(js_lexer.TCloseBrace, parseStmtOpts{})
+			closeBraceLoc := p.lexer.Loc()
+			p.lexer.Next()
+			finally = &js_ast.Finally{Loc: finallyLoc, Block: js_ast.SBlock{Stmts: stmts, CloseBraceLoc: closeBraceLoc}}
+			p.popScope()
+		}
+
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.STry{
+			BlockLoc: blockLoc,
+			Block:    js_ast.SBlock{Stmts: body, CloseBraceLoc: closeBraceLoc},
+			Catch:    catch,
+			Finally:  finally,
+		}}
+
+	case js_lexer.TFor:
+		p.pushScopeForParsePass(js_ast.ScopeBlock, loc)
+		defer p.popScope()
+
+		p.lexer.Next()
+
+		// "for await (let x of y) {}"
+		var awaitRange logger.Range
+		if p.lexer.IsContextualKeyword("await") {
+			awaitRange = p.lexer.Range()
+			if p.fnOrArrowDataParse.await != allowExpr {
+				p.log.AddError(&p.tracker, awaitRange, "Cannot use \"await\" outside an async function")
+				awaitRange = logger.Range{}
+			} else {
+				didGenerateError := false
+				if p.fnOrArrowDataParse.isTopLevel {
+					p.topLevelAwaitKeyword = awaitRange
+				}
+				if !didGenerateError && p.options.unsupportedJSFeatures.Has(compat.AsyncAwait) && p.options.unsupportedJSFeatures.Has(compat.Generator) {
+					// If for-await loops aren't supported, then we only support lowering
+					// if either async/await or generators is supported. Otherwise we
+					// cannot lower for-await loops.
+					p.markSyntaxFeature(compat.ForAwait, awaitRange)
+				}
+			}
+			p.lexer.Next()
+		}
+
+		p.lexer.Expect(js_lexer.TOpenParen)
+
+		var initOrNil js_ast.Stmt
+		var testOrNil js_ast.Expr
+		var updateOrNil js_ast.Expr
+
+		// "in" expressions aren't allowed here
+		p.allowIn = false
+
+		var badLetRange logger.Range
+		if p.lexer.IsContextualKeyword("let") {
+			badLetRange = p.lexer.Range()
+		}
+		decls := []js_ast.Decl{}
+		initLoc := p.lexer.Loc()
+		isVar := false
+		switch p.lexer.Token {
+		case js_lexer.TVar:
+			isVar = true
+			p.lexer.Next()
+			decls = p.parseAndDeclareDecls(ast.SymbolHoisted, parseStmtOpts{})
+			initOrNil = js_ast.Stmt{Loc: initLoc, Data: &js_ast.SLocal{Kind: js_ast.LocalVar, Decls: decls}}
+
+		case js_lexer.TConst:
+			p.markSyntaxFeature(compat.ConstAndLet, p.lexer.Range())
+			p.lexer.Next()
+			decls = p.parseAndDeclareDecls(ast.SymbolConst, parseStmtOpts{})
+			initOrNil = js_ast.Stmt{Loc: initLoc, Data: &js_ast.SLocal{Kind: js_ast.LocalConst, Decls: decls}}
+
+		case js_lexer.TSemicolon:
+
+		default:
+			var expr js_ast.Expr
+			var stmt js_ast.Stmt
+			expr, stmt, decls = p.parseExprOrLetOrUsingStmt(parseStmtOpts{
+				lexicalDecl:        lexicalDeclAllowAll,
+				isForLoopInit:      true,
+				isForAwaitLoopInit: awaitRange.Len > 0,
+			})
+			if stmt.Data != nil {
+				badLetRange = logger.Range{}
+				initOrNil = stmt
+			} else {
+				initOrNil = js_ast.Stmt{Loc: expr.Loc, Data: &js_ast.SExpr{Value: expr}}
+			}
+		}
+
+		// "in" expressions are allowed again
+		p.allowIn = true
+
+		// Detect for-of loops
+		if p.lexer.IsContextualKeyword("of") || awaitRange.Len > 0 {
+			if badLetRange.Len > 0 {
+				p.log.AddError(&p.tracker, badLetRange, "\"let\" must be wrapped in parentheses to be used as an expression here:")
+			}
+			if awaitRange.Len > 0 && !p.lexer.IsContextualKeyword("of") {
+				if initOrNil.Data != nil {
+					p.lexer.ExpectedString("\"of\"")
+				} else {
+					p.lexer.Unexpected()
+				}
+			}
+			p.forbidInitializers(decls, "of", false)
+			p.markSyntaxFeature(compat.ForOf, p.lexer.Range())
+			p.lexer.Next()
+			value := p.parseExpr(js_ast.LComma)
+			p.lexer.Expect(js_lexer.TCloseParen)
+			isSingleLineBody := !p.lexer.HasNewlineBefore && p.lexer.Token != js_lexer.TOpenBrace
+			body := p.parseStmt(parseStmtOpts{})
+			return js_ast.Stmt{Loc: loc, Data: &js_ast.SForOf{Await: awaitRange, Init: initOrNil, Value: value, Body: body, IsSingleLineBody: isSingleLineBody}}
+		}
+
+		// Detect for-in loops
+		if p.lexer.Token == js_lexer.TIn {
+			p.forbidInitializers(decls, "in", isVar)
+			if len(decls) == 1 {
+				if local, ok := initOrNil.Data.(*js_ast.SLocal); ok {
+					if local.Kind == js_ast.LocalUsing {
+						p.log.AddError(&p.tracker, js_lexer.RangeOfIdentifier(p.source, initOrNil.Loc), "\"using\" declarations are not allowed here")
+					} else if local.Kind == js_ast.LocalAwaitUsing {
+						p.log.AddError(&p.tracker, js_lexer.RangeOfIdentifier(p.source, initOrNil.Loc), "\"await using\" declarations are not allowed here")
+					}
+				}
+			}
+			p.lexer.Next()
+			value := p.parseExpr(js_ast.LLowest)
+			p.lexer.Expect(js_lexer.TCloseParen)
+			isSingleLineBody := !p.lexer.HasNewlineBefore && p.lexer.Token != js_lexer.TOpenBrace
+			body := p.parseStmt(parseStmtOpts{})
+			return js_ast.Stmt{Loc: loc, Data: &js_ast.SForIn{Init: initOrNil, Value: value, Body: body, IsSingleLineBody: isSingleLineBody}}
+		}
+
+		p.lexer.Expect(js_lexer.TSemicolon)
+
+		// "await using" declarations are only allowed in for-of loops
+		if local, ok := initOrNil.Data.(*js_ast.SLocal); ok && local.Kind == js_ast.LocalAwaitUsing {
+			p.log.AddError(&p.tracker, js_lexer.RangeOfIdentifier(p.source, initOrNil.Loc), "\"await using\" declarations are not allowed here")
+		}
+
+		// Only require "const" statement initializers when we know we're a normal for loop
+		if local, ok := initOrNil.Data.(*js_ast.SLocal); ok && (local.Kind == js_ast.LocalConst || local.Kind == js_ast.LocalUsing) {
+			p.requireInitializers(local.Kind, decls)
+		}
+
+		if p.lexer.Token != js_lexer.TSemicolon {
+			testOrNil = p.parseExpr(js_ast.LLowest)
+		}
+
+		p.lexer.Expect(js_lexer.TSemicolon)
+
+		if p.lexer.Token != js_lexer.TCloseParen {
+			updateOrNil = p.parseExpr(js_ast.LLowest)
+		}
+
+		p.lexer.Expect(js_lexer.TCloseParen)
+		isSingleLineBody := !p.lexer.HasNewlineBefore && p.lexer.Token != js_lexer.TOpenBrace
+		body := p.parseStmt(parseStmtOpts{})
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.SFor{
+			InitOrNil:        initOrNil,
+			TestOrNil:        testOrNil,
+			UpdateOrNil:      updateOrNil,
+			Body:             body,
+			IsSingleLineBody: isSingleLineBody,
+		}}
+
+	case js_lexer.TImport:
+		previousImportStatementKeyword := p.esmImportStatementKeyword
+		p.esmImportStatementKeyword = p.lexer.Range()
+		p.lexer.Next()
+		stmt := js_ast.SImport{}
+		wasOriginallyBareImport := false
+
+		// "export import foo = bar"
+		// "import foo = bar" in a namespace
+		if (opts.isExport || (opts.isNamespaceScope && !opts.isTypeScriptDeclare)) && p.lexer.Token != js_lexer.TIdentifier {
+			p.lexer.Expected(js_lexer.TIdentifier)
+		}
+
+	syntaxBeforePath:
+		switch p.lexer.Token {
+		case js_lexer.TOpenParen, js_lexer.TDot:
+			// "import('path')"
+			// "import.meta"
+			p.esmImportStatementKeyword = previousImportStatementKeyword // This wasn't an ESM import statement after all
+			expr := p.parseSuffix(p.parseImportExpr(loc, js_ast.LLowest), js_ast.LLowest, nil, 0)
+			p.lexer.ExpectOrInsertSemicolon()
+			return js_ast.Stmt{Loc: loc, Data: &js_ast.SExpr{Value: expr}}
+
+		case js_lexer.TStringLiteral, js_lexer.TNoSubstitutionTemplateLiteral:
+			// "import 'path'"
+			if !opts.isModuleScope && (!opts.isNamespaceScope || !opts.isTypeScriptDeclare) {
+				p.lexer.Unexpected()
+				return js_ast.Stmt{}
+			}
+
+			wasOriginallyBareImport = true
+
+		case js_lexer.TAsterisk:
+			// "import * as ns from 'path'"
+			if !opts.isModuleScope && (!opts.isNamespaceScope || !opts.isTypeScriptDeclare) {
+				p.lexer.Unexpected()
+				return js_ast.Stmt{}
+			}
+
+			p.lexer.Next()
+			p.lexer.ExpectContextualKeyword("as")
+			stmt.NamespaceRef = p.storeNameInRef(p.lexer.Identifier)
+			starLoc := p.lexer.Loc()
+			stmt.StarNameLoc = &starLoc
+			p.lexer.Expect(js_lexer.TIdentifier)
+			p.lexer.ExpectContextualKeyword("from")
+
+		case js_lexer.TOpenBrace:
+			// "import {item1, item2} from 'path'"
+			if !opts.isModuleScope && (!opts.isNamespaceScope || !opts.isTypeScriptDeclare) {
+				p.lexer.Unexpected()
+				return js_ast.Stmt{}
+			}
+
+			items, isSingleLine := p.parseImportClause()
+			stmt.Items = &items
+			stmt.IsSingleLine = isSingleLine
+			p.lexer.ExpectContextualKeyword("from")
+
+		case js_lexer.TIdentifier:
+			// "import defaultItem from 'path'"
+			// "import foo = bar"
+			if !opts.isModuleScope && !opts.isNamespaceScope {
+				p.lexer.Unexpected()
+				return js_ast.Stmt{}
+			}
+
+			defaultName := p.lexer.Identifier
+			stmt.DefaultName = &ast.LocRef{Loc: p.lexer.Loc(), Ref: p.storeNameInRef(defaultName)}
+			p.lexer.Next()
+
+			if p.options.ts.Parse {
+				// Skip over type-only imports
+				if defaultName.String == "type" {
+					switch p.lexer.Token {
+					case js_lexer.TIdentifier:
+						nameSubstring := p.lexer.Identifier
+						nameLoc := p.lexer.Loc()
+						p.lexer.Next()
+						if p.lexer.Token == js_lexer.TEquals {
+							// "import type foo = require('bar');"
+							// "import type foo = bar.baz;"
+							opts.isTypeScriptDeclare = true
+							return p.parseTypeScriptImportEqualsStmt(loc, opts, nameLoc, nameSubstring.String)
+						} else if p.lexer.Token == js_lexer.TStringLiteral && nameSubstring.String == "from" {
+							// "import type from 'bar';"
+							break syntaxBeforePath
+						} else {
+							// "import type foo from 'bar';"
+							p.lexer.ExpectContextualKeyword("from")
+							p.parsePath()
+							p.lexer.ExpectOrInsertSemicolon()
+							return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptShared}
+						}
+
+					case js_lexer.TAsterisk:
+						// "import type * as foo from 'bar';"
+						p.lexer.Next()
+						p.lexer.ExpectContextualKeyword("as")
+						p.lexer.Expect(js_lexer.TIdentifier)
+						p.lexer.ExpectContextualKeyword("from")
+						p.parsePath()
+						p.lexer.ExpectOrInsertSemicolon()
+						return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptShared}
+
+					case js_lexer.TOpenBrace:
+						// "import type {foo} from 'bar';"
+						p.parseImportClause()
+						p.lexer.ExpectContextualKeyword("from")
+						p.parsePath()
+						p.lexer.ExpectOrInsertSemicolon()
+						return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptShared}
+					}
+				}
+
+				// Parse TypeScript import assignment statements
+				if p.lexer.Token == js_lexer.TEquals || opts.isExport || (opts.isNamespaceScope && !opts.isTypeScriptDeclare) {
+					p.esmImportStatementKeyword = previousImportStatementKeyword // This wasn't an ESM import statement after all
+					return p.parseTypeScriptImportEqualsStmt(loc, opts, stmt.DefaultName.Loc, defaultName.String)
+				}
+			}
+
+			if p.lexer.Token == js_lexer.TComma {
+				p.lexer.Next()
+				switch p.lexer.Token {
+				case js_lexer.TAsterisk:
+					// "import defaultItem, * as ns from 'path'"
+					p.lexer.Next()
+					p.lexer.ExpectContextualKeyword("as")
+					stmt.NamespaceRef = p.storeNameInRef(p.lexer.Identifier)
+					starLoc := p.lexer.Loc()
+					stmt.StarNameLoc = &starLoc
+					p.lexer.Expect(js_lexer.TIdentifier)
+
+				case js_lexer.TOpenBrace:
+					// "import defaultItem, {item1, item2} from 'path'"
+					items, isSingleLine := p.parseImportClause()
+					stmt.Items = &items
+					stmt.IsSingleLine = isSingleLine
+
+				default:
+					p.lexer.Unexpected()
+				}
+			}
+
+			p.lexer.ExpectContextualKeyword("from")
+
+		default:
+			p.lexer.Unexpected()
+			return js_ast.Stmt{}
+		}
+
+		pathLoc, pathText, assertOrWith, flags := p.parsePath()
+		p.lexer.ExpectOrInsertSemicolon()
+
+		// If TypeScript's "preserveValueImports": true setting is active, TypeScript's
+		// "importsNotUsedAsValues": "preserve" setting is NOT active, and the import
+		// clause is present and empty (or is non-empty but filled with type-only
+		// items), then the import statement should still be removed entirely to match
+		// the behavior of the TypeScript compiler:
+		//
+		//   // Keep these
+		//   import 'x'
+		//   import { y } from 'x'
+		//   import { y, type z } from 'x'
+		//
+		//   // Remove these
+		//   import {} from 'x'
+		//   import { type y } from 'x'
+		//
+		//   // Remove the items from these
+		//   import d, {} from 'x'
+		//   import d, { type y } from 'x'
+		//
+		if p.options.ts.Parse && p.options.ts.Config.UnusedImportFlags() == config.TSUnusedImport_KeepValues && stmt.Items != nil && len(*stmt.Items) == 0 {
+			if stmt.DefaultName == nil {
+				return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptShared}
+			}
+			stmt.Items = nil
+		}
+
+		if wasOriginallyBareImport {
+			flags |= ast.WasOriginallyBareImport
+		}
+		stmt.ImportRecordIndex = p.addImportRecord(ast.ImportStmt, pathLoc, pathText, assertOrWith, flags)
+
+		if stmt.StarNameLoc != nil {
+			name := p.loadNameFromRef(stmt.NamespaceRef)
+			stmt.NamespaceRef = p.declareSymbol(ast.SymbolImport, *stmt.StarNameLoc, name)
+		} else {
+			// Generate a symbol for the namespace
+			name := "import_" + js_ast.GenerateNonUniqueNameFromPath(pathText)
+			stmt.NamespaceRef = p.newSymbol(ast.SymbolOther, name)
+			p.currentScope.Generated = append(p.currentScope.Generated, stmt.NamespaceRef)
+		}
+		itemRefs := make(map[string]ast.LocRef)
+
+		// Link the default item to the namespace
+		if stmt.DefaultName != nil {
+			name := p.loadNameFromRef(stmt.DefaultName.Ref)
+			ref := p.declareSymbol(ast.SymbolImport, stmt.DefaultName.Loc, name)
+			p.isImportItem[ref] = true
+			stmt.DefaultName.Ref = ref
+		}
+
+		// Link each import item to the namespace
+		if stmt.Items != nil {
+			for i, item := range *stmt.Items {
+				name := p.loadNameFromRef(item.Name.Ref)
+				ref := p.declareSymbol(ast.SymbolImport, item.Name.Loc, name)
+				p.checkForUnrepresentableIdentifier(item.AliasLoc, item.Alias)
+				p.isImportItem[ref] = true
+				(*stmt.Items)[i].Name.Ref = ref
+				itemRefs[item.Alias] = ast.LocRef{Loc: item.Name.Loc, Ref: ref}
+			}
+		}
+
+		// Track the items for this namespace
+		p.importItemsForNamespace[stmt.NamespaceRef] = namespaceImportItems{
+			entries:           itemRefs,
+			importRecordIndex: stmt.ImportRecordIndex,
+		}
+
+		// Import statements anywhere in the file disable top-level const
+		// local prefix because import cycles can be used to trigger TDZ
+		p.currentScope.IsAfterConstLocalPrefix = true
+		return js_ast.Stmt{Loc: loc, Data: &stmt}
+
+	case js_lexer.TBreak:
+		p.lexer.Next()
+		name := p.parseLabelName()
+		p.lexer.ExpectOrInsertSemicolon()
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.SBreak{Label: name}}
+
+	case js_lexer.TContinue:
+		p.lexer.Next()
+		name := p.parseLabelName()
+		p.lexer.ExpectOrInsertSemicolon()
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.SContinue{Label: name}}
+
+	case js_lexer.TReturn:
+		if p.fnOrArrowDataParse.isReturnDisallowed {
+			p.log.AddError(&p.tracker, p.lexer.Range(), "A return statement cannot be used here:")
+		}
+		p.lexer.Next()
+		var value js_ast.Expr
+		if p.lexer.Token != js_lexer.TSemicolon &&
+			!p.lexer.HasNewlineBefore &&
+			p.lexer.Token != js_lexer.TCloseBrace &&
+			p.lexer.Token != js_lexer.TEndOfFile {
+			value = p.parseExpr(js_ast.LLowest)
+		}
+		p.latestReturnHadSemicolon = p.lexer.Token == js_lexer.TSemicolon
+		p.lexer.ExpectOrInsertSemicolon()
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.SReturn{ValueOrNil: value}}
+
+	case js_lexer.TThrow:
+		p.lexer.Next()
+		if p.lexer.HasNewlineBefore {
+			endLoc := logger.Loc{Start: loc.Start + 5}
+			p.log.AddError(&p.tracker, logger.Range{Loc: endLoc},
+				"Unexpected newline after \"throw\"")
+			return js_ast.Stmt{Loc: loc, Data: &js_ast.SThrow{Value: js_ast.Expr{Loc: endLoc, Data: js_ast.ENullShared}}}
+		}
+		expr := p.parseExpr(js_ast.LLowest)
+		p.lexer.ExpectOrInsertSemicolon()
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.SThrow{Value: expr}}
+
+	case js_lexer.TDebugger:
+		p.lexer.Next()
+		p.lexer.ExpectOrInsertSemicolon()
+		return js_ast.Stmt{Loc: loc, Data: js_ast.SDebuggerShared}
+
+	case js_lexer.TOpenBrace:
+		p.pushScopeForParsePass(js_ast.ScopeBlock, loc)
+		defer p.popScope()
+
+		p.lexer.Next()
+		stmts := p.parseStmtsUpTo(js_lexer.TCloseBrace, parseStmtOpts{})
+		closeBraceLoc := p.lexer.Loc()
+		p.lexer.Next()
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.SBlock{Stmts: stmts, CloseBraceLoc: closeBraceLoc}}
+
+	default:
+		isIdentifier := p.lexer.Token == js_lexer.TIdentifier
+		nameRange := p.lexer.Range()
+		name := p.lexer.Identifier.String
+
+		// Parse either an async function, an async expression, or a normal expression
+		var expr js_ast.Expr
+		if isIdentifier && p.lexer.Raw() == "async" {
+			p.lexer.Next()
+			if p.lexer.Token == js_lexer.TFunction && !p.lexer.HasNewlineBefore {
+				p.lexer.Next()
+				return p.parseFnStmt(nameRange.Loc, opts, true /* isAsync */, nameRange)
+			}
+			expr = p.parseSuffix(p.parseAsyncPrefixExpr(nameRange, js_ast.LLowest, 0), js_ast.LLowest, nil, 0)
+		} else {
+			var stmt js_ast.Stmt
+			expr, stmt, _ = p.parseExprOrLetOrUsingStmt(opts)
+			if stmt.Data != nil {
+				p.lexer.ExpectOrInsertSemicolon()
+				return stmt
+			}
+		}
+
+		if isIdentifier {
+			if ident, ok := expr.Data.(*js_ast.EIdentifier); ok {
+				if p.lexer.Token == js_lexer.TColon && opts.deferredDecorators == nil {
+					p.pushScopeForParsePass(js_ast.ScopeLabel, loc)
+					defer p.popScope()
+
+					// Parse a labeled statement
+					p.lexer.Next()
+					name := ast.LocRef{Loc: expr.Loc, Ref: ident.Ref}
+					nestedOpts := parseStmtOpts{}
+					if opts.lexicalDecl == lexicalDeclAllowAll || opts.lexicalDecl == lexicalDeclAllowFnInsideLabel {
+						nestedOpts.lexicalDecl = lexicalDeclAllowFnInsideLabel
+					}
+					isSingleLineStmt := !p.lexer.HasNewlineBefore && p.lexer.Token != js_lexer.TOpenBrace
+					stmt := p.parseStmt(nestedOpts)
+					return js_ast.Stmt{Loc: loc, Data: &js_ast.SLabel{Name: name, Stmt: stmt, IsSingleLineStmt: isSingleLineStmt}}
+				}
+
+				if p.options.ts.Parse {
+					switch name {
+					case "type":
+						if !p.lexer.HasNewlineBefore && p.lexer.Token == js_lexer.TIdentifier {
+							// "type Foo = any"
+							p.skipTypeScriptTypeStmt(parseStmtOpts{isModuleScope: opts.isModuleScope})
+							return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptShared}
+						}
+
+					case "namespace", "module":
+						// "namespace Foo {}"
+						// "module Foo {}"
+						// "declare module 'fs' {}"
+						// "declare module 'fs';"
+						if !p.lexer.HasNewlineBefore && (opts.isModuleScope || opts.isNamespaceScope) && (p.lexer.Token == js_lexer.TIdentifier ||
+							(p.lexer.Token == js_lexer.TStringLiteral && opts.isTypeScriptDeclare)) {
+							return p.parseTypeScriptNamespaceStmt(loc, opts)
+						}
+
+					case "interface":
+						// "interface Foo {}"
+						// "export default interface Foo {}"
+						// "export default interface \n Foo {}"
+						if !p.lexer.HasNewlineBefore || opts.isExportDefault {
+							p.skipTypeScriptInterfaceStmt(parseStmtOpts{isModuleScope: opts.isModuleScope})
+							return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptShared}
+						}
+
+						// "interface \n Foo {}"
+						// "export interface \n Foo {}"
+						if opts.isExport {
+							p.log.AddError(&p.tracker, nameRange, "Unexpected \"interface\"")
+							panic(js_lexer.LexerPanic{})
+						}
+
+					case "abstract":
+						if !p.lexer.HasNewlineBefore && p.lexer.Token == js_lexer.TClass {
+							return p.parseClassStmt(loc, opts)
+						}
+
+					case "global":
+						// "declare module 'fs' { global { namespace NodeJS {} } }"
+						if opts.isNamespaceScope && opts.isTypeScriptDeclare && p.lexer.Token == js_lexer.TOpenBrace {
+							p.lexer.Next()
+							p.parseStmtsUpTo(js_lexer.TCloseBrace, opts)
+							p.lexer.Next()
+							return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptShared}
+						}
+
+					case "declare":
+						if !p.lexer.HasNewlineBefore {
+							opts.lexicalDecl = lexicalDeclAllowAll
+							opts.isTypeScriptDeclare = true
+
+							// "declare global { ... }"
+							if p.lexer.IsContextualKeyword("global") {
+								p.lexer.Next()
+								p.lexer.Expect(js_lexer.TOpenBrace)
+								p.parseStmtsUpTo(js_lexer.TCloseBrace, opts)
+								p.lexer.Next()
+								return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptShared}
+							}
+
+							// "declare const x: any"
+							scopeIndex := len(p.scopesInOrder)
+							oldLexer := p.lexer
+							stmt := p.parseStmt(opts)
+							typeDeclarationData := js_ast.STypeScriptShared
+							switch s := stmt.Data.(type) {
+							case *js_ast.SEmpty:
+								return js_ast.Stmt{Loc: loc, Data: &js_ast.SExpr{Value: expr}}
+
+							case *js_ast.STypeScript:
+								// Type declarations are expected. Propagate the "declare class"
+								// status in case our caller is a decorator that needs to know
+								// this was a "declare class" statement.
+								typeDeclarationData = s
+
+							case *js_ast.SLocal:
+								// This is also a type declaration (but doesn't use "STypeScript"
+								// because we need to be able to handle namespace exports below)
+
+							default:
+								// Anything that we don't expect is a syntax error. For example,
+								// we consider this a syntax error:
+								//
+								//   declare let declare: any, foo: any
+								//   declare foo
+								//
+								// Strangely TypeScript allows this code starting with version
+								// 4.4, but I assume this is a bug. This bug was reported here:
+								// https://github.com/microsoft/TypeScript/issues/54602
+								p.lexer = oldLexer
+								p.lexer.Unexpected()
+							}
+							p.discardScopesUpTo(scopeIndex)
+
+							// Unlike almost all uses of "declare", statements that use
+							// "export declare" with "var/let/const" inside a namespace affect
+							// code generation. They cause any declared bindings to be
+							// considered exports of the namespace. Identifier references to
+							// those names must be converted into property accesses off the
+							// namespace object:
+							//
+							//   namespace ns {
+							//     export declare const x
+							//     export function y() { return x }
+							//   }
+							//
+							//   (ns as any).x = 1
+							//   console.log(ns.y())
+							//
+							// In this example, "return x" must be replaced with "return ns.x".
+							// This is handled by replacing each "export declare" statement
+							// inside a namespace with an "export var" statement containing all
+							// of the declared bindings. That "export var" statement will later
+							// cause identifiers to be transformed into property accesses.
+							if opts.isNamespaceScope && opts.isExport {
+								var decls []js_ast.Decl
+								if s, ok := stmt.Data.(*js_ast.SLocal); ok {
+									js_ast.ForEachIdentifierBindingInDecls(s.Decls, func(loc logger.Loc, b *js_ast.BIdentifier) {
+										decls = append(decls, js_ast.Decl{Binding: js_ast.Binding{Loc: loc, Data: b}})
+									})
+								}
+								if len(decls) > 0 {
+									return js_ast.Stmt{Loc: loc, Data: &js_ast.SLocal{
+										Kind:     js_ast.LocalVar,
+										IsExport: true,
+										Decls:    decls,
+									}}
+								}
+							}
+
+							return js_ast.Stmt{Loc: loc, Data: typeDeclarationData}
+						}
+					}
+				}
+			}
+		}
+
+		p.lexer.ExpectOrInsertSemicolon()
+		return js_ast.Stmt{Loc: loc, Data: &js_ast.SExpr{Value: expr}}
+	}
+}
+
+func (p *parser) addImportRecord(kind ast.ImportKind, pathRange logger.Range, text string, assertOrWith *ast.ImportAssertOrWith, flags ast.ImportRecordFlags) uint32 {
+	index := uint32(len(p.importRecords))
+	p.importRecords = append(p.importRecords, ast.ImportRecord{
+		Kind:         kind,
+		Range:        pathRange,
+		Path:         logger.Path{Text: text},
+		AssertOrWith: assertOrWith,
+		Flags:        flags,
+	})
+	return index
+}
+
+func (p *parser) parseFnBody(data fnOrArrowDataParse) js_ast.FnBody {
+	oldFnOrArrowData := p.fnOrArrowDataParse
+	oldAllowIn := p.allowIn
+	p.fnOrArrowDataParse = data
+	p.allowIn = true
+
+	loc := p.lexer.Loc()
+	p.pushScopeForParsePass(js_ast.ScopeFunctionBody, loc)
+	defer p.popScope()
+
+	p.lexer.Expect(js_lexer.TOpenBrace)
+	stmts := p.parseStmtsUpTo(js_lexer.TCloseBrace, parseStmtOpts{
+		allowDirectivePrologue: true,
+	})
+	closeBraceLoc := p.lexer.Loc()
+	p.lexer.Next()
+
+	p.allowIn = oldAllowIn
+	p.fnOrArrowDataParse = oldFnOrArrowData
+	return js_ast.FnBody{Loc: loc, Block: js_ast.SBlock{Stmts: stmts, CloseBraceLoc: closeBraceLoc}}
+}
+
+func (p *parser) forbidLexicalDecl(loc logger.Loc) {
+	r := js_lexer.RangeOfIdentifier(p.source, loc)
+	p.log.AddError(&p.tracker, r, "Cannot use a declaration in a single-statement context")
+}
+
+func (p *parser) parseStmtsUpTo(end js_lexer.T, opts parseStmtOpts) []js_ast.Stmt {
+	stmts := []js_ast.Stmt{}
+	returnWithoutSemicolonStart := int32(-1)
+	opts.lexicalDecl = lexicalDeclAllowAll
+	isDirectivePrologue := opts.allowDirectivePrologue
+
+	for {
+		// Preserve some statement-level comments
+		comments := p.lexer.LegalCommentsBeforeToken
+		if len(comments) > 0 {
+			for _, comment := range comments {
+				stmts = append(stmts, js_ast.Stmt{
+					Loc: comment.Loc,
+					Data: &js_ast.SComment{
+						Text:           p.source.CommentTextWithoutIndent(comment),
+						IsLegalComment: true,
+					},
+				})
+			}
+		}
+
+		if p.lexer.Token == end {
+			break
+		}
+
+		stmt := p.parseStmt(opts)
+
+		// Skip TypeScript types entirely
+		if p.options.ts.Parse {
+			if _, ok := stmt.Data.(*js_ast.STypeScript); ok {
+				continue
+			}
+		}
+
+		// Parse one or more directives at the beginning
+		if isDirectivePrologue {
+			isDirectivePrologue = false
+			if expr, ok := stmt.Data.(*js_ast.SExpr); ok {
+				if str, ok := expr.Value.Data.(*js_ast.EString); ok && !str.PreferTemplate {
+					stmt.Data = &js_ast.SDirective{Value: str.Value, LegacyOctalLoc: str.LegacyOctalLoc}
+					isDirectivePrologue = true
+
+					if helpers.UTF16EqualsString(str.Value, "use strict") {
+						// Track "use strict" directives
+						p.currentScope.StrictMode = js_ast.ExplicitStrictMode
+						p.currentScope.UseStrictLoc = expr.Value.Loc
+
+						// Inside a function, strict mode actually propagates from the child
+						// scope to the parent scope:
+						//
+						//   // This is a syntax error
+						//   function fn(arguments) {
+						//     "use strict";
+						//   }
+						//
+						if p.currentScope.Kind == js_ast.ScopeFunctionBody &&
+							p.currentScope.Parent.Kind == js_ast.ScopeFunctionArgs &&
+							p.currentScope.Parent.StrictMode == js_ast.SloppyMode {
+							p.currentScope.Parent.StrictMode = js_ast.ExplicitStrictMode
+							p.currentScope.Parent.UseStrictLoc = expr.Value.Loc
+						}
+					} else if helpers.UTF16EqualsString(str.Value, "use asm") {
+						// Deliberately remove "use asm" directives. The asm.js subset of
+						// JavaScript has complicated validation rules that are triggered
+						// by this directive. This parser is not designed with asm.js in
+						// mind and round-tripping asm.js code through esbuild will very
+						// likely cause it to no longer validate as asm.js. When this
+						// happens, V8 prints a warning and people don't like seeing the
+						// warning.
+						//
+						// We deliberately do not attempt to preserve the validity of
+						// asm.js code because it's a complicated legacy format and it's
+						// obsolete now that WebAssembly exists. By removing this directive
+						// it will just become normal JavaScript, which will work fine and
+						// won't generate a warning (but will run slower). We don't generate
+						// a warning ourselves in this case because there isn't necessarily
+						// anything easy and actionable that the user can do to fix this.
+						stmt.Data = &js_ast.SEmpty{}
+					}
+				}
+			}
+		}
+
+		stmts = append(stmts, stmt)
+
+		// Warn about ASI and return statements. Here's an example of code with
+		// this problem: https://github.com/rollup/rollup/issues/3729
+		if !p.suppressWarningsAboutWeirdCode {
+			if s, ok := stmt.Data.(*js_ast.SReturn); ok && s.ValueOrNil.Data == nil && !p.latestReturnHadSemicolon {
+				returnWithoutSemicolonStart = stmt.Loc.Start
+			} else {
+				if returnWithoutSemicolonStart != -1 {
+					if _, ok := stmt.Data.(*js_ast.SExpr); ok {
+						p.log.AddID(logger.MsgID_JS_SemicolonAfterReturn, logger.Warning, &p.tracker, logger.Range{Loc: logger.Loc{Start: returnWithoutSemicolonStart + 6}},
+							"The following expression is not returned because of an automatically-inserted semicolon")
+					}
+				}
+				returnWithoutSemicolonStart = -1
+			}
+		}
+	}
+
+	return stmts
+}
+
+type generateTempRefArg uint8
+
+const (
+	tempRefNeedsDeclare generateTempRefArg = iota
+	tempRefNoDeclare
+
+	// This is used when the generated temporary may a) be used inside of a loop
+	// body and b) may be used inside of a closure. In that case we can't use
+	// "var" for the temporary and we can't declare the temporary at the top of
+	// the enclosing function. Instead, we need to use "let" and we need to
+	// declare the temporary in the enclosing block (so it's inside of the loop
+	// body).
+	tempRefNeedsDeclareMayBeCapturedInsideLoop
+)
+
+func (p *parser) generateTempRef(declare generateTempRefArg, optionalName string) ast.Ref {
+	scope := p.currentScope
+
+	if declare != tempRefNeedsDeclareMayBeCapturedInsideLoop {
+		for !scope.Kind.StopsHoisting() {
+			scope = scope.Parent
+		}
+	}
+
+	if optionalName == "" {
+		optionalName = "_" + ast.DefaultNameMinifierJS.NumberToMinifiedName(p.tempRefCount)
+		p.tempRefCount++
+	}
+	ref := p.newSymbol(ast.SymbolOther, optionalName)
+
+	if declare == tempRefNeedsDeclareMayBeCapturedInsideLoop && !scope.Kind.StopsHoisting() {
+		p.tempLetsToDeclare = append(p.tempLetsToDeclare, ref)
+	} else if declare != tempRefNoDeclare {
+		p.tempRefsToDeclare = append(p.tempRefsToDeclare, tempRef{ref: ref})
+	}
+
+	scope.Generated = append(scope.Generated, ref)
+	return ref
+}
+
+func (p *parser) generateTopLevelTempRef() ast.Ref {
+	ref := p.newSymbol(ast.SymbolOther, "_"+ast.DefaultNameMinifierJS.NumberToMinifiedName(p.topLevelTempRefCount))
+	p.topLevelTempRefsToDeclare = append(p.topLevelTempRefsToDeclare, tempRef{ref: ref})
+	p.moduleScope.Generated = append(p.moduleScope.Generated, ref)
+	p.topLevelTempRefCount++
+	return ref
+}
+
+func (p *parser) pushScopeForVisitPass(kind js_ast.ScopeKind, loc logger.Loc) {
+	order := p.scopesInOrder[0]
+
+	// Sanity-check that the scopes generated by the first and second passes match
+	if order.loc != loc || order.scope.Kind != kind {
+		panic(fmt.Sprintf("Expected scope (%d, %d) in %s, found scope (%d, %d)",
+			kind, loc.Start,
+			p.source.PrettyPath,
+			order.scope.Kind, order.loc.Start))
+	}
+
+	p.scopesInOrder = p.scopesInOrder[1:]
+	p.currentScope = order.scope
+	p.scopesForCurrentPart = append(p.scopesForCurrentPart, order.scope)
+}
+
+type findSymbolResult struct {
+	ref               ast.Ref
+	declareLoc        logger.Loc
+	isInsideWithScope bool
+}
+
+func (p *parser) findSymbol(loc logger.Loc, name string) findSymbolResult {
+	var ref ast.Ref
+	var declareLoc logger.Loc
+	isInsideWithScope := false
+	didForbidArguments := false
+	s := p.currentScope
+
+	for {
+		// Track if we're inside a "with" statement body
+		if s.Kind == js_ast.ScopeWith {
+			isInsideWithScope = true
+		}
+
+		// Forbid referencing "arguments" inside class bodies
+		if s.ForbidArguments && name == "arguments" && !didForbidArguments {
+			r := js_lexer.RangeOfIdentifier(p.source, loc)
+			p.log.AddError(&p.tracker, r, fmt.Sprintf("Cannot access %q here:", name))
+			didForbidArguments = true
+		}
+
+		// Is the symbol a member of this scope?
+		if member, ok := s.Members[name]; ok {
+			ref = member.Ref
+			declareLoc = member.Loc
+			break
+		}
+
+		// Is the symbol a member of this scope's TypeScript namespace?
+		if tsNamespace := s.TSNamespace; tsNamespace != nil {
+			if member, ok := tsNamespace.ExportedMembers[name]; ok && tsNamespace.IsEnumScope == member.IsEnumValue {
+				// If this is an identifier from a sibling TypeScript namespace, then we're
+				// going to have to generate a property access instead of a simple reference.
+				// Lazily-generate an identifier that represents this property access.
+				cache := tsNamespace.LazilyGeneratedProperyAccesses
+				if cache == nil {
+					cache = make(map[string]ast.Ref)
+					tsNamespace.LazilyGeneratedProperyAccesses = cache
+				}
+				ref, ok = cache[name]
+				if !ok {
+					ref = p.newSymbol(ast.SymbolOther, name)
+					p.symbols[ref.InnerIndex].NamespaceAlias = &ast.NamespaceAlias{
+						NamespaceRef: tsNamespace.ArgRef,
+						Alias:        name,
+					}
+					cache[name] = ref
+				}
+				declareLoc = member.Loc
+				break
+			}
+		}
+
+		s = s.Parent
+		if s == nil {
+			// Allocate an "unbound" symbol
+			p.checkForUnrepresentableIdentifier(loc, name)
+			ref = p.newSymbol(ast.SymbolUnbound, name)
+			declareLoc = loc
+			p.moduleScope.Members[name] = js_ast.ScopeMember{Ref: ref, Loc: logger.Loc{Start: -1}}
+			break
+		}
+	}
+
+	// If we had to pass through a "with" statement body to get to the symbol
+	// declaration, then this reference could potentially also refer to a
+	// property on the target object of the "with" statement. We must not rename
+	// it or we risk changing the behavior of the code.
+	if isInsideWithScope {
+		p.symbols[ref.InnerIndex].Flags |= ast.MustNotBeRenamed
+	}
+
+	// Track how many times we've referenced this symbol
+	p.recordUsage(ref)
+	return findSymbolResult{ref, declareLoc, isInsideWithScope}
+}
+
+func (p *parser) findLabelSymbol(loc logger.Loc, name string) (ref ast.Ref, isLoop bool, ok bool) {
+	for s := p.currentScope; s != nil && !s.Kind.StopsHoisting(); s = s.Parent {
+		if s.Kind == js_ast.ScopeLabel && name == p.symbols[s.Label.Ref.InnerIndex].OriginalName {
+			// Track how many times we've referenced this symbol
+			p.recordUsage(s.Label.Ref)
+			ref = s.Label.Ref
+			isLoop = s.LabelStmtIsLoop
+			ok = true
+			return
+		}
+	}
+
+	r := js_lexer.RangeOfIdentifier(p.source, loc)
+	p.log.AddError(&p.tracker, r, fmt.Sprintf("There is no containing label named %q", name))
+
+	// Allocate an "unbound" symbol
+	ref = p.newSymbol(ast.SymbolUnbound, name)
+
+	// Track how many times we've referenced this symbol
+	p.recordUsage(ref)
+	return
+}
+
+func findIdentifiers(binding js_ast.Binding, identifiers []js_ast.Decl) []js_ast.Decl {
+	switch b := binding.Data.(type) {
+	case *js_ast.BIdentifier:
+		identifiers = append(identifiers, js_ast.Decl{Binding: binding})
+
+	case *js_ast.BArray:
+		for _, item := range b.Items {
+			identifiers = findIdentifiers(item.Binding, identifiers)
+		}
+
+	case *js_ast.BObject:
+		for _, property := range b.Properties {
+			identifiers = findIdentifiers(property.Value, identifiers)
+		}
+	}
+
+	return identifiers
+}
+
+// If this is in a dead branch, then we want to trim as much dead code as we
+// can. Everything can be trimmed except for hoisted declarations ("var" and
+// "function"), which affect the parent scope. For example:
+//
+//	function foo() {
+//	  if (false) { var x; }
+//	  x = 1;
+//	}
+//
+// We can't trim the entire branch as dead or calling foo() will incorrectly
+// assign to a global variable instead.
+func shouldKeepStmtInDeadControlFlow(stmt js_ast.Stmt) bool {
+	switch s := stmt.Data.(type) {
+	case *js_ast.SEmpty, *js_ast.SExpr, *js_ast.SThrow, *js_ast.SReturn,
+		*js_ast.SBreak, *js_ast.SContinue, *js_ast.SClass, *js_ast.SDebugger:
+		// Omit these statements entirely
+		return false
+
+	case *js_ast.SLocal:
+		if s.Kind != js_ast.LocalVar {
+			// Omit these statements entirely
+			return false
+		}
+
+		// Omit everything except the identifiers
+		identifiers := []js_ast.Decl{}
+		for _, decl := range s.Decls {
+			identifiers = findIdentifiers(decl.Binding, identifiers)
+		}
+		if len(identifiers) == 0 {
+			return false
+		}
+		s.Decls = identifiers
+		return true
+
+	case *js_ast.SBlock:
+		for _, child := range s.Stmts {
+			if shouldKeepStmtInDeadControlFlow(child) {
+				return true
+			}
+		}
+		return false
+
+	case *js_ast.SIf:
+		return shouldKeepStmtInDeadControlFlow(s.Yes) || (s.NoOrNil.Data != nil && shouldKeepStmtInDeadControlFlow(s.NoOrNil))
+
+	case *js_ast.SWhile:
+		return shouldKeepStmtInDeadControlFlow(s.Body)
+
+	case *js_ast.SDoWhile:
+		return shouldKeepStmtInDeadControlFlow(s.Body)
+
+	case *js_ast.SFor:
+		return (s.InitOrNil.Data != nil && shouldKeepStmtInDeadControlFlow(s.InitOrNil)) || shouldKeepStmtInDeadControlFlow(s.Body)
+
+	case *js_ast.SForIn:
+		return shouldKeepStmtInDeadControlFlow(s.Init) || shouldKeepStmtInDeadControlFlow(s.Body)
+
+	case *js_ast.SForOf:
+		return shouldKeepStmtInDeadControlFlow(s.Init) || shouldKeepStmtInDeadControlFlow(s.Body)
+
+	case *js_ast.SLabel:
+		return shouldKeepStmtInDeadControlFlow(s.Stmt)
+
+	default:
+		// Everything else must be kept
+		return true
+	}
+}
+
+type prependTempRefsOpts struct {
+	fnBodyLoc *logger.Loc
+	kind      stmtsKind
+}
+
+func (p *parser) visitStmtsAndPrependTempRefs(stmts []js_ast.Stmt, opts prependTempRefsOpts) []js_ast.Stmt {
+	oldTempRefs := p.tempRefsToDeclare
+	oldTempRefCount := p.tempRefCount
+	p.tempRefsToDeclare = nil
+	p.tempRefCount = 0
+
+	stmts = p.visitStmts(stmts, opts.kind)
+
+	// Prepend values for "this" and "arguments"
+	if opts.fnBodyLoc != nil {
+		// Capture "this"
+		if ref := p.fnOnlyDataVisit.thisCaptureRef; ref != nil {
+			p.tempRefsToDeclare = append(p.tempRefsToDeclare, tempRef{
+				ref:        *ref,
+				valueOrNil: js_ast.Expr{Loc: *opts.fnBodyLoc, Data: js_ast.EThisShared},
+			})
+			p.currentScope.Generated = append(p.currentScope.Generated, *ref)
+		}
+
+		// Capture "arguments"
+		if ref := p.fnOnlyDataVisit.argumentsCaptureRef; ref != nil {
+			p.tempRefsToDeclare = append(p.tempRefsToDeclare, tempRef{
+				ref:        *ref,
+				valueOrNil: js_ast.Expr{Loc: *opts.fnBodyLoc, Data: &js_ast.EIdentifier{Ref: *p.fnOnlyDataVisit.argumentsRef}},
+			})
+			p.currentScope.Generated = append(p.currentScope.Generated, *ref)
+		}
+	}
+
+	// There may also be special top-level-only temporaries to declare
+	if p.currentScope == p.moduleScope && p.topLevelTempRefsToDeclare != nil {
+		p.tempRefsToDeclare = append(p.tempRefsToDeclare, p.topLevelTempRefsToDeclare...)
+		p.topLevelTempRefsToDeclare = nil
+	}
+
+	// Prepend the generated temporary variables to the beginning of the statement list
+	decls := []js_ast.Decl{}
+	for _, temp := range p.tempRefsToDeclare {
+		if p.symbols[temp.ref.InnerIndex].UseCountEstimate > 0 {
+			decls = append(decls, js_ast.Decl{Binding: js_ast.Binding{Data: &js_ast.BIdentifier{Ref: temp.ref}}, ValueOrNil: temp.valueOrNil})
+			p.recordDeclaredSymbol(temp.ref)
+		}
+	}
+	if len(decls) > 0 {
+		// Skip past leading directives and comments
+		split := 0
+		for split < len(stmts) {
+			switch stmts[split].Data.(type) {
+			case *js_ast.SComment, *js_ast.SDirective:
+				split++
+				continue
+			}
+			break
+		}
+		stmts = append(
+			append(
+				append(
+					[]js_ast.Stmt{},
+					stmts[:split]...),
+				js_ast.Stmt{Data: &js_ast.SLocal{Kind: js_ast.LocalVar, Decls: decls}}),
+			stmts[split:]...)
+	}
+
+	p.tempRefsToDeclare = oldTempRefs
+	p.tempRefCount = oldTempRefCount
+	return stmts
+}
+
+type stmtsKind uint8
+
+const (
+	stmtsNormal stmtsKind = iota
+	stmtsSwitch
+	stmtsLoopBody
+	stmtsFnBody
+)
+
+func (p *parser) visitStmts(stmts []js_ast.Stmt, kind stmtsKind) []js_ast.Stmt {
+	// Save the current control-flow liveness. This represents if we are
+	// currently inside an "if (false) { ... }" block.
+	oldIsControlFlowDead := p.isControlFlowDead
+
+	oldTempLetsToDeclare := p.tempLetsToDeclare
+	p.tempLetsToDeclare = nil
+
+	// Visit all statements first
+	visited := make([]js_ast.Stmt, 0, len(stmts))
+	var before []js_ast.Stmt
+	var after []js_ast.Stmt
+	var preprocessedEnums map[int][]js_ast.Stmt
+	if p.scopesInOrderForEnum != nil {
+		// Preprocess TypeScript enums to improve code generation. Otherwise
+		// uses of an enum before that enum has been declared won't be inlined:
+		//
+		//   console.log(Foo.FOO) // We want "FOO" to be inlined here
+		//   const enum Foo { FOO = 0 }
+		//
+		// The TypeScript compiler itself contains code with this pattern, so
+		// it's important to implement this optimization.
+		for i, stmt := range stmts {
+			if _, ok := stmt.Data.(*js_ast.SEnum); ok {
+				if preprocessedEnums == nil {
+					preprocessedEnums = make(map[int][]js_ast.Stmt)
+				}
+				oldScopesInOrder := p.scopesInOrder
+				p.scopesInOrder = p.scopesInOrderForEnum[stmt.Loc]
+				preprocessedEnums[i] = p.visitAndAppendStmt(nil, stmt)
+				p.scopesInOrder = oldScopesInOrder
+			}
+		}
+	}
+	for i, stmt := range stmts {
+		switch s := stmt.Data.(type) {
+		case *js_ast.SExportEquals:
+			// TypeScript "export = value;" becomes "module.exports = value;". This
+			// must happen at the end after everything is parsed because TypeScript
+			// moves this statement to the end when it generates code.
+			after = p.visitAndAppendStmt(after, stmt)
+			continue
+
+		case *js_ast.SFunction:
+			// Manually hoist block-level function declarations to preserve semantics.
+			// This is only done for function declarations that are not generators
+			// or async functions, since this is a backwards-compatibility hack from
+			// Annex B of the JavaScript standard.
+			if !p.currentScope.Kind.StopsHoisting() && p.symbols[int(s.Fn.Name.Ref.InnerIndex)].Kind == ast.SymbolHoistedFunction {
+				before = p.visitAndAppendStmt(before, stmt)
+				continue
+			}
+
+		case *js_ast.SEnum:
+			visited = append(visited, preprocessedEnums[i]...)
+			p.scopesInOrder = p.scopesInOrder[len(p.scopesInOrderForEnum[stmt.Loc]):]
+			continue
+		}
+		visited = p.visitAndAppendStmt(visited, stmt)
+	}
+
+	// This is used for temporary variables that could be captured in a closure,
+	// and therefore need to be generated inside the nearest enclosing block in
+	// case they are generated inside a loop.
+	if len(p.tempLetsToDeclare) > 0 {
+		decls := make([]js_ast.Decl, 0, len(p.tempLetsToDeclare))
+		for _, ref := range p.tempLetsToDeclare {
+			decls = append(decls, js_ast.Decl{Binding: js_ast.Binding{Data: &js_ast.BIdentifier{Ref: ref}}})
+		}
+		before = append(before, js_ast.Stmt{Data: &js_ast.SLocal{Kind: js_ast.LocalLet, Decls: decls}})
+	}
+	p.tempLetsToDeclare = oldTempLetsToDeclare
+
+	// Transform block-level function declarations into variable declarations
+	if len(before) > 0 {
+		var letDecls []js_ast.Decl
+		var varDecls []js_ast.Decl
+		var nonFnStmts []js_ast.Stmt
+		fnStmts := make(map[ast.Ref]int)
+		for _, stmt := range before {
+			s, ok := stmt.Data.(*js_ast.SFunction)
+			if !ok {
+				// We may get non-function statements here in certain scenarios such as when "KeepNames" is enabled
+				nonFnStmts = append(nonFnStmts, stmt)
+				continue
+			}
+
+			// This transformation of function declarations in nested scopes is
+			// intended to preserve the hoisting semantics of the original code. In
+			// JavaScript, function hoisting works differently in strict mode vs.
+			// sloppy mode code. We want the code we generate to use the semantics of
+			// the original environment, not the generated environment. However, if
+			// direct "eval" is present then it's not possible to preserve the
+			// semantics because we need two identifiers to do that and direct "eval"
+			// means neither identifier can be renamed to something else. So in that
+			// case we give up and do not preserve the semantics of the original code.
+			if p.currentScope.ContainsDirectEval {
+				if hoistedRef, ok := p.hoistedRefForSloppyModeBlockFn[s.Fn.Name.Ref]; ok {
+					// Merge the two identifiers back into a single one
+					p.symbols[hoistedRef.InnerIndex].Link = s.Fn.Name.Ref
+				}
+				nonFnStmts = append(nonFnStmts, stmt)
+				continue
+			}
+
+			index, ok := fnStmts[s.Fn.Name.Ref]
+			if !ok {
+				index = len(letDecls)
+				fnStmts[s.Fn.Name.Ref] = index
+				letDecls = append(letDecls, js_ast.Decl{Binding: js_ast.Binding{
+					Loc: s.Fn.Name.Loc, Data: &js_ast.BIdentifier{Ref: s.Fn.Name.Ref}}})
+
+				// Also write the function to the hoisted sibling symbol if applicable
+				if hoistedRef, ok := p.hoistedRefForSloppyModeBlockFn[s.Fn.Name.Ref]; ok {
+					p.recordDeclaredSymbol(hoistedRef)
+					p.recordUsage(s.Fn.Name.Ref)
+					varDecls = append(varDecls, js_ast.Decl{
+						Binding:    js_ast.Binding{Loc: s.Fn.Name.Loc, Data: &js_ast.BIdentifier{Ref: hoistedRef}},
+						ValueOrNil: js_ast.Expr{Loc: s.Fn.Name.Loc, Data: &js_ast.EIdentifier{Ref: s.Fn.Name.Ref}},
+					})
+				}
+			}
+
+			// The last function statement for a given symbol wins
+			s.Fn.Name = nil
+			letDecls[index].ValueOrNil = js_ast.Expr{Loc: stmt.Loc, Data: &js_ast.EFunction{Fn: s.Fn}}
+		}
+
+		// Reuse memory from "before"
+		before = before[:0]
+		kind := js_ast.LocalLet
+		if p.options.unsupportedJSFeatures.Has(compat.ConstAndLet) {
+			kind = js_ast.LocalVar
+		}
+		if len(letDecls) > 0 {
+			before = append(before, js_ast.Stmt{Loc: letDecls[0].ValueOrNil.Loc, Data: &js_ast.SLocal{Kind: kind, Decls: letDecls}})
+		}
+		if len(varDecls) > 0 {
+			// Potentially relocate "var" declarations to the top level
+			if assign, ok := p.maybeRelocateVarsToTopLevel(varDecls, relocateVarsNormal); ok {
+				if assign.Data != nil {
+					before = append(before, assign)
+				}
+			} else {
+				before = append(before, js_ast.Stmt{Loc: varDecls[0].ValueOrNil.Loc, Data: &js_ast.SLocal{Kind: js_ast.LocalVar, Decls: varDecls}})
+			}
+		}
+		before = append(before, nonFnStmts...)
+		visited = append(before, visited...)
+	}
+
+	// Move TypeScript "export =" statements to the end
+	visited = append(visited, after...)
+
+	// Restore the current control-flow liveness if it was changed inside the
+	// loop above. This is important because the caller will not restore it.
+	p.isControlFlowDead = oldIsControlFlowDead
+
+	// Lower using declarations
+	if kind != stmtsSwitch && p.shouldLowerUsingDeclarations(visited) {
+		ctx := p.lowerUsingDeclarationContext()
+		ctx.scanStmts(p, visited)
+		visited = ctx.finalize(p, visited, p.currentScope.Parent == nil)
+	}
+
+	// Stop now if we're not mangling
+	if !p.options.minifySyntax {
+		return visited
+	}
+
+	// If this is in a dead branch, trim as much dead code as we can
+	if p.isControlFlowDead {
+		end := 0
+		for _, stmt := range visited {
+			if !shouldKeepStmtInDeadControlFlow(stmt) {
+				continue
+			}
+
+			// Merge adjacent var statements
+			if s, ok := stmt.Data.(*js_ast.SLocal); ok && s.Kind == js_ast.LocalVar && end > 0 {
+				prevStmt := visited[end-1]
+				if prevS, ok := prevStmt.Data.(*js_ast.SLocal); ok && prevS.Kind == js_ast.LocalVar && s.IsExport == prevS.IsExport {
+					prevS.Decls = append(prevS.Decls, s.Decls...)
+					continue
+				}
+			}
+
+			visited[end] = stmt
+			end++
+		}
+		return visited[:end]
+	}
+
+	return p.mangleStmts(visited, kind)
+}
+
+func (p *parser) mangleStmts(stmts []js_ast.Stmt, kind stmtsKind) []js_ast.Stmt {
+	// Remove inlined constants now that we know whether any of these statements
+	// contained a direct eval() or not. This can't be done earlier when we
+	// encounter the constant because we haven't encountered the eval() yet.
+	// Inlined constants are not removed if they are in a top-level scope or
+	// if they are exported (which could be in a nested TypeScript namespace).
+	if p.currentScope.Parent != nil && !p.currentScope.ContainsDirectEval {
+		for i, stmt := range stmts {
+			switch s := stmt.Data.(type) {
+			case *js_ast.SEmpty, *js_ast.SComment, *js_ast.SDirective, *js_ast.SDebugger, *js_ast.STypeScript:
+				continue
+
+			case *js_ast.SLocal:
+				if !s.IsExport {
+					end := 0
+					for _, d := range s.Decls {
+						if id, ok := d.Binding.Data.(*js_ast.BIdentifier); ok {
+							if _, ok := p.constValues[id.Ref]; ok && p.symbols[id.Ref.InnerIndex].UseCountEstimate == 0 {
+								continue
+							}
+						}
+						s.Decls[end] = d
+						end++
+					}
+					if end == 0 {
+						stmts[i].Data = js_ast.SEmptyShared
+					} else {
+						s.Decls = s.Decls[:end]
+					}
+				}
+				continue
+			}
+			break
+		}
+	}
+
+	// Merge adjacent statements during mangling
+	result := make([]js_ast.Stmt, 0, len(stmts))
+	isControlFlowDead := false
+	for i, stmt := range stmts {
+		if isControlFlowDead && !shouldKeepStmtInDeadControlFlow(stmt) {
+			// Strip unnecessary statements if the control flow is dead here
+			continue
+		}
+
+		// Inline single-use variable declarations where possible:
+		//
+		//   // Before
+		//   let x = fn();
+		//   return x.y();
+		//
+		//   // After
+		//   return fn().y();
+		//
+		// The declaration must not be exported. We can't just check for the
+		// "export" keyword because something might do "export {id};" later on.
+		// Instead we just ignore all top-level declarations for now. That means
+		// this optimization currently only applies in nested scopes.
+		//
+		// Ignore declarations if the scope is shadowed by a direct "eval" call.
+		// The eval'd code may indirectly reference this symbol and the actual
+		// use count may be greater than 1.
+		if p.currentScope != p.moduleScope && !p.currentScope.ContainsDirectEval {
+			// Keep inlining variables until a failure or until there are none left.
+			// That handles cases like this:
+			//
+			//   // Before
+			//   let x = fn();
+			//   let y = x.prop;
+			//   return y;
+			//
+			//   // After
+			//   return fn().prop;
+			//
+			for len(result) > 0 {
+				// Ignore "var" declarations since those have function-level scope and
+				// we may not have visited all of their uses yet by this point. We
+				// should have visited all the uses of "let" and "const" declarations
+				// by now since they are scoped to this block which we just finished
+				// visiting.
+				if prevS, ok := result[len(result)-1].Data.(*js_ast.SLocal); ok && prevS.Kind != js_ast.LocalVar {
+					last := prevS.Decls[len(prevS.Decls)-1]
+
+					// The binding must be an identifier that is only used once.
+					// Ignore destructuring bindings since that's not the simple case.
+					// Destructuring bindings could potentially execute side-effecting
+					// code which would invalidate reordering.
+					if id, ok := last.Binding.Data.(*js_ast.BIdentifier); ok {
+						// Don't do this if "__name" was called on this symbol. In that
+						// case there is actually more than one use even though it says
+						// there is only one. The "__name" use isn't counted so that
+						// tree shaking still works when names are kept.
+						if symbol := p.symbols[id.Ref.InnerIndex]; symbol.UseCountEstimate == 1 && !symbol.Flags.Has(ast.DidKeepName) {
+							replacement := last.ValueOrNil
+
+							// The variable must be initialized, since we will be substituting
+							// the value into the usage.
+							if replacement.Data == nil {
+								replacement = js_ast.Expr{Loc: last.Binding.Loc, Data: js_ast.EUndefinedShared}
+							}
+
+							// Try to substitute the identifier with the initializer. This will
+							// fail if something with side effects is in between the declaration
+							// and the usage.
+							if p.substituteSingleUseSymbolInStmt(stmt, id.Ref, replacement) {
+								// Remove the previous declaration, since the substitution was
+								// successful.
+								if len(prevS.Decls) == 1 {
+									result = result[:len(result)-1]
+								} else {
+									prevS.Decls = prevS.Decls[:len(prevS.Decls)-1]
+								}
+
+								// Loop back to try again
+								continue
+							}
+						}
+					}
+				}
+
+				// Substitution failed so stop trying
+				break
+			}
+		}
+
+		switch s := stmt.Data.(type) {
+		case *js_ast.SEmpty:
+			// Strip empty statements
+			continue
+
+		case *js_ast.SLocal:
+			// Merge adjacent local statements
+			if len(result) > 0 {
+				prevStmt := result[len(result)-1]
+				if prevS, ok := prevStmt.Data.(*js_ast.SLocal); ok && s.Kind == prevS.Kind && s.IsExport == prevS.IsExport {
+					prevS.Decls = append(prevS.Decls, s.Decls...)
+					continue
+				}
+			}
+
+		case *js_ast.SExpr:
+			// Merge adjacent expression statements
+			if len(result) > 0 {
+				prevStmt := result[len(result)-1]
+				if prevS, ok := prevStmt.Data.(*js_ast.SExpr); ok {
+					if !s.IsFromClassOrFnThatCanBeRemovedIfUnused {
+						prevS.IsFromClassOrFnThatCanBeRemovedIfUnused = false
+					}
+					prevS.Value = js_ast.JoinWithComma(prevS.Value, s.Value)
+					continue
+				}
+			}
+
+		case *js_ast.SSwitch:
+			// Absorb a previous expression statement
+			if len(result) > 0 {
+				prevStmt := result[len(result)-1]
+				if prevS, ok := prevStmt.Data.(*js_ast.SExpr); ok {
+					s.Test = js_ast.JoinWithComma(prevS.Value, s.Test)
+					result = result[:len(result)-1]
+				}
+			}
+
+		case *js_ast.SIf:
+			// Absorb a previous expression statement
+			if len(result) > 0 {
+				prevStmt := result[len(result)-1]
+				if prevS, ok := prevStmt.Data.(*js_ast.SExpr); ok {
+					s.Test = js_ast.JoinWithComma(prevS.Value, s.Test)
+					result = result[:len(result)-1]
+				}
+			}
+
+			if isJumpStatement(s.Yes.Data) {
+				optimizeImplicitJump := false
+
+				// Absorb a previous if statement
+				if len(result) > 0 {
+					prevStmt := result[len(result)-1]
+					if prevS, ok := prevStmt.Data.(*js_ast.SIf); ok && prevS.NoOrNil.Data == nil && jumpStmtsLookTheSame(prevS.Yes.Data, s.Yes.Data) {
+						// "if (a) break c; if (b) break c;" => "if (a || b) break c;"
+						// "if (a) continue c; if (b) continue c;" => "if (a || b) continue c;"
+						// "if (a) return c; if (b) return c;" => "if (a || b) return c;"
+						// "if (a) throw c; if (b) throw c;" => "if (a || b) throw c;"
+						s.Test = js_ast.JoinWithLeftAssociativeOp(js_ast.BinOpLogicalOr, prevS.Test, s.Test)
+						result = result[:len(result)-1]
+					}
+				}
+
+				// "while (x) { if (y) continue; z(); }" => "while (x) { if (!y) z(); }"
+				// "while (x) { if (y) continue; else z(); w(); }" => "while (x) { if (!y) { z(); w(); } }" => "for (; x;) !y && (z(), w());"
+				if kind == stmtsLoopBody {
+					if continueS, ok := s.Yes.Data.(*js_ast.SContinue); ok && continueS.Label == nil {
+						optimizeImplicitJump = true
+					}
+				}
+
+				// "let x = () => { if (y) return; z(); };" => "let x = () => { if (!y) z(); };"
+				// "let x = () => { if (y) return; else z(); w(); };" => "let x = () => { if (!y) { z(); w(); } };" => "let x = () => { !y && (z(), w()); };"
+				if kind == stmtsFnBody {
+					if returnS, ok := s.Yes.Data.(*js_ast.SReturn); ok && returnS.ValueOrNil.Data == nil {
+						optimizeImplicitJump = true
+					}
+				}
+
+				if optimizeImplicitJump {
+					var body []js_ast.Stmt
+					if s.NoOrNil.Data != nil {
+						body = append(body, s.NoOrNil)
+					}
+					body = append(body, stmts[i+1:]...)
+
+					// Don't do this transformation if the branch condition could
+					// potentially access symbols declared later on on this scope below.
+					// If so, inverting the branch condition and nesting statements after
+					// this in a block would break that access which is a behavior change.
+					//
+					//   // This transformation is incorrect
+					//   if (a()) return; function a() {}
+					//   if (!a()) { function a() {} }
+					//
+					//   // This transformation is incorrect
+					//   if (a(() => b)) return; let b;
+					//   if (a(() => b)) { let b; }
+					//
+					canMoveBranchConditionOutsideScope := true
+					for _, stmt := range body {
+						if statementCaresAboutScope(stmt) {
+							canMoveBranchConditionOutsideScope = false
+							break
+						}
+					}
+
+					if canMoveBranchConditionOutsideScope {
+						body = p.mangleStmts(body, kind)
+						bodyLoc := s.Yes.Loc
+						if len(body) > 0 {
+							bodyLoc = body[0].Loc
+						}
+						return p.mangleIf(result, stmt.Loc, &js_ast.SIf{
+							Test: p.astHelpers.SimplifyBooleanExpr(js_ast.Not(s.Test)),
+							Yes:  stmtsToSingleStmt(bodyLoc, body, logger.Loc{}),
+						})
+					}
+				}
+
+				if s.NoOrNil.Data != nil {
+					// "if (a) return b; else if (c) return d; else return e;" => "if (a) return b; if (c) return d; return e;"
+					for {
+						result = append(result, stmt)
+						stmt = s.NoOrNil
+						s.NoOrNil = js_ast.Stmt{}
+						var ok bool
+						s, ok = stmt.Data.(*js_ast.SIf)
+						if !ok || !isJumpStatement(s.Yes.Data) || s.NoOrNil.Data == nil {
+							break
+						}
+					}
+					result = appendIfOrLabelBodyPreservingScope(result, stmt)
+					if isJumpStatement(stmt.Data) {
+						isControlFlowDead = true
+					}
+					continue
+				}
+			}
+
+		case *js_ast.SReturn:
+			// Merge return statements with the previous expression statement
+			if len(result) > 0 && s.ValueOrNil.Data != nil {
+				prevStmt := result[len(result)-1]
+				if prevS, ok := prevStmt.Data.(*js_ast.SExpr); ok {
+					result[len(result)-1] = js_ast.Stmt{Loc: prevStmt.Loc,
+						Data: &js_ast.SReturn{ValueOrNil: js_ast.JoinWithComma(prevS.Value, s.ValueOrNil)}}
+					continue
+				}
+			}
+
+			isControlFlowDead = true
+
+		case *js_ast.SThrow:
+			// Merge throw statements with the previous expression statement
+			if len(result) > 0 {
+				prevStmt := result[len(result)-1]
+				if prevS, ok := prevStmt.Data.(*js_ast.SExpr); ok {
+					result[len(result)-1] = js_ast.Stmt{Loc: prevStmt.Loc, Data: &js_ast.SThrow{Value: js_ast.JoinWithComma(prevS.Value, s.Value)}}
+					continue
+				}
+			}
+
+			isControlFlowDead = true
+
+		case *js_ast.SBreak, *js_ast.SContinue:
+			isControlFlowDead = true
+
+		case *js_ast.SFor:
+			if len(result) > 0 {
+				prevStmt := result[len(result)-1]
+				if prevS, ok := prevStmt.Data.(*js_ast.SExpr); ok {
+					// Insert the previous expression into the for loop initializer
+					if s.InitOrNil.Data == nil {
+						result[len(result)-1] = stmt
+						s.InitOrNil = js_ast.Stmt{Loc: prevStmt.Loc, Data: &js_ast.SExpr{Value: prevS.Value}}
+						continue
+					} else if s2, ok := s.InitOrNil.Data.(*js_ast.SExpr); ok {
+						result[len(result)-1] = stmt
+						s.InitOrNil = js_ast.Stmt{Loc: prevStmt.Loc, Data: &js_ast.SExpr{Value: js_ast.JoinWithComma(prevS.Value, s2.Value)}}
+						continue
+					}
+				} else {
+					// Insert the previous variable declaration into the for loop
+					// initializer if it's a "var" declaration, since the scope
+					// doesn't matter due to scope hoisting
+					if s.InitOrNil.Data == nil {
+						if s2, ok := prevStmt.Data.(*js_ast.SLocal); ok && s2.Kind == js_ast.LocalVar && !s2.IsExport {
+							result[len(result)-1] = stmt
+							s.InitOrNil = prevStmt
+							continue
+						}
+					} else {
+						if s2, ok := prevStmt.Data.(*js_ast.SLocal); ok && s2.Kind == js_ast.LocalVar && !s2.IsExport {
+							if s3, ok := s.InitOrNil.Data.(*js_ast.SLocal); ok && s3.Kind == js_ast.LocalVar {
+								result[len(result)-1] = stmt
+								s.InitOrNil.Data = &js_ast.SLocal{Kind: js_ast.LocalVar, Decls: append(s2.Decls, s3.Decls...)}
+								continue
+							}
+						}
+					}
+				}
+			}
+
+		case *js_ast.STry:
+			// Drop an unused identifier binding if the optional catch binding feature is supported
+			if !p.options.unsupportedJSFeatures.Has(compat.OptionalCatchBinding) && s.Catch != nil {
+				if id, ok := s.Catch.BindingOrNil.Data.(*js_ast.BIdentifier); ok {
+					if symbol := p.symbols[id.Ref.InnerIndex]; symbol.UseCountEstimate == 0 {
+						if symbol.Link != ast.InvalidRef {
+							// We cannot transform "try { x() } catch (y) { var y = 1 }" into
+							// "try { x() } catch { var y = 1 }" even though "y" is never used
+							// because the hoisted variable "y" would have different values
+							// after the statement ends due to a strange JavaScript quirk:
+							//
+							//   try { x() } catch (y) { var y = 1 }
+							//   console.log(y) // undefined
+							//
+							//   try { x() } catch { var y = 1 }
+							//   console.log(y) // 1
+							//
+						} else if p.currentScope.ContainsDirectEval {
+							// We cannot transform "try { x() } catch (y) { eval('z = y') }"
+							// into "try { x() } catch { eval('z = y') }" because the variable
+							// "y" is actually still used.
+						} else {
+							// "try { x() } catch (y) {}" => "try { x() } catch {}"
+							s.Catch.BindingOrNil.Data = nil
+						}
+					}
+				}
+			}
+		}
+
+		result = append(result, stmt)
+	}
+
+	// Drop a trailing unconditional jump statement if applicable
+	if len(result) > 0 {
+		switch kind {
+		case stmtsLoopBody:
+			// "while (x) { y(); continue; }" => "while (x) { y(); }"
+			if continueS, ok := result[len(result)-1].Data.(*js_ast.SContinue); ok && continueS.Label == nil {
+				result = result[:len(result)-1]
+			}
+
+		case stmtsFnBody:
+			// "function f() { x(); return; }" => "function f() { x(); }"
+			if returnS, ok := result[len(result)-1].Data.(*js_ast.SReturn); ok && returnS.ValueOrNil.Data == nil {
+				result = result[:len(result)-1]
+			}
+		}
+	}
+
+	// Merge certain statements in reverse order
+	if len(result) >= 2 {
+		lastStmt := result[len(result)-1]
+
+		if lastReturn, ok := lastStmt.Data.(*js_ast.SReturn); ok {
+			// "if (a) return b; if (c) return d; return e;" => "return a ? b : c ? d : e;"
+		returnLoop:
+			for len(result) >= 2 {
+				prevIndex := len(result) - 2
+				prevStmt := result[prevIndex]
+
+				switch prevS := prevStmt.Data.(type) {
+				case *js_ast.SExpr:
+					// This return statement must have a value
+					if lastReturn.ValueOrNil.Data == nil {
+						break returnLoop
+					}
+
+					// "a(); return b;" => "return a(), b;"
+					lastReturn = &js_ast.SReturn{ValueOrNil: js_ast.JoinWithComma(prevS.Value, lastReturn.ValueOrNil)}
+
+					// Merge the last two statements
+					lastStmt = js_ast.Stmt{Loc: prevStmt.Loc, Data: lastReturn}
+					result[prevIndex] = lastStmt
+					result = result[:len(result)-1]
+
+				case *js_ast.SIf:
+					// The previous statement must be an if statement with no else clause
+					if prevS.NoOrNil.Data != nil {
+						break returnLoop
+					}
+
+					// The then clause must be a return
+					prevReturn, ok := prevS.Yes.Data.(*js_ast.SReturn)
+					if !ok {
+						break returnLoop
+					}
+
+					// Handle some or all of the values being undefined
+					left := prevReturn.ValueOrNil
+					right := lastReturn.ValueOrNil
+					if left.Data == nil {
+						// "if (a) return; return b;" => "return a ? void 0 : b;"
+						left = js_ast.Expr{Loc: prevS.Yes.Loc, Data: js_ast.EUndefinedShared}
+					}
+					if right.Data == nil {
+						// "if (a) return a; return;" => "return a ? b : void 0;"
+						right = js_ast.Expr{Loc: lastStmt.Loc, Data: js_ast.EUndefinedShared}
+					}
+
+					// "if (!a) return b; return c;" => "return a ? c : b;"
+					if not, ok := prevS.Test.Data.(*js_ast.EUnary); ok && not.Op == js_ast.UnOpNot {
+						prevS.Test = not.Value
+						left, right = right, left
+					}
+
+					if comma, ok := prevS.Test.Data.(*js_ast.EBinary); ok && comma.Op == js_ast.BinOpComma {
+						// "if (a, b) return c; return d;" => "return a, b ? c : d;"
+						lastReturn = &js_ast.SReturn{ValueOrNil: js_ast.JoinWithComma(comma.Left,
+							p.astHelpers.MangleIfExpr(comma.Right.Loc, &js_ast.EIf{Test: comma.Right, Yes: left, No: right}, p.options.unsupportedJSFeatures))}
+					} else {
+						// "if (a) return b; return c;" => "return a ? b : c;"
+						lastReturn = &js_ast.SReturn{ValueOrNil: p.astHelpers.MangleIfExpr(
+							prevS.Test.Loc, &js_ast.EIf{Test: prevS.Test, Yes: left, No: right}, p.options.unsupportedJSFeatures)}
+					}
+
+					// Merge the last two statements
+					lastStmt = js_ast.Stmt{Loc: prevStmt.Loc, Data: lastReturn}
+					result[prevIndex] = lastStmt
+					result = result[:len(result)-1]
+
+				default:
+					break returnLoop
+				}
+			}
+		} else if lastThrow, ok := lastStmt.Data.(*js_ast.SThrow); ok {
+			// "if (a) throw b; if (c) throw d; throw e;" => "throw a ? b : c ? d : e;"
+		throwLoop:
+			for len(result) >= 2 {
+				prevIndex := len(result) - 2
+				prevStmt := result[prevIndex]
+
+				switch prevS := prevStmt.Data.(type) {
+				case *js_ast.SExpr:
+					// "a(); throw b;" => "throw a(), b;"
+					lastThrow = &js_ast.SThrow{Value: js_ast.JoinWithComma(prevS.Value, lastThrow.Value)}
+
+					// Merge the last two statements
+					lastStmt = js_ast.Stmt{Loc: prevStmt.Loc, Data: lastThrow}
+					result[prevIndex] = lastStmt
+					result = result[:len(result)-1]
+
+				case *js_ast.SIf:
+					// The previous statement must be an if statement with no else clause
+					if prevS.NoOrNil.Data != nil {
+						break throwLoop
+					}
+
+					// The then clause must be a throw
+					prevThrow, ok := prevS.Yes.Data.(*js_ast.SThrow)
+					if !ok {
+						break throwLoop
+					}
+
+					left := prevThrow.Value
+					right := lastThrow.Value
+
+					// "if (!a) throw b; throw c;" => "throw a ? c : b;"
+					if not, ok := prevS.Test.Data.(*js_ast.EUnary); ok && not.Op == js_ast.UnOpNot {
+						prevS.Test = not.Value
+						left, right = right, left
+					}
+
+					// Merge the last two statements
+					if comma, ok := prevS.Test.Data.(*js_ast.EBinary); ok && comma.Op == js_ast.BinOpComma {
+						// "if (a, b) return c; return d;" => "return a, b ? c : d;"
+						lastThrow = &js_ast.SThrow{Value: js_ast.JoinWithComma(comma.Left,
+							p.astHelpers.MangleIfExpr(comma.Right.Loc, &js_ast.EIf{Test: comma.Right, Yes: left, No: right}, p.options.unsupportedJSFeatures))}
+					} else {
+						// "if (a) return b; return c;" => "return a ? b : c;"
+						lastThrow = &js_ast.SThrow{
+							Value: p.astHelpers.MangleIfExpr(prevS.Test.Loc, &js_ast.EIf{Test: prevS.Test, Yes: left, No: right}, p.options.unsupportedJSFeatures)}
+					}
+					lastStmt = js_ast.Stmt{Loc: prevStmt.Loc, Data: lastThrow}
+					result[prevIndex] = lastStmt
+					result = result[:len(result)-1]
+
+				default:
+					break throwLoop
+				}
+			}
+		}
+	}
+
+	return result
+}
+
+func (p *parser) substituteSingleUseSymbolInStmt(stmt js_ast.Stmt, ref ast.Ref, replacement js_ast.Expr) bool {
+	var expr *js_ast.Expr
+
+	switch s := stmt.Data.(type) {
+	case *js_ast.SExpr:
+		expr = &s.Value
+	case *js_ast.SThrow:
+		expr = &s.Value
+	case *js_ast.SReturn:
+		expr = &s.ValueOrNil
+	case *js_ast.SIf:
+		expr = &s.Test
+	case *js_ast.SSwitch:
+		expr = &s.Test
+	case *js_ast.SLocal:
+		// Only try substituting into the initializer for the first declaration
+		if first := &s.Decls[0]; first.ValueOrNil.Data != nil {
+			// Make sure there isn't destructuring, which could evaluate code
+			if _, ok := first.Binding.Data.(*js_ast.BIdentifier); ok {
+				expr = &first.ValueOrNil
+			}
+		}
+	}
+
+	if expr != nil {
+		// Only continue trying to insert this replacement into sub-expressions
+		// after the first one if the replacement has no side effects:
+		//
+		//   // Substitution is ok
+		//   let replacement = 123;
+		//   return x + replacement;
+		//
+		//   // Substitution is not ok because "fn()" may change "x"
+		//   let replacement = fn();
+		//   return x + replacement;
+		//
+		//   // Substitution is not ok because "x == x" may change "x" due to "valueOf()" evaluation
+		//   let replacement = [x];
+		//   return (x == x) + replacement;
+		//
+		replacementCanBeRemoved := p.astHelpers.ExprCanBeRemovedIfUnused(replacement)
+
+		if new, status := p.substituteSingleUseSymbolInExpr(*expr, ref, replacement, replacementCanBeRemoved); status == substituteSuccess {
+			*expr = new
+			return true
+		}
+	}
+
+	return false
+}
+
+type substituteStatus uint8
+
+const (
+	substituteContinue substituteStatus = iota
+	substituteSuccess
+	substituteFailure
+)
+
+func (p *parser) substituteSingleUseSymbolInExpr(
+	expr js_ast.Expr,
+	ref ast.Ref,
+	replacement js_ast.Expr,
+	replacementCanBeRemoved bool,
+) (js_ast.Expr, substituteStatus) {
+	switch e := expr.Data.(type) {
+	case *js_ast.EIdentifier:
+		if e.Ref == ref {
+			p.ignoreUsage(ref)
+			return replacement, substituteSuccess
+		}
+
+	case *js_ast.ESpread:
+		if value, status := p.substituteSingleUseSymbolInExpr(e.Value, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+			e.Value = value
+			return expr, status
+		}
+
+	case *js_ast.EAwait:
+		if value, status := p.substituteSingleUseSymbolInExpr(e.Value, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+			e.Value = value
+			return expr, status
+		}
+
+	case *js_ast.EYield:
+		if e.ValueOrNil.Data != nil {
+			if value, status := p.substituteSingleUseSymbolInExpr(e.ValueOrNil, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+				e.ValueOrNil = value
+				return expr, status
+			}
+		}
+
+	case *js_ast.EImportCall:
+		if value, status := p.substituteSingleUseSymbolInExpr(e.Expr, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+			e.Expr = value
+			return expr, status
+		}
+
+		// The "import()" expression has side effects but the side effects are
+		// always asynchronous so there is no way for the side effects to modify
+		// the replacement value. So it's ok to reorder the replacement value
+		// past the "import()" expression assuming everything else checks out.
+		if replacementCanBeRemoved && p.astHelpers.ExprCanBeRemovedIfUnused(e.Expr) {
+			return expr, substituteContinue
+		}
+
+	case *js_ast.EUnary:
+		switch e.Op {
+		case js_ast.UnOpPreInc, js_ast.UnOpPostInc, js_ast.UnOpPreDec, js_ast.UnOpPostDec, js_ast.UnOpDelete:
+			// Do not substitute into an assignment position
+
+		default:
+			if value, status := p.substituteSingleUseSymbolInExpr(e.Value, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+				e.Value = value
+				return expr, status
+			}
+		}
+
+	case *js_ast.EDot:
+		if value, status := p.substituteSingleUseSymbolInExpr(e.Target, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+			e.Target = value
+			return expr, status
+		}
+
+	case *js_ast.EBinary:
+		// Do not substitute into an assignment position
+		if e.Op.BinaryAssignTarget() == js_ast.AssignTargetNone {
+			if value, status := p.substituteSingleUseSymbolInExpr(e.Left, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+				e.Left = value
+				return expr, status
+			}
+		} else if !p.astHelpers.ExprCanBeRemovedIfUnused(e.Left) {
+			// Do not reorder past a side effect in an assignment target, as that may
+			// change the replacement value. For example, "fn()" may change "a" here:
+			//
+			//   let a = 1;
+			//   foo[fn()] = a;
+			//
+			return expr, substituteFailure
+		} else if e.Op.BinaryAssignTarget() == js_ast.AssignTargetUpdate && !replacementCanBeRemoved {
+			// If this is a read-modify-write assignment and the replacement has side
+			// effects, don't reorder it past the assignment target. The assignment
+			// target is being read so it may be changed by the side effect. For
+			// example, "fn()" may change "foo" here:
+			//
+			//   let a = fn();
+			//   foo += a;
+			//
+			return expr, substituteFailure
+		}
+
+		// If we get here then it should be safe to attempt to substitute the
+		// replacement past the left operand into the right operand.
+		if value, status := p.substituteSingleUseSymbolInExpr(e.Right, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+			e.Right = value
+			return expr, status
+		}
+
+	case *js_ast.EIf:
+		if value, status := p.substituteSingleUseSymbolInExpr(e.Test, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+			e.Test = value
+			return expr, status
+		}
+
+		// Do not substitute our unconditionally-executed value into a branch
+		// unless the value itself has no side effects
+		if replacementCanBeRemoved {
+			// Unlike other branches in this function such as "a && b" or "a?.[b]",
+			// the "a ? b : c" form has potential code evaluation along both control
+			// flow paths. Handle this by allowing substitution into either branch.
+			// Side effects in one branch should not prevent the substitution into
+			// the other branch.
+
+			yesValue, yesStatus := p.substituteSingleUseSymbolInExpr(e.Yes, ref, replacement, replacementCanBeRemoved)
+			if yesStatus == substituteSuccess {
+				e.Yes = yesValue
+				return expr, yesStatus
+			}
+
+			noValue, noStatus := p.substituteSingleUseSymbolInExpr(e.No, ref, replacement, replacementCanBeRemoved)
+			if noStatus == substituteSuccess {
+				e.No = noValue
+				return expr, noStatus
+			}
+
+			// Side effects in either branch should stop us from continuing to try to
+			// substitute the replacement after the control flow branches merge again.
+			if yesStatus != substituteContinue || noStatus != substituteContinue {
+				return expr, substituteFailure
+			}
+		}
+
+	case *js_ast.EIndex:
+		if value, status := p.substituteSingleUseSymbolInExpr(e.Target, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+			e.Target = value
+			return expr, status
+		}
+
+		// Do not substitute our unconditionally-executed value into a branch
+		// unless the value itself has no side effects
+		if replacementCanBeRemoved || e.OptionalChain == js_ast.OptionalChainNone {
+			if value, status := p.substituteSingleUseSymbolInExpr(e.Index, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+				e.Index = value
+				return expr, status
+			}
+		}
+
+	case *js_ast.ECall:
+		// Don't substitute something into a call target that could change "this"
+		_, isDot := replacement.Data.(*js_ast.EDot)
+		_, isIndex := replacement.Data.(*js_ast.EIndex)
+		if isDot || isIndex {
+			if id, ok := e.Target.Data.(*js_ast.EIdentifier); ok && id.Ref == ref {
+				break
+			}
+		}
+
+		if value, status := p.substituteSingleUseSymbolInExpr(e.Target, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+			e.Target = value
+			return expr, status
+		}
+
+		// Do not substitute our unconditionally-executed value into a branch
+		// unless the value itself has no side effects
+		if replacementCanBeRemoved || e.OptionalChain == js_ast.OptionalChainNone {
+			for i, arg := range e.Args {
+				if value, status := p.substituteSingleUseSymbolInExpr(arg, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+					e.Args[i] = value
+					return expr, status
+				}
+			}
+		}
+
+	case *js_ast.EArray:
+		for i, item := range e.Items {
+			if value, status := p.substituteSingleUseSymbolInExpr(item, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+				e.Items[i] = value
+				return expr, status
+			}
+		}
+
+	case *js_ast.EObject:
+		for i, property := range e.Properties {
+			// Check the key
+			if property.Flags.Has(js_ast.PropertyIsComputed) {
+				if value, status := p.substituteSingleUseSymbolInExpr(property.Key, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+					e.Properties[i].Key = value
+					return expr, status
+				}
+
+				// Stop now because both computed keys and property spread have side effects
+				return expr, substituteFailure
+			}
+
+			// Check the value
+			if property.ValueOrNil.Data != nil {
+				if value, status := p.substituteSingleUseSymbolInExpr(property.ValueOrNil, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+					e.Properties[i].ValueOrNil = value
+					return expr, status
+				}
+			}
+		}
+
+	case *js_ast.ETemplate:
+		if e.TagOrNil.Data != nil {
+			if value, status := p.substituteSingleUseSymbolInExpr(e.TagOrNil, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+				e.TagOrNil = value
+				return expr, status
+			}
+		}
+
+		for i, part := range e.Parts {
+			if value, status := p.substituteSingleUseSymbolInExpr(part.Value, ref, replacement, replacementCanBeRemoved); status != substituteContinue {
+				e.Parts[i].Value = value
+
+				// If we substituted a primitive, merge it into the template
+				if js_ast.IsPrimitiveLiteral(value.Data) {
+					expr = js_ast.InlinePrimitivesIntoTemplate(expr.Loc, e)
+				}
+				return expr, status
+			}
+		}
+	}
+
+	// If both the replacement and this expression have no observable side
+	// effects, then we can reorder the replacement past this expression
+	if replacementCanBeRemoved && p.astHelpers.ExprCanBeRemovedIfUnused(expr) {
+		return expr, substituteContinue
+	}
+
+	// We can always reorder past primitive values
+	if js_ast.IsPrimitiveLiteral(expr.Data) || js_ast.IsPrimitiveLiteral(replacement.Data) {
+		return expr, substituteContinue
+	}
+
+	// Otherwise we should stop trying to substitute past this point
+	return expr, substituteFailure
+}
+
+func (p *parser) visitLoopBody(stmt js_ast.Stmt) js_ast.Stmt {
+	oldIsInsideLoop := p.fnOrArrowDataVisit.isInsideLoop
+	p.fnOrArrowDataVisit.isInsideLoop = true
+	p.loopBody = stmt.Data
+	stmt = p.visitSingleStmt(stmt, stmtsLoopBody)
+	p.fnOrArrowDataVisit.isInsideLoop = oldIsInsideLoop
+	return stmt
+}
+
+func (p *parser) visitSingleStmt(stmt js_ast.Stmt, kind stmtsKind) js_ast.Stmt {
+	// To reduce stack depth, special-case blocks and process their children directly
+	if block, ok := stmt.Data.(*js_ast.SBlock); ok {
+		p.pushScopeForVisitPass(js_ast.ScopeBlock, stmt.Loc)
+		block.Stmts = p.visitStmts(block.Stmts, kind)
+		p.popScope()
+		if p.options.minifySyntax {
+			stmt = stmtsToSingleStmt(stmt.Loc, block.Stmts, block.CloseBraceLoc)
+		}
+		return stmt
+	}
+
+	// Introduce a fake block scope for function declarations inside if statements
+	fn, ok := stmt.Data.(*js_ast.SFunction)
+	hasIfScope := ok && fn.Fn.HasIfScope
+	if hasIfScope {
+		p.pushScopeForVisitPass(js_ast.ScopeBlock, stmt.Loc)
+		if p.isStrictMode() {
+			p.markStrictModeFeature(ifElseFunctionStmt, js_lexer.RangeOfIdentifier(p.source, stmt.Loc), "")
+		}
+	}
+
+	stmts := p.visitStmts([]js_ast.Stmt{stmt}, kind)
+
+	// Balance the fake block scope introduced above
+	if hasIfScope {
+		p.popScope()
+	}
+
+	return stmtsToSingleStmt(stmt.Loc, stmts, logger.Loc{})
+}
+
+// One statement could potentially expand to several statements
+func stmtsToSingleStmt(loc logger.Loc, stmts []js_ast.Stmt, closeBraceLoc logger.Loc) js_ast.Stmt {
+	if len(stmts) == 0 {
+		return js_ast.Stmt{Loc: loc, Data: js_ast.SEmptyShared}
+	}
+	if len(stmts) == 1 && !statementCaresAboutScope(stmts[0]) {
+		return stmts[0]
+	}
+	return js_ast.Stmt{Loc: loc, Data: &js_ast.SBlock{Stmts: stmts, CloseBraceLoc: closeBraceLoc}}
+}
+
+func (p *parser) visitForLoopInit(stmt js_ast.Stmt, isInOrOf bool) js_ast.Stmt {
+	switch s := stmt.Data.(type) {
+	case *js_ast.SExpr:
+		assignTarget := js_ast.AssignTargetNone
+		if isInOrOf {
+			assignTarget = js_ast.AssignTargetReplace
+		}
+		p.stmtExprValue = s.Value.Data
+		s.Value, _ = p.visitExprInOut(s.Value, exprIn{assignTarget: assignTarget})
+
+	case *js_ast.SLocal:
+		for i := range s.Decls {
+			d := &s.Decls[i]
+			p.visitBinding(d.Binding, bindingOpts{})
+			if d.ValueOrNil.Data != nil {
+				d.ValueOrNil = p.visitExpr(d.ValueOrNil)
+			}
+		}
+		s.Decls = p.lowerObjectRestInDecls(s.Decls)
+		s.Kind = p.selectLocalKind(s.Kind)
+
+	default:
+		panic("Internal error")
+	}
+
+	return stmt
+}
+
+func (p *parser) recordDeclaredSymbol(ref ast.Ref) {
+	p.declaredSymbols = append(p.declaredSymbols, js_ast.DeclaredSymbol{
+		Ref:        ref,
+		IsTopLevel: p.currentScope == p.moduleScope,
+	})
+}
+
+type bindingOpts struct {
+	duplicateArgCheck map[string]logger.Range
+}
+
+func (p *parser) visitBinding(binding js_ast.Binding, opts bindingOpts) {
+	switch b := binding.Data.(type) {
+	case *js_ast.BMissing:
+
+	case *js_ast.BIdentifier:
+		p.recordDeclaredSymbol(b.Ref)
+		name := p.symbols[b.Ref.InnerIndex].OriginalName
+		p.validateDeclaredSymbolName(binding.Loc, name)
+		if opts.duplicateArgCheck != nil {
+			r := js_lexer.RangeOfIdentifier(p.source, binding.Loc)
+			if firstRange := opts.duplicateArgCheck[name]; firstRange.Len > 0 {
+				p.log.AddErrorWithNotes(&p.tracker, r,
+					fmt.Sprintf("%q cannot be bound multiple times in the same parameter list", name),
+					[]logger.MsgData{p.tracker.MsgData(firstRange, fmt.Sprintf("The name %q was originally bound here:", name))})
+			} else {
+				opts.duplicateArgCheck[name] = r
+			}
+		}
+
+	case *js_ast.BArray:
+		for i := range b.Items {
+			item := &b.Items[i]
+			p.visitBinding(item.Binding, opts)
+			if item.DefaultValueOrNil.Data != nil {
+				// Propagate the name to keep from the binding into the initializer
+				if id, ok := item.Binding.Data.(*js_ast.BIdentifier); ok {
+					p.nameToKeep = p.symbols[id.Ref.InnerIndex].OriginalName
+					p.nameToKeepIsFor = item.DefaultValueOrNil.Data
+				}
+
+				item.DefaultValueOrNil = p.visitExpr(item.DefaultValueOrNil)
+			}
+		}
+
+	case *js_ast.BObject:
+		for i, property := range b.Properties {
+			if !property.IsSpread {
+				property.Key, _ = p.visitExprInOut(property.Key, exprIn{
+					shouldMangleStringsAsProps: true,
+				})
+			}
+			p.visitBinding(property.Value, opts)
+			if property.DefaultValueOrNil.Data != nil {
+				// Propagate the name to keep from the binding into the initializer
+				if id, ok := property.Value.Data.(*js_ast.BIdentifier); ok {
+					p.nameToKeep = p.symbols[id.Ref.InnerIndex].OriginalName
+					p.nameToKeepIsFor = property.DefaultValueOrNil.Data
+				}
+
+				property.DefaultValueOrNil = p.visitExpr(property.DefaultValueOrNil)
+			}
+			b.Properties[i] = property
+		}
+
+	default:
+		panic("Internal error")
+	}
+}
+
+func statementCaresAboutScope(stmt js_ast.Stmt) bool {
+	switch s := stmt.Data.(type) {
+	case *js_ast.SBlock, *js_ast.SEmpty, *js_ast.SDebugger, *js_ast.SExpr, *js_ast.SIf,
+		*js_ast.SFor, *js_ast.SForIn, *js_ast.SForOf, *js_ast.SDoWhile, *js_ast.SWhile,
+		*js_ast.SWith, *js_ast.STry, *js_ast.SSwitch, *js_ast.SReturn, *js_ast.SThrow,
+		*js_ast.SBreak, *js_ast.SContinue, *js_ast.SDirective, *js_ast.SLabel:
+		return false
+
+	case *js_ast.SLocal:
+		return s.Kind != js_ast.LocalVar
+
+	default:
+		return true
+	}
+}
+
+func dropFirstStatement(body js_ast.Stmt, replaceOrNil js_ast.Stmt) js_ast.Stmt {
+	if block, ok := body.Data.(*js_ast.SBlock); ok && len(block.Stmts) > 0 {
+		if replaceOrNil.Data != nil {
+			block.Stmts[0] = replaceOrNil
+		} else if len(block.Stmts) == 2 && !statementCaresAboutScope(block.Stmts[1]) {
+			return block.Stmts[1]
+		} else {
+			block.Stmts = block.Stmts[1:]
+		}
+		return body
+	}
+	if replaceOrNil.Data != nil {
+		return replaceOrNil
+	}
+	return js_ast.Stmt{Loc: body.Loc, Data: js_ast.SEmptyShared}
+}
+
+func mangleFor(s *js_ast.SFor) {
+	// Get the first statement in the loop
+	first := s.Body
+	if block, ok := first.Data.(*js_ast.SBlock); ok && len(block.Stmts) > 0 {
+		first = block.Stmts[0]
+	}
+
+	if ifS, ok := first.Data.(*js_ast.SIf); ok {
+		// "for (;;) if (x) break;" => "for (; !x;) ;"
+		// "for (; a;) if (x) break;" => "for (; a && !x;) ;"
+		// "for (;;) if (x) break; else y();" => "for (; !x;) y();"
+		// "for (; a;) if (x) break; else y();" => "for (; a && !x;) y();"
+		if breakS, ok := ifS.Yes.Data.(*js_ast.SBreak); ok && breakS.Label == nil {
+			var not js_ast.Expr
+			if unary, ok := ifS.Test.Data.(*js_ast.EUnary); ok && unary.Op == js_ast.UnOpNot {
+				not = unary.Value
+			} else {
+				not = js_ast.Not(ifS.Test)
+			}
+			if s.TestOrNil.Data != nil {
+				s.TestOrNil = js_ast.Expr{Loc: s.TestOrNil.Loc, Data: &js_ast.EBinary{
+					Op:    js_ast.BinOpLogicalAnd,
+					Left:  s.TestOrNil,
+					Right: not,
+				}}
+			} else {
+				s.TestOrNil = not
+			}
+			s.Body = dropFirstStatement(s.Body, ifS.NoOrNil)
+			return
+		}
+
+		// "for (;;) if (x) y(); else break;" => "for (; x;) y();"
+		// "for (; a;) if (x) y(); else break;" => "for (; a && x;) y();"
+		if ifS.NoOrNil.Data != nil {
+			if breakS, ok := ifS.NoOrNil.Data.(*js_ast.SBreak); ok && breakS.Label == nil {
+				if s.TestOrNil.Data != nil {
+					s.TestOrNil = js_ast.Expr{Loc: s.TestOrNil.Loc, Data: &js_ast.EBinary{
+						Op:    js_ast.BinOpLogicalAnd,
+						Left:  s.TestOrNil,
+						Right: ifS.Test,
+					}}
+				} else {
+					s.TestOrNil = ifS.Test
+				}
+				s.Body = dropFirstStatement(s.Body, ifS.Yes)
+				return
+			}
+		}
+	}
+}
+
+func appendIfOrLabelBodyPreservingScope(stmts []js_ast.Stmt, body js_ast.Stmt) []js_ast.Stmt {
+	if block, ok := body.Data.(*js_ast.SBlock); ok {
+		keepBlock := false
+		for _, stmt := range block.Stmts {
+			if statementCaresAboutScope(stmt) {
+				keepBlock = true
+				break
+			}
+		}
+		if !keepBlock {
+			return append(stmts, block.Stmts...)
+		}
+	}
+
+	if statementCaresAboutScope(body) {
+		return append(stmts, js_ast.Stmt{Loc: body.Loc, Data: &js_ast.SBlock{Stmts: []js_ast.Stmt{body}}})
+	}
+
+	return append(stmts, body)
+}
+
+func (p *parser) mangleIf(stmts []js_ast.Stmt, loc logger.Loc, s *js_ast.SIf) []js_ast.Stmt {
+	// Constant folding using the test expression
+	if boolean, sideEffects, ok := js_ast.ToBooleanWithSideEffects(s.Test.Data); ok {
+		if boolean {
+			// The test is truthy
+			if s.NoOrNil.Data == nil || !shouldKeepStmtInDeadControlFlow(s.NoOrNil) {
+				// We can drop the "no" branch
+				if sideEffects == js_ast.CouldHaveSideEffects {
+					// Keep the condition if it could have side effects (but is still known to be truthy)
+					if test := p.astHelpers.SimplifyUnusedExpr(s.Test, p.options.unsupportedJSFeatures); test.Data != nil {
+						stmts = append(stmts, js_ast.Stmt{Loc: s.Test.Loc, Data: &js_ast.SExpr{Value: test}})
+					}
+				}
+				return appendIfOrLabelBodyPreservingScope(stmts, s.Yes)
+			} else {
+				// We have to keep the "no" branch
+			}
+		} else {
+			// The test is falsy
+			if !shouldKeepStmtInDeadControlFlow(s.Yes) {
+				// We can drop the "yes" branch
+				if sideEffects == js_ast.CouldHaveSideEffects {
+					// Keep the condition if it could have side effects (but is still known to be falsy)
+					if test := p.astHelpers.SimplifyUnusedExpr(s.Test, p.options.unsupportedJSFeatures); test.Data != nil {
+						stmts = append(stmts, js_ast.Stmt{Loc: s.Test.Loc, Data: &js_ast.SExpr{Value: test}})
+					}
+				}
+				if s.NoOrNil.Data == nil {
+					return stmts
+				}
+				return appendIfOrLabelBodyPreservingScope(stmts, s.NoOrNil)
+			} else {
+				// We have to keep the "yes" branch
+			}
+		}
+
+		// Use "1" and "0" instead of "true" and "false" to be shorter
+		if sideEffects == js_ast.NoSideEffects {
+			if boolean {
+				s.Test.Data = &js_ast.ENumber{Value: 1}
+			} else {
+				s.Test.Data = &js_ast.ENumber{Value: 0}
+			}
+		}
+	}
+
+	var expr js_ast.Expr
+
+	if yes, ok := s.Yes.Data.(*js_ast.SExpr); ok {
+		// "yes" is an expression
+		if s.NoOrNil.Data == nil {
+			if not, ok := s.Test.Data.(*js_ast.EUnary); ok && not.Op == js_ast.UnOpNot {
+				// "if (!a) b();" => "a || b();"
+				expr = js_ast.JoinWithLeftAssociativeOp(js_ast.BinOpLogicalOr, not.Value, yes.Value)
+			} else {
+				// "if (a) b();" => "a && b();"
+				expr = js_ast.JoinWithLeftAssociativeOp(js_ast.BinOpLogicalAnd, s.Test, yes.Value)
+			}
+		} else if no, ok := s.NoOrNil.Data.(*js_ast.SExpr); ok {
+			// "if (a) b(); else c();" => "a ? b() : c();"
+			expr = p.astHelpers.MangleIfExpr(loc, &js_ast.EIf{
+				Test: s.Test,
+				Yes:  yes.Value,
+				No:   no.Value,
+			}, p.options.unsupportedJSFeatures)
+		}
+	} else if _, ok := s.Yes.Data.(*js_ast.SEmpty); ok {
+		// "yes" is missing
+		if s.NoOrNil.Data == nil {
+			// "yes" and "no" are both missing
+			if p.astHelpers.ExprCanBeRemovedIfUnused(s.Test) {
+				// "if (1) {}" => ""
+				return stmts
+			} else {
+				// "if (a) {}" => "a;"
+				expr = s.Test
+			}
+		} else if no, ok := s.NoOrNil.Data.(*js_ast.SExpr); ok {
+			if not, ok := s.Test.Data.(*js_ast.EUnary); ok && not.Op == js_ast.UnOpNot {
+				// "if (!a) {} else b();" => "a && b();"
+				expr = js_ast.JoinWithLeftAssociativeOp(js_ast.BinOpLogicalAnd, not.Value, no.Value)
+			} else {
+				// "if (a) {} else b();" => "a || b();"
+				expr = js_ast.JoinWithLeftAssociativeOp(js_ast.BinOpLogicalOr, s.Test, no.Value)
+			}
+		} else {
+			// "yes" is missing and "no" is not missing (and is not an expression)
+			if not, ok := s.Test.Data.(*js_ast.EUnary); ok && not.Op == js_ast.UnOpNot {
+				// "if (!a) {} else throw b;" => "if (a) throw b;"
+				s.Test = not.Value
+				s.Yes = s.NoOrNil
+				s.NoOrNil = js_ast.Stmt{}
+			} else {
+				// "if (a) {} else throw b;" => "if (!a) throw b;"
+				s.Test = js_ast.Not(s.Test)
+				s.Yes = s.NoOrNil
+				s.NoOrNil = js_ast.Stmt{}
+			}
+		}
+	} else {
+		// "yes" is not missing (and is not an expression)
+		if s.NoOrNil.Data != nil {
+			// "yes" is not missing (and is not an expression) and "no" is not missing
+			if not, ok := s.Test.Data.(*js_ast.EUnary); ok && not.Op == js_ast.UnOpNot {
+				// "if (!a) return b; else return c;" => "if (a) return c; else return b;"
+				s.Test = not.Value
+				s.Yes, s.NoOrNil = s.NoOrNil, s.Yes
+			}
+		} else {
+			// "no" is missing
+			if s2, ok := s.Yes.Data.(*js_ast.SIf); ok && s2.NoOrNil.Data == nil {
+				// "if (a) if (b) return c;" => "if (a && b) return c;"
+				s.Test = js_ast.JoinWithLeftAssociativeOp(js_ast.BinOpLogicalAnd, s.Test, s2.Test)
+				s.Yes = s2.Yes
+			}
+		}
+	}
+
+	// Return an expression if we replaced the if statement with an expression above
+	if expr.Data != nil {
+		expr = p.astHelpers.SimplifyUnusedExpr(expr, p.options.unsupportedJSFeatures)
+		return append(stmts, js_ast.Stmt{Loc: loc, Data: &js_ast.SExpr{Value: expr}})
+	}
+
+	return append(stmts, js_ast.Stmt{Loc: loc, Data: s})
+}
+
+func (p *parser) keepExprSymbolName(value js_ast.Expr, name string) js_ast.Expr {
+	value = p.callRuntime(value.Loc, "__name", []js_ast.Expr{value,
+		{Loc: value.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16(name)}},
+	})
+
+	// Make sure tree shaking removes this if the function is never used
+	value.Data.(*js_ast.ECall).CanBeUnwrappedIfUnused = true
+	return value
+}
+
+func (p *parser) keepClassOrFnSymbolName(loc logger.Loc, expr js_ast.Expr, name string) js_ast.Stmt {
+	return js_ast.Stmt{Loc: loc, Data: &js_ast.SExpr{
+		Value: p.callRuntime(loc, "__name", []js_ast.Expr{
+			expr,
+			{Loc: loc, Data: &js_ast.EString{Value: helpers.StringToUTF16(name)}},
+		}),
+		IsFromClassOrFnThatCanBeRemovedIfUnused: true,
+	}}
+}
+
+func (p *parser) visitAndAppendStmt(stmts []js_ast.Stmt, stmt js_ast.Stmt) []js_ast.Stmt {
+	// By default any statement ends the const local prefix
+	wasAfterAfterConstLocalPrefix := p.currentScope.IsAfterConstLocalPrefix
+	p.currentScope.IsAfterConstLocalPrefix = true
+
+	switch s := stmt.Data.(type) {
+	case *js_ast.SEmpty, *js_ast.SComment:
+		// Comments do not end the const local prefix
+		p.currentScope.IsAfterConstLocalPrefix = wasAfterAfterConstLocalPrefix
+
+	case *js_ast.SDebugger:
+		// Debugger statements do not end the const local prefix
+		p.currentScope.IsAfterConstLocalPrefix = wasAfterAfterConstLocalPrefix
+
+		if p.options.dropDebugger {
+			return stmts
+		}
+
+	case *js_ast.STypeScript:
+		// Type annotations do not end the const local prefix
+		p.currentScope.IsAfterConstLocalPrefix = wasAfterAfterConstLocalPrefix
+
+		// Erase TypeScript constructs from the output completely
+		return stmts
+
+	case *js_ast.SDirective:
+		// Directives do not end the const local prefix
+		p.currentScope.IsAfterConstLocalPrefix = wasAfterAfterConstLocalPrefix
+
+		if p.isStrictMode() && s.LegacyOctalLoc.Start > 0 {
+			p.markStrictModeFeature(legacyOctalEscape, p.source.RangeOfLegacyOctalEscape(s.LegacyOctalLoc), "")
+		}
+
+	case *js_ast.SImport:
+		p.recordDeclaredSymbol(s.NamespaceRef)
+
+		if s.DefaultName != nil {
+			p.recordDeclaredSymbol(s.DefaultName.Ref)
+		}
+
+		if s.Items != nil {
+			for _, item := range *s.Items {
+				p.recordDeclaredSymbol(item.Name.Ref)
+			}
+		}
+
+	case *js_ast.SExportClause:
+		// "export {foo}"
+		end := 0
+		for _, item := range s.Items {
+			name := p.loadNameFromRef(item.Name.Ref)
+			ref := p.findSymbol(item.AliasLoc, name).ref
+
+			if p.symbols[ref.InnerIndex].Kind == ast.SymbolUnbound {
+				// Silently strip exports of non-local symbols in TypeScript, since
+				// those likely correspond to type-only exports. But report exports of
+				// non-local symbols as errors in JavaScript.
+				if !p.options.ts.Parse {
+					r := js_lexer.RangeOfIdentifier(p.source, item.Name.Loc)
+					p.log.AddError(&p.tracker, r, fmt.Sprintf("%q is not declared in this file", name))
+				}
+				continue
+			}
+
+			item.Name.Ref = ref
+			s.Items[end] = item
+			end++
+		}
+
+		// Note: do not remove empty export statements since TypeScript uses them as module markers
+		s.Items = s.Items[:end]
+
+	case *js_ast.SExportFrom:
+		// "export {foo} from 'path'"
+		name := p.loadNameFromRef(s.NamespaceRef)
+		s.NamespaceRef = p.newSymbol(ast.SymbolOther, name)
+		p.currentScope.Generated = append(p.currentScope.Generated, s.NamespaceRef)
+		p.recordDeclaredSymbol(s.NamespaceRef)
+
+		// This is a re-export and the symbols created here are used to reference
+		// names in another file. This means the symbols are really aliases.
+		for i, item := range s.Items {
+			name := p.loadNameFromRef(item.Name.Ref)
+			ref := p.newSymbol(ast.SymbolOther, name)
+			p.currentScope.Generated = append(p.currentScope.Generated, ref)
+			p.recordDeclaredSymbol(ref)
+			s.Items[i].Name.Ref = ref
+		}
+
+	case *js_ast.SExportStar:
+		// "export * from 'path'"
+		// "export * as ns from 'path'"
+		name := p.loadNameFromRef(s.NamespaceRef)
+		s.NamespaceRef = p.newSymbol(ast.SymbolOther, name)
+		p.currentScope.Generated = append(p.currentScope.Generated, s.NamespaceRef)
+		p.recordDeclaredSymbol(s.NamespaceRef)
+
+		// "export * as ns from 'path'"
+		if s.Alias != nil {
+			// "import * as ns from 'path'"
+			// "export {ns}"
+			if p.options.unsupportedJSFeatures.Has(compat.ExportStarAs) {
+				p.recordUsage(s.NamespaceRef)
+				return append(stmts,
+					js_ast.Stmt{Loc: stmt.Loc, Data: &js_ast.SImport{
+						NamespaceRef:      s.NamespaceRef,
+						StarNameLoc:       &s.Alias.Loc,
+						ImportRecordIndex: s.ImportRecordIndex,
+					}},
+					js_ast.Stmt{Loc: stmt.Loc, Data: &js_ast.SExportClause{
+						Items: []js_ast.ClauseItem{{
+							Alias:        s.Alias.OriginalName,
+							OriginalName: s.Alias.OriginalName,
+							AliasLoc:     s.Alias.Loc,
+							Name:         ast.LocRef{Loc: s.Alias.Loc, Ref: s.NamespaceRef},
+						}},
+						IsSingleLine: true,
+					}},
+				)
+			}
+		}
+
+	case *js_ast.SExportDefault:
+		p.recordDeclaredSymbol(s.DefaultName.Ref)
+
+		switch s2 := s.Value.Data.(type) {
+		case *js_ast.SExpr:
+			// Propagate the name to keep from the export into the value
+			p.nameToKeep = "default"
+			p.nameToKeepIsFor = s2.Value.Data
+
+			s2.Value = p.visitExpr(s2.Value)
+
+			// Discard type-only export default statements
+			if p.options.ts.Parse {
+				if id, ok := s2.Value.Data.(*js_ast.EIdentifier); ok {
+					symbol := p.symbols[id.Ref.InnerIndex]
+					if symbol.Kind == ast.SymbolUnbound && p.localTypeNames[symbol.OriginalName] {
+						return stmts
+					}
+				}
+			}
+
+			// If there are lowered "using" declarations, change this into a "var"
+			if p.currentScope.Parent == nil && p.willWrapModuleInTryCatchForUsing {
+				stmts = append(stmts,
+					js_ast.Stmt{Loc: stmt.Loc, Data: &js_ast.SLocal{
+						Decls: []js_ast.Decl{{
+							Binding:    js_ast.Binding{Loc: s.DefaultName.Loc, Data: &js_ast.BIdentifier{Ref: s.DefaultName.Ref}},
+							ValueOrNil: s2.Value,
+						}},
+					}},
+					js_ast.Stmt{Loc: stmt.Loc, Data: &js_ast.SExportClause{Items: []js_ast.ClauseItem{{
+						Alias:    "default",
+						AliasLoc: s.DefaultName.Loc,
+						Name:     s.DefaultName,
+					}}}},
+				)
+				break
+			}
+
+			stmts = append(stmts, stmt)
+
+		case *js_ast.SFunction:
+			// If we need to preserve the name but there is no name, generate a name
+			var name string
+			if p.options.keepNames {
+				if s2.Fn.Name == nil {
+					clone := s.DefaultName
+					s2.Fn.Name = &clone
+					name = "default"
+				} else {
+					name = p.symbols[s2.Fn.Name.Ref.InnerIndex].OriginalName
+				}
+			}
+
+			p.visitFn(&s2.Fn, s2.Fn.OpenParenLoc, visitFnOpts{})
+			stmts = append(stmts, stmt)
+
+			// Optionally preserve the name
+			if p.options.keepNames {
+				p.symbols[s2.Fn.Name.Ref.InnerIndex].Flags |= ast.DidKeepName
+				fn := js_ast.Expr{Loc: s2.Fn.Name.Loc, Data: &js_ast.EIdentifier{Ref: s2.Fn.Name.Ref}}
+				stmts = append(stmts, p.keepClassOrFnSymbolName(s2.Fn.Name.Loc, fn, name))
+			}
+
+		case *js_ast.SClass:
+			result := p.visitClass(s.Value.Loc, &s2.Class, s.DefaultName.Ref, "default")
+
+			// Lower class field syntax for browsers that don't support it
+			classStmts, _ := p.lowerClass(stmt, js_ast.Expr{}, result, "")
+
+			// Remember if the class was side-effect free before lowering
+			if result.canBeRemovedIfUnused {
+				for _, classStmt := range classStmts {
+					if s2, ok := classStmt.Data.(*js_ast.SExpr); ok {
+						s2.IsFromClassOrFnThatCanBeRemovedIfUnused = true
+					}
+				}
+			}
+
+			stmts = append(stmts, classStmts...)
+
+		default:
+			panic("Internal error")
+		}
+
+		// Use a more friendly name than "default" now that "--keep-names" has
+		// been applied and has made sure to enforce the name "default"
+		if p.symbols[s.DefaultName.Ref.InnerIndex].OriginalName == "default" {
+			p.symbols[s.DefaultName.Ref.InnerIndex].OriginalName = p.source.IdentifierName + "_default"
+		}
+
+		return stmts
+
+	case *js_ast.SExportEquals:
+		// "module.exports = value"
+		stmts = append(stmts, js_ast.AssignStmt(
+			js_ast.Expr{Loc: stmt.Loc, Data: &js_ast.EDot{
+				Target:  js_ast.Expr{Loc: stmt.Loc, Data: &js_ast.EIdentifier{Ref: p.moduleRef}},
+				Name:    "exports",
+				NameLoc: stmt.Loc,
+			}},
+			p.visitExpr(s.Value),
+		))
+		p.recordUsage(p.moduleRef)
+		return stmts
+
+	case *js_ast.SBreak:
+		if s.Label != nil {
+			name := p.loadNameFromRef(s.Label.Ref)
+			s.Label.Ref, _, _ = p.findLabelSymbol(s.Label.Loc, name)
+		} else if !p.fnOrArrowDataVisit.isInsideLoop && !p.fnOrArrowDataVisit.isInsideSwitch {
+			r := js_lexer.RangeOfIdentifier(p.source, stmt.Loc)
+			p.log.AddError(&p.tracker, r, "Cannot use \"break\" here:")
+		}
+
+	case *js_ast.SContinue:
+		if s.Label != nil {
+			name := p.loadNameFromRef(s.Label.Ref)
+			var isLoop, ok bool
+			s.Label.Ref, isLoop, ok = p.findLabelSymbol(s.Label.Loc, name)
+			if ok && !isLoop {
+				r := js_lexer.RangeOfIdentifier(p.source, s.Label.Loc)
+				p.log.AddError(&p.tracker, r, fmt.Sprintf("Cannot continue to label \"%s\"", name))
+			}
+		} else if !p.fnOrArrowDataVisit.isInsideLoop {
+			r := js_lexer.RangeOfIdentifier(p.source, stmt.Loc)
+			p.log.AddError(&p.tracker, r, "Cannot use \"continue\" here:")
+		}
+
+	case *js_ast.SLabel:
+		// Forbid functions inside labels in strict mode
+		if p.isStrictMode() {
+			if _, ok := s.Stmt.Data.(*js_ast.SFunction); ok {
+				p.markStrictModeFeature(labelFunctionStmt, js_lexer.RangeOfIdentifier(p.source, s.Stmt.Loc), "")
+			}
+		}
+
+		p.pushScopeForVisitPass(js_ast.ScopeLabel, stmt.Loc)
+		name := p.loadNameFromRef(s.Name.Ref)
+		if js_lexer.StrictModeReservedWords[name] {
+			p.markStrictModeFeature(reservedWord, js_lexer.RangeOfIdentifier(p.source, s.Name.Loc), name)
+		}
+		ref := p.newSymbol(ast.SymbolLabel, name)
+		s.Name.Ref = ref
+
+		// Duplicate labels are an error
+		for scope := p.currentScope.Parent; scope != nil; scope = scope.Parent {
+			if scope.Label.Ref != ast.InvalidRef && name == p.symbols[scope.Label.Ref.InnerIndex].OriginalName {
+				p.log.AddErrorWithNotes(&p.tracker, js_lexer.RangeOfIdentifier(p.source, s.Name.Loc),
+					fmt.Sprintf("Duplicate label %q", name),
+					[]logger.MsgData{p.tracker.MsgData(js_lexer.RangeOfIdentifier(p.source, scope.Label.Loc),
+						fmt.Sprintf("The original label %q is here:", name))})
+				break
+			}
+			if scope.Kind == js_ast.ScopeFunctionBody {
+				// Labels are only visible within the function they are defined in.
+				break
+			}
+		}
+
+		p.currentScope.Label = ast.LocRef{Loc: s.Name.Loc, Ref: ref}
+		switch s.Stmt.Data.(type) {
+		case *js_ast.SFor, *js_ast.SForIn, *js_ast.SForOf, *js_ast.SWhile, *js_ast.SDoWhile:
+			p.currentScope.LabelStmtIsLoop = true
+		}
+
+		// If we're dropping this statement, consider control flow to be dead
+		_, shouldDropLabel := p.dropLabelsMap[name]
+		old := p.isControlFlowDead
+		if shouldDropLabel {
+			p.isControlFlowDead = true
+		}
+
+		s.Stmt = p.visitSingleStmt(s.Stmt, stmtsNormal)
+		p.popScope()
+
+		// Drop this entire statement if requested
+		if shouldDropLabel {
+			p.isControlFlowDead = old
+			return stmts
+		}
+
+		if p.options.minifySyntax {
+			// Optimize "x: break x" which some people apparently write by hand
+			if child, ok := s.Stmt.Data.(*js_ast.SBreak); ok && child.Label != nil && child.Label.Ref == s.Name.Ref {
+				return stmts
+			}
+
+			// Remove the label if it's not necessary
+			if p.symbols[ref.InnerIndex].UseCountEstimate == 0 {
+				return appendIfOrLabelBodyPreservingScope(stmts, s.Stmt)
+			}
+		}
+
+		// Handle "for await" that has been lowered by moving this label inside the "try"
+		if try, ok := s.Stmt.Data.(*js_ast.STry); ok && len(try.Block.Stmts) > 0 {
+			if _, ok := try.Block.Stmts[0].Data.(*js_ast.SFor); ok {
+				try.Block.Stmts[0] = js_ast.Stmt{Loc: stmt.Loc, Data: &js_ast.SLabel{
+					Stmt:             try.Block.Stmts[0],
+					Name:             s.Name,
+					IsSingleLineStmt: s.IsSingleLineStmt,
+				}}
+				return append(stmts, s.Stmt)
+			}
+		}
+
+	case *js_ast.SLocal:
+		// Silently remove unsupported top-level "await" in dead code branches
+		if s.Kind == js_ast.LocalAwaitUsing && p.fnOrArrowDataVisit.isOutsideFnOrArrow {
+			if p.isControlFlowDead && (p.options.unsupportedJSFeatures.Has(compat.TopLevelAwait) || !p.options.outputFormat.KeepESMImportExportSyntax()) {
+				s.Kind = js_ast.LocalUsing
+			} else {
+				p.liveTopLevelAwaitKeyword = logger.Range{Loc: stmt.Loc, Len: 5}
+				p.markSyntaxFeature(compat.TopLevelAwait, logger.Range{Loc: stmt.Loc, Len: 5})
+			}
+		}
+
+		// Local statements do not end the const local prefix
+		p.currentScope.IsAfterConstLocalPrefix = wasAfterAfterConstLocalPrefix
+
+		for i := range s.Decls {
+			d := &s.Decls[i]
+			p.visitBinding(d.Binding, bindingOpts{})
+
+			// Visit the initializer
+			if d.ValueOrNil.Data != nil {
+				// Fold numeric constants in the initializer
+				oldShouldFoldTypeScriptConstantExpressions := p.shouldFoldTypeScriptConstantExpressions
+				p.shouldFoldTypeScriptConstantExpressions = p.options.minifySyntax && !p.currentScope.IsAfterConstLocalPrefix
+
+				// Propagate the name to keep from the binding into the initializer
+				if id, ok := d.Binding.Data.(*js_ast.BIdentifier); ok {
+					p.nameToKeep = p.symbols[id.Ref.InnerIndex].OriginalName
+					p.nameToKeepIsFor = d.ValueOrNil.Data
+				}
+
+				d.ValueOrNil = p.visitExpr(d.ValueOrNil)
+
+				p.shouldFoldTypeScriptConstantExpressions = oldShouldFoldTypeScriptConstantExpressions
+
+				// Initializing to undefined is implicit, but be careful to not
+				// accidentally cause a syntax error or behavior change by removing
+				// the value
+				//
+				// Good:
+				//   "let a = undefined;" => "let a;"
+				//
+				// Bad (a syntax error):
+				//   "let {} = undefined;" => "let {};"
+				//
+				// Bad (a behavior change):
+				//   "a = 123; var a = undefined;" => "a = 123; var a;"
+				//
+				if p.options.minifySyntax && s.Kind == js_ast.LocalLet {
+					if _, ok := d.Binding.Data.(*js_ast.BIdentifier); ok {
+						if _, ok := d.ValueOrNil.Data.(*js_ast.EUndefined); ok {
+							d.ValueOrNil = js_ast.Expr{}
+						}
+					}
+				}
+
+				// Yarn's PnP data may be stored in a variable: https://github.com/yarnpkg/berry/pull/4320
+				if p.options.decodeHydrateRuntimeStateYarnPnP {
+					if str, ok := d.ValueOrNil.Data.(*js_ast.EString); ok {
+						if id, ok := d.Binding.Data.(*js_ast.BIdentifier); ok {
+							if p.stringLocalsForYarnPnP == nil {
+								p.stringLocalsForYarnPnP = make(map[ast.Ref]stringLocalForYarnPnP)
+							}
+							p.stringLocalsForYarnPnP[id.Ref] = stringLocalForYarnPnP{value: str.Value, loc: d.ValueOrNil.Loc}
+						}
+					}
+				}
+			}
+
+			// Attempt to continue the const local prefix
+			if p.options.minifySyntax && !p.currentScope.IsAfterConstLocalPrefix {
+				if id, ok := d.Binding.Data.(*js_ast.BIdentifier); ok {
+					if s.Kind == js_ast.LocalConst && d.ValueOrNil.Data != nil {
+						if value := js_ast.ExprToConstValue(d.ValueOrNil); value.Kind != js_ast.ConstValueNone {
+							if p.constValues == nil {
+								p.constValues = make(map[ast.Ref]js_ast.ConstValue)
+							}
+							p.constValues[id.Ref] = value
+							continue
+						}
+					}
+
+					if d.ValueOrNil.Data != nil && !isSafeForConstLocalPrefix(d.ValueOrNil) {
+						p.currentScope.IsAfterConstLocalPrefix = true
+					}
+				} else {
+					// A non-identifier binding ends the const local prefix
+					p.currentScope.IsAfterConstLocalPrefix = true
+				}
+			}
+		}
+
+		// Handle being exported inside a namespace
+		if s.IsExport && p.enclosingNamespaceArgRef != nil {
+			wrapIdentifier := func(loc logger.Loc, ref ast.Ref) js_ast.Expr {
+				p.recordUsage(*p.enclosingNamespaceArgRef)
+				return js_ast.Expr{Loc: loc, Data: p.dotOrMangledPropVisit(
+					js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: *p.enclosingNamespaceArgRef}},
+					p.symbols[ref.InnerIndex].OriginalName,
+					loc,
+				)}
+			}
+			for _, decl := range s.Decls {
+				if decl.ValueOrNil.Data != nil {
+					target := js_ast.ConvertBindingToExpr(decl.Binding, wrapIdentifier)
+					if result, ok := p.lowerAssign(target, decl.ValueOrNil, objRestReturnValueIsUnused); ok {
+						target = result
+					} else {
+						target = js_ast.Assign(target, decl.ValueOrNil)
+					}
+					stmts = append(stmts, js_ast.Stmt{Loc: stmt.Loc, Data: &js_ast.SExpr{Value: target}})
+				}
+			}
+			return stmts
+		}
+
+		s.Decls = p.lowerObjectRestInDecls(s.Decls)
+
+		// Optimization: Avoid unnecessary "using" machinery by changing ones
+		// initialized to "null" or "undefined" into a normal variable. Note that
+		// "await using" still needs the "await", so we can't do it for those.
+		if p.options.minifySyntax && s.Kind == js_ast.LocalUsing {
+			s.Kind = js_ast.LocalConst
+			for _, decl := range s.Decls {
+				if t := js_ast.KnownPrimitiveType(decl.ValueOrNil.Data); t != js_ast.PrimitiveNull && t != js_ast.PrimitiveUndefined {
+					s.Kind = js_ast.LocalUsing
+					break
+				}
+			}
+		}
+
+		s.Kind = p.selectLocalKind(s.Kind)
+
+		// Potentially relocate "var" declarations to the top level
+		if s.Kind == js_ast.LocalVar {
+			if assign, ok := p.maybeRelocateVarsToTopLevel(s.Decls, relocateVarsNormal); ok {
+				if assign.Data != nil {
+					stmts = append(stmts, assign)
+				}
+				return stmts
+			}
+		}
+
+	case *js_ast.SExpr:
+		shouldTrimUnsightlyPrimitives := !p.options.minifySyntax && !isUnsightlyPrimitive(s.Value.Data)
+		p.stmtExprValue = s.Value.Data
+		s.Value = p.visitExpr(s.Value)
+
+		// Expressions that have been simplified down to a single primitive don't
+		// have any effect, and are automatically removed during minification.
+		// However, some people are really bothered by seeing them. Remove them
+		// so we don't bother these people.
+		if shouldTrimUnsightlyPrimitives && isUnsightlyPrimitive(s.Value.Data) {
+			return stmts
+		}
+
+		// Trim expressions without side effects
+		if p.options.minifySyntax {
+			s.Value = p.astHelpers.SimplifyUnusedExpr(s.Value, p.options.unsupportedJSFeatures)
+			if s.Value.Data == nil {
+				return stmts
+			}
+		}
+
+	case *js_ast.SThrow:
+		s.Value = p.visitExpr(s.Value)
+
+	case *js_ast.SReturn:
+		// Forbid top-level return inside modules with ECMAScript syntax
+		if p.fnOrArrowDataVisit.isOutsideFnOrArrow {
+			if p.isFileConsideredESM {
+				_, notes := p.whyESModule()
+				p.log.AddErrorWithNotes(&p.tracker, js_lexer.RangeOfIdentifier(p.source, stmt.Loc),
+					"Top-level return cannot be used inside an ECMAScript module", notes)
+			} else {
+				p.hasTopLevelReturn = true
+			}
+		}
+
+		if s.ValueOrNil.Data != nil {
+			s.ValueOrNil = p.visitExpr(s.ValueOrNil)
+
+			// Returning undefined is implicit except when inside an async generator
+			// function, where "return undefined" behaves like "return await undefined"
+			// but just "return" has no "await".
+			if p.options.minifySyntax && (!p.fnOrArrowDataVisit.isAsync || !p.fnOrArrowDataVisit.isGenerator) {
+				if _, ok := s.ValueOrNil.Data.(*js_ast.EUndefined); ok {
+					s.ValueOrNil = js_ast.Expr{}
+				}
+			}
+		}
+
+	case *js_ast.SBlock:
+		p.pushScopeForVisitPass(js_ast.ScopeBlock, stmt.Loc)
+
+		// Pass the "is loop body" status on to the direct children of a block used
+		// as a loop body. This is used to enable optimizations specific to the
+		// topmost scope in a loop body block.
+		if p.loopBody == s {
+			s.Stmts = p.visitStmts(s.Stmts, stmtsLoopBody)
+		} else {
+			s.Stmts = p.visitStmts(s.Stmts, stmtsNormal)
+		}
+
+		p.popScope()
+
+		if p.options.minifySyntax {
+			if len(s.Stmts) == 1 && !statementCaresAboutScope(s.Stmts[0]) {
+				// Unwrap blocks containing a single statement
+				stmt = s.Stmts[0]
+			} else if len(s.Stmts) == 0 {
+				// Trim empty blocks
+				stmt = js_ast.Stmt{Loc: stmt.Loc, Data: js_ast.SEmptyShared}
+			}
+		}
+
+	case *js_ast.SWith:
+		p.markStrictModeFeature(withStatement, js_lexer.RangeOfIdentifier(p.source, stmt.Loc), "")
+		s.Value = p.visitExpr(s.Value)
+		p.pushScopeForVisitPass(js_ast.ScopeWith, s.BodyLoc)
+		s.Body = p.visitSingleStmt(s.Body, stmtsNormal)
+		p.popScope()
+
+	case *js_ast.SWhile:
+		s.Test = p.visitExpr(s.Test)
+		s.Body = p.visitLoopBody(s.Body)
+
+		if p.options.minifySyntax {
+			s.Test = p.astHelpers.SimplifyBooleanExpr(s.Test)
+
+			// A true value is implied
+			testOrNil := s.Test
+			if boolean, sideEffects, ok := js_ast.ToBooleanWithSideEffects(s.Test.Data); ok && boolean && sideEffects == js_ast.NoSideEffects {
+				testOrNil = js_ast.Expr{}
+			}
+
+			// "while (a) {}" => "for (;a;) {}"
+			forS := &js_ast.SFor{TestOrNil: testOrNil, Body: s.Body, IsSingleLineBody: s.IsSingleLineBody}
+			mangleFor(forS)
+			stmt = js_ast.Stmt{Loc: stmt.Loc, Data: forS}
+		}
+
+	case *js_ast.SDoWhile:
+		s.Body = p.visitLoopBody(s.Body)
+		s.Test = p.visitExpr(s.Test)
+
+		if p.options.minifySyntax {
+			s.Test = p.astHelpers.SimplifyBooleanExpr(s.Test)
+		}
+
+	case *js_ast.SIf:
+		s.Test = p.visitExpr(s.Test)
+
+		if p.options.minifySyntax {
+			s.Test = p.astHelpers.SimplifyBooleanExpr(s.Test)
+		}
+
+		// Fold constants
+		boolean, _, ok := js_ast.ToBooleanWithSideEffects(s.Test.Data)
+
+		// Mark the control flow as dead if the branch is never taken
+		if ok && !boolean {
+			old := p.isControlFlowDead
+			p.isControlFlowDead = true
+			s.Yes = p.visitSingleStmt(s.Yes, stmtsNormal)
+			p.isControlFlowDead = old
+		} else {
+			s.Yes = p.visitSingleStmt(s.Yes, stmtsNormal)
+		}
+
+		// The "else" clause is optional
+		if s.NoOrNil.Data != nil {
+			// Mark the control flow as dead if the branch is never taken
+			if ok && boolean {
+				old := p.isControlFlowDead
+				p.isControlFlowDead = true
+				s.NoOrNil = p.visitSingleStmt(s.NoOrNil, stmtsNormal)
+				p.isControlFlowDead = old
+			} else {
+				s.NoOrNil = p.visitSingleStmt(s.NoOrNil, stmtsNormal)
+			}
+
+			// Trim unnecessary "else" clauses
+			if p.options.minifySyntax {
+				if _, ok := s.NoOrNil.Data.(*js_ast.SEmpty); ok {
+					s.NoOrNil = js_ast.Stmt{}
+				}
+			}
+		}
+
+		if p.options.minifySyntax {
+			return p.mangleIf(stmts, stmt.Loc, s)
+		}
+
+	case *js_ast.SFor:
+		p.pushScopeForVisitPass(js_ast.ScopeBlock, stmt.Loc)
+		if s.InitOrNil.Data != nil {
+			p.visitForLoopInit(s.InitOrNil, false)
+		}
+
+		if s.TestOrNil.Data != nil {
+			s.TestOrNil = p.visitExpr(s.TestOrNil)
+
+			if p.options.minifySyntax {
+				s.TestOrNil = p.astHelpers.SimplifyBooleanExpr(s.TestOrNil)
+
+				// A true value is implied
+				if boolean, sideEffects, ok := js_ast.ToBooleanWithSideEffects(s.TestOrNil.Data); ok && boolean && sideEffects == js_ast.NoSideEffects {
+					s.TestOrNil = js_ast.Expr{}
+				}
+			}
+		}
+
+		if s.UpdateOrNil.Data != nil {
+			s.UpdateOrNil = p.visitExpr(s.UpdateOrNil)
+		}
+		s.Body = p.visitLoopBody(s.Body)
+
+		// Potentially relocate "var" declarations to the top level. Note that this
+		// must be done inside the scope of the for loop or they won't be relocated.
+		if s.InitOrNil.Data != nil {
+			if init, ok := s.InitOrNil.Data.(*js_ast.SLocal); ok && init.Kind == js_ast.LocalVar {
+				if assign, ok := p.maybeRelocateVarsToTopLevel(init.Decls, relocateVarsNormal); ok {
+					if assign.Data != nil {
+						s.InitOrNil = assign
+					} else {
+						s.InitOrNil = js_ast.Stmt{}
+					}
+				}
+			}
+		}
+
+		p.popScope()
+
+		if p.options.minifySyntax {
+			mangleFor(s)
+		}
+
+	case *js_ast.SForIn:
+		p.pushScopeForVisitPass(js_ast.ScopeBlock, stmt.Loc)
+		p.visitForLoopInit(s.Init, true)
+		s.Value = p.visitExpr(s.Value)
+		s.Body = p.visitLoopBody(s.Body)
+
+		// Check for a variable initializer
+		if local, ok := s.Init.Data.(*js_ast.SLocal); ok && local.Kind == js_ast.LocalVar && len(local.Decls) == 1 {
+			decl := &local.Decls[0]
+			if id, ok := decl.Binding.Data.(*js_ast.BIdentifier); ok && decl.ValueOrNil.Data != nil {
+				p.markStrictModeFeature(forInVarInit, p.source.RangeOfOperatorBefore(decl.ValueOrNil.Loc, "="), "")
+
+				// Lower for-in variable initializers in case the output is used in strict mode
+				stmts = append(stmts, js_ast.Stmt{Loc: stmt.Loc, Data: &js_ast.SExpr{Value: js_ast.Assign(
+					js_ast.Expr{Loc: decl.Binding.Loc, Data: &js_ast.EIdentifier{Ref: id.Ref}},
+					decl.ValueOrNil,
+				)}})
+				decl.ValueOrNil = js_ast.Expr{}
+			}
+		}
+
+		// Potentially relocate "var" declarations to the top level. Note that this
+		// must be done inside the scope of the for loop or they won't be relocated.
+		if init, ok := s.Init.Data.(*js_ast.SLocal); ok && init.Kind == js_ast.LocalVar {
+			if replacement, ok := p.maybeRelocateVarsToTopLevel(init.Decls, relocateVarsForInOrForOf); ok {
+				s.Init = replacement
+			}
+		}
+
+		p.popScope()
+
+		p.lowerObjectRestInForLoopInit(s.Init, &s.Body)
+
+	case *js_ast.SForOf:
+		// Silently remove unsupported top-level "await" in dead code branches
+		if s.Await.Len > 0 && p.fnOrArrowDataVisit.isOutsideFnOrArrow {
+			if p.isControlFlowDead && (p.options.unsupportedJSFeatures.Has(compat.TopLevelAwait) || !p.options.outputFormat.KeepESMImportExportSyntax()) {
+				s.Await = logger.Range{}
+			} else {
+				p.liveTopLevelAwaitKeyword = s.Await
+				p.markSyntaxFeature(compat.TopLevelAwait, s.Await)
+			}
+		}
+
+		p.pushScopeForVisitPass(js_ast.ScopeBlock, stmt.Loc)
+		p.visitForLoopInit(s.Init, true)
+		s.Value = p.visitExpr(s.Value)
+		s.Body = p.visitLoopBody(s.Body)
+
+		// Potentially relocate "var" declarations to the top level. Note that this
+		// must be done inside the scope of the for loop or they won't be relocated.
+		if init, ok := s.Init.Data.(*js_ast.SLocal); ok && init.Kind == js_ast.LocalVar {
+			if replacement, ok := p.maybeRelocateVarsToTopLevel(init.Decls, relocateVarsForInOrForOf); ok {
+				s.Init = replacement
+			}
+		}
+
+		// Handle "for (using x of y)" and "for (await using x of y)"
+		if local, ok := s.Init.Data.(*js_ast.SLocal); ok {
+			if local.Kind == js_ast.LocalUsing && p.options.unsupportedJSFeatures.Has(compat.Using) {
+				p.lowerUsingDeclarationInForOf(s.Init.Loc, local, &s.Body)
+			} else if local.Kind == js_ast.LocalAwaitUsing {
+				if p.fnOrArrowDataVisit.isOutsideFnOrArrow {
+					if p.isControlFlowDead && (p.options.unsupportedJSFeatures.Has(compat.TopLevelAwait) || !p.options.outputFormat.KeepESMImportExportSyntax()) {
+						// Silently remove unsupported top-level "await" in dead code branches
+						local.Kind = js_ast.LocalUsing
+					} else {
+						p.liveTopLevelAwaitKeyword = logger.Range{Loc: s.Init.Loc, Len: 5}
+						p.markSyntaxFeature(compat.TopLevelAwait, p.liveTopLevelAwaitKeyword)
+					}
+					if p.options.unsupportedJSFeatures.Has(compat.Using) {
+						p.lowerUsingDeclarationInForOf(s.Init.Loc, local, &s.Body)
+					}
+				} else if p.options.unsupportedJSFeatures.Has(compat.Using) || p.options.unsupportedJSFeatures.Has(compat.AsyncAwait) ||
+					(p.options.unsupportedJSFeatures.Has(compat.AsyncGenerator) && p.fnOrArrowDataVisit.isGenerator) {
+					p.lowerUsingDeclarationInForOf(s.Init.Loc, local, &s.Body)
+				}
+			}
+		}
+
+		p.popScope()
+
+		p.lowerObjectRestInForLoopInit(s.Init, &s.Body)
+
+		// Lower "for await" if it's unsupported if it's in a lowered async generator
+		if s.Await.Len > 0 && (p.options.unsupportedJSFeatures.Has(compat.ForAwait) ||
+			(p.options.unsupportedJSFeatures.Has(compat.AsyncGenerator) && p.fnOrArrowDataVisit.isGenerator)) {
+			return p.lowerForAwaitLoop(stmt.Loc, s, stmts)
+		}
+
+	case *js_ast.STry:
+		p.pushScopeForVisitPass(js_ast.ScopeBlock, stmt.Loc)
+		if p.fnOrArrowDataVisit.tryBodyCount == 0 {
+			if s.Catch != nil {
+				p.fnOrArrowDataVisit.tryCatchLoc = s.Catch.Loc
+			} else {
+				p.fnOrArrowDataVisit.tryCatchLoc = stmt.Loc
+			}
+		}
+		p.fnOrArrowDataVisit.tryBodyCount++
+		s.Block.Stmts = p.visitStmts(s.Block.Stmts, stmtsNormal)
+		p.fnOrArrowDataVisit.tryBodyCount--
+		p.popScope()
+
+		if s.Catch != nil {
+			p.pushScopeForVisitPass(js_ast.ScopeCatchBinding, s.Catch.Loc)
+			if s.Catch.BindingOrNil.Data != nil {
+				p.visitBinding(s.Catch.BindingOrNil, bindingOpts{})
+			}
+
+			p.pushScopeForVisitPass(js_ast.ScopeBlock, s.Catch.BlockLoc)
+			s.Catch.Block.Stmts = p.visitStmts(s.Catch.Block.Stmts, stmtsNormal)
+			p.popScope()
+
+			p.lowerObjectRestInCatchBinding(s.Catch)
+			p.popScope()
+		}
+
+		if s.Finally != nil {
+			p.pushScopeForVisitPass(js_ast.ScopeBlock, s.Finally.Loc)
+			s.Finally.Block.Stmts = p.visitStmts(s.Finally.Block.Stmts, stmtsNormal)
+			p.popScope()
+		}
+
+	case *js_ast.SSwitch:
+		s.Test = p.visitExpr(s.Test)
+		p.pushScopeForVisitPass(js_ast.ScopeBlock, s.BodyLoc)
+		oldIsInsideSwitch := p.fnOrArrowDataVisit.isInsideSwitch
+		p.fnOrArrowDataVisit.isInsideSwitch = true
+		for i, c := range s.Cases {
+			if c.ValueOrNil.Data != nil {
+				c.ValueOrNil = p.visitExpr(c.ValueOrNil)
+				p.warnAboutEqualityCheck("case", c.ValueOrNil, c.ValueOrNil.Loc)
+				p.warnAboutTypeofAndString(s.Test, c.ValueOrNil, onlyCheckOriginalOrder)
+			}
+			c.Body = p.visitStmts(c.Body, stmtsSwitch)
+
+			// Make sure the assignment to the body above is preserved
+			s.Cases[i] = c
+		}
+		p.fnOrArrowDataVisit.isInsideSwitch = oldIsInsideSwitch
+		p.popScope()
+
+		// Check for duplicate case values
+		p.duplicateCaseChecker.reset()
+		for _, c := range s.Cases {
+			if c.ValueOrNil.Data != nil {
+				p.duplicateCaseChecker.check(p, c.ValueOrNil)
+			}
+		}
+
+		// Unwrap switch statements in dead code
+		if p.options.minifySyntax && p.isControlFlowDead {
+			for _, c := range s.Cases {
+				stmts = append(stmts, c.Body...)
+			}
+			return stmts
+		}
+
+		// "using" declarations inside switch statements must be special-cased
+		if lowered := p.maybeLowerUsingDeclarationsInSwitch(stmt.Loc, s); lowered != nil {
+			return append(stmts, lowered...)
+		}
+
+	case *js_ast.SFunction:
+		p.visitFn(&s.Fn, s.Fn.OpenParenLoc, visitFnOpts{})
+
+		// Strip this function declaration if it was overwritten
+		if p.symbols[s.Fn.Name.Ref.InnerIndex].Flags.Has(ast.RemoveOverwrittenFunctionDeclaration) && !s.IsExport {
+			return stmts
+		}
+
+		if p.options.minifySyntax && !s.Fn.IsGenerator && !s.Fn.IsAsync && !s.Fn.HasRestArg && s.Fn.Name != nil {
+			if len(s.Fn.Body.Block.Stmts) == 0 {
+				// Mark if this function is an empty function
+				hasSideEffectFreeArguments := true
+				for _, arg := range s.Fn.Args {
+					if _, ok := arg.Binding.Data.(*js_ast.BIdentifier); !ok {
+						hasSideEffectFreeArguments = false
+						break
+					}
+				}
+				if hasSideEffectFreeArguments {
+					p.symbols[s.Fn.Name.Ref.InnerIndex].Flags |= ast.IsEmptyFunction
+				}
+			} else if len(s.Fn.Args) == 1 && len(s.Fn.Body.Block.Stmts) == 1 {
+				// Mark if this function is an identity function
+				if arg := s.Fn.Args[0]; arg.DefaultOrNil.Data == nil {
+					if id, ok := arg.Binding.Data.(*js_ast.BIdentifier); ok {
+						if ret, ok := s.Fn.Body.Block.Stmts[0].Data.(*js_ast.SReturn); ok {
+							if retID, ok := ret.ValueOrNil.Data.(*js_ast.EIdentifier); ok && id.Ref == retID.Ref {
+								p.symbols[s.Fn.Name.Ref.InnerIndex].Flags |= ast.IsIdentityFunction
+							}
+						}
+					}
+				}
+			}
+		}
+
+		// Handle exporting this function from a namespace
+		if s.IsExport && p.enclosingNamespaceArgRef != nil {
+			s.IsExport = false
+			stmts = append(stmts, stmt, js_ast.AssignStmt(
+				js_ast.Expr{Loc: stmt.Loc, Data: p.dotOrMangledPropVisit(
+					js_ast.Expr{Loc: stmt.Loc, Data: &js_ast.EIdentifier{Ref: *p.enclosingNamespaceArgRef}},
+					p.symbols[s.Fn.Name.Ref.InnerIndex].OriginalName,
+					s.Fn.Name.Loc,
+				)},
+				js_ast.Expr{Loc: s.Fn.Name.Loc, Data: &js_ast.EIdentifier{Ref: s.Fn.Name.Ref}},
+			))
+		} else {
+			stmts = append(stmts, stmt)
+		}
+
+		// Optionally preserve the name
+		if p.options.keepNames {
+			symbol := &p.symbols[s.Fn.Name.Ref.InnerIndex]
+			symbol.Flags |= ast.DidKeepName
+			fn := js_ast.Expr{Loc: s.Fn.Name.Loc, Data: &js_ast.EIdentifier{Ref: s.Fn.Name.Ref}}
+			stmts = append(stmts, p.keepClassOrFnSymbolName(s.Fn.Name.Loc, fn, symbol.OriginalName))
+		}
+		return stmts
+
+	case *js_ast.SClass:
+		result := p.visitClass(stmt.Loc, &s.Class, ast.InvalidRef, "")
+
+		// Remove the export flag inside a namespace
+		var nameToExport string
+		wasExportInsideNamespace := s.IsExport && p.enclosingNamespaceArgRef != nil
+		if wasExportInsideNamespace {
+			nameToExport = p.symbols[s.Class.Name.Ref.InnerIndex].OriginalName
+			s.IsExport = false
+		}
+
+		// Lower class field syntax for browsers that don't support it
+		classStmts, _ := p.lowerClass(stmt, js_ast.Expr{}, result, "")
+
+		// Remember if the class was side-effect free before lowering
+		if result.canBeRemovedIfUnused {
+			for _, classStmt := range classStmts {
+				if s2, ok := classStmt.Data.(*js_ast.SExpr); ok {
+					s2.IsFromClassOrFnThatCanBeRemovedIfUnused = true
+				}
+			}
+		}
+
+		stmts = append(stmts, classStmts...)
+
+		// Handle exporting this class from a namespace
+		if wasExportInsideNamespace {
+			stmts = append(stmts, js_ast.AssignStmt(
+				js_ast.Expr{Loc: stmt.Loc, Data: p.dotOrMangledPropVisit(
+					js_ast.Expr{Loc: stmt.Loc, Data: &js_ast.EIdentifier{Ref: *p.enclosingNamespaceArgRef}},
+					nameToExport,
+					s.Class.Name.Loc,
+				)},
+				js_ast.Expr{Loc: s.Class.Name.Loc, Data: &js_ast.EIdentifier{Ref: s.Class.Name.Ref}},
+			))
+		}
+
+		return stmts
+
+	case *js_ast.SEnum:
+		// Do not end the const local prefix after TypeScript enums. We process
+		// them first within their scope so that they are inlined into all code in
+		// that scope. We don't want that to cause the const local prefix to end.
+		p.currentScope.IsAfterConstLocalPrefix = wasAfterAfterConstLocalPrefix
+
+		// Track cross-module enum constants during bundling
+		var tsTopLevelEnumValues map[string]js_ast.TSEnumValue
+		if p.currentScope == p.moduleScope && p.options.mode == config.ModeBundle {
+			tsTopLevelEnumValues = make(map[string]js_ast.TSEnumValue)
+		}
+
+		p.recordDeclaredSymbol(s.Name.Ref)
+		p.pushScopeForVisitPass(js_ast.ScopeEntry, stmt.Loc)
+		p.recordDeclaredSymbol(s.Arg)
+
+		// Scan ahead for any variables inside this namespace. This must be done
+		// ahead of time before visiting any statements inside the namespace
+		// because we may end up visiting the uses before the declarations.
+		// We need to convert the uses into property accesses on the namespace.
+		for _, value := range s.Values {
+			if value.Ref != ast.InvalidRef {
+				p.isExportedInsideNamespace[value.Ref] = s.Arg
+			}
+		}
+
+		// Values without initializers are initialized to one more than the
+		// previous value if the previous value is numeric. Otherwise values
+		// without initializers are initialized to undefined.
+		nextNumericValue := float64(0)
+		hasNumericValue := true
+		valueExprs := []js_ast.Expr{}
+		allValuesArePure := true
+
+		// Update the exported members of this enum as we constant fold each one
+		exportedMembers := p.currentScope.TSNamespace.ExportedMembers
+
+		// We normally don't fold numeric constants because they might increase code
+		// size, but it's important to fold numeric constants inside enums since
+		// that's what the TypeScript compiler does.
+		oldShouldFoldTypeScriptConstantExpressions := p.shouldFoldTypeScriptConstantExpressions
+		p.shouldFoldTypeScriptConstantExpressions = true
+
+		// Create an assignment for each enum value
+		for _, value := range s.Values {
+			name := helpers.UTF16ToString(value.Name)
+			var assignTarget js_ast.Expr
+			hasStringValue := false
+
+			if value.ValueOrNil.Data != nil {
+				value.ValueOrNil = p.visitExpr(value.ValueOrNil)
+				hasNumericValue = false
+
+				// "See through" any wrapped comments
+				underlyingValue := value.ValueOrNil
+				if inlined, ok := value.ValueOrNil.Data.(*js_ast.EInlinedEnum); ok {
+					underlyingValue = inlined.Value
+				}
+
+				switch e := underlyingValue.Data.(type) {
+				case *js_ast.ENumber:
+					if tsTopLevelEnumValues != nil {
+						tsTopLevelEnumValues[name] = js_ast.TSEnumValue{Number: e.Value}
+					}
+					member := exportedMembers[name]
+					member.Data = &js_ast.TSNamespaceMemberEnumNumber{Value: e.Value}
+					exportedMembers[name] = member
+					p.refToTSNamespaceMemberData[value.Ref] = member.Data
+					hasNumericValue = true
+					nextNumericValue = e.Value + 1
+
+				case *js_ast.EString:
+					if tsTopLevelEnumValues != nil {
+						tsTopLevelEnumValues[name] = js_ast.TSEnumValue{String: e.Value}
+					}
+					member := exportedMembers[name]
+					member.Data = &js_ast.TSNamespaceMemberEnumString{Value: e.Value}
+					exportedMembers[name] = member
+					p.refToTSNamespaceMemberData[value.Ref] = member.Data
+					hasStringValue = true
+
+				default:
+					if js_ast.KnownPrimitiveType(underlyingValue.Data) == js_ast.PrimitiveString {
+						hasStringValue = true
+					}
+					if !p.astHelpers.ExprCanBeRemovedIfUnused(underlyingValue) {
+						allValuesArePure = false
+					}
+				}
+			} else if hasNumericValue {
+				if tsTopLevelEnumValues != nil {
+					tsTopLevelEnumValues[name] = js_ast.TSEnumValue{Number: nextNumericValue}
+				}
+				member := exportedMembers[name]
+				member.Data = &js_ast.TSNamespaceMemberEnumNumber{Value: nextNumericValue}
+				exportedMembers[name] = member
+				p.refToTSNamespaceMemberData[value.Ref] = member.Data
+				value.ValueOrNil = js_ast.Expr{Loc: value.Loc, Data: &js_ast.ENumber{Value: nextNumericValue}}
+				nextNumericValue++
+			} else {
+				value.ValueOrNil = js_ast.Expr{Loc: value.Loc, Data: js_ast.EUndefinedShared}
+			}
+
+			if p.options.minifySyntax && js_ast.IsIdentifier(name) {
+				// "Enum.Name = value"
+				assignTarget = js_ast.Assign(
+					js_ast.Expr{Loc: value.Loc, Data: &js_ast.EDot{
+						Target:  js_ast.Expr{Loc: value.Loc, Data: &js_ast.EIdentifier{Ref: s.Arg}},
+						Name:    name,
+						NameLoc: value.Loc,
+					}},
+					value.ValueOrNil,
+				)
+			} else {
+				// "Enum['Name'] = value"
+				assignTarget = js_ast.Assign(
+					js_ast.Expr{Loc: value.Loc, Data: &js_ast.EIndex{
+						Target: js_ast.Expr{Loc: value.Loc, Data: &js_ast.EIdentifier{Ref: s.Arg}},
+						Index:  js_ast.Expr{Loc: value.Loc, Data: &js_ast.EString{Value: value.Name}},
+					}},
+					value.ValueOrNil,
+				)
+			}
+			p.recordUsage(s.Arg)
+
+			// String-valued enums do not form a two-way map
+			if hasStringValue {
+				valueExprs = append(valueExprs, assignTarget)
+			} else {
+				// "Enum[assignTarget] = 'Name'"
+				valueExprs = append(valueExprs, js_ast.Assign(
+					js_ast.Expr{Loc: value.Loc, Data: &js_ast.EIndex{
+						Target: js_ast.Expr{Loc: value.Loc, Data: &js_ast.EIdentifier{Ref: s.Arg}},
+						Index:  assignTarget,
+					}},
+					js_ast.Expr{Loc: value.Loc, Data: &js_ast.EString{Value: value.Name}},
+				))
+				p.recordUsage(s.Arg)
+			}
+		}
+
+		p.popScope()
+		p.shouldFoldTypeScriptConstantExpressions = oldShouldFoldTypeScriptConstantExpressions
+
+		// Track all exported top-level enums for cross-module inlining
+		if tsTopLevelEnumValues != nil {
+			if p.tsEnums == nil {
+				p.tsEnums = make(map[ast.Ref]map[string]js_ast.TSEnumValue)
+			}
+			p.tsEnums[s.Name.Ref] = tsTopLevelEnumValues
+		}
+
+		// Wrap this enum definition in a closure
+		stmts = p.generateClosureForTypeScriptEnum(
+			stmts, stmt.Loc, s.IsExport, s.Name.Loc, s.Name.Ref, s.Arg, valueExprs, allValuesArePure)
+		return stmts
+
+	case *js_ast.SNamespace:
+		p.recordDeclaredSymbol(s.Name.Ref)
+
+		// Scan ahead for any variables inside this namespace. This must be done
+		// ahead of time before visiting any statements inside the namespace
+		// because we may end up visiting the uses before the declarations.
+		// We need to convert the uses into property accesses on the namespace.
+		for _, childStmt := range s.Stmts {
+			if local, ok := childStmt.Data.(*js_ast.SLocal); ok {
+				if local.IsExport {
+					js_ast.ForEachIdentifierBindingInDecls(local.Decls, func(loc logger.Loc, b *js_ast.BIdentifier) {
+						p.isExportedInsideNamespace[b.Ref] = s.Arg
+					})
+				}
+			}
+		}
+
+		oldEnclosingNamespaceArgRef := p.enclosingNamespaceArgRef
+		p.enclosingNamespaceArgRef = &s.Arg
+		p.pushScopeForVisitPass(js_ast.ScopeEntry, stmt.Loc)
+		p.recordDeclaredSymbol(s.Arg)
+		stmtsInsideNamespace := p.visitStmtsAndPrependTempRefs(s.Stmts, prependTempRefsOpts{kind: stmtsFnBody})
+		p.popScope()
+		p.enclosingNamespaceArgRef = oldEnclosingNamespaceArgRef
+
+		// Generate a closure for this namespace
+		stmts = p.generateClosureForTypeScriptNamespaceOrEnum(
+			stmts, stmt.Loc, s.IsExport, s.Name.Loc, s.Name.Ref, s.Arg, stmtsInsideNamespace)
+		return stmts
+
+	default:
+		panic("Internal error")
+	}
+
+	stmts = append(stmts, stmt)
+	return stmts
+}
+
+func isUnsightlyPrimitive(data js_ast.E) bool {
+	switch data.(type) {
+	case *js_ast.EBoolean, *js_ast.ENull, *js_ast.EUndefined, *js_ast.ENumber, *js_ast.EBigInt, *js_ast.EString:
+		return true
+	}
+	return false
+}
+
+// If we encounter a variable initializer that could possibly trigger access to
+// a constant declared later on, then we need to end the const local prefix.
+// We want to avoid situations like this:
+//
+//	const x = y; // This is supposed to throw due to TDZ
+//	const y = 1;
+//
+// or this:
+//
+//	const x = 1;
+//	const y = foo(); // This is supposed to throw due to TDZ
+//	const z = 2;
+//	const foo = () => z;
+//
+// But a situation like this is ok:
+//
+//	const x = 1;
+//	const y = [() => x + z];
+//	const z = 2;
+func isSafeForConstLocalPrefix(expr js_ast.Expr) bool {
+	switch e := expr.Data.(type) {
+	case *js_ast.EMissing, *js_ast.EString, *js_ast.ERegExp, *js_ast.EBigInt, *js_ast.EFunction, *js_ast.EArrow:
+		return true
+
+	case *js_ast.EArray:
+		for _, item := range e.Items {
+			if !isSafeForConstLocalPrefix(item) {
+				return false
+			}
+		}
+		return true
+
+	case *js_ast.EObject:
+		// For now just allow "{}" and forbid everything else
+		return len(e.Properties) == 0
+	}
+
+	return false
+}
+
+type relocateVarsMode uint8
+
+const (
+	relocateVarsNormal relocateVarsMode = iota
+	relocateVarsForInOrForOf
+)
+
+// If we are currently in a hoisted child of the module scope, relocate these
+// declarations to the top level and return an equivalent assignment statement.
+// Make sure to check that the declaration kind is "var" before calling this.
+// And make sure to check that the returned statement is not the zero value.
+//
+// This is done to make it easier to traverse top-level declarations in the linker
+// during bundling. Now it is sufficient to just scan the top-level statements
+// instead of having to traverse recursively into the statement tree.
+func (p *parser) maybeRelocateVarsToTopLevel(decls []js_ast.Decl, mode relocateVarsMode) (js_ast.Stmt, bool) {
+	// Only do this when bundling, and not when the scope is already top-level
+	if p.options.mode != config.ModeBundle || p.currentScope == p.moduleScope {
+		return js_ast.Stmt{}, false
+	}
+
+	// Only do this if we're not inside a function
+	scope := p.currentScope
+	for !scope.Kind.StopsHoisting() {
+		scope = scope.Parent
+	}
+	if scope != p.moduleScope {
+		return js_ast.Stmt{}, false
+	}
+
+	// Convert the declarations to assignments
+	wrapIdentifier := func(loc logger.Loc, ref ast.Ref) js_ast.Expr {
+		p.relocatedTopLevelVars = append(p.relocatedTopLevelVars, ast.LocRef{Loc: loc, Ref: ref})
+		p.recordUsage(ref)
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: ref}}
+	}
+	var value js_ast.Expr
+	for _, decl := range decls {
+		binding := js_ast.ConvertBindingToExpr(decl.Binding, wrapIdentifier)
+		if decl.ValueOrNil.Data != nil {
+			value = js_ast.JoinWithComma(value, js_ast.Assign(binding, decl.ValueOrNil))
+		} else if mode == relocateVarsForInOrForOf {
+			value = js_ast.JoinWithComma(value, binding)
+		}
+	}
+	if value.Data == nil {
+		// If none of the variables had any initializers, just remove the declarations
+		return js_ast.Stmt{}, true
+	}
+	return js_ast.Stmt{Loc: value.Loc, Data: &js_ast.SExpr{Value: value}}, true
+}
+
+func (p *parser) markExprAsParenthesized(value js_ast.Expr, openParenLoc logger.Loc, isAsync bool) {
+	// Don't lose comments due to parentheses. For example, we don't want to lose
+	// the comment here:
+	//
+	//   ( /* comment */ (foo) );
+	//
+	if !isAsync {
+		if comments, ok := p.exprComments[openParenLoc]; ok {
+			delete(p.exprComments, openParenLoc)
+			p.exprComments[value.Loc] = append(comments, p.exprComments[value.Loc]...)
+		}
+	}
+
+	switch e := value.Data.(type) {
+	case *js_ast.EArray:
+		e.IsParenthesized = true
+	case *js_ast.EObject:
+		e.IsParenthesized = true
+	}
+}
+
+func (p *parser) maybeTransposeIfExprChain(expr js_ast.Expr, visit func(js_ast.Expr) js_ast.Expr) js_ast.Expr {
+	if e, ok := expr.Data.(*js_ast.EIf); ok {
+		e.Yes = p.maybeTransposeIfExprChain(e.Yes, visit)
+		e.No = p.maybeTransposeIfExprChain(e.No, visit)
+		return expr
+	}
+	return visit(expr)
+}
+
+func (p *parser) iifeCanBeRemovedIfUnused(args []js_ast.Arg, body js_ast.FnBody) bool {
+	for _, arg := range args {
+		if arg.DefaultOrNil.Data != nil && !p.astHelpers.ExprCanBeRemovedIfUnused(arg.DefaultOrNil) {
+			// The default value has a side effect
+			return false
+		}
+
+		if _, ok := arg.Binding.Data.(*js_ast.BIdentifier); !ok {
+			// Destructuring is a side effect (due to property access)
+			return false
+		}
+	}
+
+	// Check whether any statements have side effects or not. Consider return
+	// statements as not having side effects because if the IIFE can be removed
+	// then we know the return value is unused, so we know that returning the
+	// value has no side effects.
+	return p.astHelpers.StmtsCanBeRemovedIfUnused(body.Block.Stmts, js_ast.ReturnCanBeRemovedIfUnused)
+}
+
+type captureValueMode uint8
+
+const (
+	valueDefinitelyNotMutated captureValueMode = iota
+	valueCouldBeMutated
+)
+
+// This is a helper function to use when you need to capture a value that may
+// have side effects so you can use it multiple times. It guarantees that the
+// side effects take place exactly once.
+//
+// Example usage:
+//
+//	// "value" => "value + value"
+//	// "value()" => "(_a = value(), _a + _a)"
+//	valueFunc, wrapFunc := p.captureValueWithPossibleSideEffects(loc, 2, value)
+//	return wrapFunc(js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+//	  Op: js_ast.BinOpAdd,
+//	  Left: valueFunc(),
+//	  Right: valueFunc(),
+//	}})
+//
+// This returns a function for generating references instead of a raw reference
+// because AST nodes are supposed to be unique in memory, not aliases of other
+// AST nodes. That way you can mutate one during lowering without having to
+// worry about messing up other nodes.
+func (p *parser) captureValueWithPossibleSideEffects(
+	loc logger.Loc, // The location to use for the generated references
+	count int, // The expected number of references to generate
+	value js_ast.Expr, // The value that might have side effects
+	mode captureValueMode, // Say if "value" might be mutated and must be captured
+) (
+	func() js_ast.Expr, // Generates reference expressions "_a"
+	func(js_ast.Expr) js_ast.Expr, // Call this on the final expression
+) {
+	wrapFunc := func(expr js_ast.Expr) js_ast.Expr {
+		// Make sure side effects still happen if no expression was generated
+		if expr.Data == nil {
+			return value
+		}
+		return expr
+	}
+
+	// Referencing certain expressions more than once has no side effects, so we
+	// can just create them inline without capturing them in a temporary variable
+	var valueFunc func() js_ast.Expr
+	switch e := value.Data.(type) {
+	case *js_ast.ENull:
+		valueFunc = func() js_ast.Expr { return js_ast.Expr{Loc: loc, Data: js_ast.ENullShared} }
+	case *js_ast.EUndefined:
+		valueFunc = func() js_ast.Expr { return js_ast.Expr{Loc: loc, Data: js_ast.EUndefinedShared} }
+	case *js_ast.EThis:
+		valueFunc = func() js_ast.Expr { return js_ast.Expr{Loc: loc, Data: js_ast.EThisShared} }
+	case *js_ast.EBoolean:
+		valueFunc = func() js_ast.Expr { return js_ast.Expr{Loc: loc, Data: &js_ast.EBoolean{Value: e.Value}} }
+	case *js_ast.ENumber:
+		valueFunc = func() js_ast.Expr { return js_ast.Expr{Loc: loc, Data: &js_ast.ENumber{Value: e.Value}} }
+	case *js_ast.EBigInt:
+		valueFunc = func() js_ast.Expr { return js_ast.Expr{Loc: loc, Data: &js_ast.EBigInt{Value: e.Value}} }
+	case *js_ast.EString:
+		valueFunc = func() js_ast.Expr { return js_ast.Expr{Loc: loc, Data: &js_ast.EString{Value: e.Value}} }
+	case *js_ast.EPrivateIdentifier:
+		valueFunc = func() js_ast.Expr { return js_ast.Expr{Loc: loc, Data: &js_ast.EPrivateIdentifier{Ref: e.Ref}} }
+	case *js_ast.EIdentifier:
+		if mode == valueDefinitelyNotMutated {
+			valueFunc = func() js_ast.Expr {
+				// Make sure we record this usage in the usage count so that duplicating
+				// a single-use reference means it's no longer considered a single-use
+				// reference. Otherwise the single-use reference inlining code may
+				// incorrectly inline the initializer into the first reference, leaving
+				// the second reference without a definition.
+				p.recordUsage(e.Ref)
+				return js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: e.Ref}}
+			}
+		}
+	}
+	if valueFunc != nil {
+		return valueFunc, wrapFunc
+	}
+
+	// We don't need to worry about side effects if the value won't be used
+	// multiple times. This special case lets us avoid generating a temporary
+	// reference.
+	if count < 2 {
+		return func() js_ast.Expr {
+			return value
+		}, wrapFunc
+	}
+
+	// Otherwise, fall back to generating a temporary reference
+	tempRef := ast.InvalidRef
+
+	// If we're in a function argument scope, then we won't be able to generate
+	// symbols in this scope to store stuff, since there's nowhere to put the
+	// variable declaration. We don't want to put the variable declaration
+	// outside the function since some code in the argument list may cause the
+	// function to be reentrant, and we can't put the variable declaration in
+	// the function body since that's not accessible by the argument list.
+	//
+	// Instead, we use an immediately-invoked arrow function to create a new
+	// symbol inline by introducing a new scope. Make sure to only use it for
+	// symbol declaration and still initialize the variable inline to preserve
+	// side effect order.
+	if p.currentScope.Kind == js_ast.ScopeFunctionArgs {
+		return func() js_ast.Expr {
+				if tempRef == ast.InvalidRef {
+					tempRef = p.generateTempRef(tempRefNoDeclare, "")
+
+					// Assign inline so the order of side effects remains the same
+					p.recordUsage(tempRef)
+					return js_ast.Assign(js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: tempRef}}, value)
+				}
+				p.recordUsage(tempRef)
+				return js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: tempRef}}
+			}, func(expr js_ast.Expr) js_ast.Expr {
+				// Make sure side effects still happen if no expression was generated
+				if expr.Data == nil {
+					return value
+				}
+
+				// Generate a new variable using an arrow function to avoid messing with "this"
+				return js_ast.Expr{Loc: loc, Data: &js_ast.ECall{
+					Target: js_ast.Expr{Loc: loc, Data: &js_ast.EArrow{
+						Args:       []js_ast.Arg{{Binding: js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: tempRef}}}},
+						PreferExpr: true,
+						Body:       js_ast.FnBody{Loc: loc, Block: js_ast.SBlock{Stmts: []js_ast.Stmt{{Loc: loc, Data: &js_ast.SReturn{ValueOrNil: expr}}}}},
+					}},
+				}}
+			}
+	}
+
+	return func() js_ast.Expr {
+		if tempRef == ast.InvalidRef {
+			tempRef = p.generateTempRef(tempRefNeedsDeclare, "")
+			p.recordUsage(tempRef)
+			return js_ast.Assign(js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: tempRef}}, value)
+		}
+		p.recordUsage(tempRef)
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: tempRef}}
+	}, wrapFunc
+}
+
+func (p *parser) visitDecorators(decorators []js_ast.Decorator, decoratorScope *js_ast.Scope) []js_ast.Decorator {
+	if decorators != nil {
+		// Decorators cause us to temporarily revert to the scope that encloses the
+		// class declaration, since that's where the generated code for decorators
+		// will be inserted. I believe this currently only matters for parameter
+		// decorators, where the scope should not be within the argument list.
+		oldScope := p.currentScope
+		p.currentScope = decoratorScope
+
+		for i, decorator := range decorators {
+			decorators[i].Value = p.visitExpr(decorator.Value)
+		}
+
+		// Avoid "popScope" because this decorator scope is not hierarchical
+		p.currentScope = oldScope
+	}
+
+	return decorators
+}
+
+type visitClassResult struct {
+	bodyScope         *js_ast.Scope
+	innerClassNameRef ast.Ref
+	superCtorRef      ast.Ref
+
+	// If true, the class was determined to be safe to remove if the class is
+	// never used (i.e. the class definition is side-effect free). This is
+	// determined after visiting but before lowering since lowering may generate
+	// class mutations that cannot be automatically analyzed as side-effect free.
+	canBeRemovedIfUnused bool
+}
+
+func (p *parser) visitClass(nameScopeLoc logger.Loc, class *js_ast.Class, defaultNameRef ast.Ref, nameToKeep string) (result visitClassResult) {
+	class.Decorators = p.visitDecorators(class.Decorators, p.currentScope)
+
+	if class.Name != nil {
+		p.recordDeclaredSymbol(class.Name.Ref)
+		if p.options.keepNames {
+			nameToKeep = p.symbols[class.Name.Ref.InnerIndex].OriginalName
+		}
+	}
+
+	// Replace "this" with a reference to the class inside static field
+	// initializers if static fields are being lowered, since that relocates the
+	// field initializers outside of the class body and "this" will no longer
+	// reference the same thing.
+	classLoweringInfo := p.computeClassLoweringInfo(class)
+	recomputeClassLoweringInfo := false
+
+	// Sometimes we need to lower private members even though they are supported.
+	// This flags them for lowering so that we lower references to them as we
+	// traverse the class body.
+	//
+	// We don't need to worry about possible references to the class shadowing
+	// symbol inside the class body changing our decision to lower private members
+	// later on because that shouldn't be possible.
+	if classLoweringInfo.lowerAllStaticFields {
+		for _, prop := range class.Properties {
+			// We need to lower all private members if fields of that type are lowered,
+			// not just private fields (methods and accessors too):
+			//
+			//   class Foo {
+			//     get #foo() {}
+			//     static bar = new Foo().#foo
+			//   }
+			//
+			// We can't transform that to this:
+			//
+			//   class Foo {
+			//     get #foo() {}
+			//   }
+			//   Foo.bar = new Foo().#foo;
+			//
+			// The private getter must be lowered too.
+			if private, ok := prop.Key.Data.(*js_ast.EPrivateIdentifier); ok {
+				p.symbols[private.Ref.InnerIndex].Flags |= ast.PrivateSymbolMustBeLowered
+				recomputeClassLoweringInfo = true
+			}
+		}
+	}
+
+	// Conservatively lower all private names that have been used in a private
+	// brand check anywhere in the file. See the comment on this map for details.
+	if p.lowerAllOfThesePrivateNames != nil {
+		for _, prop := range class.Properties {
+			if private, ok := prop.Key.Data.(*js_ast.EPrivateIdentifier); ok {
+				if symbol := &p.symbols[private.Ref.InnerIndex]; p.lowerAllOfThesePrivateNames[symbol.OriginalName] {
+					symbol.Flags |= ast.PrivateSymbolMustBeLowered
+					recomputeClassLoweringInfo = true
+				}
+			}
+		}
+	}
+
+	// If we changed private symbol lowering decisions, then recompute class
+	// lowering info because that may have changed other decisions too
+	if recomputeClassLoweringInfo {
+		classLoweringInfo = p.computeClassLoweringInfo(class)
+	}
+
+	p.pushScopeForVisitPass(js_ast.ScopeClassName, nameScopeLoc)
+	oldEnclosingClassKeyword := p.enclosingClassKeyword
+	p.enclosingClassKeyword = class.ClassKeyword
+	p.currentScope.RecursiveSetStrictMode(js_ast.ImplicitStrictModeClass)
+	if class.Name != nil {
+		p.validateDeclaredSymbolName(class.Name.Loc, p.symbols[class.Name.Ref.InnerIndex].OriginalName)
+	}
+
+	// Create the "__super" symbol if necessary. This will cause us to replace
+	// all "super()" call expressions with a call to this symbol, which will
+	// then be inserted into the "constructor" method.
+	result.superCtorRef = ast.InvalidRef
+	if classLoweringInfo.shimSuperCtorCalls {
+		result.superCtorRef = p.newSymbol(ast.SymbolOther, "__super")
+		p.currentScope.Generated = append(p.currentScope.Generated, result.superCtorRef)
+		p.recordDeclaredSymbol(result.superCtorRef)
+	}
+	oldSuperCtorRef := p.superCtorRef
+	p.superCtorRef = result.superCtorRef
+
+	// Insert an immutable inner name that spans the whole class to match
+	// JavaScript's semantics specifically the "CreateImmutableBinding" here:
+	// https://262.ecma-international.org/6.0/#sec-runtime-semantics-classdefinitionevaluation
+	// The class body (and extends clause) "captures" the original value of the
+	// class name. This matters for class statements because the symbol can be
+	// re-assigned to something else later. The captured values must be the
+	// original value of the name, not the re-assigned value. Use "const" for
+	// this symbol to match JavaScript run-time semantics. You are not allowed
+	// to assign to this symbol (it throws a TypeError).
+	if class.Name != nil {
+		name := p.symbols[class.Name.Ref.InnerIndex].OriginalName
+		result.innerClassNameRef = p.newSymbol(ast.SymbolConst, "_"+name)
+		p.currentScope.Members[name] = js_ast.ScopeMember{Loc: class.Name.Loc, Ref: result.innerClassNameRef}
+	} else {
+		name := "_this"
+		if defaultNameRef != ast.InvalidRef {
+			name = "_" + p.source.IdentifierName + "_default"
+		}
+		result.innerClassNameRef = p.newSymbol(ast.SymbolConst, name)
+	}
+	p.recordDeclaredSymbol(result.innerClassNameRef)
+
+	if class.ExtendsOrNil.Data != nil {
+		class.ExtendsOrNil = p.visitExpr(class.ExtendsOrNil)
+	}
+
+	// A scope is needed for private identifiers
+	p.pushScopeForVisitPass(js_ast.ScopeClassBody, class.BodyLoc)
+	result.bodyScope = p.currentScope
+
+	for i := range class.Properties {
+		property := &class.Properties[i]
+
+		if property.Kind == js_ast.PropertyClassStaticBlock {
+			oldFnOrArrowData := p.fnOrArrowDataVisit
+			oldFnOnlyDataVisit := p.fnOnlyDataVisit
+
+			p.fnOrArrowDataVisit = fnOrArrowDataVisit{}
+			p.fnOnlyDataVisit = fnOnlyDataVisit{
+				isThisNested:           true,
+				isNewTargetAllowed:     true,
+				isInStaticClassContext: true,
+				innerClassNameRef:      &result.innerClassNameRef,
+			}
+
+			if classLoweringInfo.lowerAllStaticFields {
+				// Need to lower "this" and "super" since they won't be valid outside the class body
+				p.fnOnlyDataVisit.shouldReplaceThisWithInnerClassNameRef = true
+				p.fnOrArrowDataVisit.shouldLowerSuperPropertyAccess = true
+			}
+
+			p.pushScopeForVisitPass(js_ast.ScopeClassStaticInit, property.ClassStaticBlock.Loc)
+
+			// Make it an error to use "arguments" in a static class block
+			p.currentScope.ForbidArguments = true
+
+			property.ClassStaticBlock.Block.Stmts = p.visitStmts(property.ClassStaticBlock.Block.Stmts, stmtsFnBody)
+			p.popScope()
+
+			p.fnOrArrowDataVisit = oldFnOrArrowData
+			p.fnOnlyDataVisit = oldFnOnlyDataVisit
+			continue
+		}
+
+		property.Decorators = p.visitDecorators(property.Decorators, result.bodyScope)
+
+		// Visit the property key
+		if private, ok := property.Key.Data.(*js_ast.EPrivateIdentifier); ok {
+			// Special-case private identifiers here
+			p.recordDeclaredSymbol(private.Ref)
+		} else {
+			// It's forbidden to reference the class name in a computed key
+			if property.Flags.Has(js_ast.PropertyIsComputed) && class.Name != nil {
+				p.symbols[result.innerClassNameRef.InnerIndex].Kind = ast.SymbolClassInComputedPropertyKey
+			}
+
+			key, _ := p.visitExprInOut(property.Key, exprIn{
+				shouldMangleStringsAsProps: true,
+			})
+			property.Key = key
+
+			// Re-allow using the class name after visiting a computed key
+			if property.Flags.Has(js_ast.PropertyIsComputed) && class.Name != nil {
+				p.symbols[result.innerClassNameRef.InnerIndex].Kind = ast.SymbolConst
+			}
+
+			if p.options.minifySyntax {
+				if inlined, ok := key.Data.(*js_ast.EInlinedEnum); ok {
+					switch inlined.Value.Data.(type) {
+					case *js_ast.EString, *js_ast.ENumber:
+						key.Data = inlined.Value.Data
+						property.Key.Data = key.Data
+					}
+				}
+				switch k := key.Data.(type) {
+				case *js_ast.ENumber, *js_ast.ENameOfSymbol:
+					// "class { [123] }" => "class { 123 }"
+					property.Flags &= ^js_ast.PropertyIsComputed
+				case *js_ast.EString:
+					if numberValue, ok := js_ast.StringToEquivalentNumberValue(k.Value); ok && numberValue >= 0 {
+						// "class { '123' }" => "class { 123 }"
+						property.Key.Data = &js_ast.ENumber{Value: numberValue}
+						property.Flags &= ^js_ast.PropertyIsComputed
+					} else if property.Flags.Has(js_ast.PropertyIsComputed) {
+						// "class {['x'] = y}" => "class {'x' = y}"
+						isInvalidConstructor := false
+						if helpers.UTF16EqualsString(k.Value, "constructor") {
+							if !property.Kind.IsMethodDefinition() {
+								// "constructor" is an invalid name for both instance and static fields
+								isInvalidConstructor = true
+							} else if !property.Flags.Has(js_ast.PropertyIsStatic) {
+								// Calling an instance method "constructor" is problematic so avoid that too
+								isInvalidConstructor = true
+							}
+						}
+
+						// A static property must not be called "prototype"
+						isInvalidPrototype := property.Flags.Has(js_ast.PropertyIsStatic) && helpers.UTF16EqualsString(k.Value, "prototype")
+
+						if !isInvalidConstructor && !isInvalidPrototype {
+							property.Flags &= ^js_ast.PropertyIsComputed
+						}
+					}
+				}
+			}
+		}
+
+		// Make it an error to use "arguments" in a class body
+		p.currentScope.ForbidArguments = true
+
+		// The value of "this" and "super" is shadowed inside property values
+		oldFnOnlyDataVisit := p.fnOnlyDataVisit
+		oldShouldLowerSuperPropertyAccess := p.fnOrArrowDataVisit.shouldLowerSuperPropertyAccess
+		p.fnOrArrowDataVisit.shouldLowerSuperPropertyAccess = false
+		p.fnOnlyDataVisit.shouldReplaceThisWithInnerClassNameRef = false
+		p.fnOnlyDataVisit.isThisNested = true
+		p.fnOnlyDataVisit.isNewTargetAllowed = true
+		p.fnOnlyDataVisit.isInStaticClassContext = property.Flags.Has(js_ast.PropertyIsStatic)
+		p.fnOnlyDataVisit.innerClassNameRef = &result.innerClassNameRef
+
+		// We need to explicitly assign the name to the property initializer if it
+		// will be transformed such that it is no longer an inline initializer.
+		nameToKeep := ""
+		isLoweredPrivateMethod := false
+		if private, ok := property.Key.Data.(*js_ast.EPrivateIdentifier); ok {
+			if !property.Kind.IsMethodDefinition() || p.privateSymbolNeedsToBeLowered(private) {
+				nameToKeep = p.symbols[private.Ref.InnerIndex].OriginalName
+			}
+
+			// Lowered private methods (both instance and static) are initialized
+			// outside of the class body, so we must rewrite "super" property
+			// accesses inside them. Lowered private instance fields are initialized
+			// inside the constructor where "super" is valid, so those don't need to
+			// be rewritten.
+			if property.Kind.IsMethodDefinition() && p.privateSymbolNeedsToBeLowered(private) {
+				isLoweredPrivateMethod = true
+			}
+		} else if !property.Kind.IsMethodDefinition() && !property.Flags.Has(js_ast.PropertyIsComputed) {
+			if str, ok := property.Key.Data.(*js_ast.EString); ok {
+				nameToKeep = helpers.UTF16ToString(str.Value)
+			}
+		}
+
+		// Handle methods
+		if property.ValueOrNil.Data != nil {
+			p.propMethodDecoratorScope = result.bodyScope
+
+			// Propagate the name to keep from the method into the initializer
+			if nameToKeep != "" {
+				p.nameToKeep = nameToKeep
+				p.nameToKeepIsFor = property.ValueOrNil.Data
+			}
+
+			// Propagate whether we're in a derived class constructor
+			if class.ExtendsOrNil.Data != nil && !property.Flags.Has(js_ast.PropertyIsComputed) {
+				if str, ok := property.Key.Data.(*js_ast.EString); ok && helpers.UTF16EqualsString(str.Value, "constructor") {
+					p.propDerivedCtorValue = property.ValueOrNil.Data
+				}
+			}
+
+			property.ValueOrNil, _ = p.visitExprInOut(property.ValueOrNil, exprIn{
+				isMethod:               true,
+				isLoweredPrivateMethod: isLoweredPrivateMethod,
+			})
+		}
+
+		// Handle initialized fields
+		if property.InitializerOrNil.Data != nil {
+			if property.Flags.Has(js_ast.PropertyIsStatic) && classLoweringInfo.lowerAllStaticFields {
+				// Need to lower "this" and "super" since they won't be valid outside the class body
+				p.fnOnlyDataVisit.shouldReplaceThisWithInnerClassNameRef = true
+				p.fnOrArrowDataVisit.shouldLowerSuperPropertyAccess = true
+			}
+
+			// Propagate the name to keep from the field into the initializer
+			if nameToKeep != "" {
+				p.nameToKeep = nameToKeep
+				p.nameToKeepIsFor = property.InitializerOrNil.Data
+			}
+
+			property.InitializerOrNil = p.visitExpr(property.InitializerOrNil)
+		}
+
+		// Restore "this" so it will take the inherited value in property keys
+		p.fnOnlyDataVisit = oldFnOnlyDataVisit
+		p.fnOrArrowDataVisit.shouldLowerSuperPropertyAccess = oldShouldLowerSuperPropertyAccess
+
+		// Restore the ability to use "arguments" in decorators and computed properties
+		p.currentScope.ForbidArguments = false
+	}
+
+	// Check for and warn about duplicate keys in class bodies
+	if !p.suppressWarningsAboutWeirdCode {
+		p.warnAboutDuplicateProperties(class.Properties, duplicatePropertiesInClass)
+	}
+
+	// Analyze side effects before adding the name keeping call
+	result.canBeRemovedIfUnused = p.astHelpers.ClassCanBeRemovedIfUnused(*class)
+
+	// Implement name keeping using a static block at the start of the class body
+	if p.options.keepNames && nameToKeep != "" {
+		propertyPreventsKeepNames := false
+		for _, prop := range class.Properties {
+			// A static property called "name" shadows the automatically-generated name
+			if prop.Flags.Has(js_ast.PropertyIsStatic) {
+				if str, ok := prop.Key.Data.(*js_ast.EString); ok && helpers.UTF16EqualsString(str.Value, "name") {
+					propertyPreventsKeepNames = true
+					break
+				}
+			}
+		}
+		if !propertyPreventsKeepNames {
+			var this js_ast.Expr
+			if classLoweringInfo.lowerAllStaticFields {
+				p.recordUsage(result.innerClassNameRef)
+				this = js_ast.Expr{Loc: class.BodyLoc, Data: &js_ast.EIdentifier{Ref: result.innerClassNameRef}}
+			} else {
+				this = js_ast.Expr{Loc: class.BodyLoc, Data: js_ast.EThisShared}
+			}
+			properties := make([]js_ast.Property, 0, 1+len(class.Properties))
+			properties = append(properties, js_ast.Property{
+				Kind: js_ast.PropertyClassStaticBlock,
+				ClassStaticBlock: &js_ast.ClassStaticBlock{Loc: class.BodyLoc, Block: js_ast.SBlock{Stmts: []js_ast.Stmt{
+					p.keepClassOrFnSymbolName(class.BodyLoc, this, nameToKeep),
+				}}},
+			})
+			class.Properties = append(properties, class.Properties...)
+		}
+	}
+
+	p.enclosingClassKeyword = oldEnclosingClassKeyword
+	p.superCtorRef = oldSuperCtorRef
+	p.popScope()
+
+	if p.symbols[result.innerClassNameRef.InnerIndex].UseCountEstimate == 0 {
+		// Don't generate a shadowing name if one isn't needed
+		result.innerClassNameRef = ast.InvalidRef
+	} else if class.Name == nil {
+		// If there was originally no class name but something inside needed one
+		// (e.g. there was a static property initializer that referenced "this"),
+		// populate the class name. If this is an "export default class" statement,
+		// use the existing default name so that things will work as expected if
+		// this is turned into a regular class statement later on.
+		classNameRef := defaultNameRef
+		if classNameRef == ast.InvalidRef {
+			classNameRef = p.newSymbol(ast.SymbolOther, "_this")
+			p.currentScope.Generated = append(p.currentScope.Generated, classNameRef)
+			p.recordDeclaredSymbol(classNameRef)
+		}
+		class.Name = &ast.LocRef{Loc: nameScopeLoc, Ref: classNameRef}
+	}
+
+	p.popScope()
+
+	// Sanity check that the class lowering info hasn't changed before and after
+	// visiting. The class transform relies on this because lowering assumes that
+	// must be able to expect that visiting has done certain things.
+	if classLoweringInfo != p.computeClassLoweringInfo(class) {
+		panic("Internal error")
+	}
+
+	return
+}
+
+func isSimpleParameterList(args []js_ast.Arg, hasRestArg bool) bool {
+	if hasRestArg {
+		return false
+	}
+	for _, arg := range args {
+		if _, ok := arg.Binding.Data.(*js_ast.BIdentifier); !ok || arg.DefaultOrNil.Data != nil {
+			return false
+		}
+	}
+	return true
+}
+
+func fnBodyContainsUseStrict(body []js_ast.Stmt) (logger.Loc, bool) {
+	for _, stmt := range body {
+		switch s := stmt.Data.(type) {
+		case *js_ast.SComment:
+			continue
+		case *js_ast.SDirective:
+			if helpers.UTF16EqualsString(s.Value, "use strict") {
+				return stmt.Loc, true
+			}
+		default:
+			return logger.Loc{}, false
+		}
+	}
+	return logger.Loc{}, false
+}
+
+type visitArgsOpts struct {
+	body           []js_ast.Stmt
+	decoratorScope *js_ast.Scope
+	hasRestArg     bool
+
+	// This is true if the function is an arrow function or a method
+	isUniqueFormalParameters bool
+}
+
+func (p *parser) visitArgs(args []js_ast.Arg, opts visitArgsOpts) {
+	var duplicateArgCheck map[string]logger.Range
+	useStrictLoc, hasUseStrict := fnBodyContainsUseStrict(opts.body)
+	hasSimpleArgs := isSimpleParameterList(args, opts.hasRestArg)
+
+	// Section 15.2.1 Static Semantics: Early Errors: "It is a Syntax Error if
+	// FunctionBodyContainsUseStrict of FunctionBody is true and
+	// IsSimpleParameterList of FormalParameters is false."
+	if hasUseStrict && !hasSimpleArgs {
+		p.log.AddError(&p.tracker, p.source.RangeOfString(useStrictLoc),
+			"Cannot use a \"use strict\" directive in a function with a non-simple parameter list")
+	}
+
+	// Section 15.1.1 Static Semantics: Early Errors: "Multiple occurrences of
+	// the same BindingIdentifier in a FormalParameterList is only allowed for
+	// functions which have simple parameter lists and which are not defined in
+	// strict mode code."
+	if opts.isUniqueFormalParameters || hasUseStrict || !hasSimpleArgs || p.isStrictMode() {
+		duplicateArgCheck = make(map[string]logger.Range)
+	}
+
+	for i := range args {
+		arg := &args[i]
+		arg.Decorators = p.visitDecorators(arg.Decorators, opts.decoratorScope)
+		p.visitBinding(arg.Binding, bindingOpts{
+			duplicateArgCheck: duplicateArgCheck,
+		})
+		if arg.DefaultOrNil.Data != nil {
+			arg.DefaultOrNil = p.visitExpr(arg.DefaultOrNil)
+		}
+	}
+}
+
+func (p *parser) isDotOrIndexDefineMatch(expr js_ast.Expr, parts []string) bool {
+	switch e := expr.Data.(type) {
+	case *js_ast.EDot:
+		if len(parts) > 1 {
+			// Intermediates must be dot expressions
+			last := len(parts) - 1
+			return parts[last] == e.Name && p.isDotOrIndexDefineMatch(e.Target, parts[:last])
+		}
+
+	case *js_ast.EIndex:
+		if len(parts) > 1 {
+			if str, ok := e.Index.Data.(*js_ast.EString); ok {
+				// Intermediates must be dot expressions
+				last := len(parts) - 1
+				return parts[last] == helpers.UTF16ToString(str.Value) && p.isDotOrIndexDefineMatch(e.Target, parts[:last])
+			}
+		}
+
+	case *js_ast.EThis:
+		// Allow matching on top-level "this"
+		if !p.fnOnlyDataVisit.isThisNested {
+			return len(parts) == 1 && parts[0] == "this"
+		}
+
+	case *js_ast.EImportMeta:
+		// Allow matching on "import.meta"
+		return len(parts) == 2 && parts[0] == "import" && parts[1] == "meta"
+
+	case *js_ast.EIdentifier:
+		// The last expression must be an identifier
+		if len(parts) == 1 {
+			// The name must match
+			name := p.loadNameFromRef(e.Ref)
+			if name != parts[0] {
+				return false
+			}
+
+			result := p.findSymbol(expr.Loc, name)
+
+			// The "findSymbol" function also marks this symbol as used. But that's
+			// never what we want here because we're just peeking to see what kind of
+			// symbol it is to see if it's a match. If it's not a match, it will be
+			// re-resolved again later and marked as used there. So we don't want to
+			// mark it as used twice.
+			p.ignoreUsage(result.ref)
+
+			// We must not be in a "with" statement scope
+			if result.isInsideWithScope {
+				return false
+			}
+
+			// The last symbol must be unbound or injected
+			return p.symbols[result.ref.InnerIndex].Kind.IsUnboundOrInjected()
+		}
+	}
+
+	return false
+}
+
+func (p *parser) instantiateDefineExpr(loc logger.Loc, expr config.DefineExpr, opts identifierOpts) js_ast.Expr {
+	if expr.Constant != nil {
+		return js_ast.Expr{Loc: loc, Data: expr.Constant}
+	}
+
+	if expr.InjectedDefineIndex.IsValid() {
+		ref := p.injectedDefineSymbols[expr.InjectedDefineIndex.GetIndex()]
+		p.recordUsage(ref)
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: ref}}
+	}
+
+	parts := expr.Parts
+	if len(parts) == 0 {
+		return js_ast.Expr{}
+	}
+
+	// Check both user-specified defines and known globals
+	if opts.matchAgainstDefines {
+		// Make sure define resolution is not recursive
+		opts.matchAgainstDefines = false
+
+		// Substitute user-specified defines
+		if defines, ok := p.options.defines.DotDefines[parts[len(parts)-1]]; ok {
+			for _, define := range defines {
+				if define.Data.DefineExpr != nil && helpers.StringArraysEqual(define.Parts, parts) {
+					return p.instantiateDefineExpr(loc, *define.Data.DefineExpr, opts)
+				}
+			}
+		}
+	}
+
+	// Check injected dot names
+	if names, ok := p.injectedDotNames[parts[len(parts)-1]]; ok {
+		for _, name := range names {
+			if helpers.StringArraysEqual(name.parts, parts) {
+				return p.instantiateInjectDotName(loc, name, opts.assignTarget)
+			}
+		}
+	}
+
+	// Generate an identifier for the first part
+	var value js_ast.Expr
+	firstPart := parts[0]
+	parts = parts[1:]
+	switch firstPart {
+	case "NaN":
+		value = js_ast.Expr{Loc: loc, Data: &js_ast.ENumber{Value: math.NaN()}}
+
+	case "Infinity":
+		value = js_ast.Expr{Loc: loc, Data: &js_ast.ENumber{Value: math.Inf(1)}}
+
+	case "null":
+		value = js_ast.Expr{Loc: loc, Data: js_ast.ENullShared}
+
+	case "undefined":
+		value = js_ast.Expr{Loc: loc, Data: js_ast.EUndefinedShared}
+
+	case "this":
+		if thisValue, ok := p.valueForThis(loc, false /* shouldLog */, js_ast.AssignTargetNone, false, false); ok {
+			value = thisValue
+		} else {
+			value = js_ast.Expr{Loc: loc, Data: js_ast.EThisShared}
+		}
+
+	default:
+		if firstPart == "import" && len(parts) > 0 && parts[0] == "meta" {
+			if importMeta, ok := p.valueForImportMeta(loc); ok {
+				value = importMeta
+			} else {
+				value = js_ast.Expr{Loc: loc, Data: &js_ast.EImportMeta{}}
+			}
+			parts = parts[1:]
+			break
+		}
+
+		result := p.findSymbol(loc, firstPart)
+		value = p.handleIdentifier(loc, &js_ast.EIdentifier{
+			Ref:                   result.ref,
+			MustKeepDueToWithStmt: result.isInsideWithScope,
+
+			// Enable tree shaking
+			CanBeRemovedIfUnused: true,
+		}, opts)
+	}
+
+	// Build up a chain of property access expressions for subsequent parts
+	for _, part := range parts {
+		if expr, ok := p.maybeRewritePropertyAccess(loc, js_ast.AssignTargetNone, false, value, part, loc, false, false, false); ok {
+			value = expr
+		} else if p.isMangledProp(part) {
+			value = js_ast.Expr{Loc: loc, Data: &js_ast.EIndex{
+				Target: value,
+				Index:  js_ast.Expr{Loc: loc, Data: &js_ast.ENameOfSymbol{Ref: p.symbolForMangledProp(part)}},
+			}}
+		} else {
+			value = js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+				Target:  value,
+				Name:    part,
+				NameLoc: loc,
+
+				// Enable tree shaking
+				CanBeRemovedIfUnused: true,
+			}}
+		}
+	}
+
+	return value
+}
+
+func (p *parser) instantiateInjectDotName(loc logger.Loc, name injectedDotName, assignTarget js_ast.AssignTarget) js_ast.Expr {
+	// Note: We don't need to "ignoreRef" on the underlying identifier
+	// because we have only parsed it but not visited it yet
+	ref := p.injectedDefineSymbols[name.injectedDefineIndex]
+	p.recordUsage(ref)
+
+	if assignTarget != js_ast.AssignTargetNone {
+		if where, ok := p.injectedSymbolSources[ref]; ok {
+			r := js_lexer.RangeOfIdentifier(p.source, loc)
+			tracker := logger.MakeLineColumnTracker(&where.source)
+			joined := strings.Join(name.parts, ".")
+			p.log.AddErrorWithNotes(&p.tracker, r,
+				fmt.Sprintf("Cannot assign to %q because it's an import from an injected file", joined),
+				[]logger.MsgData{tracker.MsgData(js_lexer.RangeOfIdentifier(where.source, where.loc),
+					fmt.Sprintf("The symbol %q was exported from %q here:", joined, where.source.PrettyPath))})
+		}
+	}
+
+	return js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: ref}}
+}
+
+func (p *parser) checkForUnrepresentableIdentifier(loc logger.Loc, name string) {
+	if p.options.asciiOnly && p.options.unsupportedJSFeatures.Has(compat.UnicodeEscapes) &&
+		helpers.ContainsNonBMPCodePoint(name) {
+		if p.unrepresentableIdentifiers == nil {
+			p.unrepresentableIdentifiers = make(map[string]bool)
+		}
+		if !p.unrepresentableIdentifiers[name] {
+			p.unrepresentableIdentifiers[name] = true
+			where := config.PrettyPrintTargetEnvironment(p.options.originalTargetEnv, p.options.unsupportedJSFeatureOverridesMask)
+			r := js_lexer.RangeOfIdentifier(p.source, loc)
+			p.log.AddError(&p.tracker, r, fmt.Sprintf("%q cannot be escaped in %s but you "+
+				"can set the charset to \"utf8\" to allow unescaped Unicode characters", name, where))
+		}
+	}
+}
+
+type typeofStringOrder uint8
+
+const (
+	onlyCheckOriginalOrder typeofStringOrder = iota
+	checkBothOrders
+)
+
+func (p *parser) warnAboutTypeofAndString(a js_ast.Expr, b js_ast.Expr, order typeofStringOrder) {
+	if order == checkBothOrders {
+		if _, ok := a.Data.(*js_ast.EString); ok {
+			a, b = b, a
+		}
+	}
+
+	if typeof, ok := a.Data.(*js_ast.EUnary); ok && typeof.Op == js_ast.UnOpTypeof {
+		if str, ok := b.Data.(*js_ast.EString); ok {
+			value := helpers.UTF16ToString(str.Value)
+			switch value {
+			case "undefined", "object", "boolean", "number", "bigint", "string", "symbol", "function", "unknown":
+			default:
+				// Warn about typeof comparisons with values that will never be
+				// returned. Here's an example of code with this problem:
+				// https://github.com/olifolkerd/tabulator/issues/2962
+				r := p.source.RangeOfString(b.Loc)
+				text := fmt.Sprintf("The \"typeof\" operator will never evaluate to %q", value)
+				kind := logger.Warning
+				if p.suppressWarningsAboutWeirdCode {
+					kind = logger.Debug
+				}
+				var notes []logger.MsgData
+				if value == "null" {
+					notes = append(notes, logger.MsgData{
+						Text: "The expression \"typeof x\" actually evaluates to \"object\" in JavaScript, not \"null\". " +
+							"You need to use \"x === null\" to test for null.",
+					})
+				}
+				p.log.AddIDWithNotes(logger.MsgID_JS_ImpossibleTypeof, kind, &p.tracker, r, text, notes)
+			}
+		}
+	}
+}
+
+func (p *parser) warnAboutEqualityCheck(op string, value js_ast.Expr, afterOpLoc logger.Loc) bool {
+	switch e := value.Data.(type) {
+	case *js_ast.ENumber:
+		// "0 === -0" is true in JavaScript. Here's an example of code with this
+		// problem: https://github.com/mrdoob/three.js/pull/11183
+		if e.Value == 0 && math.Signbit(e.Value) {
+			r := logger.Range{Loc: value.Loc, Len: 0}
+			if int(r.Loc.Start) < len(p.source.Contents) && p.source.Contents[r.Loc.Start] == '-' {
+				zeroRange := p.source.RangeOfNumber(logger.Loc{Start: r.Loc.Start + 1})
+				r.Len = zeroRange.Len + 1
+			}
+			text := fmt.Sprintf("Comparison with -0 using the %q operator will also match 0", op)
+			if op == "case" {
+				text = "Comparison with -0 using a case clause will also match 0"
+			}
+			kind := logger.Warning
+			if p.suppressWarningsAboutWeirdCode {
+				kind = logger.Debug
+			}
+			p.log.AddIDWithNotes(logger.MsgID_JS_EqualsNegativeZero, kind, &p.tracker, r, text,
+				[]logger.MsgData{{Text: "Floating-point equality is defined such that 0 and -0 are equal, so \"x === -0\" returns true for both 0 and -0. " +
+					"You need to use \"Object.is(x, -0)\" instead to test for -0."}})
+			return true
+		}
+
+		// "NaN === NaN" is false in JavaScript
+		if math.IsNaN(e.Value) {
+			text := fmt.Sprintf("Comparison with NaN using the %q operator here is always %v", op, op[0] == '!')
+			if op == "case" {
+				text = "This case clause will never be evaluated because equality with NaN is always false"
+			}
+			r := p.source.RangeOfOperatorBefore(afterOpLoc, op)
+			kind := logger.Warning
+			if p.suppressWarningsAboutWeirdCode {
+				kind = logger.Debug
+			}
+			p.log.AddIDWithNotes(logger.MsgID_JS_EqualsNaN, kind, &p.tracker, r, text,
+				[]logger.MsgData{{Text: "Floating-point equality is defined such that NaN is never equal to anything, so \"x === NaN\" always returns false. " +
+					"You need to use \"Number.isNaN(x)\" instead to test for NaN."}})
+			return true
+		}
+
+	case *js_ast.EArray, *js_ast.EArrow, *js_ast.EClass,
+		*js_ast.EFunction, *js_ast.EObject, *js_ast.ERegExp:
+		// This warning only applies to strict equality because loose equality can
+		// cause string conversions. For example, "x == []" is true if x is the
+		// empty string. Here's an example of code with this problem:
+		// https://github.com/aws/aws-sdk-js/issues/3325
+		if len(op) > 2 {
+			text := fmt.Sprintf("Comparison using the %q operator here is always %v", op, op[0] == '!')
+			if op == "case" {
+				text = "This case clause will never be evaluated because the comparison is always false"
+			}
+			r := p.source.RangeOfOperatorBefore(afterOpLoc, op)
+			kind := logger.Warning
+			if p.suppressWarningsAboutWeirdCode {
+				kind = logger.Debug
+			}
+			p.log.AddIDWithNotes(logger.MsgID_JS_EqualsNewObject, kind, &p.tracker, r, text,
+				[]logger.MsgData{{Text: "Equality with a new object is always false in JavaScript because the equality operator tests object identity. " +
+					"You need to write code to compare the contents of the object instead. " +
+					"For example, use \"Array.isArray(x) && x.length === 0\" instead of \"x === []\" to test for an empty array."}})
+			return true
+		}
+	}
+
+	return false
+}
+
+// EDot nodes represent a property access. This function may return an
+// expression to replace the property access with. It assumes that the
+// target of the EDot expression has already been visited.
+func (p *parser) maybeRewritePropertyAccess(
+	loc logger.Loc,
+	assignTarget js_ast.AssignTarget,
+	isDeleteTarget bool,
+	target js_ast.Expr,
+	name string,
+	nameLoc logger.Loc,
+	isCallTarget bool,
+	isTemplateTag bool,
+	preferQuotedKey bool,
+) (js_ast.Expr, bool) {
+	if id, ok := target.Data.(*js_ast.EIdentifier); ok {
+		// Rewrite property accesses on explicit namespace imports as an identifier.
+		// This lets us replace them easily in the printer to rebind them to
+		// something else without paying the cost of a whole-tree traversal during
+		// module linking just to rewrite these EDot expressions.
+		if p.options.mode == config.ModeBundle {
+			if importItems, ok := p.importItemsForNamespace[id.Ref]; ok {
+				// Cache translation so each property access resolves to the same import
+				item, ok := importItems.entries[name]
+				if !ok {
+					// Replace non-default imports with "undefined" for JSON import assertions
+					if record := &p.importRecords[importItems.importRecordIndex]; (record.Flags&ast.AssertTypeJSON) != 0 && name != "default" {
+						kind := logger.Warning
+						if p.suppressWarningsAboutWeirdCode {
+							kind = logger.Debug
+						}
+						p.log.AddIDWithNotes(logger.MsgID_JS_AssertTypeJSON, kind, &p.tracker, js_lexer.RangeOfIdentifier(p.source, nameLoc),
+							fmt.Sprintf("Non-default import %q is undefined with a JSON import assertion", name),
+							p.notesForAssertTypeJSON(record, name))
+						p.ignoreUsage(id.Ref)
+						return js_ast.Expr{Loc: loc, Data: js_ast.EUndefinedShared}, true
+					}
+
+					// Generate a new import item symbol in the module scope
+					item = ast.LocRef{Loc: nameLoc, Ref: p.newSymbol(ast.SymbolImport, name)}
+					p.moduleScope.Generated = append(p.moduleScope.Generated, item.Ref)
+
+					// Link the namespace import and the import item together
+					importItems.entries[name] = item
+					p.isImportItem[item.Ref] = true
+
+					symbol := &p.symbols[item.Ref.InnerIndex]
+					if p.options.mode == config.ModePassThrough {
+						// Make sure the printer prints this as a property access
+						symbol.NamespaceAlias = &ast.NamespaceAlias{
+							NamespaceRef: id.Ref,
+							Alias:        name,
+						}
+					} else {
+						// Mark this as generated in case it's missing. We don't want to
+						// generate errors for missing import items that are automatically
+						// generated.
+						symbol.ImportItemStatus = ast.ImportItemGenerated
+					}
+				}
+
+				// Undo the usage count for the namespace itself. This is used later
+				// to detect whether the namespace symbol has ever been "captured"
+				// or whether it has just been used to read properties off of.
+				//
+				// The benefit of doing this is that if both this module and the
+				// imported module end up in the same module group and the namespace
+				// symbol has never been captured, then we don't need to generate
+				// any code for the namespace at all.
+				p.ignoreUsage(id.Ref)
+
+				// Track how many times we've referenced this symbol
+				p.recordUsage(item.Ref)
+				return p.handleIdentifier(nameLoc, &js_ast.EIdentifier{Ref: item.Ref}, identifierOpts{
+					assignTarget:    assignTarget,
+					isCallTarget:    isCallTarget,
+					isDeleteTarget:  isDeleteTarget,
+					preferQuotedKey: preferQuotedKey,
+
+					// If this expression is used as the target of a call expression, make
+					// sure the value of "this" is preserved.
+					wasOriginallyIdentifier: false,
+				}), true
+			}
+
+			// Rewrite "module.require()" to "require()" for Webpack compatibility.
+			// See https://github.com/webpack/webpack/pull/7750 for more info.
+			if isCallTarget && id.Ref == p.moduleRef && name == "require" {
+				p.ignoreUsage(p.moduleRef)
+
+				// This uses "require" instead of a reference to our "__require"
+				// function so that the code coming up that detects calls to
+				// "require" will recognize it.
+				p.recordUsage(p.requireRef)
+				return js_ast.Expr{Loc: nameLoc, Data: &js_ast.EIdentifier{Ref: p.requireRef}}, true
+			}
+		}
+	}
+
+	// Attempt to simplify statically-determined object literal property accesses
+	if !isCallTarget && !isTemplateTag && p.options.minifySyntax && assignTarget == js_ast.AssignTargetNone {
+		if object, ok := target.Data.(*js_ast.EObject); ok {
+			var replace js_ast.Expr
+			hasProtoNull := false
+			isUnsafe := false
+
+			// Check that doing this is safe
+			for _, prop := range object.Properties {
+				// "{ ...a }.a" must be preserved
+				// "new ({ a() {} }.a)" must throw
+				// "{ get a() {} }.a" must be preserved
+				// "{ set a(b) {} }.a = 1" must be preserved
+				// "{ a: 1, [String.fromCharCode(97)]: 2 }.a" must be 2
+				if prop.Kind == js_ast.PropertySpread || prop.Flags.Has(js_ast.PropertyIsComputed) || prop.Kind.IsMethodDefinition() {
+					isUnsafe = true
+					break
+				}
+
+				// Do not attempt to compare against numeric keys
+				key, ok := prop.Key.Data.(*js_ast.EString)
+				if !ok {
+					isUnsafe = true
+					break
+				}
+
+				// The "__proto__" key has special behavior
+				if helpers.UTF16EqualsString(key.Value, "__proto__") {
+					if _, ok := prop.ValueOrNil.Data.(*js_ast.ENull); ok {
+						// Replacing "{__proto__: null}.a" with undefined should be safe
+						hasProtoNull = true
+					}
+				}
+
+				// This entire object literal must have no side effects
+				if !p.astHelpers.ExprCanBeRemovedIfUnused(prop.ValueOrNil) {
+					isUnsafe = true
+					break
+				}
+
+				// Note that we need to take the last value if there are duplicate keys
+				if helpers.UTF16EqualsString(key.Value, name) {
+					replace = prop.ValueOrNil
+				}
+			}
+
+			if !isUnsafe {
+				// If the key was found, return the value for that key. Note
+				// that "{__proto__: null}.__proto__" is undefined, not null.
+				if replace.Data != nil && name != "__proto__" {
+					return replace, true
+				}
+
+				// We can only return "undefined" when a key is missing if the prototype is null
+				if hasProtoNull {
+					return js_ast.Expr{Loc: target.Loc, Data: js_ast.EUndefinedShared}, true
+				}
+			}
+		}
+	}
+
+	// Handle references to namespaces or namespace members
+	if target.Data == p.tsNamespaceTarget && assignTarget == js_ast.AssignTargetNone && !isDeleteTarget {
+		if ns, ok := p.tsNamespaceMemberData.(*js_ast.TSNamespaceMemberNamespace); ok {
+			if member, ok := ns.ExportedMembers[name]; ok {
+				switch m := member.Data.(type) {
+				case *js_ast.TSNamespaceMemberEnumNumber:
+					p.ignoreUsageOfIdentifierInDotChain(target)
+					return p.wrapInlinedEnum(js_ast.Expr{Loc: loc, Data: &js_ast.ENumber{Value: m.Value}}, name), true
+
+				case *js_ast.TSNamespaceMemberEnumString:
+					p.ignoreUsageOfIdentifierInDotChain(target)
+					return p.wrapInlinedEnum(js_ast.Expr{Loc: loc, Data: &js_ast.EString{Value: m.Value}}, name), true
+
+				case *js_ast.TSNamespaceMemberNamespace:
+					// If this isn't a constant, return a clone of this property access
+					// but with the namespace member data associated with it so that
+					// more property accesses off of this property access are recognized.
+					if preferQuotedKey || !js_ast.IsIdentifier(name) {
+						p.tsNamespaceTarget = &js_ast.EIndex{
+							Target: target,
+							Index:  js_ast.Expr{Loc: nameLoc, Data: &js_ast.EString{Value: helpers.StringToUTF16(name)}},
+						}
+					} else {
+						p.tsNamespaceTarget = p.dotOrMangledPropVisit(target, name, nameLoc)
+					}
+					p.tsNamespaceMemberData = member.Data
+					return js_ast.Expr{Loc: loc, Data: p.tsNamespaceTarget}, true
+				}
+			}
+		}
+	}
+
+	// Symbol uses due to a property access off of an imported symbol are tracked
+	// specially. This lets us do tree shaking for cross-file TypeScript enums.
+	if p.options.mode == config.ModeBundle && !p.isControlFlowDead {
+		if id, ok := target.Data.(*js_ast.EImportIdentifier); ok {
+			// Remove the normal symbol use
+			use := p.symbolUses[id.Ref]
+			use.CountEstimate--
+			if use.CountEstimate == 0 {
+				delete(p.symbolUses, id.Ref)
+			} else {
+				p.symbolUses[id.Ref] = use
+			}
+
+			// Add a special symbol use instead
+			if p.importSymbolPropertyUses == nil {
+				p.importSymbolPropertyUses = make(map[ast.Ref]map[string]js_ast.SymbolUse)
+			}
+			properties := p.importSymbolPropertyUses[id.Ref]
+			if properties == nil {
+				properties = make(map[string]js_ast.SymbolUse)
+				p.importSymbolPropertyUses[id.Ref] = properties
+			}
+			use = properties[name]
+			use.CountEstimate++
+			properties[name] = use
+		}
+	}
+
+	// Minify "foo".length
+	if p.options.minifySyntax && assignTarget == js_ast.AssignTargetNone {
+		switch t := target.Data.(type) {
+		case *js_ast.EString:
+			if name == "length" {
+				return js_ast.Expr{Loc: loc, Data: &js_ast.ENumber{Value: float64(len(t.Value))}}, true
+			}
+		case *js_ast.EInlinedEnum:
+			if s, ok := t.Value.Data.(*js_ast.EString); ok && name == "length" {
+				return js_ast.Expr{Loc: loc, Data: &js_ast.ENumber{Value: float64(len(s.Value))}}, true
+			}
+		}
+	}
+
+	return js_ast.Expr{}, false
+}
+
+type exprIn struct {
+	isMethod               bool
+	isLoweredPrivateMethod bool
+
+	// This tells us if there are optional chain expressions (EDot, EIndex, or
+	// ECall) that are chained on to this expression. Because of the way the AST
+	// works, chaining expressions on to this expression means they are our
+	// parent expressions.
+	//
+	// Some examples:
+	//
+	//   a?.b.c  // EDot
+	//   a?.b[c] // EIndex
+	//   a?.b()  // ECall
+	//
+	// Note that this is false if our parent is a node with a OptionalChain
+	// value of OptionalChainStart. That means it's the start of a new chain, so
+	// it's not considered part of this one.
+	//
+	// Some examples:
+	//
+	//   a?.b?.c   // EDot
+	//   a?.b?.[c] // EIndex
+	//   a?.b?.()  // ECall
+	//
+	// Also note that this is false if our parent is a node with a OptionalChain
+	// value of OptionalChainNone. That means it's outside parentheses, which
+	// means it's no longer part of the chain.
+	//
+	// Some examples:
+	//
+	//   (a?.b).c  // EDot
+	//   (a?.b)[c] // EIndex
+	//   (a?.b)()  // ECall
+	//
+	hasChainParent bool
+
+	// If our parent is an ECall node with an OptionalChain value of
+	// OptionalChainStart, then we will need to store the value for the "this" of
+	// that call somewhere if the current expression is an optional chain that
+	// ends in a property access. That's because the value for "this" will be
+	// used twice: once for the inner optional chain and once for the outer
+	// optional chain.
+	//
+	// Example:
+	//
+	//   // Original
+	//   a?.b?.();
+	//
+	//   // Lowered
+	//   var _a;
+	//   (_a = a == null ? void 0 : a.b) == null ? void 0 : _a.call(a);
+	//
+	// In the example above we need to store "a" as the value for "this" so we
+	// can substitute it back in when we call "_a" if "_a" is indeed present.
+	// See also "thisArgFunc" and "thisArgWrapFunc" in "exprOut".
+	storeThisArgForParentOptionalChain bool
+
+	// If true, string literals that match the current property mangling pattern
+	// should be turned into ENameOfSymbol expressions, which will cause us to
+	// rename them in the linker.
+	shouldMangleStringsAsProps bool
+
+	// Certain substitutions of identifiers are disallowed for assignment targets.
+	// For example, we shouldn't transform "undefined = 1" into "void 0 = 1". This
+	// isn't something real-world code would do but it matters for conformance
+	// tests.
+	assignTarget js_ast.AssignTarget
+}
+
+type exprOut struct {
+	// If our parent is an ECall node with an OptionalChain value of
+	// OptionalChainContinue, then we may need to return the value for "this"
+	// from this node or one of this node's children so that the parent that is
+	// the end of the optional chain can use it.
+	//
+	// Example:
+	//
+	//   // Original
+	//   a?.b?.().c();
+	//
+	//   // Lowered
+	//   var _a;
+	//   (_a = a == null ? void 0 : a.b) == null ? void 0 : _a.call(a).c();
+	//
+	// The value "_a" for "this" must be passed all the way up to the call to
+	// ".c()" which is where the optional chain is lowered. From there it must
+	// be substituted as the value for "this" in the call to ".b?.()". See also
+	// "storeThisArgForParentOptionalChain" in "exprIn".
+	thisArgFunc     func() js_ast.Expr
+	thisArgWrapFunc func(js_ast.Expr) js_ast.Expr
+
+	// True if the child node is an optional chain node (EDot, EIndex, or ECall
+	// with an IsOptionalChain value of true)
+	childContainsOptionalChain bool
+
+	// If true and this is used as a call target, the whole call expression
+	// must be replaced with undefined.
+	methodCallMustBeReplacedWithUndefined bool
+}
+
+func (p *parser) visitExpr(expr js_ast.Expr) js_ast.Expr {
+	expr, _ = p.visitExprInOut(expr, exprIn{})
+	return expr
+}
+
+func (p *parser) valueForThis(
+	loc logger.Loc,
+	shouldLog bool,
+	assignTarget js_ast.AssignTarget,
+	isCallTarget bool,
+	isDeleteTarget bool,
+) (js_ast.Expr, bool) {
+	// Substitute "this" if we're inside a static class context
+	if p.fnOnlyDataVisit.shouldReplaceThisWithInnerClassNameRef {
+		p.recordUsage(*p.fnOnlyDataVisit.innerClassNameRef)
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: *p.fnOnlyDataVisit.innerClassNameRef}}, true
+	}
+
+	// Is this a top-level use of "this"?
+	if !p.fnOnlyDataVisit.isThisNested {
+		// Substitute user-specified defines
+		if data, ok := p.options.defines.IdentifierDefines["this"]; ok {
+			if data.DefineExpr != nil {
+				return p.instantiateDefineExpr(loc, *data.DefineExpr, identifierOpts{
+					assignTarget:   assignTarget,
+					isCallTarget:   isCallTarget,
+					isDeleteTarget: isDeleteTarget,
+				}), true
+			}
+		}
+
+		// Otherwise, replace top-level "this" with either "undefined" or "exports"
+		if p.isFileConsideredToHaveESMExports {
+			// Warn about "this" becoming undefined, but only once per file
+			if shouldLog && !p.messageAboutThisIsUndefined && !p.fnOnlyDataVisit.silenceMessageAboutThisBeingUndefined {
+				p.messageAboutThisIsUndefined = true
+				kind := logger.Debug
+				data := p.tracker.MsgData(js_lexer.RangeOfIdentifier(p.source, loc),
+					"Top-level \"this\" will be replaced with undefined since this file is an ECMAScript module")
+				data.Location.Suggestion = "undefined"
+				_, notes := p.whyESModule()
+				p.log.AddMsgID(logger.MsgID_JS_ThisIsUndefinedInESM, logger.Msg{Kind: kind, Data: data, Notes: notes})
+			}
+
+			// In an ES6 module, "this" is supposed to be undefined. Instead of
+			// doing this at runtime using "fn.call(undefined)", we do it at
+			// compile time using expression substitution here.
+			return js_ast.Expr{Loc: loc, Data: js_ast.EUndefinedShared}, true
+		} else if p.options.mode != config.ModePassThrough {
+			// In a CommonJS module, "this" is supposed to be the same as "exports".
+			// Instead of doing this at runtime using "fn.call(module.exports)", we
+			// do it at compile time using expression substitution here.
+			p.recordUsage(p.exportsRef)
+			return js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: p.exportsRef}}, true
+		}
+	}
+
+	return js_ast.Expr{}, false
+}
+
+func (p *parser) valueForImportMeta(loc logger.Loc) (js_ast.Expr, bool) {
+	if p.options.unsupportedJSFeatures.Has(compat.ImportMeta) ||
+		(p.options.mode != config.ModePassThrough && !p.options.outputFormat.KeepESMImportExportSyntax()) {
+		// Generate the variable if it doesn't exist yet
+		if p.importMetaRef == ast.InvalidRef {
+			p.importMetaRef = p.newSymbol(ast.SymbolOther, "import_meta")
+			p.moduleScope.Generated = append(p.moduleScope.Generated, p.importMetaRef)
+		}
+
+		// Replace "import.meta" with a reference to the symbol
+		p.recordUsage(p.importMetaRef)
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: p.importMetaRef}}, true
+	}
+
+	return js_ast.Expr{}, false
+}
+
+func locAfterOp(e *js_ast.EBinary) logger.Loc {
+	if e.Left.Loc.Start < e.Right.Loc.Start {
+		return e.Right.Loc
+	} else {
+		// Handle the case when we have transposed the operands
+		return e.Left.Loc
+	}
+}
+
+// This function exists to tie all of these checks together in one place
+func isEvalOrArguments(name string) bool {
+	return name == "eval" || name == "arguments"
+}
+
+func (p *parser) reportPrivateNameUsage(name string) {
+	if p.parseExperimentalDecoratorNesting > 0 {
+		if p.lowerAllOfThesePrivateNames == nil {
+			p.lowerAllOfThesePrivateNames = make(map[string]bool)
+		}
+		p.lowerAllOfThesePrivateNames[name] = true
+	}
+}
+
+func (p *parser) isValidAssignmentTarget(expr js_ast.Expr) bool {
+	switch e := expr.Data.(type) {
+	case *js_ast.EIdentifier:
+		if p.isStrictMode() {
+			if name := p.loadNameFromRef(e.Ref); isEvalOrArguments(name) {
+				return false
+			}
+		}
+		return true
+	case *js_ast.EDot:
+		return e.OptionalChain == js_ast.OptionalChainNone
+	case *js_ast.EIndex:
+		return e.OptionalChain == js_ast.OptionalChainNone
+
+	// Don't worry about recursive checking for objects and arrays. This will
+	// already be handled naturally by passing down the assign target flag.
+	case *js_ast.EObject:
+		return !e.IsParenthesized
+	case *js_ast.EArray:
+		return !e.IsParenthesized
+	}
+	return false
+}
+
+func containsClosingScriptTag(text string) bool {
+	for {
+		i := strings.Index(text, "</")
+		if i < 0 {
+			break
+		}
+		text = text[i+2:]
+		if len(text) >= 6 && strings.EqualFold(text[:6], "script") {
+			return true
+		}
+	}
+	return false
+}
+
+func (p *parser) isUnsupportedRegularExpression(loc logger.Loc, value string) (pattern string, flags string, isUnsupported bool) {
+	var what string
+	var r logger.Range
+
+	end := strings.LastIndexByte(value, '/')
+	pattern = value[1:end]
+	flags = value[end+1:]
+	isUnicode := strings.IndexByte(flags, 'u') >= 0
+	parenDepth := 0
+	i := 0
+
+	// Do a simple scan for unsupported features assuming the regular expression
+	// is valid. This doesn't do a full validation of the regular expression
+	// because regular expression grammar is complicated. If it contains a syntax
+	// error that we don't catch, then we will just generate output code with a
+	// syntax error. Garbage in, garbage out.
+pattern:
+	for i < len(pattern) {
+		c := pattern[i]
+		i++
+
+		switch c {
+		case '[':
+		class:
+			for i < len(pattern) {
+				c := pattern[i]
+				i++
+
+				switch c {
+				case ']':
+					break class
+
+				case '\\':
+					i++ // Skip the escaped character
+				}
+			}
+
+		case '(':
+			tail := pattern[i:]
+
+			if strings.HasPrefix(tail, "?<=") || strings.HasPrefix(tail, "?<!") {
+				if p.options.unsupportedJSFeatures.Has(compat.RegexpLookbehindAssertions) {
+					what = "Lookbehind assertions in regular expressions are not available"
+					r = logger.Range{Loc: logger.Loc{Start: loc.Start + int32(i) + 1}, Len: 3}
+					isUnsupported = true
+					break pattern
+				}
+			} else if strings.HasPrefix(tail, "?<") {
+				if p.options.unsupportedJSFeatures.Has(compat.RegexpNamedCaptureGroups) {
+					if end := strings.IndexByte(tail, '>'); end >= 0 {
+						what = "Named capture groups in regular expressions are not available"
+						r = logger.Range{Loc: logger.Loc{Start: loc.Start + int32(i) + 1}, Len: int32(end) + 1}
+						isUnsupported = true
+						break pattern
+					}
+				}
+			}
+
+			parenDepth++
+
+		case ')':
+			if parenDepth == 0 {
+				r := logger.Range{Loc: logger.Loc{Start: loc.Start + int32(i)}, Len: 1}
+				p.log.AddError(&p.tracker, r, "Unexpected \")\" in regular expression")
+				return
+			}
+
+			parenDepth--
+
+		case '\\':
+			tail := pattern[i:]
+
+			if isUnicode && (strings.HasPrefix(tail, "p{") || strings.HasPrefix(tail, "P{")) {
+				if p.options.unsupportedJSFeatures.Has(compat.RegexpUnicodePropertyEscapes) {
+					if end := strings.IndexByte(tail, '}'); end >= 0 {
+						what = "Unicode property escapes in regular expressions are not available"
+						r = logger.Range{Loc: logger.Loc{Start: loc.Start + int32(i)}, Len: int32(end) + 2}
+						isUnsupported = true
+						break pattern
+					}
+				}
+			}
+
+			i++ // Skip the escaped character
+		}
+	}
+
+	if !isUnsupported {
+		for i, c := range flags {
+			switch c {
+			case 'g', 'i', 'm':
+				continue // These are part of ES5 and are always supported
+
+			case 's':
+				if !p.options.unsupportedJSFeatures.Has(compat.RegexpDotAllFlag) {
+					continue // This is part of ES2018
+				}
+
+			case 'y', 'u':
+				if !p.options.unsupportedJSFeatures.Has(compat.RegexpStickyAndUnicodeFlags) {
+					continue // These are part of ES2018
+				}
+
+			case 'd':
+				if !p.options.unsupportedJSFeatures.Has(compat.RegexpMatchIndices) {
+					continue // This is part of ES2022
+				}
+
+			case 'v':
+				if !p.options.unsupportedJSFeatures.Has(compat.RegexpSetNotation) {
+					continue // This is from a proposal: https://github.com/tc39/proposal-regexp-v-flag
+				}
+
+			default:
+				// Unknown flags are never supported
+			}
+
+			r = logger.Range{Loc: logger.Loc{Start: loc.Start + int32(end+1) + int32(i)}, Len: 1}
+			what = fmt.Sprintf("The regular expression flag \"%c\" is not available", c)
+			isUnsupported = true
+			break
+		}
+	}
+
+	if isUnsupported {
+		where := config.PrettyPrintTargetEnvironment(p.options.originalTargetEnv, p.options.unsupportedJSFeatureOverridesMask)
+		p.log.AddIDWithNotes(logger.MsgID_JS_UnsupportedRegExp, logger.Debug, &p.tracker, r, fmt.Sprintf("%s in %s", what, where), []logger.MsgData{{
+			Text: "This regular expression literal has been converted to a \"new RegExp()\" constructor " +
+				"to avoid generating code with a syntax error. However, you will need to include a " +
+				"polyfill for \"RegExp\" for your code to have the correct behavior at run-time."}})
+	}
+
+	return
+}
+
+// This function takes "exprIn" as input from the caller and produces "exprOut"
+// for the caller to pass along extra data. This is mostly for optional chaining.
+func (p *parser) visitExprInOut(expr js_ast.Expr, in exprIn) (js_ast.Expr, exprOut) {
+	if in.assignTarget != js_ast.AssignTargetNone && !p.isValidAssignmentTarget(expr) {
+		p.log.AddError(&p.tracker, logger.Range{Loc: expr.Loc}, "Invalid assignment target")
+	}
+
+	// Note: Anything added before or after this switch statement will be bypassed
+	// when visiting nested "EBinary" nodes due to stack overflow mitigations for
+	// deeply-nested ASTs. If anything like that is added, care must be taken that
+	// it doesn't affect these mitigations by ensuring that the mitigations are not
+	// applied in those cases (e.g. by adding an additional conditional check).
+	switch e := expr.Data.(type) {
+	case *js_ast.ENull, *js_ast.ESuper, *js_ast.EBoolean, *js_ast.EBigInt, *js_ast.EUndefined, *js_ast.EJSXText:
+
+	case *js_ast.ENameOfSymbol:
+		e.Ref = p.symbolForMangledProp(p.loadNameFromRef(e.Ref))
+
+	case *js_ast.ERegExp:
+		// "/pattern/flags" => "new RegExp('pattern', 'flags')"
+		if pattern, flags, ok := p.isUnsupportedRegularExpression(expr.Loc, e.Value); ok {
+			args := []js_ast.Expr{{
+				Loc:  logger.Loc{Start: expr.Loc.Start + 1},
+				Data: &js_ast.EString{Value: helpers.StringToUTF16(pattern)},
+			}}
+			if flags != "" {
+				args = append(args, js_ast.Expr{
+					Loc:  logger.Loc{Start: expr.Loc.Start + int32(len(pattern)) + 2},
+					Data: &js_ast.EString{Value: helpers.StringToUTF16(flags)},
+				})
+			}
+			regExpRef := p.makeRegExpRef()
+			p.recordUsage(regExpRef)
+			return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENew{
+				Target:        js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EIdentifier{Ref: regExpRef}},
+				Args:          args,
+				CloseParenLoc: logger.Loc{Start: expr.Loc.Start + int32(len(e.Value))},
+			}}, exprOut{}
+		}
+
+	case *js_ast.ENewTarget:
+		if !p.fnOnlyDataVisit.isNewTargetAllowed {
+			p.log.AddError(&p.tracker, e.Range, "Cannot use \"new.target\" here:")
+		}
+
+	case *js_ast.EString:
+		if e.LegacyOctalLoc.Start > 0 {
+			if e.PreferTemplate {
+				p.log.AddError(&p.tracker, p.source.RangeOfLegacyOctalEscape(e.LegacyOctalLoc),
+					"Legacy octal escape sequences cannot be used in template literals")
+			} else if p.isStrictMode() {
+				p.markStrictModeFeature(legacyOctalEscape, p.source.RangeOfLegacyOctalEscape(e.LegacyOctalLoc), "")
+			}
+		}
+
+		if in.shouldMangleStringsAsProps && p.options.mangleQuoted && !e.PreferTemplate {
+			if name := helpers.UTF16ToString(e.Value); p.isMangledProp(name) {
+				return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENameOfSymbol{
+					Ref: p.symbolForMangledProp(name),
+				}}, exprOut{}
+			}
+		}
+
+	case *js_ast.ENumber:
+		if p.legacyOctalLiterals != nil && p.isStrictMode() {
+			if r, ok := p.legacyOctalLiterals[expr.Data]; ok {
+				p.markStrictModeFeature(legacyOctalLiteral, r, "")
+			}
+		}
+
+	case *js_ast.EThis:
+		isDeleteTarget := e == p.deleteTarget
+		isCallTarget := e == p.callTarget
+
+		if value, ok := p.valueForThis(expr.Loc, true /* shouldLog */, in.assignTarget, isDeleteTarget, isCallTarget); ok {
+			return value, exprOut{}
+		}
+
+		// Capture "this" inside arrow functions that will be lowered into normal
+		// function expressions for older language environments
+		if p.fnOrArrowDataVisit.isArrow && p.options.unsupportedJSFeatures.Has(compat.Arrow) && p.fnOnlyDataVisit.isThisNested {
+			return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EIdentifier{Ref: p.captureThis()}}, exprOut{}
+		}
+
+	case *js_ast.EImportMeta:
+		isDeleteTarget := e == p.deleteTarget
+		isCallTarget := e == p.callTarget
+
+		// Check both user-specified defines and known globals
+		if defines, ok := p.options.defines.DotDefines["meta"]; ok {
+			for _, define := range defines {
+				if p.isDotOrIndexDefineMatch(expr, define.Parts) {
+					// Substitute user-specified defines
+					if define.Data.DefineExpr != nil {
+						return p.instantiateDefineExpr(expr.Loc, *define.Data.DefineExpr, identifierOpts{
+							assignTarget:   in.assignTarget,
+							isCallTarget:   isCallTarget,
+							isDeleteTarget: isDeleteTarget,
+						}), exprOut{}
+					}
+				}
+			}
+		}
+
+		// Check injected dot names
+		if names, ok := p.injectedDotNames["meta"]; ok {
+			for _, name := range names {
+				if p.isDotOrIndexDefineMatch(expr, name.parts) {
+					// Note: We don't need to "ignoreRef" on the underlying identifier
+					// because we have only parsed it but not visited it yet
+					return p.instantiateInjectDotName(expr.Loc, name, in.assignTarget), exprOut{}
+				}
+			}
+		}
+
+		// Warn about "import.meta" if it's not replaced by a define
+		if p.options.unsupportedJSFeatures.Has(compat.ImportMeta) {
+			r := logger.Range{Loc: expr.Loc, Len: e.RangeLen}
+			p.markSyntaxFeature(compat.ImportMeta, r)
+		} else if p.options.mode != config.ModePassThrough && !p.options.outputFormat.KeepESMImportExportSyntax() {
+			r := logger.Range{Loc: expr.Loc, Len: e.RangeLen}
+			kind := logger.Warning
+			if p.suppressWarningsAboutWeirdCode || p.fnOrArrowDataVisit.tryBodyCount > 0 {
+				kind = logger.Debug
+			}
+			p.log.AddIDWithNotes(logger.MsgID_JS_EmptyImportMeta, kind, &p.tracker, r, fmt.Sprintf(
+				"\"import.meta\" is not available with the %q output format and will be empty", p.options.outputFormat.String()),
+				[]logger.MsgData{{Text: "You need to set the output format to \"esm\" for \"import.meta\" to work correctly."}})
+		}
+
+		// Convert "import.meta" to a variable if it's not supported in the output format
+		if importMeta, ok := p.valueForImportMeta(expr.Loc); ok {
+			return importMeta, exprOut{}
+		}
+
+	case *js_ast.ESpread:
+		e.Value = p.visitExpr(e.Value)
+
+	case *js_ast.EIdentifier:
+		isCallTarget := e == p.callTarget
+		isDeleteTarget := e == p.deleteTarget
+		name := p.loadNameFromRef(e.Ref)
+		if p.isStrictMode() && js_lexer.StrictModeReservedWords[name] {
+			p.markStrictModeFeature(reservedWord, js_lexer.RangeOfIdentifier(p.source, expr.Loc), name)
+		}
+		result := p.findSymbol(expr.Loc, name)
+		e.MustKeepDueToWithStmt = result.isInsideWithScope
+		e.Ref = result.ref
+
+		// Handle referencing a class name within that class's computed property
+		// key. This is not allowed, and must fail at run-time:
+		//
+		//   class Foo {
+		//     static foo = 'bar'
+		//     static [Foo.foo] = 'foo'
+		//   }
+		//
+		if p.symbols[result.ref.InnerIndex].Kind == ast.SymbolClassInComputedPropertyKey {
+			p.log.AddID(logger.MsgID_JS_ClassNameWillThrow, logger.Warning, &p.tracker, js_lexer.RangeOfIdentifier(p.source, expr.Loc),
+				fmt.Sprintf("Accessing class %q before initialization will throw", name))
+			return p.callRuntime(expr.Loc, "__earlyAccess", []js_ast.Expr{{Loc: expr.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16(name)}}}), exprOut{}
+		}
+
+		// Handle assigning to a constant
+		if in.assignTarget != js_ast.AssignTargetNone {
+			switch p.symbols[result.ref.InnerIndex].Kind {
+			case ast.SymbolConst:
+				r := js_lexer.RangeOfIdentifier(p.source, expr.Loc)
+				notes := []logger.MsgData{p.tracker.MsgData(js_lexer.RangeOfIdentifier(p.source, result.declareLoc),
+					fmt.Sprintf("The symbol %q was declared a constant here:", name))}
+
+				// Make this an error when bundling because we may need to convert this
+				// "const" into a "var" during bundling. Also make this an error when
+				// the constant is inlined because we will otherwise generate code with
+				// a syntax error.
+				if _, isInlinedConstant := p.constValues[result.ref]; isInlinedConstant || p.options.mode == config.ModeBundle ||
+					(p.currentScope.Parent == nil && p.willWrapModuleInTryCatchForUsing) {
+					p.log.AddErrorWithNotes(&p.tracker, r,
+						fmt.Sprintf("Cannot assign to %q because it is a constant", name), notes)
+				} else {
+					p.log.AddIDWithNotes(logger.MsgID_JS_AssignToConstant, logger.Warning, &p.tracker, r,
+						fmt.Sprintf("This assignment will throw because %q is a constant", name), notes)
+				}
+
+			case ast.SymbolInjected:
+				if where, ok := p.injectedSymbolSources[result.ref]; ok {
+					r := js_lexer.RangeOfIdentifier(p.source, expr.Loc)
+					tracker := logger.MakeLineColumnTracker(&where.source)
+					p.log.AddErrorWithNotes(&p.tracker, r,
+						fmt.Sprintf("Cannot assign to %q because it's an import from an injected file", name),
+						[]logger.MsgData{tracker.MsgData(js_lexer.RangeOfIdentifier(where.source, where.loc),
+							fmt.Sprintf("The symbol %q was exported from %q here:", name, where.source.PrettyPath))})
+				}
+			}
+		}
+
+		// Substitute user-specified defines for unbound or injected symbols
+		methodCallMustBeReplacedWithUndefined := false
+		if p.symbols[e.Ref.InnerIndex].Kind.IsUnboundOrInjected() && !result.isInsideWithScope && e != p.deleteTarget {
+			if data, ok := p.options.defines.IdentifierDefines[name]; ok {
+				if data.DefineExpr != nil {
+					new := p.instantiateDefineExpr(expr.Loc, *data.DefineExpr, identifierOpts{
+						assignTarget:   in.assignTarget,
+						isCallTarget:   isCallTarget,
+						isDeleteTarget: isDeleteTarget,
+					})
+					if in.assignTarget == js_ast.AssignTargetNone || defineValueCanBeUsedInAssignTarget(new.Data) {
+						p.ignoreUsage(e.Ref)
+						return new, exprOut{}
+					} else {
+						p.logAssignToDefine(js_lexer.RangeOfIdentifier(p.source, expr.Loc), name, js_ast.Expr{})
+					}
+				}
+
+				// Copy the side effect flags over in case this expression is unused
+				if data.Flags.Has(config.CanBeRemovedIfUnused) {
+					e.CanBeRemovedIfUnused = true
+				}
+				if data.Flags.Has(config.CallCanBeUnwrappedIfUnused) && !p.options.ignoreDCEAnnotations {
+					e.CallCanBeUnwrappedIfUnused = true
+				}
+				if data.Flags.Has(config.MethodCallsMustBeReplacedWithUndefined) {
+					methodCallMustBeReplacedWithUndefined = true
+				}
+			}
+		}
+
+		return p.handleIdentifier(expr.Loc, e, identifierOpts{
+				assignTarget:            in.assignTarget,
+				isCallTarget:            isCallTarget,
+				isDeleteTarget:          isDeleteTarget,
+				wasOriginallyIdentifier: true,
+			}), exprOut{
+				methodCallMustBeReplacedWithUndefined: methodCallMustBeReplacedWithUndefined,
+			}
+
+	case *js_ast.EJSXElement:
+		propsLoc := expr.Loc
+
+		// Resolving the location index to a specific line and column in
+		// development mode is not too expensive because we seek from the
+		// previous JSX element. It amounts to at most a single additional
+		// scan over the source code. Note that this has to happen before
+		// we visit anything about this JSX element to make sure that we
+		// only ever need to scan forward, not backward.
+		var jsxSourceLine int
+		var jsxSourceColumn int
+		if p.options.jsx.Development && p.options.jsx.AutomaticRuntime {
+			for p.jsxSourceLoc < int(propsLoc.Start) {
+				r, size := utf8.DecodeRuneInString(p.source.Contents[p.jsxSourceLoc:])
+				p.jsxSourceLoc += size
+				if r == '\n' || r == '\r' || r == '\u2028' || r == '\u2029' {
+					if r == '\r' && p.jsxSourceLoc < len(p.source.Contents) && p.source.Contents[p.jsxSourceLoc] == '\n' {
+						p.jsxSourceLoc++ // Handle Windows-style CRLF newlines
+					}
+					p.jsxSourceLine++
+					p.jsxSourceColumn = 0
+				} else {
+					// Babel and TypeScript count columns in UTF-16 code units
+					if r < 0xFFFF {
+						p.jsxSourceColumn++
+					} else {
+						p.jsxSourceColumn += 2
+					}
+				}
+			}
+			jsxSourceLine = p.jsxSourceLine
+			jsxSourceColumn = p.jsxSourceColumn
+		}
+
+		if e.TagOrNil.Data != nil {
+			propsLoc = e.TagOrNil.Loc
+			e.TagOrNil = p.visitExpr(e.TagOrNil)
+			p.warnAboutImportNamespaceCall(e.TagOrNil, exprKindJSXTag)
+		}
+
+		// Visit properties
+		hasSpread := false
+		for i, property := range e.Properties {
+			if property.Kind == js_ast.PropertySpread {
+				hasSpread = true
+			} else {
+				if mangled, ok := property.Key.Data.(*js_ast.ENameOfSymbol); ok {
+					mangled.Ref = p.symbolForMangledProp(p.loadNameFromRef(mangled.Ref))
+				} else {
+					property.Key = p.visitExpr(property.Key)
+				}
+			}
+			if property.ValueOrNil.Data != nil {
+				property.ValueOrNil = p.visitExpr(property.ValueOrNil)
+			}
+			if property.InitializerOrNil.Data != nil {
+				property.InitializerOrNil = p.visitExpr(property.InitializerOrNil)
+			}
+			e.Properties[i] = property
+		}
+
+		// "{a, ...{b, c}, d}" => "{a, b, c, d}"
+		if p.options.minifySyntax && hasSpread {
+			e.Properties = js_ast.MangleObjectSpread(e.Properties)
+		}
+
+		// Visit children
+		if len(e.NullableChildren) > 0 {
+			for i, childOrNil := range e.NullableChildren {
+				if childOrNil.Data != nil {
+					e.NullableChildren[i] = p.visitExpr(childOrNil)
+				}
+			}
+		}
+
+		if p.options.jsx.Preserve {
+			// If the tag is an identifier, mark it as needing to be upper-case
+			switch tag := e.TagOrNil.Data.(type) {
+			case *js_ast.EIdentifier:
+				p.symbols[tag.Ref.InnerIndex].Flags |= ast.MustStartWithCapitalLetterForJSX
+
+			case *js_ast.EImportIdentifier:
+				p.symbols[tag.Ref.InnerIndex].Flags |= ast.MustStartWithCapitalLetterForJSX
+			}
+		} else {
+			// Remove any nil children in the array (in place) before iterating over it
+			children := e.NullableChildren
+			{
+				end := 0
+				for _, childOrNil := range children {
+					if childOrNil.Data != nil {
+						children[end] = childOrNil
+						end++
+					}
+				}
+				children = children[:end]
+			}
+
+			// A missing tag is a fragment
+			if e.TagOrNil.Data == nil {
+				if p.options.jsx.AutomaticRuntime {
+					e.TagOrNil = p.importJSXSymbol(expr.Loc, JSXImportFragment)
+				} else {
+					e.TagOrNil = p.instantiateDefineExpr(expr.Loc, p.options.jsx.Fragment, identifierOpts{
+						wasOriginallyIdentifier: true,
+						matchAgainstDefines:     true, // Allow defines to rewrite the JSX fragment factory
+					})
+				}
+			}
+
+			shouldUseCreateElement := !p.options.jsx.AutomaticRuntime
+			if !shouldUseCreateElement {
+				// Even for runtime="automatic", <div {...props} key={key} /> is special cased to createElement
+				// See https://github.com/babel/babel/blob/e482c763466ba3f44cb9e3467583b78b7f030b4a/packages/babel-plugin-transform-react-jsx/src/create-plugin.ts#L352
+				seenPropsSpread := false
+				for _, property := range e.Properties {
+					if seenPropsSpread && property.Kind == js_ast.PropertyField {
+						if str, ok := property.Key.Data.(*js_ast.EString); ok && helpers.UTF16EqualsString(str.Value, "key") {
+							shouldUseCreateElement = true
+							break
+						}
+					} else if property.Kind == js_ast.PropertySpread {
+						seenPropsSpread = true
+					}
+				}
+			}
+
+			if shouldUseCreateElement {
+				// Arguments to createElement()
+				args := []js_ast.Expr{e.TagOrNil}
+				if len(e.Properties) > 0 {
+					args = append(args, p.lowerObjectSpread(propsLoc, &js_ast.EObject{
+						Properties:   e.Properties,
+						IsSingleLine: e.IsTagSingleLine,
+					}))
+				} else {
+					args = append(args, js_ast.Expr{Loc: propsLoc, Data: js_ast.ENullShared})
+				}
+				if len(children) > 0 {
+					args = append(args, children...)
+				}
+
+				// Call createElement()
+				var target js_ast.Expr
+				kind := js_ast.NormalCall
+				if p.options.jsx.AutomaticRuntime {
+					target = p.importJSXSymbol(expr.Loc, JSXImportCreateElement)
+				} else {
+					target = p.instantiateDefineExpr(expr.Loc, p.options.jsx.Factory, identifierOpts{
+						wasOriginallyIdentifier: true,
+						matchAgainstDefines:     true, // Allow defines to rewrite the JSX factory
+					})
+					if js_ast.IsPropertyAccess(target) {
+						kind = js_ast.TargetWasOriginallyPropertyAccess
+					}
+					p.warnAboutImportNamespaceCall(target, exprKindCall)
+				}
+				return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ECall{
+					Target:        target,
+					Args:          args,
+					CloseParenLoc: e.CloseLoc,
+					IsMultiLine:   !e.IsTagSingleLine,
+					Kind:          kind,
+
+					// Enable tree shaking
+					CanBeUnwrappedIfUnused: !p.options.ignoreDCEAnnotations && !p.options.jsx.SideEffects,
+				}}, exprOut{}
+			} else {
+				// Arguments to jsx()
+				args := []js_ast.Expr{e.TagOrNil}
+
+				// Props argument
+				properties := make([]js_ast.Property, 0, len(e.Properties)+1)
+
+				// For jsx(), "key" is passed in as a separate argument, so filter it out
+				// from the props here. Also, check for __source and __self, which might have
+				// been added by some upstream plugin. Their presence here would represent a
+				// configuration error.
+				hasKey := false
+				keyProperty := js_ast.Expr{Loc: expr.Loc, Data: js_ast.EUndefinedShared}
+				for _, property := range e.Properties {
+					if str, ok := property.Key.Data.(*js_ast.EString); ok {
+						propName := helpers.UTF16ToString(str.Value)
+						switch propName {
+						case "key":
+							if boolean, ok := property.ValueOrNil.Data.(*js_ast.EBoolean); ok && boolean.Value && property.Flags.Has(js_ast.PropertyWasShorthand) {
+								r := js_lexer.RangeOfIdentifier(p.source, property.Loc)
+								msg := logger.Msg{
+									Kind:  logger.Error,
+									Data:  p.tracker.MsgData(r, "Please provide an explicit value for \"key\":"),
+									Notes: []logger.MsgData{{Text: "Using \"key\" as a shorthand for \"key={true}\" is not allowed when using React's \"automatic\" JSX transform."}},
+								}
+								msg.Data.Location.Suggestion = "key={true}"
+								p.log.AddMsg(msg)
+							} else {
+								keyProperty = property.ValueOrNil
+								hasKey = true
+							}
+							continue
+
+						case "__source", "__self":
+							r := js_lexer.RangeOfIdentifier(p.source, property.Loc)
+							p.log.AddErrorWithNotes(&p.tracker, r,
+								fmt.Sprintf("Duplicate \"%s\" prop found:", propName),
+								[]logger.MsgData{{Text: "Both \"__source\" and \"__self\" are set automatically by esbuild when using React's \"automatic\" JSX transform. " +
+									"This duplicate prop may have come from a plugin."}})
+							continue
+						}
+					}
+					properties = append(properties, property)
+				}
+
+				isStaticChildren := len(children) > 1
+
+				// Children are passed in as an explicit prop
+				if len(children) > 0 {
+					childrenValue := children[0]
+
+					if len(children) > 1 {
+						childrenValue.Data = &js_ast.EArray{Items: children}
+					} else if _, ok := childrenValue.Data.(*js_ast.ESpread); ok {
+						// TypeScript considers spread children to be static, but Babel considers
+						// it to be an error ("Spread children are not supported in React.").
+						// We'll follow TypeScript's behavior here because spread children may be
+						// valid with non-React source runtimes.
+						childrenValue.Data = &js_ast.EArray{Items: []js_ast.Expr{childrenValue}}
+						isStaticChildren = true
+					}
+
+					properties = append(properties, js_ast.Property{
+						Key: js_ast.Expr{
+							Data: &js_ast.EString{Value: helpers.StringToUTF16("children")},
+							Loc:  childrenValue.Loc,
+						},
+						ValueOrNil: childrenValue,
+						Kind:       js_ast.PropertyField,
+						Loc:        childrenValue.Loc,
+					})
+				}
+
+				args = append(args, p.lowerObjectSpread(propsLoc, &js_ast.EObject{
+					Properties:   properties,
+					IsSingleLine: e.IsTagSingleLine,
+				}))
+
+				// "key"
+				if hasKey || p.options.jsx.Development {
+					args = append(args, keyProperty)
+				}
+
+				if p.options.jsx.Development {
+					// "isStaticChildren"
+					args = append(args, js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EBoolean{Value: isStaticChildren}})
+
+					// "__source"
+					args = append(args, js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EObject{
+						Properties: []js_ast.Property{
+							{
+								Kind:       js_ast.PropertyField,
+								Key:        js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16("fileName")}},
+								ValueOrNil: js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16(p.source.PrettyPath)}},
+							},
+							{
+								Kind:       js_ast.PropertyField,
+								Key:        js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16("lineNumber")}},
+								ValueOrNil: js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENumber{Value: float64(jsxSourceLine + 1)}}, // 1-based lines
+							},
+							{
+								Kind:       js_ast.PropertyField,
+								Key:        js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16("columnNumber")}},
+								ValueOrNil: js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENumber{Value: float64(jsxSourceColumn + 1)}}, // 1-based columns
+							},
+						},
+					}})
+
+					// "__self"
+					__self := js_ast.Expr{Loc: expr.Loc, Data: js_ast.EThisShared}
+					{
+						if p.fnOnlyDataVisit.shouldReplaceThisWithInnerClassNameRef {
+							// Substitute "this" if we're inside a static class context
+							p.recordUsage(*p.fnOnlyDataVisit.innerClassNameRef)
+							__self.Data = &js_ast.EIdentifier{Ref: *p.fnOnlyDataVisit.innerClassNameRef}
+						} else if !p.fnOnlyDataVisit.isThisNested && p.options.mode != config.ModePassThrough {
+							// Replace top-level "this" with "undefined" if there's an output format
+							__self.Data = js_ast.EUndefinedShared
+						} else if p.fnOrArrowDataVisit.isDerivedClassCtor {
+							// We can't use "this" here in case it comes before "super()"
+							__self.Data = js_ast.EUndefinedShared
+						}
+					}
+					if _, ok := __self.Data.(*js_ast.EUndefined); !ok {
+						// Omit "__self" entirely if it's undefined
+						args = append(args, __self)
+					}
+				}
+
+				jsx := JSXImportJSX
+				if isStaticChildren {
+					jsx = JSXImportJSXS
+				}
+
+				return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ECall{
+					Target:        p.importJSXSymbol(expr.Loc, jsx),
+					Args:          args,
+					CloseParenLoc: e.CloseLoc,
+					IsMultiLine:   !e.IsTagSingleLine,
+
+					// Enable tree shaking
+					CanBeUnwrappedIfUnused: !p.options.ignoreDCEAnnotations && !p.options.jsx.SideEffects,
+				}}, exprOut{}
+			}
+		}
+
+	case *js_ast.ETemplate:
+		if e.LegacyOctalLoc.Start > 0 {
+			p.log.AddError(&p.tracker, p.source.RangeOfLegacyOctalEscape(e.LegacyOctalLoc),
+				"Legacy octal escape sequences cannot be used in template literals")
+		}
+
+		var tagThisFunc func() js_ast.Expr
+		var tagWrapFunc func(js_ast.Expr) js_ast.Expr
+
+		if e.TagOrNil.Data != nil {
+			// Capture the value for "this" if the tag is a lowered optional chain.
+			// We'll need to manually apply this value later to preserve semantics.
+			tagIsLoweredOptionalChain := false
+			if p.options.unsupportedJSFeatures.Has(compat.OptionalChain) {
+				switch target := e.TagOrNil.Data.(type) {
+				case *js_ast.EDot:
+					tagIsLoweredOptionalChain = target.OptionalChain != js_ast.OptionalChainNone
+				case *js_ast.EIndex:
+					tagIsLoweredOptionalChain = target.OptionalChain != js_ast.OptionalChainNone
+				}
+			}
+
+			p.templateTag = e.TagOrNil.Data
+			tag, tagOut := p.visitExprInOut(e.TagOrNil, exprIn{storeThisArgForParentOptionalChain: tagIsLoweredOptionalChain})
+			e.TagOrNil = tag
+			tagThisFunc = tagOut.thisArgFunc
+			tagWrapFunc = tagOut.thisArgWrapFunc
+
+			// Copy the call side effect flag over if this is a known target
+			if id, ok := tag.Data.(*js_ast.EIdentifier); ok && p.symbols[id.Ref.InnerIndex].Flags.Has(ast.CallCanBeUnwrappedIfUnused) {
+				e.CanBeUnwrappedIfUnused = true
+			}
+
+			// The value of "this" must be manually preserved for private member
+			// accesses inside template tag expressions such as "this.#foo``".
+			// The private member "this.#foo" must see the value of "this".
+			if target, loc, private := p.extractPrivateIndex(e.TagOrNil); private != nil {
+				// "foo.#bar`123`" => "__privateGet(_a = foo, #bar).bind(_a)`123`"
+				targetFunc, targetWrapFunc := p.captureValueWithPossibleSideEffects(target.Loc, 2, target, valueCouldBeMutated)
+				e.TagOrNil = targetWrapFunc(js_ast.Expr{Loc: target.Loc, Data: &js_ast.ECall{
+					Target: js_ast.Expr{Loc: target.Loc, Data: &js_ast.EDot{
+						Target:  p.lowerPrivateGet(targetFunc(), loc, private),
+						Name:    "bind",
+						NameLoc: target.Loc,
+					}},
+					Args: []js_ast.Expr{targetFunc()},
+					Kind: js_ast.TargetWasOriginallyPropertyAccess,
+				}})
+			}
+		}
+
+		for i, part := range e.Parts {
+			e.Parts[i].Value = p.visitExpr(part.Value)
+		}
+
+		// When mangling, inline string values into the template literal. Note that
+		// it may no longer be a template literal after this point (it may turn into
+		// a plain string literal instead).
+		if p.shouldFoldTypeScriptConstantExpressions || p.options.minifySyntax {
+			expr = js_ast.InlinePrimitivesIntoTemplate(expr.Loc, e)
+		}
+
+		shouldLowerTemplateLiteral := p.options.unsupportedJSFeatures.Has(compat.TemplateLiteral)
+
+		// If the tag was originally an optional chaining property access, then
+		// we'll need to lower this template literal as well to preserve the value
+		// for "this".
+		if tagThisFunc != nil {
+			shouldLowerTemplateLiteral = true
+		}
+
+		// Lower tagged template literals that include "</script"
+		// since we won't be able to escape it without lowering it
+		if !shouldLowerTemplateLiteral && !p.options.unsupportedJSFeatures.Has(compat.InlineScript) && e.TagOrNil.Data != nil {
+			if containsClosingScriptTag(e.HeadRaw) {
+				shouldLowerTemplateLiteral = true
+			} else {
+				for _, part := range e.Parts {
+					if containsClosingScriptTag(part.TailRaw) {
+						shouldLowerTemplateLiteral = true
+						break
+					}
+				}
+			}
+		}
+
+		// Convert template literals to older syntax if this is still a template literal
+		if shouldLowerTemplateLiteral {
+			if e, ok := expr.Data.(*js_ast.ETemplate); ok {
+				return p.lowerTemplateLiteral(expr.Loc, e, tagThisFunc, tagWrapFunc), exprOut{}
+			}
+		}
+
+	case *js_ast.EBinary:
+		// The handling of binary expressions is convoluted because we're using
+		// iteration on the heap instead of recursion on the call stack to avoid
+		// stack overflow for deeply-nested ASTs. See the comment before the
+		// definition of "binaryExprVisitor" for details.
+		v := binaryExprVisitor{
+			e:   e,
+			loc: expr.Loc,
+			in:  in,
+		}
+
+		// Everything uses a single stack to reduce allocation overhead. This stack
+		// should almost always be very small, and almost all visits should reuse
+		// existing memory without allocating anything.
+		stackBottom := len(p.binaryExprStack)
+
+		// Iterate down into the AST along the left node of the binary operation.
+		// Continue iterating until we encounter something that's not a binary node.
+		for {
+			// Check whether this node is a special case. If it is, a result will be
+			// provided which ends our iteration. Otherwise, the visitor object will
+			// be prepared for visiting.
+			if result := v.checkAndPrepare(p); result.Data != nil {
+				expr = result
+				break
+			}
+
+			// Grab the arguments to our nested "visitExprInOut" call for the left
+			// node. We only care about deeply-nested left nodes because most binary
+			// operators in JavaScript are left-associative and the problematic edge
+			// cases we're trying to avoid crashing on have lots of left-associative
+			// binary operators chained together without parentheses (e.g. "1+2+...").
+			left := v.e.Left
+			leftIn := v.leftIn
+			leftBinary, ok := left.Data.(*js_ast.EBinary)
+
+			// Stop iterating if iteration doesn't apply to the left node. This checks
+			// the assignment target because "visitExprInOut" has additional behavior
+			// in that case that we don't want to miss (before the top-level "switch"
+			// statement).
+			if !ok || leftIn.assignTarget != js_ast.AssignTargetNone {
+				v.e.Left, _ = p.visitExprInOut(left, leftIn)
+				expr = v.visitRightAndFinish(p)
+				break
+			}
+
+			// Note that we only append to the stack (and therefore allocate memory
+			// on the heap) when there are nested binary expressions. A single binary
+			// expression doesn't add anything to the stack.
+			p.binaryExprStack = append(p.binaryExprStack, v)
+			v = binaryExprVisitor{
+				e:   leftBinary,
+				loc: left.Loc,
+				in:  leftIn,
+			}
+		}
+
+		// Process all binary operations from the deepest-visited node back toward
+		// our original top-level binary operation.
+		for {
+			n := len(p.binaryExprStack) - 1
+			if n < stackBottom {
+				break
+			}
+			v := p.binaryExprStack[n]
+			p.binaryExprStack = p.binaryExprStack[:n]
+			v.e.Left = expr
+			expr = v.visitRightAndFinish(p)
+		}
+
+		return expr, exprOut{}
+
+	case *js_ast.EDot:
+		isDeleteTarget := e == p.deleteTarget
+		isCallTarget := e == p.callTarget
+		isTemplateTag := e == p.templateTag
+
+		// Check both user-specified defines and known globals
+		if defines, ok := p.options.defines.DotDefines[e.Name]; ok {
+			for _, define := range defines {
+				if p.isDotOrIndexDefineMatch(expr, define.Parts) {
+					// Substitute user-specified defines
+					if define.Data.DefineExpr != nil {
+						new := p.instantiateDefineExpr(expr.Loc, *define.Data.DefineExpr, identifierOpts{
+							assignTarget:   in.assignTarget,
+							isCallTarget:   isCallTarget,
+							isDeleteTarget: isDeleteTarget,
+						})
+						if in.assignTarget == js_ast.AssignTargetNone || defineValueCanBeUsedInAssignTarget(new.Data) {
+							// Note: We don't need to "ignoreRef" on the underlying identifier
+							// because we have only parsed it but not visited it yet
+							return new, exprOut{}
+						} else {
+							r := logger.Range{Loc: expr.Loc, Len: js_lexer.RangeOfIdentifier(p.source, e.NameLoc).End() - expr.Loc.Start}
+							p.logAssignToDefine(r, "", expr)
+						}
+					}
+
+					// Copy the side effect flags over in case this expression is unused
+					if define.Data.Flags.Has(config.CanBeRemovedIfUnused) {
+						e.CanBeRemovedIfUnused = true
+					}
+					if define.Data.Flags.Has(config.CallCanBeUnwrappedIfUnused) && !p.options.ignoreDCEAnnotations {
+						e.CallCanBeUnwrappedIfUnused = true
+					}
+					if define.Data.Flags.Has(config.IsSymbolInstance) {
+						e.IsSymbolInstance = true
+					}
+					break
+				}
+			}
+		}
+
+		// Check injected dot names
+		if names, ok := p.injectedDotNames[e.Name]; ok {
+			for _, name := range names {
+				if p.isDotOrIndexDefineMatch(expr, name.parts) {
+					// Note: We don't need to "ignoreRef" on the underlying identifier
+					// because we have only parsed it but not visited it yet
+					return p.instantiateInjectDotName(expr.Loc, name, in.assignTarget), exprOut{}
+				}
+			}
+		}
+
+		// Track ".then().catch()" chains
+		if isCallTarget && p.thenCatchChain.nextTarget == e {
+			if e.Name == "catch" {
+				p.thenCatchChain = thenCatchChain{
+					nextTarget: e.Target.Data,
+					hasCatch:   true,
+					catchLoc:   e.NameLoc,
+				}
+			} else if e.Name == "then" {
+				p.thenCatchChain = thenCatchChain{
+					nextTarget: e.Target.Data,
+					hasCatch:   p.thenCatchChain.hasCatch || p.thenCatchChain.hasMultipleArgs,
+					catchLoc:   p.thenCatchChain.catchLoc,
+				}
+			}
+		}
+
+		p.dotOrIndexTarget = e.Target.Data
+		target, out := p.visitExprInOut(e.Target, exprIn{
+			hasChainParent: e.OptionalChain == js_ast.OptionalChainContinue,
+		})
+		e.Target = target
+
+		// Lower "super.prop" if necessary
+		if e.OptionalChain == js_ast.OptionalChainNone && in.assignTarget == js_ast.AssignTargetNone &&
+			!isCallTarget && p.shouldLowerSuperPropertyAccess(e.Target) {
+			// "super.foo" => "__superGet('foo')"
+			key := js_ast.Expr{Loc: e.NameLoc, Data: &js_ast.EString{Value: helpers.StringToUTF16(e.Name)}}
+			value := p.lowerSuperPropertyGet(expr.Loc, key)
+			if isTemplateTag {
+				value.Data = &js_ast.ECall{
+					Target: js_ast.Expr{Loc: value.Loc, Data: &js_ast.EDot{
+						Target:  value,
+						Name:    "bind",
+						NameLoc: value.Loc,
+					}},
+					Args: []js_ast.Expr{{Loc: value.Loc, Data: js_ast.EThisShared}},
+					Kind: js_ast.TargetWasOriginallyPropertyAccess,
+				}
+			}
+			return value, exprOut{}
+		}
+
+		// Lower optional chaining if we're the top of the chain
+		containsOptionalChain := e.OptionalChain == js_ast.OptionalChainStart ||
+			(e.OptionalChain == js_ast.OptionalChainContinue && out.childContainsOptionalChain)
+		if containsOptionalChain && !in.hasChainParent {
+			return p.lowerOptionalChain(expr, in, out)
+		}
+
+		// Potentially rewrite this property access
+		out = exprOut{
+			childContainsOptionalChain:            containsOptionalChain,
+			methodCallMustBeReplacedWithUndefined: out.methodCallMustBeReplacedWithUndefined,
+			thisArgFunc:                           out.thisArgFunc,
+			thisArgWrapFunc:                       out.thisArgWrapFunc,
+		}
+		if !in.hasChainParent {
+			out.thisArgFunc = nil
+			out.thisArgWrapFunc = nil
+		}
+		if e.OptionalChain == js_ast.OptionalChainNone {
+			if value, ok := p.maybeRewritePropertyAccess(expr.Loc, in.assignTarget,
+				isDeleteTarget, e.Target, e.Name, e.NameLoc, isCallTarget, isTemplateTag, false); ok {
+				return value, out
+			}
+		}
+		return js_ast.Expr{Loc: expr.Loc, Data: e}, out
+
+	case *js_ast.EIndex:
+		isCallTarget := e == p.callTarget
+		isTemplateTag := e == p.templateTag
+		isDeleteTarget := e == p.deleteTarget
+
+		// Check both user-specified defines and known globals
+		if str, ok := e.Index.Data.(*js_ast.EString); ok {
+			if defines, ok := p.options.defines.DotDefines[helpers.UTF16ToString(str.Value)]; ok {
+				for _, define := range defines {
+					if p.isDotOrIndexDefineMatch(expr, define.Parts) {
+						// Substitute user-specified defines
+						if define.Data.DefineExpr != nil {
+							new := p.instantiateDefineExpr(expr.Loc, *define.Data.DefineExpr, identifierOpts{
+								assignTarget:   in.assignTarget,
+								isCallTarget:   isCallTarget,
+								isDeleteTarget: isDeleteTarget,
+							})
+							if in.assignTarget == js_ast.AssignTargetNone || defineValueCanBeUsedInAssignTarget(new.Data) {
+								// Note: We don't need to "ignoreRef" on the underlying identifier
+								// because we have only parsed it but not visited it yet
+								return new, exprOut{}
+							} else {
+								r := logger.Range{Loc: expr.Loc}
+								afterIndex := logger.Loc{Start: p.source.RangeOfString(e.Index.Loc).End()}
+								if closeBracket := p.source.RangeOfOperatorAfter(afterIndex, "]"); closeBracket.Len > 0 {
+									r.Len = closeBracket.End() - r.Loc.Start
+								}
+								p.logAssignToDefine(r, "", expr)
+							}
+						}
+
+						// Copy the side effect flags over in case this expression is unused
+						if define.Data.Flags.Has(config.CanBeRemovedIfUnused) {
+							e.CanBeRemovedIfUnused = true
+						}
+						if define.Data.Flags.Has(config.CallCanBeUnwrappedIfUnused) && !p.options.ignoreDCEAnnotations {
+							e.CallCanBeUnwrappedIfUnused = true
+						}
+						if define.Data.Flags.Has(config.IsSymbolInstance) {
+							e.IsSymbolInstance = true
+						}
+						break
+					}
+				}
+			}
+		}
+
+		// "a['b']" => "a.b"
+		if p.options.minifySyntax {
+			if str, ok := e.Index.Data.(*js_ast.EString); ok && js_ast.IsIdentifierUTF16(str.Value) {
+				dot := p.dotOrMangledPropParse(e.Target, js_lexer.MaybeSubstring{String: helpers.UTF16ToString(str.Value)}, e.Index.Loc, e.OptionalChain, wasOriginallyIndex)
+				if isCallTarget {
+					p.callTarget = dot
+				}
+				if isTemplateTag {
+					p.templateTag = dot
+				}
+				if isDeleteTarget {
+					p.deleteTarget = dot
+				}
+				return p.visitExprInOut(js_ast.Expr{Loc: expr.Loc, Data: dot}, in)
+			}
+		}
+
+		p.dotOrIndexTarget = e.Target.Data
+		target, out := p.visitExprInOut(e.Target, exprIn{
+			hasChainParent: e.OptionalChain == js_ast.OptionalChainContinue,
+		})
+		e.Target = target
+
+		// Special-case private identifiers
+		if private, ok := e.Index.Data.(*js_ast.EPrivateIdentifier); ok {
+			name := p.loadNameFromRef(private.Ref)
+			result := p.findSymbol(e.Index.Loc, name)
+			private.Ref = result.ref
+
+			// Unlike regular identifiers, there are no unbound private identifiers
+			kind := p.symbols[result.ref.InnerIndex].Kind
+			if !kind.IsPrivate() {
+				r := logger.Range{Loc: e.Index.Loc, Len: int32(len(name))}
+				p.log.AddError(&p.tracker, r, fmt.Sprintf("Private name %q must be declared in an enclosing class", name))
+			} else {
+				var r logger.Range
+				var text string
+				if in.assignTarget != js_ast.AssignTargetNone && (kind == ast.SymbolPrivateMethod || kind == ast.SymbolPrivateStaticMethod) {
+					r = logger.Range{Loc: e.Index.Loc, Len: int32(len(name))}
+					text = fmt.Sprintf("Writing to read-only method %q will throw", name)
+				} else if in.assignTarget != js_ast.AssignTargetNone && (kind == ast.SymbolPrivateGet || kind == ast.SymbolPrivateStaticGet) {
+					r = logger.Range{Loc: e.Index.Loc, Len: int32(len(name))}
+					text = fmt.Sprintf("Writing to getter-only property %q will throw", name)
+				} else if in.assignTarget != js_ast.AssignTargetReplace && (kind == ast.SymbolPrivateSet || kind == ast.SymbolPrivateStaticSet) {
+					r = logger.Range{Loc: e.Index.Loc, Len: int32(len(name))}
+					text = fmt.Sprintf("Reading from setter-only property %q will throw", name)
+				}
+				if text != "" {
+					kind := logger.Warning
+					if p.suppressWarningsAboutWeirdCode {
+						kind = logger.Debug
+					}
+					p.log.AddID(logger.MsgID_JS_PrivateNameWillThrow, kind, &p.tracker, r, text)
+				}
+			}
+
+			// Lower private member access only if we're sure the target isn't needed
+			// for the value of "this" for a call expression. All other cases will be
+			// taken care of by the enclosing call expression.
+			if p.privateSymbolNeedsToBeLowered(private) && e.OptionalChain == js_ast.OptionalChainNone &&
+				in.assignTarget == js_ast.AssignTargetNone && !isCallTarget && !isTemplateTag {
+				// "foo.#bar" => "__privateGet(foo, #bar)"
+				return p.lowerPrivateGet(e.Target, e.Index.Loc, private), exprOut{}
+			}
+		} else {
+			e.Index, _ = p.visitExprInOut(e.Index, exprIn{
+				shouldMangleStringsAsProps: true,
+			})
+		}
+
+		// Lower "super[prop]" if necessary
+		if e.OptionalChain == js_ast.OptionalChainNone && in.assignTarget == js_ast.AssignTargetNone &&
+			!isCallTarget && p.shouldLowerSuperPropertyAccess(e.Target) {
+			// "super[foo]" => "__superGet(foo)"
+			value := p.lowerSuperPropertyGet(expr.Loc, e.Index)
+			if isTemplateTag {
+				value.Data = &js_ast.ECall{
+					Target: js_ast.Expr{Loc: value.Loc, Data: &js_ast.EDot{
+						Target:  value,
+						Name:    "bind",
+						NameLoc: value.Loc,
+					}},
+					Args: []js_ast.Expr{{Loc: value.Loc, Data: js_ast.EThisShared}},
+					Kind: js_ast.TargetWasOriginallyPropertyAccess,
+				}
+			}
+			return value, exprOut{}
+		}
+
+		// Lower optional chaining if we're the top of the chain
+		containsOptionalChain := e.OptionalChain == js_ast.OptionalChainStart ||
+			(e.OptionalChain == js_ast.OptionalChainContinue && out.childContainsOptionalChain)
+		if containsOptionalChain && !in.hasChainParent {
+			return p.lowerOptionalChain(expr, in, out)
+		}
+
+		// Potentially rewrite this property access
+		out = exprOut{
+			childContainsOptionalChain:            containsOptionalChain,
+			methodCallMustBeReplacedWithUndefined: out.methodCallMustBeReplacedWithUndefined,
+			thisArgFunc:                           out.thisArgFunc,
+			thisArgWrapFunc:                       out.thisArgWrapFunc,
+		}
+		if !in.hasChainParent {
+			out.thisArgFunc = nil
+			out.thisArgWrapFunc = nil
+		}
+		if str, ok := e.Index.Data.(*js_ast.EString); ok && e.OptionalChain == js_ast.OptionalChainNone {
+			preferQuotedKey := !p.options.minifySyntax
+			if value, ok := p.maybeRewritePropertyAccess(expr.Loc, in.assignTarget, isDeleteTarget,
+				e.Target, helpers.UTF16ToString(str.Value), e.Index.Loc, isCallTarget, isTemplateTag, preferQuotedKey); ok {
+				return value, out
+			}
+		}
+
+		// Create an error for assigning to an import namespace when bundling. Even
+		// though this is a run-time error, we make it a compile-time error when
+		// bundling because scope hoisting means these will no longer be run-time
+		// errors.
+		if p.options.mode == config.ModeBundle && (in.assignTarget != js_ast.AssignTargetNone || isDeleteTarget) {
+			if id, ok := e.Target.Data.(*js_ast.EIdentifier); ok && p.symbols[id.Ref.InnerIndex].Kind == ast.SymbolImport {
+				r := js_lexer.RangeOfIdentifier(p.source, e.Target.Loc)
+				p.log.AddErrorWithNotes(&p.tracker, r,
+					fmt.Sprintf("Cannot assign to property on import %q", p.symbols[id.Ref.InnerIndex].OriginalName),
+					[]logger.MsgData{{Text: "Imports are immutable in JavaScript. " +
+						"To modify the value of this import, you must export a setter function in the " +
+						"imported file and then import and call that function here instead."}})
+
+			}
+		}
+
+		if p.options.minifySyntax {
+			switch index := e.Index.Data.(type) {
+			case *js_ast.EString:
+				// "a['x' + 'y']" => "a.xy" (this is done late to allow for constant folding)
+				if js_ast.IsIdentifierUTF16(index.Value) {
+					return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EDot{
+						Target:                     e.Target,
+						Name:                       helpers.UTF16ToString(index.Value),
+						NameLoc:                    e.Index.Loc,
+						OptionalChain:              e.OptionalChain,
+						CanBeRemovedIfUnused:       e.CanBeRemovedIfUnused,
+						CallCanBeUnwrappedIfUnused: e.CallCanBeUnwrappedIfUnused,
+					}}, out
+				}
+
+				// "a['123']" => "a[123]" (this is done late to allow "'123'" to be mangled)
+				if numberValue, ok := js_ast.StringToEquivalentNumberValue(index.Value); ok {
+					e.Index.Data = &js_ast.ENumber{Value: numberValue}
+				}
+
+			case *js_ast.ENumber:
+				// "'abc'[1]" => "'b'"
+				if target, ok := e.Target.Data.(*js_ast.EString); ok {
+					if intValue := math.Floor(index.Value); index.Value == intValue && intValue >= 0 && intValue < float64(len(target.Value)) {
+						return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EString{Value: []uint16{target.Value[int(intValue)]}}}, out
+					}
+				}
+			}
+		}
+
+		return js_ast.Expr{Loc: expr.Loc, Data: e}, out
+
+	case *js_ast.EUnary:
+		switch e.Op {
+		case js_ast.UnOpTypeof:
+			e.Value, _ = p.visitExprInOut(e.Value, exprIn{assignTarget: e.Op.UnaryAssignTarget()})
+
+			// Compile-time "typeof" evaluation
+			if typeof, ok := js_ast.TypeofWithoutSideEffects(e.Value.Data); ok {
+				return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16(typeof)}}, exprOut{}
+			}
+
+		case js_ast.UnOpDelete:
+			// Warn about code that tries to do "delete super.foo"
+			var superPropLoc logger.Loc
+			switch e2 := e.Value.Data.(type) {
+			case *js_ast.EDot:
+				if _, ok := e2.Target.Data.(*js_ast.ESuper); ok {
+					superPropLoc = e2.Target.Loc
+				}
+			case *js_ast.EIndex:
+				if _, ok := e2.Target.Data.(*js_ast.ESuper); ok {
+					superPropLoc = e2.Target.Loc
+				}
+			case *js_ast.EIdentifier:
+				p.markStrictModeFeature(deleteBareName, js_lexer.RangeOfIdentifier(p.source, e.Value.Loc), "")
+			}
+			if superPropLoc.Start != 0 {
+				r := js_lexer.RangeOfIdentifier(p.source, superPropLoc)
+				text := "Attempting to delete a property of \"super\" will throw a ReferenceError"
+				kind := logger.Warning
+				if p.suppressWarningsAboutWeirdCode {
+					kind = logger.Debug
+				}
+				p.log.AddID(logger.MsgID_JS_DeleteSuperProperty, kind, &p.tracker, r, text)
+			}
+
+			p.deleteTarget = e.Value.Data
+			value, out := p.visitExprInOut(e.Value, exprIn{hasChainParent: true})
+			e.Value = value
+
+			// Lower optional chaining if present since we're guaranteed to be the
+			// end of the chain
+			if out.childContainsOptionalChain {
+				return p.lowerOptionalChain(expr, in, out)
+			}
+
+		default:
+			e.Value, _ = p.visitExprInOut(e.Value, exprIn{assignTarget: e.Op.UnaryAssignTarget()})
+
+			// Post-process the unary expression
+			switch e.Op {
+			case js_ast.UnOpNot:
+				if p.options.minifySyntax {
+					e.Value = p.astHelpers.SimplifyBooleanExpr(e.Value)
+				}
+
+				if boolean, sideEffects, ok := js_ast.ToBooleanWithSideEffects(e.Value.Data); ok && sideEffects == js_ast.NoSideEffects {
+					return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EBoolean{Value: !boolean}}, exprOut{}
+				}
+
+				if p.options.minifySyntax {
+					if result, ok := js_ast.MaybeSimplifyNot(e.Value); ok {
+						return result, exprOut{}
+					}
+				}
+
+			case js_ast.UnOpVoid:
+				if p.astHelpers.ExprCanBeRemovedIfUnused(e.Value) {
+					return js_ast.Expr{Loc: expr.Loc, Data: js_ast.EUndefinedShared}, exprOut{}
+				}
+
+			case js_ast.UnOpPos:
+				if number, ok := js_ast.ToNumberWithoutSideEffects(e.Value.Data); ok {
+					return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENumber{Value: number}}, exprOut{}
+				}
+
+			case js_ast.UnOpNeg:
+				if number, ok := js_ast.ToNumberWithoutSideEffects(e.Value.Data); ok {
+					return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENumber{Value: -number}}, exprOut{}
+				}
+
+			case js_ast.UnOpCpl:
+				if p.shouldFoldTypeScriptConstantExpressions || p.options.minifySyntax {
+					// Minification folds complement operations since they are unlikely to result in larger output
+					if number, ok := js_ast.ToNumberWithoutSideEffects(e.Value.Data); ok {
+						return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENumber{Value: float64(^js_ast.ToInt32(number))}}, exprOut{}
+					}
+				}
+
+				////////////////////////////////////////////////////////////////////////////////
+				// All assignment operators below here
+
+			case js_ast.UnOpPreDec, js_ast.UnOpPreInc, js_ast.UnOpPostDec, js_ast.UnOpPostInc:
+				if target, loc, private := p.extractPrivateIndex(e.Value); private != nil {
+					return p.lowerPrivateSetUnOp(target, loc, private, e.Op), exprOut{}
+				}
+				if property := p.extractSuperProperty(e.Value); property.Data != nil {
+					e.Value = p.callSuperPropertyWrapper(expr.Loc, property)
+				}
+			}
+		}
+
+		// "-(a, b)" => "a, -b"
+		if p.options.minifySyntax && e.Op != js_ast.UnOpDelete && e.Op != js_ast.UnOpTypeof {
+			if comma, ok := e.Value.Data.(*js_ast.EBinary); ok && comma.Op == js_ast.BinOpComma {
+				return js_ast.JoinWithComma(comma.Left, js_ast.Expr{
+					Loc: comma.Right.Loc,
+					Data: &js_ast.EUnary{
+						Op:    e.Op,
+						Value: comma.Right,
+					},
+				}), exprOut{}
+			}
+		}
+
+	case *js_ast.EIf:
+		e.Test = p.visitExpr(e.Test)
+
+		if p.options.minifySyntax {
+			e.Test = p.astHelpers.SimplifyBooleanExpr(e.Test)
+		}
+
+		// Propagate these flags into the branches
+		childIn := exprIn{
+			shouldMangleStringsAsProps: in.shouldMangleStringsAsProps,
+		}
+
+		// Fold constants
+		if boolean, sideEffects, ok := js_ast.ToBooleanWithSideEffects(e.Test.Data); !ok {
+			e.Yes, _ = p.visitExprInOut(e.Yes, childIn)
+			e.No, _ = p.visitExprInOut(e.No, childIn)
+		} else {
+			// Mark the control flow as dead if the branch is never taken
+			if boolean {
+				// "true ? live : dead"
+				e.Yes, _ = p.visitExprInOut(e.Yes, childIn)
+				old := p.isControlFlowDead
+				p.isControlFlowDead = true
+				e.No, _ = p.visitExprInOut(e.No, childIn)
+				p.isControlFlowDead = old
+
+				if p.options.minifySyntax {
+					// "(a, true) ? b : c" => "a, b"
+					if sideEffects == js_ast.CouldHaveSideEffects {
+						return js_ast.JoinWithComma(p.astHelpers.SimplifyUnusedExpr(e.Test, p.options.unsupportedJSFeatures), e.Yes), exprOut{}
+					}
+
+					return e.Yes, exprOut{}
+				}
+			} else {
+				// "false ? dead : live"
+				old := p.isControlFlowDead
+				p.isControlFlowDead = true
+				e.Yes, _ = p.visitExprInOut(e.Yes, childIn)
+				p.isControlFlowDead = old
+				e.No, _ = p.visitExprInOut(e.No, childIn)
+
+				if p.options.minifySyntax {
+					// "(a, false) ? b : c" => "a, c"
+					if sideEffects == js_ast.CouldHaveSideEffects {
+						return js_ast.JoinWithComma(p.astHelpers.SimplifyUnusedExpr(e.Test, p.options.unsupportedJSFeatures), e.No), exprOut{}
+					}
+
+					return e.No, exprOut{}
+				}
+			}
+		}
+
+		if p.options.minifySyntax {
+			return p.astHelpers.MangleIfExpr(expr.Loc, e, p.options.unsupportedJSFeatures), exprOut{}
+		}
+
+	case *js_ast.EAwait:
+		// Silently remove unsupported top-level "await" in dead code branches
+		if p.fnOrArrowDataVisit.isOutsideFnOrArrow {
+			if p.isControlFlowDead && (p.options.unsupportedJSFeatures.Has(compat.TopLevelAwait) || !p.options.outputFormat.KeepESMImportExportSyntax()) {
+				return p.visitExprInOut(e.Value, in)
+			} else {
+				p.liveTopLevelAwaitKeyword = logger.Range{Loc: expr.Loc, Len: 5}
+				p.markSyntaxFeature(compat.TopLevelAwait, logger.Range{Loc: expr.Loc, Len: 5})
+			}
+		}
+
+		p.awaitTarget = e.Value.Data
+		e.Value = p.visitExpr(e.Value)
+
+		// "await" expressions turn into "yield" expressions when lowering
+		return p.maybeLowerAwait(expr.Loc, e), exprOut{}
+
+	case *js_ast.EYield:
+		if e.ValueOrNil.Data != nil {
+			e.ValueOrNil = p.visitExpr(e.ValueOrNil)
+		}
+
+		// "yield* x" turns into "yield* __yieldStar(x)" when lowering async generator functions
+		if e.IsStar && p.options.unsupportedJSFeatures.Has(compat.AsyncGenerator) && p.fnOrArrowDataVisit.isGenerator {
+			e.ValueOrNil = p.callRuntime(expr.Loc, "__yieldStar", []js_ast.Expr{e.ValueOrNil})
+		}
+
+	case *js_ast.EArray:
+		if in.assignTarget != js_ast.AssignTargetNone {
+			if e.CommaAfterSpread.Start != 0 {
+				p.log.AddError(&p.tracker, logger.Range{Loc: e.CommaAfterSpread, Len: 1}, "Unexpected \",\" after rest pattern")
+			}
+			p.markSyntaxFeature(compat.Destructuring, logger.Range{Loc: expr.Loc, Len: 1})
+		}
+		hasSpread := false
+		for i, item := range e.Items {
+			switch e2 := item.Data.(type) {
+			case *js_ast.EMissing:
+			case *js_ast.ESpread:
+				e2.Value, _ = p.visitExprInOut(e2.Value, exprIn{assignTarget: in.assignTarget})
+				hasSpread = true
+			case *js_ast.EBinary:
+				if in.assignTarget != js_ast.AssignTargetNone && e2.Op == js_ast.BinOpAssign {
+					e2.Left, _ = p.visitExprInOut(e2.Left, exprIn{assignTarget: js_ast.AssignTargetReplace})
+
+					// Propagate the name to keep from the binding into the initializer
+					if id, ok := e2.Left.Data.(*js_ast.EIdentifier); ok {
+						p.nameToKeep = p.symbols[id.Ref.InnerIndex].OriginalName
+						p.nameToKeepIsFor = e2.Right.Data
+					}
+
+					e2.Right = p.visitExpr(e2.Right)
+				} else {
+					item, _ = p.visitExprInOut(item, exprIn{assignTarget: in.assignTarget})
+				}
+			default:
+				item, _ = p.visitExprInOut(item, exprIn{assignTarget: in.assignTarget})
+			}
+			e.Items[i] = item
+		}
+
+		// "[1, ...[2, 3], 4]" => "[1, 2, 3, 4]"
+		if p.options.minifySyntax && hasSpread && in.assignTarget == js_ast.AssignTargetNone {
+			e.Items = js_ast.InlineSpreadsOfArrayLiterals(e.Items)
+		}
+
+	case *js_ast.EObject:
+		if in.assignTarget != js_ast.AssignTargetNone {
+			if e.CommaAfterSpread.Start != 0 {
+				p.log.AddError(&p.tracker, logger.Range{Loc: e.CommaAfterSpread, Len: 1}, "Unexpected \",\" after rest pattern")
+			}
+			p.markSyntaxFeature(compat.Destructuring, logger.Range{Loc: expr.Loc, Len: 1})
+		}
+
+		hasSpread := false
+		protoRange := logger.Range{}
+		innerClassNameRef := ast.InvalidRef
+
+		for i := range e.Properties {
+			property := &e.Properties[i]
+
+			if property.Kind != js_ast.PropertySpread {
+				key := property.Key
+				if mangled, ok := key.Data.(*js_ast.ENameOfSymbol); ok {
+					mangled.Ref = p.symbolForMangledProp(p.loadNameFromRef(mangled.Ref))
+				} else {
+					key, _ = p.visitExprInOut(property.Key, exprIn{
+						shouldMangleStringsAsProps: true,
+					})
+					property.Key = key
+				}
+
+				// Forbid duplicate "__proto__" properties according to the specification
+				if !property.Flags.Has(js_ast.PropertyIsComputed) && !property.Flags.Has(js_ast.PropertyWasShorthand) &&
+					property.Kind == js_ast.PropertyField && in.assignTarget == js_ast.AssignTargetNone {
+					if str, ok := key.Data.(*js_ast.EString); ok && helpers.UTF16EqualsString(str.Value, "__proto__") {
+						r := js_lexer.RangeOfIdentifier(p.source, key.Loc)
+						if protoRange.Len > 0 {
+							p.log.AddErrorWithNotes(&p.tracker, r,
+								"Cannot specify the \"__proto__\" property more than once per object",
+								[]logger.MsgData{p.tracker.MsgData(protoRange, "The earlier \"__proto__\" property is here:")})
+						} else {
+							protoRange = r
+						}
+					}
+				}
+
+				// "{['x']: y}" => "{x: y}"
+				if p.options.minifySyntax && property.Flags.Has(js_ast.PropertyIsComputed) {
+					if inlined, ok := key.Data.(*js_ast.EInlinedEnum); ok {
+						switch inlined.Value.Data.(type) {
+						case *js_ast.EString, *js_ast.ENumber:
+							key.Data = inlined.Value.Data
+							property.Key.Data = key.Data
+						}
+					}
+					switch k := key.Data.(type) {
+					case *js_ast.ENumber, *js_ast.ENameOfSymbol:
+						property.Flags &= ^js_ast.PropertyIsComputed
+					case *js_ast.EString:
+						if !helpers.UTF16EqualsString(k.Value, "__proto__") {
+							property.Flags &= ^js_ast.PropertyIsComputed
+						}
+					}
+				}
+			} else {
+				hasSpread = true
+			}
+
+			// Extract the initializer for expressions like "({ a: b = c } = d)"
+			if in.assignTarget != js_ast.AssignTargetNone && property.InitializerOrNil.Data == nil && property.ValueOrNil.Data != nil {
+				if binary, ok := property.ValueOrNil.Data.(*js_ast.EBinary); ok && binary.Op == js_ast.BinOpAssign {
+					property.InitializerOrNil = binary.Right
+					property.ValueOrNil = binary.Left
+				}
+			}
+
+			if property.ValueOrNil.Data != nil {
+				oldIsInStaticClassContext := p.fnOnlyDataVisit.isInStaticClassContext
+				oldInnerClassNameRef := p.fnOnlyDataVisit.innerClassNameRef
+
+				// If this is an async method and async methods are unsupported,
+				// generate a temporary variable in case this async method contains a
+				// "super" property reference. If that happens, the "super" expression
+				// must be lowered which will need a reference to this object literal.
+				if property.Kind == js_ast.PropertyMethod && p.options.unsupportedJSFeatures.Has(compat.AsyncAwait) {
+					if fn, ok := property.ValueOrNil.Data.(*js_ast.EFunction); ok && fn.Fn.IsAsync {
+						if innerClassNameRef == ast.InvalidRef {
+							innerClassNameRef = p.generateTempRef(tempRefNeedsDeclareMayBeCapturedInsideLoop, "")
+						}
+						p.fnOnlyDataVisit.isInStaticClassContext = true
+						p.fnOnlyDataVisit.innerClassNameRef = &innerClassNameRef
+					}
+				}
+
+				// Propagate the name to keep from the property into the value
+				if str, ok := property.Key.Data.(*js_ast.EString); ok {
+					p.nameToKeep = helpers.UTF16ToString(str.Value)
+					p.nameToKeepIsFor = property.ValueOrNil.Data
+				}
+
+				property.ValueOrNil, _ = p.visitExprInOut(property.ValueOrNil, exprIn{
+					isMethod:     property.Kind.IsMethodDefinition(),
+					assignTarget: in.assignTarget,
+				})
+
+				p.fnOnlyDataVisit.innerClassNameRef = oldInnerClassNameRef
+				p.fnOnlyDataVisit.isInStaticClassContext = oldIsInStaticClassContext
+			}
+
+			if property.InitializerOrNil.Data != nil {
+				// Propagate the name to keep from the binding into the initializer
+				if id, ok := property.ValueOrNil.Data.(*js_ast.EIdentifier); ok {
+					p.nameToKeep = p.symbols[id.Ref.InnerIndex].OriginalName
+					p.nameToKeepIsFor = property.InitializerOrNil.Data
+				}
+
+				property.InitializerOrNil = p.visitExpr(property.InitializerOrNil)
+			}
+
+			// "{ '123': 4 }" => "{ 123: 4 }" (this is done late to allow "'123'" to be mangled)
+			if p.options.minifySyntax {
+				if str, ok := property.Key.Data.(*js_ast.EString); ok {
+					if numberValue, ok := js_ast.StringToEquivalentNumberValue(str.Value); ok && numberValue >= 0 {
+						property.Key.Data = &js_ast.ENumber{Value: numberValue}
+					}
+				}
+			}
+		}
+
+		// Check for and warn about duplicate keys in object literals
+		if !p.suppressWarningsAboutWeirdCode {
+			p.warnAboutDuplicateProperties(e.Properties, duplicatePropertiesInObject)
+		}
+
+		if in.assignTarget == js_ast.AssignTargetNone {
+			// "{a, ...{b, c}, d}" => "{a, b, c, d}"
+			if p.options.minifySyntax && hasSpread {
+				e.Properties = js_ast.MangleObjectSpread(e.Properties)
+			}
+
+			// Object expressions represent both object literals and binding patterns.
+			// Only lower object spread if we're an object literal, not a binding pattern.
+			value := p.lowerObjectSpread(expr.Loc, e)
+
+			// If we generated and used the temporary variable for a lowered "super"
+			// property reference inside a lowered "async" method, then initialize
+			// the temporary with this object literal.
+			if innerClassNameRef != ast.InvalidRef && p.symbols[innerClassNameRef.InnerIndex].UseCountEstimate > 0 {
+				p.recordUsage(innerClassNameRef)
+				value = js_ast.Assign(js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EIdentifier{Ref: innerClassNameRef}}, value)
+			}
+
+			return value, exprOut{}
+		}
+
+	case *js_ast.EImportCall:
+		isAwaitTarget := e == p.awaitTarget
+		isThenCatchTarget := e == p.thenCatchChain.nextTarget && p.thenCatchChain.hasCatch
+		e.Expr = p.visitExpr(e.Expr)
+
+		var assertOrWith *ast.ImportAssertOrWith
+		var flags ast.ImportRecordFlags
+		if e.OptionsOrNil.Data != nil {
+			e.OptionsOrNil = p.visitExpr(e.OptionsOrNil)
+
+			// If there's an additional argument, this can't be split because the
+			// additional argument requires evaluation and our AST nodes can't be
+			// reused in different places in the AST (e.g. function scopes must be
+			// unique). Also the additional argument may have side effects and we
+			// don't currently account for that.
+			why := "the second argument was not an object literal"
+			whyLoc := e.OptionsOrNil.Loc
+
+			// However, make a special case for an additional argument that contains
+			// only an "assert" or a "with" clause. In that case we can split this
+			// AST node.
+			if object, ok := e.OptionsOrNil.Data.(*js_ast.EObject); ok {
+				if len(object.Properties) == 1 {
+					if prop := object.Properties[0]; prop.Kind == js_ast.PropertyField && !prop.Flags.Has(js_ast.PropertyIsComputed) {
+						if str, ok := prop.Key.Data.(*js_ast.EString); ok && (helpers.UTF16EqualsString(str.Value, "assert") || helpers.UTF16EqualsString(str.Value, "with")) {
+							keyword := ast.WithKeyword
+							if helpers.UTF16EqualsString(str.Value, "assert") {
+								keyword = ast.AssertKeyword
+							}
+							if value, ok := prop.ValueOrNil.Data.(*js_ast.EObject); ok {
+								entries := []ast.AssertOrWithEntry{}
+								for _, p := range value.Properties {
+									if p.Kind == js_ast.PropertyField && !p.Flags.Has(js_ast.PropertyIsComputed) {
+										if key, ok := p.Key.Data.(*js_ast.EString); ok {
+											if value, ok := p.ValueOrNil.Data.(*js_ast.EString); ok {
+												entries = append(entries, ast.AssertOrWithEntry{
+													Key:             key.Value,
+													KeyLoc:          p.Key.Loc,
+													Value:           value.Value,
+													ValueLoc:        p.ValueOrNil.Loc,
+													PreferQuotedKey: p.Flags.Has(js_ast.PropertyPreferQuotedKey),
+												})
+												if keyword == ast.AssertKeyword && helpers.UTF16EqualsString(key.Value, "type") && helpers.UTF16EqualsString(value.Value, "json") {
+													flags |= ast.AssertTypeJSON
+												}
+												continue
+											} else {
+												why = fmt.Sprintf("the value for the property %q was not a string literal",
+													helpers.UTF16ToString(key.Value))
+												whyLoc = p.ValueOrNil.Loc
+											}
+										} else {
+											why = "this property was not a string literal"
+											whyLoc = p.Key.Loc
+										}
+									} else {
+										why = "this property was invalid"
+										whyLoc = p.Key.Loc
+									}
+									entries = nil
+									break
+								}
+								if entries != nil {
+									if keyword == ast.AssertKeyword {
+										p.maybeWarnAboutAssertKeyword(prop.Key.Loc)
+									}
+									assertOrWith = &ast.ImportAssertOrWith{
+										Entries:            entries,
+										Keyword:            keyword,
+										KeywordLoc:         prop.Key.Loc,
+										InnerOpenBraceLoc:  prop.ValueOrNil.Loc,
+										InnerCloseBraceLoc: value.CloseBraceLoc,
+										OuterOpenBraceLoc:  e.OptionsOrNil.Loc,
+										OuterCloseBraceLoc: object.CloseBraceLoc,
+									}
+									why = ""
+								}
+							} else {
+								why = "the value for \"assert\" was not an object literal"
+								whyLoc = prop.ValueOrNil.Loc
+							}
+						} else {
+							why = "this property was not called \"assert\" or \"with\""
+							whyLoc = prop.Key.Loc
+						}
+					} else {
+						why = "this property was invalid"
+						whyLoc = prop.Key.Loc
+					}
+				} else {
+					why = "the second argument was not an object literal with a single property called \"assert\" or \"with\""
+					whyLoc = e.OptionsOrNil.Loc
+				}
+			}
+
+			// Handle the case that isn't just an import assertion or attribute clause
+			if why != "" {
+				// Only warn when bundling
+				if p.options.mode == config.ModeBundle {
+					text := "This \"import()\" was not recognized because " + why
+					kind := logger.Warning
+					if p.suppressWarningsAboutWeirdCode {
+						kind = logger.Debug
+					}
+					p.log.AddID(logger.MsgID_JS_UnsupportedDynamicImport, kind, &p.tracker, logger.Range{Loc: whyLoc}, text)
+				}
+
+				// If import assertions and/attributes are both not supported in the
+				// target platform, then "import()" cannot accept a second argument
+				// and keeping them would be a syntax error, so we need to get rid of
+				// them. We can't just not print them because they may have important
+				// side effects. Attempt to discard them without changing side effects
+				// and generate an error if that isn't possible.
+				if p.options.unsupportedJSFeatures.Has(compat.ImportAssertions) && p.options.unsupportedJSFeatures.Has(compat.ImportAttributes) {
+					if p.astHelpers.ExprCanBeRemovedIfUnused(e.OptionsOrNil) {
+						e.OptionsOrNil = js_ast.Expr{}
+					} else {
+						p.markSyntaxFeature(compat.ImportAttributes, logger.Range{Loc: e.OptionsOrNil.Loc})
+					}
+				}
+
+				// Stop now so we don't try to split "?:" expressions below and
+				// potentially end up with an AST node reused multiple times
+				break
+			}
+		}
+
+		return p.maybeTransposeIfExprChain(e.Expr, func(arg js_ast.Expr) js_ast.Expr {
+			// The argument must be a string
+			if str, ok := arg.Data.(*js_ast.EString); ok {
+				// Ignore calls to import() if the control flow is provably dead here.
+				// We don't want to spend time scanning the required files if they will
+				// never be used.
+				if p.isControlFlowDead {
+					return js_ast.Expr{Loc: arg.Loc, Data: js_ast.ENullShared}
+				}
+
+				importRecordIndex := p.addImportRecord(ast.ImportDynamic, p.source.RangeOfString(arg.Loc), helpers.UTF16ToString(str.Value), assertOrWith, flags)
+				if isAwaitTarget && p.fnOrArrowDataVisit.tryBodyCount != 0 {
+					record := &p.importRecords[importRecordIndex]
+					record.Flags |= ast.HandlesImportErrors
+					record.ErrorHandlerLoc = p.fnOrArrowDataVisit.tryCatchLoc
+				} else if isThenCatchTarget {
+					record := &p.importRecords[importRecordIndex]
+					record.Flags |= ast.HandlesImportErrors
+					record.ErrorHandlerLoc = p.thenCatchChain.catchLoc
+				}
+				p.importRecordsForCurrentPart = append(p.importRecordsForCurrentPart, importRecordIndex)
+				return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EImportString{
+					ImportRecordIndex: importRecordIndex,
+					CloseParenLoc:     e.CloseParenLoc,
+				}}
+			}
+
+			// Handle glob patterns
+			if p.options.mode == config.ModeBundle {
+				if value := p.handleGlobPattern(arg, ast.ImportDynamic, "globImport", assertOrWith); value.Data != nil {
+					return value
+				}
+			}
+
+			// Use a debug log so people can see this if they want to
+			r := js_lexer.RangeOfIdentifier(p.source, expr.Loc)
+			p.log.AddID(logger.MsgID_JS_UnsupportedDynamicImport, logger.Debug, &p.tracker, r,
+				"This \"import\" expression will not be bundled because the argument is not a string literal")
+
+			// We need to convert this into a call to "require()" if ES6 syntax is
+			// not supported in the current output format. The full conversion:
+			//
+			//   Before:
+			//     import(foo)
+			//
+			//   After:
+			//     Promise.resolve().then(() => __toESM(require(foo)))
+			//
+			// This is normally done by the printer since we don't know during the
+			// parsing stage whether this module is external or not. However, it's
+			// guaranteed to be external if the argument isn't a string. We handle
+			// this case here instead of in the printer because both the printer
+			// and the linker currently need an import record to handle this case
+			// correctly, and you need a string literal to get an import record.
+			if p.options.unsupportedJSFeatures.Has(compat.DynamicImport) {
+				var then js_ast.Expr
+				value := p.callRuntime(arg.Loc, "__toESM", []js_ast.Expr{{Loc: expr.Loc, Data: &js_ast.ECall{
+					Target:        p.valueToSubstituteForRequire(expr.Loc),
+					Args:          []js_ast.Expr{arg},
+					CloseParenLoc: e.CloseParenLoc,
+				}}})
+				body := js_ast.FnBody{Loc: expr.Loc, Block: js_ast.SBlock{Stmts: []js_ast.Stmt{{Loc: expr.Loc, Data: &js_ast.SReturn{ValueOrNil: value}}}}}
+				if p.options.unsupportedJSFeatures.Has(compat.Arrow) {
+					then = js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EFunction{Fn: js_ast.Fn{Body: body}}}
+				} else {
+					then = js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EArrow{Body: body, PreferExpr: true}}
+				}
+				return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ECall{
+					Target: js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EDot{
+						Target: js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ECall{
+							Target: js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EDot{
+								Target:  js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EIdentifier{Ref: p.makePromiseRef()}},
+								Name:    "resolve",
+								NameLoc: expr.Loc,
+							}},
+							Kind: js_ast.TargetWasOriginallyPropertyAccess,
+						}},
+						Name:    "then",
+						NameLoc: expr.Loc,
+					}},
+					Args: []js_ast.Expr{then},
+					Kind: js_ast.TargetWasOriginallyPropertyAccess,
+				}}
+			}
+
+			return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EImportCall{
+				Expr:          arg,
+				OptionsOrNil:  e.OptionsOrNil,
+				CloseParenLoc: e.CloseParenLoc,
+			}}
+		}), exprOut{}
+
+	case *js_ast.ECall:
+		p.callTarget = e.Target.Data
+
+		// Track ".then().catch()" chains
+		p.thenCatchChain = thenCatchChain{
+			nextTarget:      e.Target.Data,
+			hasMultipleArgs: len(e.Args) >= 2,
+			hasCatch:        p.thenCatchChain.nextTarget == e && p.thenCatchChain.hasCatch,
+			catchLoc:        p.thenCatchChain.catchLoc,
+		}
+		if p.thenCatchChain.hasMultipleArgs {
+			p.thenCatchChain.catchLoc = e.Args[1].Loc
+		}
+
+		// Prepare to recognize "require.resolve()" and "Object.create" calls
+		couldBeRequireResolve := false
+		couldBeObjectCreate := false
+		if len(e.Args) == 1 {
+			if dot, ok := e.Target.Data.(*js_ast.EDot); ok && dot.OptionalChain == js_ast.OptionalChainNone {
+				if p.options.mode != config.ModePassThrough && dot.Name == "resolve" {
+					couldBeRequireResolve = true
+				} else if dot.Name == "create" {
+					couldBeObjectCreate = true
+				}
+			}
+		}
+
+		wasIdentifierBeforeVisit := false
+		isParenthesizedOptionalChain := false
+		switch e2 := e.Target.Data.(type) {
+		case *js_ast.EIdentifier:
+			wasIdentifierBeforeVisit = true
+		case *js_ast.EDot:
+			isParenthesizedOptionalChain = e.OptionalChain == js_ast.OptionalChainNone && e2.OptionalChain != js_ast.OptionalChainNone
+		case *js_ast.EIndex:
+			isParenthesizedOptionalChain = e.OptionalChain == js_ast.OptionalChainNone && e2.OptionalChain != js_ast.OptionalChainNone
+		}
+		target, out := p.visitExprInOut(e.Target, exprIn{
+			hasChainParent: e.OptionalChain == js_ast.OptionalChainContinue,
+
+			// Signal to our child if this is an ECall at the start of an optional
+			// chain. If so, the child will need to stash the "this" context for us
+			// that we need for the ".call(this, ...args)".
+			storeThisArgForParentOptionalChain: e.OptionalChain == js_ast.OptionalChainStart || isParenthesizedOptionalChain,
+		})
+		e.Target = target
+		p.warnAboutImportNamespaceCall(e.Target, exprKindCall)
+
+		hasSpread := false
+		oldIsControlFlowDead := p.isControlFlowDead
+
+		// If we're removing this call, don't count any arguments as symbol uses
+		if out.methodCallMustBeReplacedWithUndefined {
+			if js_ast.IsPropertyAccess(e.Target) {
+				p.isControlFlowDead = true
+			} else {
+				out.methodCallMustBeReplacedWithUndefined = false
+			}
+		}
+
+		// Visit the arguments
+		for i, arg := range e.Args {
+			arg = p.visitExpr(arg)
+			if _, ok := arg.Data.(*js_ast.ESpread); ok {
+				hasSpread = true
+			}
+			e.Args[i] = arg
+		}
+
+		// Mark side-effect free IIFEs with "/* @__PURE__ */"
+		if !e.CanBeUnwrappedIfUnused {
+			switch target := e.Target.Data.(type) {
+			case *js_ast.EArrow:
+				if !target.IsAsync && p.iifeCanBeRemovedIfUnused(target.Args, target.Body) {
+					e.CanBeUnwrappedIfUnused = true
+				}
+			case *js_ast.EFunction:
+				if !target.Fn.IsAsync && !target.Fn.IsGenerator && p.iifeCanBeRemovedIfUnused(target.Fn.Args, target.Fn.Body) {
+					e.CanBeUnwrappedIfUnused = true
+				}
+			}
+		}
+
+		// Our hack for reading Yarn PnP files is implemented here:
+		if p.options.decodeHydrateRuntimeStateYarnPnP {
+			if id, ok := e.Target.Data.(*js_ast.EIdentifier); ok && p.symbols[id.Ref.InnerIndex].OriginalName == "hydrateRuntimeState" && len(e.Args) >= 1 {
+				switch arg := e.Args[0].Data.(type) {
+				case *js_ast.EObject:
+					// "hydrateRuntimeState(<object literal>)"
+					if arg := e.Args[0]; isValidJSON(arg) {
+						p.manifestForYarnPnP = arg
+					}
+
+				case *js_ast.ECall:
+					// "hydrateRuntimeState(JSON.parse(<something>))"
+					if len(arg.Args) == 1 {
+						if dot, ok := arg.Target.Data.(*js_ast.EDot); ok && dot.Name == "parse" {
+							if id, ok := dot.Target.Data.(*js_ast.EIdentifier); ok {
+								if symbol := &p.symbols[id.Ref.InnerIndex]; symbol.Kind == ast.SymbolUnbound && symbol.OriginalName == "JSON" {
+									arg := arg.Args[0]
+									switch a := arg.Data.(type) {
+									case *js_ast.EString:
+										// "hydrateRuntimeState(JSON.parse(<string literal>))"
+										source := logger.Source{KeyPath: p.source.KeyPath, Contents: helpers.UTF16ToString(a.Value)}
+										stringInJSTable := logger.GenerateStringInJSTable(p.source.Contents, arg.Loc, source.Contents)
+										log := logger.NewStringInJSLog(p.log, &p.tracker, stringInJSTable)
+										p.manifestForYarnPnP, _ = ParseJSON(log, source, JSONOptions{})
+										remapExprLocsInJSON(&p.manifestForYarnPnP, stringInJSTable)
+
+									case *js_ast.EIdentifier:
+										// "hydrateRuntimeState(JSON.parse(<identifier>))"
+										if data, ok := p.stringLocalsForYarnPnP[a.Ref]; ok {
+											source := logger.Source{KeyPath: p.source.KeyPath, Contents: helpers.UTF16ToString(data.value)}
+											stringInJSTable := logger.GenerateStringInJSTable(p.source.Contents, data.loc, source.Contents)
+											log := logger.NewStringInJSLog(p.log, &p.tracker, stringInJSTable)
+											p.manifestForYarnPnP, _ = ParseJSON(log, source, JSONOptions{})
+											remapExprLocsInJSON(&p.manifestForYarnPnP, stringInJSTable)
+										}
+									}
+								}
+							}
+						}
+					}
+				}
+			}
+		}
+
+		// Stop now if this call must be removed
+		if out.methodCallMustBeReplacedWithUndefined {
+			p.isControlFlowDead = oldIsControlFlowDead
+			return js_ast.Expr{Loc: expr.Loc, Data: js_ast.EUndefinedShared}, exprOut{}
+		}
+
+		// "foo(1, ...[2, 3], 4)" => "foo(1, 2, 3, 4)"
+		if p.options.minifySyntax && hasSpread {
+			e.Args = js_ast.InlineSpreadsOfArrayLiterals(e.Args)
+		}
+
+		switch t := target.Data.(type) {
+		case *js_ast.EImportIdentifier:
+			// If this function is inlined, allow it to be tree-shaken
+			if p.options.minifySyntax && !p.isControlFlowDead {
+				p.convertSymbolUseToCall(t.Ref, len(e.Args) == 1 && !hasSpread)
+			}
+
+		case *js_ast.EIdentifier:
+			// Detect if this is a direct eval. Note that "(1 ? eval : 0)(x)" will
+			// become "eval(x)" after we visit the target due to dead code elimination,
+			// but that doesn't mean it should become a direct eval.
+			//
+			// Note that "eval?.(x)" is considered an indirect eval. There was debate
+			// about this after everyone implemented it as a direct eval, but the
+			// language committee said it was indirect and everyone had to change it:
+			// https://github.com/tc39/ecma262/issues/2062.
+			if e.OptionalChain == js_ast.OptionalChainNone {
+				symbol := p.symbols[t.Ref.InnerIndex]
+				if wasIdentifierBeforeVisit && symbol.OriginalName == "eval" {
+					e.Kind = js_ast.DirectEval
+
+					// Pessimistically assume that if this looks like a CommonJS module
+					// (e.g. no "export" keywords), a direct call to "eval" means that
+					// code could potentially access "module" or "exports".
+					if p.options.mode == config.ModeBundle && !p.isFileConsideredToHaveESMExports {
+						p.recordUsage(p.moduleRef)
+						p.recordUsage(p.exportsRef)
+					}
+
+					// Mark this scope and all parent scopes as containing a direct eval.
+					// This will prevent us from renaming any symbols.
+					for s := p.currentScope; s != nil; s = s.Parent {
+						s.ContainsDirectEval = true
+					}
+
+					// Warn when direct eval is used in an ESM file. There is no way we
+					// can guarantee that this will work correctly for top-level imported
+					// and exported symbols due to scope hoisting. Except don't warn when
+					// this code is in a 3rd-party library because there's nothing people
+					// will be able to do about the warning.
+					text := "Using direct eval with a bundler is not recommended and may cause problems"
+					kind := logger.Debug
+					if p.options.mode == config.ModeBundle && p.isFileConsideredESM && !p.suppressWarningsAboutWeirdCode {
+						kind = logger.Warning
+					}
+					p.log.AddIDWithNotes(logger.MsgID_JS_DirectEval, kind, &p.tracker, js_lexer.RangeOfIdentifier(p.source, e.Target.Loc), text,
+						[]logger.MsgData{{Text: "You can read more about direct eval and bundling here: https://esbuild.github.io/link/direct-eval"}})
+				} else if symbol.Flags.Has(ast.CallCanBeUnwrappedIfUnused) {
+					// Automatically add a "/* @__PURE__ */" comment to file-local calls
+					// of functions declared with a "/* @__NO_SIDE_EFFECTS__ */" comment
+					t.CallCanBeUnwrappedIfUnused = true
+				}
+			}
+
+			// Optimize references to global constructors
+			if p.options.minifySyntax && t.CanBeRemovedIfUnused && len(e.Args) <= 1 && !hasSpread {
+				if symbol := &p.symbols[t.Ref.InnerIndex]; symbol.Kind == ast.SymbolUnbound {
+					// Note: We construct expressions by assigning to "expr.Data" so
+					// that the source map position for the constructor is preserved
+					switch symbol.OriginalName {
+					case "Boolean":
+						if len(e.Args) == 0 {
+							return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EBoolean{Value: false}}, exprOut{}
+						} else {
+							expr.Data = &js_ast.EUnary{Value: p.astHelpers.SimplifyBooleanExpr(e.Args[0]), Op: js_ast.UnOpNot}
+							return js_ast.Not(expr), exprOut{}
+						}
+
+					case "Number":
+						if len(e.Args) == 0 {
+							return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENumber{Value: 0}}, exprOut{}
+						} else {
+							arg := e.Args[0]
+
+							switch js_ast.KnownPrimitiveType(arg.Data) {
+							case js_ast.PrimitiveNumber:
+								return arg, exprOut{}
+
+							case
+								js_ast.PrimitiveUndefined, // NaN
+								js_ast.PrimitiveNull,      // 0
+								js_ast.PrimitiveBoolean,   // 0 or 1
+								js_ast.PrimitiveString:    // StringToNumber
+								if number, ok := js_ast.ToNumberWithoutSideEffects(arg.Data); ok {
+									expr.Data = &js_ast.ENumber{Value: number}
+								} else {
+									expr.Data = &js_ast.EUnary{Value: arg, Op: js_ast.UnOpPos}
+								}
+								return expr, exprOut{}
+							}
+						}
+
+					case "String":
+						if len(e.Args) == 0 {
+							return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EString{Value: nil}}, exprOut{}
+						} else {
+							arg := e.Args[0]
+
+							switch js_ast.KnownPrimitiveType(arg.Data) {
+							case js_ast.PrimitiveString:
+								return arg, exprOut{}
+							}
+						}
+
+					case "BigInt":
+						if len(e.Args) == 1 {
+							arg := e.Args[0]
+
+							switch js_ast.KnownPrimitiveType(arg.Data) {
+							case js_ast.PrimitiveBigInt:
+								return arg, exprOut{}
+							}
+						}
+					}
+				}
+			}
+
+			// Copy the call side effect flag over if this is a known target
+			if t.CallCanBeUnwrappedIfUnused {
+				e.CanBeUnwrappedIfUnused = true
+			}
+
+			// If this function is inlined, allow it to be tree-shaken
+			if p.options.minifySyntax && !p.isControlFlowDead {
+				p.convertSymbolUseToCall(t.Ref, len(e.Args) == 1 && !hasSpread)
+			}
+
+		case *js_ast.EDot:
+			// Recognize "require.resolve()" calls
+			if couldBeRequireResolve && t.Name == "resolve" {
+				if id, ok := t.Target.Data.(*js_ast.EIdentifier); ok && id.Ref == p.requireRef {
+					p.ignoreUsage(p.requireRef)
+					return p.maybeTransposeIfExprChain(e.Args[0], func(arg js_ast.Expr) js_ast.Expr {
+						if str, ok := e.Args[0].Data.(*js_ast.EString); ok {
+							// Ignore calls to require.resolve() if the control flow is provably
+							// dead here. We don't want to spend time scanning the required files
+							// if they will never be used.
+							if p.isControlFlowDead {
+								return js_ast.Expr{Loc: expr.Loc, Data: js_ast.ENullShared}
+							}
+
+							importRecordIndex := p.addImportRecord(ast.ImportRequireResolve, p.source.RangeOfString(e.Args[0].Loc), helpers.UTF16ToString(str.Value), nil, 0)
+							if p.fnOrArrowDataVisit.tryBodyCount != 0 {
+								record := &p.importRecords[importRecordIndex]
+								record.Flags |= ast.HandlesImportErrors
+								record.ErrorHandlerLoc = p.fnOrArrowDataVisit.tryCatchLoc
+							}
+							p.importRecordsForCurrentPart = append(p.importRecordsForCurrentPart, importRecordIndex)
+
+							// Create a new expression to represent the operation
+							return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ERequireResolveString{
+								ImportRecordIndex: importRecordIndex,
+								CloseParenLoc:     e.CloseParenLoc,
+							}}
+						}
+
+						// Otherwise just return a clone of the "require.resolve()" call
+						return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ECall{
+							Target: js_ast.Expr{Loc: e.Target.Loc, Data: &js_ast.EDot{
+								Target:  p.valueToSubstituteForRequire(t.Target.Loc),
+								Name:    t.Name,
+								NameLoc: t.NameLoc,
+							}},
+							Args:          []js_ast.Expr{arg},
+							Kind:          e.Kind,
+							CloseParenLoc: e.CloseParenLoc,
+						}}
+					}), exprOut{}
+				}
+			}
+
+			// Recognize "Object.create()" calls
+			if couldBeObjectCreate && t.Name == "create" {
+				if id, ok := t.Target.Data.(*js_ast.EIdentifier); ok {
+					if symbol := &p.symbols[id.Ref.InnerIndex]; symbol.Kind == ast.SymbolUnbound && symbol.OriginalName == "Object" {
+						switch e.Args[0].Data.(type) {
+						case *js_ast.ENull, *js_ast.EObject:
+							// Mark "Object.create(null)" and "Object.create({})" as pure
+							e.CanBeUnwrappedIfUnused = true
+						}
+					}
+				}
+			}
+
+			if p.options.minifySyntax {
+				switch t.Name {
+				case "charCodeAt":
+					// Recognize "charCodeAt()" calls
+					if str, ok := t.Target.Data.(*js_ast.EString); ok && len(e.Args) <= 1 {
+						index := 0
+						hasIndex := false
+						if len(e.Args) == 0 {
+							hasIndex = true
+						} else if num, ok := e.Args[0].Data.(*js_ast.ENumber); ok && num.Value == math.Trunc(num.Value) && math.Abs(num.Value) <= 0x7FFF_FFFF {
+							index = int(num.Value)
+							hasIndex = true
+						}
+						if hasIndex {
+							if index >= 0 && index < len(str.Value) {
+								return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENumber{Value: float64(str.Value[index])}}, exprOut{}
+							} else {
+								return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENumber{Value: math.NaN()}}, exprOut{}
+							}
+						}
+					}
+
+				case "fromCharCode":
+					// Recognize "fromCharCode()" calls
+					if id, ok := t.Target.Data.(*js_ast.EIdentifier); ok {
+						if symbol := &p.symbols[id.Ref.InnerIndex]; symbol.Kind == ast.SymbolUnbound && symbol.OriginalName == "String" {
+							charCodes := make([]uint16, 0, len(e.Args))
+							for _, arg := range e.Args {
+								arg, ok := js_ast.ToNumberWithoutSideEffects(arg.Data)
+								if !ok {
+									break
+								}
+								charCodes = append(charCodes, uint16(js_ast.ToInt32(arg)))
+							}
+							if len(charCodes) == len(e.Args) {
+								return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EString{Value: charCodes}}, exprOut{}
+							}
+						}
+					}
+
+				case "toString":
+					switch target := t.Target.Data.(type) {
+					case *js_ast.ENumber:
+						radix := 0
+						if len(e.Args) == 0 {
+							radix = 10
+						} else if len(e.Args) == 1 {
+							if num, ok := e.Args[0].Data.(*js_ast.ENumber); ok && num.Value == math.Trunc(num.Value) && num.Value >= 2 && num.Value <= 36 {
+								radix = int(num.Value)
+							}
+						}
+						if radix != 0 {
+							if str, ok := js_ast.TryToStringOnNumberSafely(target.Value, radix); ok {
+								return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16(str)}}, exprOut{}
+							}
+						}
+
+					case *js_ast.ERegExp:
+						if len(e.Args) == 0 {
+							return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16(target.Value)}}, exprOut{}
+						}
+
+					case *js_ast.EBoolean:
+						if len(e.Args) == 0 {
+							if target.Value {
+								return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16("true")}}, exprOut{}
+							} else {
+								return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16("false")}}, exprOut{}
+							}
+						}
+
+					case *js_ast.EString:
+						if len(e.Args) == 0 {
+							return t.Target, exprOut{}
+						}
+					}
+				}
+			}
+
+			// Copy the call side effect flag over if this is a known target
+			if t.CallCanBeUnwrappedIfUnused {
+				e.CanBeUnwrappedIfUnused = true
+			}
+
+		case *js_ast.EIndex:
+			// Copy the call side effect flag over if this is a known target
+			if t.CallCanBeUnwrappedIfUnused {
+				e.CanBeUnwrappedIfUnused = true
+			}
+
+		case *js_ast.ESuper:
+			// If we're shimming "super()" calls, replace this call with "__super()"
+			if p.superCtorRef != ast.InvalidRef {
+				p.recordUsage(p.superCtorRef)
+				target.Data = &js_ast.EIdentifier{Ref: p.superCtorRef}
+				e.Target.Data = target.Data
+			}
+		}
+
+		// Handle parenthesized optional chains
+		if isParenthesizedOptionalChain && out.thisArgFunc != nil && out.thisArgWrapFunc != nil {
+			return p.lowerParenthesizedOptionalChain(expr.Loc, e, out), exprOut{}
+		}
+
+		// Lower optional chaining if we're the top of the chain
+		containsOptionalChain := e.OptionalChain == js_ast.OptionalChainStart ||
+			(e.OptionalChain == js_ast.OptionalChainContinue && out.childContainsOptionalChain)
+		if containsOptionalChain && !in.hasChainParent {
+			return p.lowerOptionalChain(expr, in, out)
+		}
+
+		// If this is a plain call expression (instead of an optional chain), lower
+		// private member access in the call target now if there is one
+		if !containsOptionalChain {
+			if target, loc, private := p.extractPrivateIndex(e.Target); private != nil {
+				// "foo.#bar(123)" => "__privateGet(_a = foo, #bar).call(_a, 123)"
+				targetFunc, targetWrapFunc := p.captureValueWithPossibleSideEffects(target.Loc, 2, target, valueCouldBeMutated)
+				return targetWrapFunc(js_ast.Expr{Loc: target.Loc, Data: &js_ast.ECall{
+					Target: js_ast.Expr{Loc: target.Loc, Data: &js_ast.EDot{
+						Target:  p.lowerPrivateGet(targetFunc(), loc, private),
+						Name:    "call",
+						NameLoc: target.Loc,
+					}},
+					Args:                   append([]js_ast.Expr{targetFunc()}, e.Args...),
+					CanBeUnwrappedIfUnused: e.CanBeUnwrappedIfUnused,
+					Kind:                   js_ast.TargetWasOriginallyPropertyAccess,
+				}}), exprOut{}
+			}
+			p.maybeLowerSuperPropertyGetInsideCall(e)
+		}
+
+		// Track calls to require() so we can use them while bundling
+		if p.options.mode != config.ModePassThrough && e.OptionalChain == js_ast.OptionalChainNone {
+			if id, ok := e.Target.Data.(*js_ast.EIdentifier); ok && id.Ref == p.requireRef {
+				// Heuristic: omit warnings inside try/catch blocks because presumably
+				// the try/catch statement is there to handle the potential run-time
+				// error from the unbundled require() call failing.
+				omitWarnings := p.fnOrArrowDataVisit.tryBodyCount != 0
+
+				if p.options.mode != config.ModePassThrough {
+					// There must be one argument
+					if len(e.Args) == 1 {
+						p.ignoreUsage(p.requireRef)
+						return p.maybeTransposeIfExprChain(e.Args[0], func(arg js_ast.Expr) js_ast.Expr {
+							// The argument must be a string
+							if str, ok := arg.Data.(*js_ast.EString); ok {
+								// Ignore calls to require() if the control flow is provably dead here.
+								// We don't want to spend time scanning the required files if they will
+								// never be used.
+								if p.isControlFlowDead {
+									return js_ast.Expr{Loc: expr.Loc, Data: js_ast.ENullShared}
+								}
+
+								importRecordIndex := p.addImportRecord(ast.ImportRequire, p.source.RangeOfString(arg.Loc), helpers.UTF16ToString(str.Value), nil, 0)
+								if p.fnOrArrowDataVisit.tryBodyCount != 0 {
+									record := &p.importRecords[importRecordIndex]
+									record.Flags |= ast.HandlesImportErrors
+									record.ErrorHandlerLoc = p.fnOrArrowDataVisit.tryCatchLoc
+								}
+								p.importRecordsForCurrentPart = append(p.importRecordsForCurrentPart, importRecordIndex)
+
+								// Currently "require" is not converted into "import" for ESM
+								if p.options.mode != config.ModeBundle && p.options.outputFormat == config.FormatESModule && !omitWarnings {
+									r := js_lexer.RangeOfIdentifier(p.source, e.Target.Loc)
+									p.log.AddID(logger.MsgID_JS_UnsupportedRequireCall, logger.Warning, &p.tracker, r, "Converting \"require\" to \"esm\" is currently not supported")
+								}
+
+								// Create a new expression to represent the operation
+								return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ERequireString{
+									ImportRecordIndex: importRecordIndex,
+									CloseParenLoc:     e.CloseParenLoc,
+								}}
+							}
+
+							// Handle glob patterns
+							if p.options.mode == config.ModeBundle {
+								if value := p.handleGlobPattern(arg, ast.ImportRequire, "globRequire", nil); value.Data != nil {
+									return value
+								}
+							}
+
+							// Use a debug log so people can see this if they want to
+							r := js_lexer.RangeOfIdentifier(p.source, e.Target.Loc)
+							p.log.AddID(logger.MsgID_JS_UnsupportedRequireCall, logger.Debug, &p.tracker, r,
+								"This call to \"require\" will not be bundled because the argument is not a string literal")
+
+							// Otherwise just return a clone of the "require()" call
+							return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ECall{
+								Target:        p.valueToSubstituteForRequire(e.Target.Loc),
+								Args:          []js_ast.Expr{arg},
+								CloseParenLoc: e.CloseParenLoc,
+							}}
+						}), exprOut{}
+					} else {
+						// Use a debug log so people can see this if they want to
+						r := js_lexer.RangeOfIdentifier(p.source, e.Target.Loc)
+						p.log.AddIDWithNotes(logger.MsgID_JS_UnsupportedRequireCall, logger.Debug, &p.tracker, r,
+							fmt.Sprintf("This call to \"require\" will not be bundled because it has %d arguments", len(e.Args)),
+							[]logger.MsgData{{Text: "To be bundled by esbuild, a \"require\" call must have exactly 1 argument."}})
+					}
+
+					return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ECall{
+						Target:        p.valueToSubstituteForRequire(e.Target.Loc),
+						Args:          e.Args,
+						CloseParenLoc: e.CloseParenLoc,
+					}}, exprOut{}
+				}
+			}
+		}
+
+		out = exprOut{
+			childContainsOptionalChain: containsOptionalChain,
+			thisArgFunc:                out.thisArgFunc,
+			thisArgWrapFunc:            out.thisArgWrapFunc,
+		}
+		if !in.hasChainParent {
+			out.thisArgFunc = nil
+			out.thisArgWrapFunc = nil
+		}
+		return expr, out
+
+	case *js_ast.ENew:
+		hasSpread := false
+
+		e.Target = p.visitExpr(e.Target)
+		p.warnAboutImportNamespaceCall(e.Target, exprKindNew)
+
+		for i, arg := range e.Args {
+			arg = p.visitExpr(arg)
+			if _, ok := arg.Data.(*js_ast.ESpread); ok {
+				hasSpread = true
+			}
+			e.Args[i] = arg
+		}
+
+		// "new foo(1, ...[2, 3], 4)" => "new foo(1, 2, 3, 4)"
+		if p.options.minifySyntax && hasSpread {
+			e.Args = js_ast.InlineSpreadsOfArrayLiterals(e.Args)
+		}
+
+		p.maybeMarkKnownGlobalConstructorAsPure(e)
+
+	case *js_ast.EArrow:
+		// Check for a propagated name to keep from the parent context
+		var nameToKeep string
+		if p.nameToKeepIsFor == e {
+			nameToKeep = p.nameToKeep
+		}
+
+		// Prepare for suspicious logical operator checking
+		if e.PreferExpr && len(e.Args) == 1 && e.Args[0].DefaultOrNil.Data == nil && len(e.Body.Block.Stmts) == 1 {
+			if _, ok := e.Args[0].Binding.Data.(*js_ast.BIdentifier); ok {
+				if stmt, ok := e.Body.Block.Stmts[0].Data.(*js_ast.SReturn); ok {
+					if binary, ok := stmt.ValueOrNil.Data.(*js_ast.EBinary); ok && (binary.Op == js_ast.BinOpLogicalAnd || binary.Op == js_ast.BinOpLogicalOr) {
+						p.suspiciousLogicalOperatorInsideArrow = binary
+					}
+				}
+			}
+		}
+
+		asyncArrowNeedsToBeLowered := e.IsAsync && p.options.unsupportedJSFeatures.Has(compat.AsyncAwait)
+		oldFnOrArrowData := p.fnOrArrowDataVisit
+		p.fnOrArrowDataVisit = fnOrArrowDataVisit{
+			isArrow:                        true,
+			isAsync:                        e.IsAsync,
+			shouldLowerSuperPropertyAccess: oldFnOrArrowData.shouldLowerSuperPropertyAccess || asyncArrowNeedsToBeLowered,
+		}
+
+		// Mark if we're inside an async arrow function. This value should be true
+		// even if we're inside multiple arrow functions and the closest inclosing
+		// arrow function isn't async, as long as at least one enclosing arrow
+		// function within the current enclosing function is async.
+		oldInsideAsyncArrowFn := p.fnOnlyDataVisit.isInsideAsyncArrowFn
+		if e.IsAsync {
+			p.fnOnlyDataVisit.isInsideAsyncArrowFn = true
+		}
+
+		p.pushScopeForVisitPass(js_ast.ScopeFunctionArgs, expr.Loc)
+		p.visitArgs(e.Args, visitArgsOpts{
+			hasRestArg:               e.HasRestArg,
+			body:                     e.Body.Block.Stmts,
+			isUniqueFormalParameters: true,
+		})
+		p.pushScopeForVisitPass(js_ast.ScopeFunctionBody, e.Body.Loc)
+		e.Body.Block.Stmts = p.visitStmtsAndPrependTempRefs(e.Body.Block.Stmts, prependTempRefsOpts{kind: stmtsFnBody})
+		p.popScope()
+		p.lowerFunction(&e.IsAsync, nil, &e.Args, e.Body.Loc, &e.Body.Block, &e.PreferExpr, &e.HasRestArg, true /* isArrow */)
+		p.popScope()
+
+		if p.options.minifySyntax && len(e.Body.Block.Stmts) == 1 {
+			if s, ok := e.Body.Block.Stmts[0].Data.(*js_ast.SReturn); ok {
+				if s.ValueOrNil.Data == nil {
+					// "() => { return }" => "() => {}"
+					e.Body.Block.Stmts = []js_ast.Stmt{}
+				} else {
+					// "() => { return x }" => "() => x"
+					e.PreferExpr = true
+				}
+			}
+		}
+
+		p.fnOnlyDataVisit.isInsideAsyncArrowFn = oldInsideAsyncArrowFn
+		p.fnOrArrowDataVisit = oldFnOrArrowData
+
+		// Convert arrow functions to function expressions when lowering
+		if p.options.unsupportedJSFeatures.Has(compat.Arrow) {
+			expr.Data = &js_ast.EFunction{Fn: js_ast.Fn{
+				Args:         e.Args,
+				Body:         e.Body,
+				ArgumentsRef: ast.InvalidRef,
+				IsAsync:      e.IsAsync,
+				HasRestArg:   e.HasRestArg,
+			}}
+		}
+
+		// Optionally preserve the name
+		if p.options.keepNames && nameToKeep != "" {
+			expr = p.keepExprSymbolName(expr, nameToKeep)
+		}
+
+	case *js_ast.EFunction:
+		// Check for a propagated name to keep from the parent context
+		var nameToKeep string
+		if p.nameToKeepIsFor == e {
+			nameToKeep = p.nameToKeep
+		}
+
+		p.visitFn(&e.Fn, expr.Loc, visitFnOpts{
+			isMethod:               in.isMethod,
+			isDerivedClassCtor:     e == p.propDerivedCtorValue,
+			isLoweredPrivateMethod: in.isLoweredPrivateMethod,
+		})
+		name := e.Fn.Name
+
+		// Remove unused function names when minifying
+		if p.options.minifySyntax && !p.currentScope.ContainsDirectEval &&
+			name != nil && p.symbols[name.Ref.InnerIndex].UseCountEstimate == 0 {
+			e.Fn.Name = nil
+		}
+
+		// Optionally preserve the name for functions, but not for methods
+		if p.options.keepNames && (!in.isMethod || in.isLoweredPrivateMethod) {
+			if name != nil {
+				expr = p.keepExprSymbolName(expr, p.symbols[name.Ref.InnerIndex].OriginalName)
+			} else if nameToKeep != "" {
+				expr = p.keepExprSymbolName(expr, nameToKeep)
+			}
+		}
+
+	case *js_ast.EClass:
+		// Check for a propagated name to keep from the parent context
+		var nameToKeep string
+		if p.nameToKeepIsFor == e {
+			nameToKeep = p.nameToKeep
+		}
+
+		result := p.visitClass(expr.Loc, &e.Class, ast.InvalidRef, nameToKeep)
+
+		// Lower class field syntax for browsers that don't support it
+		_, expr = p.lowerClass(js_ast.Stmt{}, expr, result, nameToKeep)
+
+		// We may be able to determine that a class is side-effect before lowering
+		// but not after lowering (e.g. due to "--keep-names" mutating the object).
+		// If that's the case, add a special annotation so this doesn't prevent
+		// tree-shaking from happening.
+		if result.canBeRemovedIfUnused {
+			expr.Data = &js_ast.EAnnotation{
+				Value: expr,
+				Flags: js_ast.CanBeRemovedIfUnusedFlag,
+			}
+		}
+
+	default:
+		// Note: EPrivateIdentifier should have already been handled
+		panic(fmt.Sprintf("Unexpected expression of type %T", expr.Data))
+	}
+
+	return expr, exprOut{}
+}
+
+// This exists to handle very deeply-nested ASTs. For example, the "grapheme-splitter"
+// package contains this monstrosity:
+//
+//	if (
+//	  (0x0300 <= code && code <= 0x036F) ||
+//	  (0x0483 <= code && code <= 0x0487) ||
+//	  (0x0488 <= code && code <= 0x0489) ||
+//	  (0x0591 <= code && code <= 0x05BD) ||
+//	  ... many hundreds of lines later ...
+//	) {
+//	  return;
+//	}
+//
+// If "checkAndPrepare" returns non-nil, then the return value is the final
+// expression. Otherwise, the final expression can be obtained by manually
+// visiting the left child and then calling "visitRightAndFinish":
+//
+//	if result := v.checkAndPrepare(p); result.Data != nil {
+//	  return result
+//	}
+//	v.e.Left, _ = p.visitExprInOut(v.e.Left, v.leftIn)
+//	return v.visitRightAndFinish(p)
+//
+// This code is convoluted this way so that we can use our own stack on the
+// heap instead of the call stack when there are additional levels of nesting.
+// Before this transformation, the code previously looked something like this:
+//
+//	... The code in "checkAndPrepare" ...
+//	e.Left, _ = p.visitExprInOut(e.Left, in)
+//	... The code in "visitRightAndFinish" ...
+//
+// If this code is still confusing, it may be helpful to look back in git
+// history at the commit that introduced this transformation.
+//
+// Go normally has growable call stacks so this code transformation normally
+// doesn't do anything, but WebAssembly doesn't allow stack pointer manipulation
+// so Go's WebAssembly implementation doesn't support growable call stacks and
+// is therefore vulnerable to stack overflow. So this code transformation is
+// only really relevant for esbuild's WebAssembly-based API.
+type binaryExprVisitor struct {
+	// Inputs
+	e   *js_ast.EBinary
+	loc logger.Loc
+	in  exprIn
+
+	// Input for visiting the left child
+	leftIn exprIn
+
+	// "Local variables" passed from "checkAndPrepare" to "visitRightAndFinish"
+	isStmtExpr                               bool
+	oldSilenceWarningAboutThisBeingUndefined bool
+}
+
+func (v *binaryExprVisitor) checkAndPrepare(p *parser) js_ast.Expr {
+	e := v.e
+
+	// Special-case EPrivateIdentifier to allow it here
+	if private, ok := e.Left.Data.(*js_ast.EPrivateIdentifier); ok && e.Op == js_ast.BinOpIn {
+		name := p.loadNameFromRef(private.Ref)
+		result := p.findSymbol(e.Left.Loc, name)
+		private.Ref = result.ref
+
+		// Unlike regular identifiers, there are no unbound private identifiers
+		symbol := &p.symbols[result.ref.InnerIndex]
+		if !symbol.Kind.IsPrivate() {
+			r := logger.Range{Loc: e.Left.Loc, Len: int32(len(name))}
+			p.log.AddError(&p.tracker, r, fmt.Sprintf("Private name %q must be declared in an enclosing class", name))
+		}
+
+		e.Right = p.visitExpr(e.Right)
+
+		if p.privateSymbolNeedsToBeLowered(private) {
+			return p.lowerPrivateBrandCheck(e.Right, v.loc, private)
+		}
+		return js_ast.Expr{Loc: v.loc, Data: e}
+	}
+
+	v.isStmtExpr = e == p.stmtExprValue
+	v.oldSilenceWarningAboutThisBeingUndefined = p.fnOnlyDataVisit.silenceMessageAboutThisBeingUndefined
+
+	if _, ok := e.Left.Data.(*js_ast.EThis); ok && e.Op == js_ast.BinOpLogicalAnd {
+		p.fnOnlyDataVisit.silenceMessageAboutThisBeingUndefined = true
+	}
+	v.leftIn = exprIn{
+		assignTarget:               e.Op.BinaryAssignTarget(),
+		shouldMangleStringsAsProps: e.Op == js_ast.BinOpIn,
+	}
+	return js_ast.Expr{}
+}
+
+func (v *binaryExprVisitor) visitRightAndFinish(p *parser) js_ast.Expr {
+	e := v.e
+
+	// Mark the control flow as dead if the branch is never taken
+	switch e.Op {
+	case js_ast.BinOpLogicalOr:
+		if boolean, _, ok := js_ast.ToBooleanWithSideEffects(e.Left.Data); ok && boolean {
+			// "true || dead"
+			old := p.isControlFlowDead
+			p.isControlFlowDead = true
+			e.Right = p.visitExpr(e.Right)
+			p.isControlFlowDead = old
+		} else {
+			e.Right = p.visitExpr(e.Right)
+		}
+
+	case js_ast.BinOpLogicalAnd:
+		if boolean, _, ok := js_ast.ToBooleanWithSideEffects(e.Left.Data); ok && !boolean {
+			// "false && dead"
+			old := p.isControlFlowDead
+			p.isControlFlowDead = true
+			e.Right = p.visitExpr(e.Right)
+			p.isControlFlowDead = old
+		} else {
+			e.Right = p.visitExpr(e.Right)
+		}
+
+	case js_ast.BinOpNullishCoalescing:
+		if isNullOrUndefined, _, ok := js_ast.ToNullOrUndefinedWithSideEffects(e.Left.Data); ok && !isNullOrUndefined {
+			// "notNullOrUndefined ?? dead"
+			old := p.isControlFlowDead
+			p.isControlFlowDead = true
+			e.Right = p.visitExpr(e.Right)
+			p.isControlFlowDead = old
+		} else {
+			e.Right = p.visitExpr(e.Right)
+		}
+
+	case js_ast.BinOpComma:
+		e.Right, _ = p.visitExprInOut(e.Right, exprIn{
+			shouldMangleStringsAsProps: v.in.shouldMangleStringsAsProps,
+		})
+
+	case js_ast.BinOpAssign, js_ast.BinOpLogicalOrAssign, js_ast.BinOpLogicalAndAssign, js_ast.BinOpNullishCoalescingAssign:
+		// Check for a propagated name to keep from the parent context
+		if id, ok := e.Left.Data.(*js_ast.EIdentifier); ok {
+			p.nameToKeep = p.symbols[id.Ref.InnerIndex].OriginalName
+			p.nameToKeepIsFor = e.Right.Data
+		}
+
+		e.Right = p.visitExpr(e.Right)
+
+	default:
+		e.Right = p.visitExpr(e.Right)
+	}
+	p.fnOnlyDataVisit.silenceMessageAboutThisBeingUndefined = v.oldSilenceWarningAboutThisBeingUndefined
+
+	// Always put constants consistently on the same side for equality
+	// comparisons to help improve compression. In theory, dictionary-based
+	// compression methods may already have a dictionary entry for code that
+	// is similar to previous code. Note that we can only reorder expressions
+	// that do not have any side effects.
+	//
+	// Constants are currently ordered on the right instead of the left because
+	// it results in slightly smalller gzip size on our primary benchmark
+	// (although slightly larger uncompressed size). The size difference is
+	// less than 0.1% so it really isn't that important an optimization.
+	if p.options.minifySyntax {
+		switch e.Op {
+		case js_ast.BinOpLooseEq, js_ast.BinOpLooseNe, js_ast.BinOpStrictEq, js_ast.BinOpStrictNe:
+			// "1 === x" => "x === 1"
+			if js_ast.IsPrimitiveLiteral(e.Left.Data) && !js_ast.IsPrimitiveLiteral(e.Right.Data) {
+				e.Left, e.Right = e.Right, e.Left
+			}
+		}
+	}
+
+	if p.shouldFoldTypeScriptConstantExpressions || (p.options.minifySyntax && js_ast.ShouldFoldBinaryOperatorWhenMinifying(e)) {
+		if result := js_ast.FoldBinaryOperator(v.loc, e); result.Data != nil {
+			return result
+		}
+	}
+
+	// Post-process the binary expression
+	switch e.Op {
+	case js_ast.BinOpComma:
+		// "(1, 2)" => "2"
+		// "(sideEffects(), 2)" => "(sideEffects(), 2)"
+		if p.options.minifySyntax {
+			e.Left = p.astHelpers.SimplifyUnusedExpr(e.Left, p.options.unsupportedJSFeatures)
+			if e.Left.Data == nil {
+				return e.Right
+			}
+		}
+
+	case js_ast.BinOpLooseEq:
+		if result, ok := js_ast.CheckEqualityIfNoSideEffects(e.Left.Data, e.Right.Data, js_ast.LooseEquality); ok {
+			return js_ast.Expr{Loc: v.loc, Data: &js_ast.EBoolean{Value: result}}
+		}
+		afterOpLoc := locAfterOp(e)
+		if !p.warnAboutEqualityCheck("==", e.Left, afterOpLoc) {
+			p.warnAboutEqualityCheck("==", e.Right, afterOpLoc)
+		}
+		p.warnAboutTypeofAndString(e.Left, e.Right, checkBothOrders)
+
+		if p.options.minifySyntax {
+			// "x == void 0" => "x == null"
+			if _, ok := e.Left.Data.(*js_ast.EUndefined); ok {
+				e.Left.Data = js_ast.ENullShared
+			} else if _, ok := e.Right.Data.(*js_ast.EUndefined); ok {
+				e.Right.Data = js_ast.ENullShared
+			}
+
+			if result, ok := js_ast.MaybeSimplifyEqualityComparison(v.loc, e, p.options.unsupportedJSFeatures); ok {
+				return result
+			}
+		}
+
+	case js_ast.BinOpStrictEq:
+		if result, ok := js_ast.CheckEqualityIfNoSideEffects(e.Left.Data, e.Right.Data, js_ast.StrictEquality); ok {
+			return js_ast.Expr{Loc: v.loc, Data: &js_ast.EBoolean{Value: result}}
+		}
+		afterOpLoc := locAfterOp(e)
+		if !p.warnAboutEqualityCheck("===", e.Left, afterOpLoc) {
+			p.warnAboutEqualityCheck("===", e.Right, afterOpLoc)
+		}
+		p.warnAboutTypeofAndString(e.Left, e.Right, checkBothOrders)
+
+		if p.options.minifySyntax {
+			// "typeof x === 'undefined'" => "typeof x == 'undefined'"
+			if js_ast.CanChangeStrictToLoose(e.Left, e.Right) {
+				e.Op = js_ast.BinOpLooseEq
+			}
+
+			if result, ok := js_ast.MaybeSimplifyEqualityComparison(v.loc, e, p.options.unsupportedJSFeatures); ok {
+				return result
+			}
+		}
+
+	case js_ast.BinOpLooseNe:
+		if result, ok := js_ast.CheckEqualityIfNoSideEffects(e.Left.Data, e.Right.Data, js_ast.LooseEquality); ok {
+			return js_ast.Expr{Loc: v.loc, Data: &js_ast.EBoolean{Value: !result}}
+		}
+		afterOpLoc := locAfterOp(e)
+		if !p.warnAboutEqualityCheck("!=", e.Left, afterOpLoc) {
+			p.warnAboutEqualityCheck("!=", e.Right, afterOpLoc)
+		}
+		p.warnAboutTypeofAndString(e.Left, e.Right, checkBothOrders)
+
+		if p.options.minifySyntax {
+			// "x != void 0" => "x != null"
+			if _, ok := e.Left.Data.(*js_ast.EUndefined); ok {
+				e.Left.Data = js_ast.ENullShared
+			} else if _, ok := e.Right.Data.(*js_ast.EUndefined); ok {
+				e.Right.Data = js_ast.ENullShared
+			}
+
+			if result, ok := js_ast.MaybeSimplifyEqualityComparison(v.loc, e, p.options.unsupportedJSFeatures); ok {
+				return result
+			}
+		}
+
+	case js_ast.BinOpStrictNe:
+		if result, ok := js_ast.CheckEqualityIfNoSideEffects(e.Left.Data, e.Right.Data, js_ast.StrictEquality); ok {
+			return js_ast.Expr{Loc: v.loc, Data: &js_ast.EBoolean{Value: !result}}
+		}
+		afterOpLoc := locAfterOp(e)
+		if !p.warnAboutEqualityCheck("!==", e.Left, afterOpLoc) {
+			p.warnAboutEqualityCheck("!==", e.Right, afterOpLoc)
+		}
+		p.warnAboutTypeofAndString(e.Left, e.Right, checkBothOrders)
+
+		if p.options.minifySyntax {
+			// "typeof x !== 'undefined'" => "typeof x != 'undefined'"
+			if js_ast.CanChangeStrictToLoose(e.Left, e.Right) {
+				e.Op = js_ast.BinOpLooseNe
+			}
+
+			if result, ok := js_ast.MaybeSimplifyEqualityComparison(v.loc, e, p.options.unsupportedJSFeatures); ok {
+				return result
+			}
+		}
+
+	case js_ast.BinOpNullishCoalescing:
+		if isNullOrUndefined, sideEffects, ok := js_ast.ToNullOrUndefinedWithSideEffects(e.Left.Data); ok {
+			// Warn about potential bugs
+			if !js_ast.IsPrimitiveLiteral(e.Left.Data) {
+				// "return props.flag === flag ?? true" is "return (props.flag === flag) ?? true" not "return props.flag === (flag ?? true)"
+				var which string
+				var leftIsNullOrUndefined string
+				var leftIsReturned string
+				if !isNullOrUndefined {
+					which = "left"
+					leftIsNullOrUndefined = "never"
+					leftIsReturned = "always"
+				} else {
+					which = "right"
+					leftIsNullOrUndefined = "always"
+					leftIsReturned = "never"
+				}
+				kind := logger.Warning
+				if p.suppressWarningsAboutWeirdCode {
+					kind = logger.Debug
+				}
+				rOp := p.source.RangeOfOperatorBefore(e.Right.Loc, "??")
+				rLeft := logger.Range{Loc: e.Left.Loc, Len: p.source.LocBeforeWhitespace(rOp.Loc).Start - e.Left.Loc.Start}
+				p.log.AddIDWithNotes(logger.MsgID_JS_SuspiciousNullishCoalescing, kind, &p.tracker, rOp,
+					fmt.Sprintf("The \"??\" operator here will always return the %s operand", which), []logger.MsgData{
+						p.tracker.MsgData(rLeft, fmt.Sprintf(
+							"The left operand of the \"??\" operator here will %s be null or undefined, so it will %s be returned. This usually indicates a bug in your code:",
+							leftIsNullOrUndefined, leftIsReturned))})
+			}
+
+			if !isNullOrUndefined {
+				return e.Left
+			} else if sideEffects == js_ast.NoSideEffects {
+				return e.Right
+			}
+		}
+
+		if p.options.minifySyntax {
+			// "a ?? (b ?? c)" => "a ?? b ?? c"
+			if right, ok := e.Right.Data.(*js_ast.EBinary); ok && right.Op == js_ast.BinOpNullishCoalescing {
+				e.Left = js_ast.JoinWithLeftAssociativeOp(js_ast.BinOpNullishCoalescing, e.Left, right.Left)
+				e.Right = right.Right
+			}
+		}
+
+		if p.options.unsupportedJSFeatures.Has(compat.NullishCoalescing) {
+			return p.lowerNullishCoalescing(v.loc, e.Left, e.Right)
+		}
+
+	case js_ast.BinOpLogicalOr:
+		if boolean, sideEffects, ok := js_ast.ToBooleanWithSideEffects(e.Left.Data); ok {
+			// Warn about potential bugs
+			if e == p.suspiciousLogicalOperatorInsideArrow {
+				if arrowLoc := p.source.RangeOfOperatorBefore(v.loc, "=>"); arrowLoc.Loc.Start+2 == p.source.LocBeforeWhitespace(v.loc).Start {
+					// "return foo => 1 || foo <= 0"
+					var which string
+					if boolean {
+						which = "left"
+					} else {
+						which = "right"
+					}
+					kind := logger.Warning
+					if p.suppressWarningsAboutWeirdCode {
+						kind = logger.Debug
+					}
+					note := p.tracker.MsgData(arrowLoc,
+						"The \"=>\" symbol creates an arrow function expression in JavaScript. Did you mean to use the greater-than-or-equal-to operator \">=\" here instead?")
+					note.Location.Suggestion = ">="
+					rOp := p.source.RangeOfOperatorBefore(e.Right.Loc, "||")
+					p.log.AddIDWithNotes(logger.MsgID_JS_SuspiciousLogicalOperator, kind, &p.tracker, rOp,
+						fmt.Sprintf("The \"||\" operator here will always return the %s operand", which), []logger.MsgData{note})
+				}
+			}
+
+			if boolean {
+				return e.Left
+			} else if sideEffects == js_ast.NoSideEffects {
+				return e.Right
+			}
+		}
+
+		if p.options.minifySyntax {
+			// "a || (b || c)" => "a || b || c"
+			if right, ok := e.Right.Data.(*js_ast.EBinary); ok && right.Op == js_ast.BinOpLogicalOr {
+				e.Left = js_ast.JoinWithLeftAssociativeOp(js_ast.BinOpLogicalOr, e.Left, right.Left)
+				e.Right = right.Right
+			}
+
+			// "a === null || a === undefined" => "a == null"
+			if left, right, ok := js_ast.IsBinaryNullAndUndefined(e.Left, e.Right, js_ast.BinOpStrictEq); ok {
+				e.Op = js_ast.BinOpLooseEq
+				e.Left = left
+				e.Right = right
+			}
+		}
+
+	case js_ast.BinOpLogicalAnd:
+		if boolean, sideEffects, ok := js_ast.ToBooleanWithSideEffects(e.Left.Data); ok {
+			// Warn about potential bugs
+			if e == p.suspiciousLogicalOperatorInsideArrow {
+				if arrowLoc := p.source.RangeOfOperatorBefore(v.loc, "=>"); arrowLoc.Loc.Start+2 == p.source.LocBeforeWhitespace(v.loc).Start {
+					// "return foo => 0 && foo <= 1"
+					var which string
+					if !boolean {
+						which = "left"
+					} else {
+						which = "right"
+					}
+					kind := logger.Warning
+					if p.suppressWarningsAboutWeirdCode {
+						kind = logger.Debug
+					}
+					note := p.tracker.MsgData(arrowLoc,
+						"The \"=>\" symbol creates an arrow function expression in JavaScript. Did you mean to use the greater-than-or-equal-to operator \">=\" here instead?")
+					note.Location.Suggestion = ">="
+					rOp := p.source.RangeOfOperatorBefore(e.Right.Loc, "&&")
+					p.log.AddIDWithNotes(logger.MsgID_JS_SuspiciousLogicalOperator, kind, &p.tracker, rOp,
+						fmt.Sprintf("The \"&&\" operator here will always return the %s operand", which), []logger.MsgData{note})
+				}
+			}
+
+			if !boolean {
+				return e.Left
+			} else if sideEffects == js_ast.NoSideEffects {
+				return e.Right
+			}
+		}
+
+		if p.options.minifySyntax {
+			// "a && (b && c)" => "a && b && c"
+			if right, ok := e.Right.Data.(*js_ast.EBinary); ok && right.Op == js_ast.BinOpLogicalAnd {
+				e.Left = js_ast.JoinWithLeftAssociativeOp(js_ast.BinOpLogicalAnd, e.Left, right.Left)
+				e.Right = right.Right
+			}
+
+			// "a !== null && a !== undefined" => "a != null"
+			if left, right, ok := js_ast.IsBinaryNullAndUndefined(e.Left, e.Right, js_ast.BinOpStrictNe); ok {
+				e.Op = js_ast.BinOpLooseNe
+				e.Left = left
+				e.Right = right
+			}
+		}
+
+	case js_ast.BinOpAdd:
+		// "'abc' + 'xyz'" => "'abcxyz'"
+		if result := js_ast.FoldStringAddition(e.Left, e.Right, js_ast.StringAdditionNormal); result.Data != nil {
+			return result
+		}
+
+		if left, ok := e.Left.Data.(*js_ast.EBinary); ok && left.Op == js_ast.BinOpAdd {
+			// "x + 'abc' + 'xyz'" => "x + 'abcxyz'"
+			if result := js_ast.FoldStringAddition(left.Right, e.Right, js_ast.StringAdditionWithNestedLeft); result.Data != nil {
+				return js_ast.Expr{Loc: v.loc, Data: &js_ast.EBinary{Op: left.Op, Left: left.Left, Right: result}}
+			}
+		}
+
+	case js_ast.BinOpPow:
+		// Lower the exponentiation operator for browsers that don't support it
+		if p.options.unsupportedJSFeatures.Has(compat.ExponentOperator) {
+			return p.callRuntime(v.loc, "__pow", []js_ast.Expr{e.Left, e.Right})
+		}
+
+		////////////////////////////////////////////////////////////////////////////////
+		// All assignment operators below here
+
+	case js_ast.BinOpAssign:
+		if target, loc, private := p.extractPrivateIndex(e.Left); private != nil {
+			return p.lowerPrivateSet(target, loc, private, e.Right)
+		}
+
+		if property := p.extractSuperProperty(e.Left); property.Data != nil {
+			return p.lowerSuperPropertySet(e.Left.Loc, property, e.Right)
+		}
+
+		// Lower assignment destructuring patterns for browsers that don't
+		// support them. Note that assignment expressions are used to represent
+		// initializers in binding patterns, so only do this if we're not
+		// ourselves the target of an assignment. Example: "[a = b] = c"
+		if v.in.assignTarget == js_ast.AssignTargetNone {
+			mode := objRestMustReturnInitExpr
+			if v.isStmtExpr {
+				mode = objRestReturnValueIsUnused
+			}
+			if result, ok := p.lowerAssign(e.Left, e.Right, mode); ok {
+				return result
+			}
+
+			// If CommonJS-style exports are disabled, then references to them are
+			// treated as global variable references. This is consistent with how
+			// they work in node and the browser, so it's the correct interpretation.
+			//
+			// However, people sometimes try to use both types of exports within the
+			// same module and expect it to work. We warn about this when module
+			// format conversion is enabled.
+			//
+			// Only warn about this for uses in assignment position since there are
+			// some legitimate other uses. For example, some people do "typeof module"
+			// to check for a CommonJS environment, and we shouldn't warn on that.
+			if p.options.mode != config.ModePassThrough && p.isFileConsideredToHaveESMExports && !p.isControlFlowDead {
+				if dot, ok := e.Left.Data.(*js_ast.EDot); ok {
+					var name string
+					var loc logger.Loc
+
+					switch target := dot.Target.Data.(type) {
+					case *js_ast.EIdentifier:
+						if symbol := &p.symbols[target.Ref.InnerIndex]; symbol.Kind == ast.SymbolUnbound &&
+							((symbol.OriginalName == "module" && dot.Name == "exports") || symbol.OriginalName == "exports") &&
+							!symbol.Flags.Has(ast.DidWarnAboutCommonJSInESM) {
+							// "module.exports = ..."
+							// "exports.something = ..."
+							name = symbol.OriginalName
+							loc = dot.Target.Loc
+							symbol.Flags |= ast.DidWarnAboutCommonJSInESM
+						}
+
+					case *js_ast.EDot:
+						if target.Name == "exports" {
+							if id, ok := target.Target.Data.(*js_ast.EIdentifier); ok {
+								if symbol := &p.symbols[id.Ref.InnerIndex]; symbol.Kind == ast.SymbolUnbound &&
+									symbol.OriginalName == "module" && !symbol.Flags.Has(ast.DidWarnAboutCommonJSInESM) {
+									// "module.exports.foo = ..."
+									name = symbol.OriginalName
+									loc = target.Target.Loc
+									symbol.Flags |= ast.DidWarnAboutCommonJSInESM
+								}
+							}
+						}
+					}
+
+					if name != "" {
+						kind := logger.Warning
+						if p.suppressWarningsAboutWeirdCode {
+							kind = logger.Debug
+						}
+						why, notes := p.whyESModule()
+						if why == whyESMTypeModulePackageJSON {
+							text := "Node's package format requires that CommonJS files in a \"type\": \"module\" package use the \".cjs\" file extension."
+							if p.options.ts.Parse {
+								text += " If you are using TypeScript, you can use the \".cts\" file extension with esbuild instead."
+							}
+							notes = append(notes, logger.MsgData{Text: text})
+						}
+						p.log.AddIDWithNotes(logger.MsgID_JS_CommonJSVariableInESM, kind, &p.tracker, js_lexer.RangeOfIdentifier(p.source, loc),
+							fmt.Sprintf("The CommonJS %q variable is treated as a global variable in an ECMAScript module and may not work as expected", name),
+							notes)
+					}
+				}
+			}
+		}
+
+	case js_ast.BinOpAddAssign:
+		if result := p.maybeLowerSetBinOp(e.Left, js_ast.BinOpAdd, e.Right); result.Data != nil {
+			return result
+		}
+
+	case js_ast.BinOpSubAssign:
+		if result := p.maybeLowerSetBinOp(e.Left, js_ast.BinOpSub, e.Right); result.Data != nil {
+			return result
+		}
+
+	case js_ast.BinOpMulAssign:
+		if result := p.maybeLowerSetBinOp(e.Left, js_ast.BinOpMul, e.Right); result.Data != nil {
+			return result
+		}
+
+	case js_ast.BinOpDivAssign:
+		if result := p.maybeLowerSetBinOp(e.Left, js_ast.BinOpDiv, e.Right); result.Data != nil {
+			return result
+		}
+
+	case js_ast.BinOpRemAssign:
+		if result := p.maybeLowerSetBinOp(e.Left, js_ast.BinOpRem, e.Right); result.Data != nil {
+			return result
+		}
+
+	case js_ast.BinOpPowAssign:
+		// Lower the exponentiation operator for browsers that don't support it
+		if p.options.unsupportedJSFeatures.Has(compat.ExponentOperator) {
+			return p.lowerExponentiationAssignmentOperator(v.loc, e)
+		}
+
+		if result := p.maybeLowerSetBinOp(e.Left, js_ast.BinOpPow, e.Right); result.Data != nil {
+			return result
+		}
+
+	case js_ast.BinOpShlAssign:
+		if result := p.maybeLowerSetBinOp(e.Left, js_ast.BinOpShl, e.Right); result.Data != nil {
+			return result
+		}
+
+	case js_ast.BinOpShrAssign:
+		if result := p.maybeLowerSetBinOp(e.Left, js_ast.BinOpShr, e.Right); result.Data != nil {
+			return result
+		}
+
+	case js_ast.BinOpUShrAssign:
+		if result := p.maybeLowerSetBinOp(e.Left, js_ast.BinOpUShr, e.Right); result.Data != nil {
+			return result
+		}
+
+	case js_ast.BinOpBitwiseOrAssign:
+		if result := p.maybeLowerSetBinOp(e.Left, js_ast.BinOpBitwiseOr, e.Right); result.Data != nil {
+			return result
+		}
+
+	case js_ast.BinOpBitwiseAndAssign:
+		if result := p.maybeLowerSetBinOp(e.Left, js_ast.BinOpBitwiseAnd, e.Right); result.Data != nil {
+			return result
+		}
+
+	case js_ast.BinOpBitwiseXorAssign:
+		if result := p.maybeLowerSetBinOp(e.Left, js_ast.BinOpBitwiseXor, e.Right); result.Data != nil {
+			return result
+		}
+
+	case js_ast.BinOpNullishCoalescingAssign:
+		if value, ok := p.lowerNullishCoalescingAssignmentOperator(v.loc, e); ok {
+			return value
+		}
+
+	case js_ast.BinOpLogicalAndAssign:
+		if value, ok := p.lowerLogicalAssignmentOperator(v.loc, e, js_ast.BinOpLogicalAnd); ok {
+			return value
+		}
+
+	case js_ast.BinOpLogicalOrAssign:
+		if value, ok := p.lowerLogicalAssignmentOperator(v.loc, e, js_ast.BinOpLogicalOr); ok {
+			return value
+		}
+	}
+
+	// "(a, b) + c" => "a, b + c"
+	if p.options.minifySyntax && e.Op != js_ast.BinOpComma {
+		if comma, ok := e.Left.Data.(*js_ast.EBinary); ok && comma.Op == js_ast.BinOpComma {
+			return js_ast.JoinWithComma(comma.Left, js_ast.Expr{
+				Loc: comma.Right.Loc,
+				Data: &js_ast.EBinary{
+					Op:    e.Op,
+					Left:  comma.Right,
+					Right: e.Right,
+				},
+			})
+		}
+	}
+
+	return js_ast.Expr{Loc: v.loc, Data: e}
+}
+
+func remapExprLocsInJSON(expr *js_ast.Expr, table []logger.StringInJSTableEntry) {
+	expr.Loc = logger.RemapStringInJSLoc(table, expr.Loc)
+
+	switch e := expr.Data.(type) {
+	case *js_ast.EArray:
+		e.CloseBracketLoc = logger.RemapStringInJSLoc(table, e.CloseBracketLoc)
+		for i := range e.Items {
+			remapExprLocsInJSON(&e.Items[i], table)
+		}
+
+	case *js_ast.EObject:
+		e.CloseBraceLoc = logger.RemapStringInJSLoc(table, e.CloseBraceLoc)
+		for i := range e.Properties {
+			remapExprLocsInJSON(&e.Properties[i].Key, table)
+			remapExprLocsInJSON(&e.Properties[i].ValueOrNil, table)
+		}
+	}
+}
+
+func (p *parser) handleGlobPattern(expr js_ast.Expr, kind ast.ImportKind, prefix string, assertOrWith *ast.ImportAssertOrWith) js_ast.Expr {
+	pattern, approximateRange := p.globPatternFromExpr(expr)
+	if pattern == nil {
+		return js_ast.Expr{}
+	}
+
+	var last helpers.GlobPart
+	var parts []helpers.GlobPart
+
+	for _, part := range pattern {
+		if part.isWildcard {
+			if last.Wildcard == helpers.GlobNone {
+				if !strings.HasSuffix(last.Prefix, "/") {
+					// "`a${b}c`" => "a*c"
+					last.Wildcard = helpers.GlobAllExceptSlash
+				} else {
+					// "`a/${b}c`" => "a/**/*c"
+					last.Wildcard = helpers.GlobAllIncludingSlash
+					parts = append(parts, last)
+					last = helpers.GlobPart{Prefix: "/", Wildcard: helpers.GlobAllExceptSlash}
+				}
+			}
+		} else if part.text != "" {
+			if last.Wildcard != helpers.GlobNone {
+				parts = append(parts, last)
+				last = helpers.GlobPart{}
+			}
+			last.Prefix += part.text
+		}
+	}
+
+	parts = append(parts, last)
+
+	// Don't handle this if it's a string constant
+	if len(parts) == 1 && parts[0].Wildcard == helpers.GlobNone {
+		return js_ast.Expr{}
+	}
+
+	// We currently only support relative globs
+	if prefix := parts[0].Prefix; !strings.HasPrefix(prefix, "./") && !strings.HasPrefix(prefix, "../") {
+		return js_ast.Expr{}
+	}
+
+	ref := ast.InvalidRef
+
+	// Don't generate duplicate glob imports
+outer:
+	for _, globPattern := range p.globPatternImports {
+		// Check the kind
+		if globPattern.kind != kind {
+			continue
+		}
+
+		// Check the parts
+		if len(globPattern.parts) != len(parts) {
+			continue
+		}
+		for i := range parts {
+			if globPattern.parts[i] != parts[i] {
+				continue outer
+			}
+		}
+
+		// Check the import assertions/attributes
+		if assertOrWith == nil {
+			if globPattern.assertOrWith != nil {
+				continue
+			}
+		} else {
+			if globPattern.assertOrWith == nil {
+				continue
+			}
+			if assertOrWith.Keyword != globPattern.assertOrWith.Keyword {
+				continue
+			}
+			a := assertOrWith.Entries
+			b := globPattern.assertOrWith.Entries
+			if len(a) != len(b) {
+				continue
+			}
+			for i := range a {
+				ai := a[i]
+				bi := b[i]
+				if !helpers.UTF16EqualsUTF16(ai.Key, bi.Key) || !helpers.UTF16EqualsUTF16(ai.Value, bi.Value) {
+					continue outer
+				}
+			}
+		}
+
+		// If we get here, then these are the same glob pattern
+		ref = globPattern.ref
+		break
+	}
+
+	// If there's no duplicate glob import, then generate a new glob import
+	if ref == ast.InvalidRef && prefix != "" {
+		sb := strings.Builder{}
+		sb.WriteString(prefix)
+
+		for _, part := range parts {
+			gap := true
+			for _, c := range part.Prefix {
+				if !js_ast.IsIdentifierContinue(c) {
+					gap = true
+				} else {
+					if gap {
+						sb.WriteByte('_')
+						gap = false
+					}
+					sb.WriteRune(c)
+				}
+			}
+		}
+
+		name := sb.String()
+		ref = p.newSymbol(ast.SymbolOther, name)
+		p.moduleScope.Generated = append(p.moduleScope.Generated, ref)
+
+		p.globPatternImports = append(p.globPatternImports, globPatternImport{
+			assertOrWith:     assertOrWith,
+			parts:            parts,
+			name:             name,
+			approximateRange: approximateRange,
+			ref:              ref,
+			kind:             kind,
+		})
+	}
+
+	p.recordUsage(ref)
+	return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ECall{
+		Target: js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EIdentifier{Ref: ref}},
+		Args:   []js_ast.Expr{expr},
+	}}
+}
+
+type globPart struct {
+	text       string
+	isWildcard bool
+}
+
+func (p *parser) globPatternFromExpr(expr js_ast.Expr) ([]globPart, logger.Range) {
+	switch e := expr.Data.(type) {
+	case *js_ast.EString:
+		return []globPart{{text: helpers.UTF16ToString(e.Value)}}, p.source.RangeOfString(expr.Loc)
+
+	case *js_ast.ETemplate:
+		if e.TagOrNil.Data != nil {
+			break
+		}
+
+		pattern := make([]globPart, 0, 1+2*len(e.Parts))
+		pattern = append(pattern, globPart{text: helpers.UTF16ToString(e.HeadCooked)})
+
+		for _, part := range e.Parts {
+			if partPattern, _ := p.globPatternFromExpr(part.Value); partPattern != nil {
+				pattern = append(pattern, partPattern...)
+			} else {
+				pattern = append(pattern, globPart{isWildcard: true})
+			}
+			pattern = append(pattern, globPart{text: helpers.UTF16ToString(part.TailCooked)})
+		}
+
+		if len(e.Parts) == 0 {
+			return pattern, p.source.RangeOfString(expr.Loc)
+		}
+
+		text := p.source.Contents
+		templateRange := logger.Range{Loc: e.HeadLoc}
+
+		for i := e.Parts[len(e.Parts)-1].TailLoc.Start; i < int32(len(text)); i++ {
+			c := text[i]
+			if c == '`' {
+				templateRange.Len = i + 1 - templateRange.Loc.Start
+				break
+			} else if c == '\\' {
+				i += 1
+			}
+		}
+
+		return pattern, templateRange
+
+	case *js_ast.EBinary:
+		if e.Op != js_ast.BinOpAdd {
+			break
+		}
+
+		pattern, leftRange := p.globPatternFromExpr(e.Left)
+		if pattern == nil {
+			break
+		}
+
+		if rightPattern, rightRange := p.globPatternFromExpr(e.Right); rightPattern != nil {
+			pattern = append(pattern, rightPattern...)
+			leftRange.Len = rightRange.End() - leftRange.Loc.Start
+			return pattern, leftRange
+		}
+
+		pattern = append(pattern, globPart{isWildcard: true})
+
+		// Try to extend the left range by the right operand in some common cases
+		switch right := e.Right.Data.(type) {
+		case *js_ast.EIdentifier:
+			leftRange.Len = js_lexer.RangeOfIdentifier(p.source, e.Right.Loc).End() - leftRange.Loc.Start
+
+		case *js_ast.ECall:
+			if right.CloseParenLoc.Start > 0 {
+				leftRange.Len = right.CloseParenLoc.Start + 1 - leftRange.Loc.Start
+			}
+		}
+
+		return pattern, leftRange
+	}
+
+	return nil, logger.Range{}
+}
+
+func (p *parser) convertSymbolUseToCall(ref ast.Ref, isSingleNonSpreadArgCall bool) {
+	// Remove the normal symbol use
+	use := p.symbolUses[ref]
+	use.CountEstimate--
+	if use.CountEstimate == 0 {
+		delete(p.symbolUses, ref)
+	} else {
+		p.symbolUses[ref] = use
+	}
+
+	// Add a special symbol use instead
+	if p.symbolCallUses == nil {
+		p.symbolCallUses = make(map[ast.Ref]js_ast.SymbolCallUse)
+	}
+	callUse := p.symbolCallUses[ref]
+	callUse.CallCountEstimate++
+	if isSingleNonSpreadArgCall {
+		callUse.SingleArgNonSpreadCallCountEstimate++
+	}
+	p.symbolCallUses[ref] = callUse
+}
+
+func (p *parser) warnAboutImportNamespaceCall(target js_ast.Expr, kind importNamespaceCallKind) {
+	if p.options.outputFormat != config.FormatPreserve {
+		if id, ok := target.Data.(*js_ast.EIdentifier); ok && p.importItemsForNamespace[id.Ref].entries != nil {
+			key := importNamespaceCall{
+				ref:  id.Ref,
+				kind: kind,
+			}
+			if p.importNamespaceCCMap == nil {
+				p.importNamespaceCCMap = make(map[importNamespaceCall]bool)
+			}
+
+			// Don't log a warning for the same identifier more than once
+			if _, ok := p.importNamespaceCCMap[key]; ok {
+				return
+			}
+
+			p.importNamespaceCCMap[key] = true
+			r := js_lexer.RangeOfIdentifier(p.source, target.Loc)
+
+			var notes []logger.MsgData
+			name := p.symbols[id.Ref.InnerIndex].OriginalName
+			if member, ok := p.moduleScope.Members[name]; ok && member.Ref == id.Ref {
+				if star := p.source.RangeOfOperatorBefore(member.Loc, "*"); star.Len > 0 {
+					if as := p.source.RangeOfOperatorBefore(member.Loc, "as"); as.Len > 0 && as.Loc.Start > star.Loc.Start {
+						note := p.tracker.MsgData(
+							logger.Range{Loc: star.Loc, Len: js_lexer.RangeOfIdentifier(p.source, member.Loc).End() - star.Loc.Start},
+							fmt.Sprintf("Consider changing %q to a default import instead:", name))
+						note.Location.Suggestion = name
+						notes = append(notes, note)
+					}
+				}
+			}
+
+			if p.options.ts.Parse {
+				notes = append(notes, logger.MsgData{
+					Text: "Make sure to enable TypeScript's \"esModuleInterop\" setting so that TypeScript's type checker generates an error when you try to do this. " +
+						"You can read more about this setting here: https://www.typescriptlang.org/tsconfig#esModuleInterop",
+				})
+			}
+
+			var verb string
+			var where string
+			var noun string
+
+			switch kind {
+			case exprKindCall:
+				verb = "Calling"
+				noun = "function"
+
+			case exprKindNew:
+				verb = "Constructing"
+				noun = "constructor"
+
+			case exprKindJSXTag:
+				verb = "Using"
+				where = " in a JSX expression"
+				noun = "component"
+			}
+
+			p.log.AddIDWithNotes(logger.MsgID_JS_CallImportNamespace, logger.Warning, &p.tracker, r, fmt.Sprintf(
+				"%s %q%s will crash at run-time because it's an import namespace object, not a %s",
+				verb,
+				p.symbols[id.Ref.InnerIndex].OriginalName,
+				where,
+				noun,
+			), notes)
+		}
+	}
+}
+
+func (p *parser) maybeMarkKnownGlobalConstructorAsPure(e *js_ast.ENew) {
+	if id, ok := e.Target.Data.(*js_ast.EIdentifier); ok {
+		if symbol := p.symbols[id.Ref.InnerIndex]; symbol.Kind == ast.SymbolUnbound {
+			switch symbol.OriginalName {
+			case "WeakSet", "WeakMap":
+				n := len(e.Args)
+
+				if n == 0 {
+					// "new WeakSet()" is pure
+					e.CanBeUnwrappedIfUnused = true
+					break
+				}
+
+				if n == 1 {
+					switch arg := e.Args[0].Data.(type) {
+					case *js_ast.ENull, *js_ast.EUndefined:
+						// "new WeakSet(null)" is pure
+						// "new WeakSet(void 0)" is pure
+						e.CanBeUnwrappedIfUnused = true
+
+					case *js_ast.EArray:
+						if len(arg.Items) == 0 {
+							// "new WeakSet([])" is pure
+							e.CanBeUnwrappedIfUnused = true
+						} else {
+							// "new WeakSet([x])" is impure because an exception is thrown if "x" is not an object
+						}
+
+					default:
+						// "new WeakSet(x)" is impure because the iterator for "x" could have side effects
+					}
+				}
+
+			case "Date":
+				n := len(e.Args)
+
+				if n == 0 {
+					// "new Date()" is pure
+					e.CanBeUnwrappedIfUnused = true
+					break
+				}
+
+				if n == 1 {
+					switch js_ast.KnownPrimitiveType(e.Args[0].Data) {
+					case js_ast.PrimitiveNull, js_ast.PrimitiveUndefined, js_ast.PrimitiveBoolean, js_ast.PrimitiveNumber, js_ast.PrimitiveString:
+						// "new Date('')" is pure
+						// "new Date(0)" is pure
+						// "new Date(null)" is pure
+						// "new Date(true)" is pure
+						// "new Date(false)" is pure
+						// "new Date(undefined)" is pure
+						e.CanBeUnwrappedIfUnused = true
+
+					default:
+						// "new Date(x)" is impure because converting "x" to a string could have side effects
+					}
+				}
+
+			case "Set":
+				n := len(e.Args)
+
+				if n == 0 {
+					// "new Set()" is pure
+					e.CanBeUnwrappedIfUnused = true
+					break
+				}
+
+				if n == 1 {
+					switch e.Args[0].Data.(type) {
+					case *js_ast.EArray, *js_ast.ENull, *js_ast.EUndefined:
+						// "new Set([a, b, c])" is pure
+						// "new Set(null)" is pure
+						// "new Set(void 0)" is pure
+						e.CanBeUnwrappedIfUnused = true
+
+					default:
+						// "new Set(x)" is impure because the iterator for "x" could have side effects
+					}
+				}
+
+			case "Map":
+				n := len(e.Args)
+
+				if n == 0 {
+					// "new Map()" is pure
+					e.CanBeUnwrappedIfUnused = true
+					break
+				}
+
+				if n == 1 {
+					switch arg := e.Args[0].Data.(type) {
+					case *js_ast.ENull, *js_ast.EUndefined:
+						// "new Map(null)" is pure
+						// "new Map(void 0)" is pure
+						e.CanBeUnwrappedIfUnused = true
+
+					case *js_ast.EArray:
+						allEntriesAreArrays := true
+						for _, item := range arg.Items {
+							if _, ok := item.Data.(*js_ast.EArray); !ok {
+								// "new Map([x])" is impure because "x[0]" could have side effects
+								allEntriesAreArrays = false
+								break
+							}
+						}
+
+						// "new Map([[a, b], [c, d]])" is pure
+						if allEntriesAreArrays {
+							e.CanBeUnwrappedIfUnused = true
+						}
+
+					default:
+						// "new Map(x)" is impure because the iterator for "x" could have side effects
+					}
+				}
+			}
+		}
+	}
+}
+
+type identifierOpts struct {
+	assignTarget            js_ast.AssignTarget
+	isCallTarget            bool
+	isDeleteTarget          bool
+	preferQuotedKey         bool
+	wasOriginallyIdentifier bool
+	matchAgainstDefines     bool
+}
+
+func (p *parser) handleIdentifier(loc logger.Loc, e *js_ast.EIdentifier, opts identifierOpts) js_ast.Expr {
+	ref := e.Ref
+
+	// Substitute inlined constants
+	if p.options.minifySyntax {
+		if value, ok := p.constValues[ref]; ok {
+			p.ignoreUsage(ref)
+			return js_ast.ConstValueToExpr(loc, value)
+		}
+	}
+
+	// Capture the "arguments" variable if necessary
+	if p.fnOnlyDataVisit.argumentsRef != nil && ref == *p.fnOnlyDataVisit.argumentsRef {
+		isInsideUnsupportedArrow := p.fnOrArrowDataVisit.isArrow && p.options.unsupportedJSFeatures.Has(compat.Arrow)
+		isInsideUnsupportedAsyncArrow := p.fnOnlyDataVisit.isInsideAsyncArrowFn && p.options.unsupportedJSFeatures.Has(compat.AsyncAwait)
+		if isInsideUnsupportedArrow || isInsideUnsupportedAsyncArrow {
+			return js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: p.captureArguments()}}
+		}
+	}
+
+	// Create an error for assigning to an import namespace
+	if (opts.assignTarget != js_ast.AssignTargetNone ||
+		(opts.isDeleteTarget && p.symbols[ref.InnerIndex].ImportItemStatus == ast.ImportItemGenerated)) &&
+		p.symbols[ref.InnerIndex].Kind == ast.SymbolImport {
+		r := js_lexer.RangeOfIdentifier(p.source, loc)
+
+		// Try to come up with a setter name to try to make this message more understandable
+		var setterHint string
+		originalName := p.symbols[ref.InnerIndex].OriginalName
+		if js_ast.IsIdentifier(originalName) && originalName != "_" {
+			if len(originalName) == 1 || (len(originalName) > 1 && originalName[0] < utf8.RuneSelf) {
+				setterHint = fmt.Sprintf(" (e.g. \"set%s%s\")", strings.ToUpper(originalName[:1]), originalName[1:])
+			} else {
+				setterHint = fmt.Sprintf(" (e.g. \"set_%s\")", originalName)
+			}
+		}
+
+		notes := []logger.MsgData{{Text: "Imports are immutable in JavaScript. " +
+			fmt.Sprintf("To modify the value of this import, you must export a setter function in the "+
+				"imported file%s and then import and call that function here instead.", setterHint)}}
+
+		if p.options.mode == config.ModeBundle {
+			p.log.AddErrorWithNotes(&p.tracker, r, fmt.Sprintf("Cannot assign to import %q", originalName), notes)
+		} else {
+			kind := logger.Warning
+			if p.suppressWarningsAboutWeirdCode {
+				kind = logger.Debug
+			}
+			p.log.AddIDWithNotes(logger.MsgID_JS_AssignToImport, kind, &p.tracker, r,
+				fmt.Sprintf("This assignment will throw because %q is an import", originalName), notes)
+		}
+	}
+
+	// Substitute an EImportIdentifier now if this has a namespace alias
+	if opts.assignTarget == js_ast.AssignTargetNone && !opts.isDeleteTarget {
+		symbol := &p.symbols[ref.InnerIndex]
+		if nsAlias := symbol.NamespaceAlias; nsAlias != nil {
+			data := p.dotOrMangledPropVisit(
+				js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: nsAlias.NamespaceRef}},
+				symbol.OriginalName, loc)
+
+			// Handle references to namespaces or namespace members
+			if tsMemberData, ok := p.refToTSNamespaceMemberData[nsAlias.NamespaceRef]; ok {
+				if ns, ok := tsMemberData.(*js_ast.TSNamespaceMemberNamespace); ok {
+					if member, ok := ns.ExportedMembers[nsAlias.Alias]; ok {
+						switch m := member.Data.(type) {
+						case *js_ast.TSNamespaceMemberEnumNumber:
+							return p.wrapInlinedEnum(js_ast.Expr{Loc: loc, Data: &js_ast.ENumber{Value: m.Value}}, nsAlias.Alias)
+
+						case *js_ast.TSNamespaceMemberEnumString:
+							return p.wrapInlinedEnum(js_ast.Expr{Loc: loc, Data: &js_ast.EString{Value: m.Value}}, nsAlias.Alias)
+
+						case *js_ast.TSNamespaceMemberNamespace:
+							p.tsNamespaceTarget = data
+							p.tsNamespaceMemberData = member.Data
+						}
+					}
+				}
+			}
+
+			return js_ast.Expr{Loc: loc, Data: data}
+		}
+	}
+
+	// Substitute an EImportIdentifier now if this is an import item
+	if p.isImportItem[ref] {
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EImportIdentifier{
+			Ref:                     ref,
+			PreferQuotedKey:         opts.preferQuotedKey,
+			WasOriginallyIdentifier: opts.wasOriginallyIdentifier,
+		}}
+	}
+
+	// Handle references to namespaces or namespace members
+	if tsMemberData, ok := p.refToTSNamespaceMemberData[ref]; ok {
+		switch m := tsMemberData.(type) {
+		case *js_ast.TSNamespaceMemberEnumNumber:
+			return p.wrapInlinedEnum(js_ast.Expr{Loc: loc, Data: &js_ast.ENumber{Value: m.Value}}, p.symbols[ref.InnerIndex].OriginalName)
+
+		case *js_ast.TSNamespaceMemberEnumString:
+			return p.wrapInlinedEnum(js_ast.Expr{Loc: loc, Data: &js_ast.EString{Value: m.Value}}, p.symbols[ref.InnerIndex].OriginalName)
+
+		case *js_ast.TSNamespaceMemberNamespace:
+			p.tsNamespaceTarget = e
+			p.tsNamespaceMemberData = tsMemberData
+		}
+	}
+
+	// Substitute a namespace export reference now if appropriate
+	if p.options.ts.Parse {
+		if nsRef, ok := p.isExportedInsideNamespace[ref]; ok {
+			name := p.symbols[ref.InnerIndex].OriginalName
+
+			// Otherwise, create a property access on the namespace
+			p.recordUsage(nsRef)
+			propertyAccess := p.dotOrMangledPropVisit(js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: nsRef}}, name, loc)
+			if p.tsNamespaceTarget == e {
+				p.tsNamespaceTarget = propertyAccess
+			}
+			return js_ast.Expr{Loc: loc, Data: propertyAccess}
+		}
+	}
+
+	// Swap references to the global "require" function with our "__require" stub
+	if ref == p.requireRef && !opts.isCallTarget {
+		if p.options.mode == config.ModeBundle && p.source.Index != runtime.SourceIndex && e != p.dotOrIndexTarget {
+			p.log.AddID(logger.MsgID_JS_IndirectRequire, logger.Debug, &p.tracker, js_lexer.RangeOfIdentifier(p.source, loc),
+				"Indirect calls to \"require\" will not be bundled")
+		}
+
+		return p.valueToSubstituteForRequire(loc)
+	}
+
+	// Mark any mutated symbols as mutable
+	if opts.assignTarget != js_ast.AssignTargetNone {
+		p.symbols[e.Ref.InnerIndex].Flags |= ast.CouldPotentiallyBeMutated
+	}
+
+	return js_ast.Expr{Loc: loc, Data: e}
+}
+
+type visitFnOpts struct {
+	isMethod               bool
+	isDerivedClassCtor     bool
+	isLoweredPrivateMethod bool
+}
+
+func (p *parser) visitFn(fn *js_ast.Fn, scopeLoc logger.Loc, opts visitFnOpts) {
+	var decoratorScope *js_ast.Scope
+	oldFnOrArrowData := p.fnOrArrowDataVisit
+	oldFnOnlyData := p.fnOnlyDataVisit
+	p.fnOrArrowDataVisit = fnOrArrowDataVisit{
+		isAsync:                        fn.IsAsync,
+		isGenerator:                    fn.IsGenerator,
+		isDerivedClassCtor:             opts.isDerivedClassCtor,
+		shouldLowerSuperPropertyAccess: (fn.IsAsync && p.options.unsupportedJSFeatures.Has(compat.AsyncAwait)) || opts.isLoweredPrivateMethod,
+	}
+	p.fnOnlyDataVisit = fnOnlyDataVisit{
+		isThisNested:       true,
+		isNewTargetAllowed: true,
+		argumentsRef:       &fn.ArgumentsRef,
+	}
+
+	if opts.isMethod {
+		decoratorScope = p.propMethodDecoratorScope
+		p.fnOnlyDataVisit.innerClassNameRef = oldFnOnlyData.innerClassNameRef
+		p.fnOnlyDataVisit.isInStaticClassContext = oldFnOnlyData.isInStaticClassContext
+	}
+
+	if fn.Name != nil {
+		p.recordDeclaredSymbol(fn.Name.Ref)
+	}
+
+	p.pushScopeForVisitPass(js_ast.ScopeFunctionArgs, scopeLoc)
+	p.visitArgs(fn.Args, visitArgsOpts{
+		hasRestArg:               fn.HasRestArg,
+		body:                     fn.Body.Block.Stmts,
+		isUniqueFormalParameters: fn.IsUniqueFormalParameters,
+		decoratorScope:           decoratorScope,
+	})
+	p.pushScopeForVisitPass(js_ast.ScopeFunctionBody, fn.Body.Loc)
+	if fn.Name != nil {
+		p.validateDeclaredSymbolName(fn.Name.Loc, p.symbols[fn.Name.Ref.InnerIndex].OriginalName)
+	}
+	fn.Body.Block.Stmts = p.visitStmtsAndPrependTempRefs(fn.Body.Block.Stmts, prependTempRefsOpts{fnBodyLoc: &fn.Body.Loc, kind: stmtsFnBody})
+	p.popScope()
+	p.lowerFunction(&fn.IsAsync, &fn.IsGenerator, &fn.Args, fn.Body.Loc, &fn.Body.Block, nil, &fn.HasRestArg, false /* isArrow */)
+	p.popScope()
+
+	p.fnOrArrowDataVisit = oldFnOrArrowData
+	p.fnOnlyDataVisit = oldFnOnlyData
+}
+
+func (p *parser) recordExport(loc logger.Loc, alias string, ref ast.Ref) {
+	if name, ok := p.namedExports[alias]; ok {
+		// Duplicate exports are an error
+		p.log.AddErrorWithNotes(&p.tracker, js_lexer.RangeOfIdentifier(p.source, loc),
+			fmt.Sprintf("Multiple exports with the same name %q", alias),
+			[]logger.MsgData{p.tracker.MsgData(js_lexer.RangeOfIdentifier(p.source, name.AliasLoc),
+				fmt.Sprintf("The name %q was originally exported here:", alias))})
+	} else {
+		p.namedExports[alias] = js_ast.NamedExport{AliasLoc: loc, Ref: ref}
+	}
+}
+
+type importsExportsScanResult struct {
+	stmts               []js_ast.Stmt
+	keptImportEquals    bool
+	removedImportEquals bool
+}
+
+// Returns true if this is an unused TypeScript import-equals statement
+func (p *parser) checkForUnusedTSImportEquals(s *js_ast.SLocal, result *importsExportsScanResult) bool {
+	if s.WasTSImportEquals && !s.IsExport {
+		decl := s.Decls[0]
+
+		// Skip to the underlying reference
+		value := s.Decls[0].ValueOrNil
+		for {
+			if dot, ok := value.Data.(*js_ast.EDot); ok {
+				value = dot.Target
+			} else {
+				break
+			}
+		}
+
+		// Is this an identifier reference and not a require() call?
+		valueRef := ast.InvalidRef
+		switch v := value.Data.(type) {
+		case *js_ast.EIdentifier:
+			valueRef = v.Ref
+		case *js_ast.EImportIdentifier:
+			valueRef = v.Ref
+		}
+		if valueRef != ast.InvalidRef {
+			// Is this import statement unused?
+			if ref := decl.Binding.Data.(*js_ast.BIdentifier).Ref; p.symbols[ref.InnerIndex].UseCountEstimate == 0 {
+				// Also don't count the referenced identifier
+				p.ignoreUsage(valueRef)
+
+				// Import-equals statements can come in any order. Removing one
+				// could potentially cause another one to be removable too.
+				// Continue iterating until a fixed point has been reached to make
+				// sure we get them all.
+				result.removedImportEquals = true
+				return true
+			} else {
+				result.keptImportEquals = true
+			}
+		}
+	}
+
+	return false
+}
+
+func (p *parser) scanForUnusedTSImportEquals(stmts []js_ast.Stmt) (result importsExportsScanResult) {
+	stmtsEnd := 0
+
+	for _, stmt := range stmts {
+		if s, ok := stmt.Data.(*js_ast.SLocal); ok && p.checkForUnusedTSImportEquals(s, &result) {
+			// Remove unused import-equals statements, since those likely
+			// correspond to types instead of values
+			continue
+		}
+
+		// Filter out statements we skipped over
+		stmts[stmtsEnd] = stmt
+		stmtsEnd++
+	}
+
+	result.stmts = stmts[:stmtsEnd]
+	return
+}
+
+func (p *parser) scanForImportsAndExports(stmts []js_ast.Stmt) (result importsExportsScanResult) {
+	unusedImportFlags := p.options.ts.Config.UnusedImportFlags()
+	stmtsEnd := 0
+
+	for _, stmt := range stmts {
+		switch s := stmt.Data.(type) {
+		case *js_ast.SImport:
+			record := &p.importRecords[s.ImportRecordIndex]
+
+			// We implement TypeScript's "preserveValueImports" tsconfig.json setting
+			// to support the use case of compiling partial modules for compile-to-
+			// JavaScript languages such as Svelte. These languages try to reference
+			// imports in ways that are impossible for TypeScript and esbuild to know
+			// about when they are only given a partial module to compile. Here is an
+			// example of some Svelte code that contains a TypeScript snippet:
+			//
+			//   <script lang="ts">
+			//     import Counter from './Counter.svelte';
+			//     export let name: string = 'world';
+			//   </script>
+			//   <main>
+			//     <h1>Hello {name}!</h1>
+			//     <Counter />
+			//   </main>
+			//
+			// Tools that use esbuild to compile TypeScript code inside a Svelte
+			// file like this only give esbuild the contents of the <script> tag.
+			// The "preserveValueImports" setting avoids removing unused import
+			// names, which means additional code appended after the TypeScript-
+			// to-JavaScript conversion can still access those unused imports.
+			//
+			// There are two scenarios where we don't do this:
+			//
+			//   * If we're bundling, then we know we aren't being used to compile
+			//     a partial module. The parser is seeing the entire code for the
+			//     module so it's safe to remove unused imports. And also we don't
+			//     want the linker to generate errors about missing imports if the
+			//     imported file is also in the bundle.
+			//
+			//   * If identifier minification is enabled, then using esbuild as a
+			//     partial-module transform library wouldn't work anyway because
+			//     the names wouldn't match. And that means we're minifying so the
+			//     user is expecting the output to be as small as possible. So we
+			//     should omit unused imports.
+			//
+			keepUnusedImports := p.options.ts.Parse && (unusedImportFlags&config.TSUnusedImport_KeepValues) != 0 &&
+				p.options.mode != config.ModeBundle && !p.options.minifyIdentifiers
+
+			// Forbid non-default imports for JSON import assertions
+			if (record.Flags&ast.AssertTypeJSON) != 0 && p.options.mode == config.ModeBundle && s.Items != nil {
+				for _, item := range *s.Items {
+					if p.options.ts.Parse && p.tsUseCounts[item.Name.Ref.InnerIndex] == 0 && (unusedImportFlags&config.TSUnusedImport_KeepValues) == 0 {
+						// Do not count imports that TypeScript interprets as type annotations
+						continue
+					}
+					if item.Alias != "default" {
+						p.log.AddErrorWithNotes(&p.tracker, js_lexer.RangeOfIdentifier(p.source, item.AliasLoc),
+							fmt.Sprintf("Cannot use non-default import %q with a JSON import assertion", item.Alias),
+							p.notesForAssertTypeJSON(record, item.Alias))
+					}
+				}
+			}
+
+			// TypeScript always trims unused imports. This is important for
+			// correctness since some imports might be fake (only in the type
+			// system and used for type-only imports).
+			if (p.options.minifySyntax || p.options.ts.Parse) && !keepUnusedImports {
+				foundImports := false
+				isUnusedInTypeScript := true
+
+				// Remove the default name if it's unused
+				if s.DefaultName != nil {
+					foundImports = true
+					symbol := p.symbols[s.DefaultName.Ref.InnerIndex]
+
+					// TypeScript has a separate definition of unused
+					if p.options.ts.Parse && (p.tsUseCounts[s.DefaultName.Ref.InnerIndex] != 0 || (p.options.ts.Config.UnusedImportFlags()&config.TSUnusedImport_KeepValues) != 0) {
+						isUnusedInTypeScript = false
+					}
+
+					// Remove the symbol if it's never used outside a dead code region
+					if symbol.UseCountEstimate == 0 && (p.options.ts.Parse || !p.moduleScope.ContainsDirectEval) {
+						s.DefaultName = nil
+					}
+				}
+
+				// Remove the star import if it's unused
+				if s.StarNameLoc != nil {
+					foundImports = true
+					symbol := p.symbols[s.NamespaceRef.InnerIndex]
+
+					// TypeScript has a separate definition of unused
+					if p.options.ts.Parse && (p.tsUseCounts[s.NamespaceRef.InnerIndex] != 0 || (p.options.ts.Config.UnusedImportFlags()&config.TSUnusedImport_KeepValues) != 0) {
+						isUnusedInTypeScript = false
+					}
+
+					// Remove the symbol if it's never used outside a dead code region
+					if symbol.UseCountEstimate == 0 && (p.options.ts.Parse || !p.moduleScope.ContainsDirectEval) {
+						// Make sure we don't remove this if it was used for a property
+						// access while bundling
+						if importItems, ok := p.importItemsForNamespace[s.NamespaceRef]; ok && len(importItems.entries) == 0 {
+							s.StarNameLoc = nil
+						}
+					}
+				}
+
+				// Remove items if they are unused
+				if s.Items != nil {
+					foundImports = true
+					itemsEnd := 0
+
+					for _, item := range *s.Items {
+						symbol := p.symbols[item.Name.Ref.InnerIndex]
+
+						// TypeScript has a separate definition of unused
+						if p.options.ts.Parse && (p.tsUseCounts[item.Name.Ref.InnerIndex] != 0 || (p.options.ts.Config.UnusedImportFlags()&config.TSUnusedImport_KeepValues) != 0) {
+							isUnusedInTypeScript = false
+						}
+
+						// Remove the symbol if it's never used outside a dead code region
+						if symbol.UseCountEstimate != 0 || (!p.options.ts.Parse && p.moduleScope.ContainsDirectEval) {
+							(*s.Items)[itemsEnd] = item
+							itemsEnd++
+						}
+					}
+
+					// Filter the array by taking a slice
+					if itemsEnd == 0 {
+						s.Items = nil
+					} else {
+						*s.Items = (*s.Items)[:itemsEnd]
+					}
+				}
+
+				// Omit this statement if we're parsing TypeScript and all imports are
+				// unused. Note that this is distinct from the case where there were
+				// no imports at all (e.g. "import 'foo'"). In that case we want to keep
+				// the statement because the user is clearly trying to import the module
+				// for side effects.
+				//
+				// This culling is important for correctness when parsing TypeScript
+				// because a) the TypeScript compiler does ths and we want to match it
+				// and b) this may be a fake module that only exists in the type system
+				// and doesn't actually exist in reality.
+				//
+				// We do not want to do this culling in JavaScript though because the
+				// module may have side effects even if all imports are unused.
+				if p.options.ts.Parse && foundImports && isUnusedInTypeScript && (unusedImportFlags&config.TSUnusedImport_KeepStmt) == 0 {
+					// Ignore import records with a pre-filled source index. These are
+					// for injected files and we definitely do not want to trim these.
+					if !record.SourceIndex.IsValid() && !record.CopySourceIndex.IsValid() {
+						record.Flags |= ast.IsUnused
+						continue
+					}
+				}
+			}
+
+			if p.options.mode != config.ModePassThrough {
+				if s.StarNameLoc != nil {
+					// "importItemsForNamespace" has property accesses off the namespace
+					if importItems, ok := p.importItemsForNamespace[s.NamespaceRef]; ok && len(importItems.entries) > 0 {
+						// Sort keys for determinism
+						sorted := make([]string, 0, len(importItems.entries))
+						for alias := range importItems.entries {
+							sorted = append(sorted, alias)
+						}
+						sort.Strings(sorted)
+
+						// Create named imports for these property accesses. This will
+						// cause missing imports to generate useful warnings.
+						//
+						// It will also improve bundling efficiency for internal imports
+						// by still converting property accesses off the namespace into
+						// bare identifiers even if the namespace is still needed.
+						for _, alias := range sorted {
+							name := importItems.entries[alias]
+							p.namedImports[name.Ref] = js_ast.NamedImport{
+								Alias:             alias,
+								AliasLoc:          name.Loc,
+								NamespaceRef:      s.NamespaceRef,
+								ImportRecordIndex: s.ImportRecordIndex,
+							}
+
+							// Make sure the printer prints this as a property access
+							p.symbols[name.Ref.InnerIndex].NamespaceAlias = &ast.NamespaceAlias{
+								NamespaceRef: s.NamespaceRef,
+								Alias:        alias,
+							}
+
+							// Also record these automatically-generated top-level namespace alias symbols
+							p.declaredSymbols = append(p.declaredSymbols, js_ast.DeclaredSymbol{
+								Ref:        name.Ref,
+								IsTopLevel: true,
+							})
+						}
+					}
+				}
+
+				if s.DefaultName != nil {
+					p.namedImports[s.DefaultName.Ref] = js_ast.NamedImport{
+						Alias:             "default",
+						AliasLoc:          s.DefaultName.Loc,
+						NamespaceRef:      s.NamespaceRef,
+						ImportRecordIndex: s.ImportRecordIndex,
+					}
+				}
+
+				if s.StarNameLoc != nil {
+					p.namedImports[s.NamespaceRef] = js_ast.NamedImport{
+						AliasIsStar:       true,
+						AliasLoc:          *s.StarNameLoc,
+						NamespaceRef:      ast.InvalidRef,
+						ImportRecordIndex: s.ImportRecordIndex,
+					}
+				}
+
+				if s.Items != nil {
+					for _, item := range *s.Items {
+						p.namedImports[item.Name.Ref] = js_ast.NamedImport{
+							Alias:             item.Alias,
+							AliasLoc:          item.AliasLoc,
+							NamespaceRef:      s.NamespaceRef,
+							ImportRecordIndex: s.ImportRecordIndex,
+						}
+					}
+				}
+			}
+
+			p.importRecordsForCurrentPart = append(p.importRecordsForCurrentPart, s.ImportRecordIndex)
+
+			if s.StarNameLoc != nil {
+				record.Flags |= ast.ContainsImportStar
+			}
+
+			if s.DefaultName != nil {
+				record.Flags |= ast.ContainsDefaultAlias
+			} else if s.Items != nil {
+				for _, item := range *s.Items {
+					if item.Alias == "default" {
+						record.Flags |= ast.ContainsDefaultAlias
+					} else if item.Alias == "__esModule" {
+						record.Flags |= ast.ContainsESModuleAlias
+					}
+				}
+			}
+
+		case *js_ast.SFunction:
+			if s.IsExport {
+				p.recordExport(s.Fn.Name.Loc, p.symbols[s.Fn.Name.Ref.InnerIndex].OriginalName, s.Fn.Name.Ref)
+			}
+
+		case *js_ast.SClass:
+			if s.IsExport {
+				p.recordExport(s.Class.Name.Loc, p.symbols[s.Class.Name.Ref.InnerIndex].OriginalName, s.Class.Name.Ref)
+			}
+
+		case *js_ast.SLocal:
+			if s.IsExport {
+				js_ast.ForEachIdentifierBindingInDecls(s.Decls, func(loc logger.Loc, b *js_ast.BIdentifier) {
+					p.recordExport(loc, p.symbols[b.Ref.InnerIndex].OriginalName, b.Ref)
+				})
+			}
+
+			// Remove unused import-equals statements, since those likely
+			// correspond to types instead of values
+			if p.checkForUnusedTSImportEquals(s, &result) {
+				continue
+			}
+
+		case *js_ast.SExportDefault:
+			p.recordExport(s.DefaultName.Loc, "default", s.DefaultName.Ref)
+
+		case *js_ast.SExportClause:
+			for _, item := range s.Items {
+				p.recordExport(item.AliasLoc, item.Alias, item.Name.Ref)
+			}
+
+		case *js_ast.SExportStar:
+			record := &p.importRecords[s.ImportRecordIndex]
+			p.importRecordsForCurrentPart = append(p.importRecordsForCurrentPart, s.ImportRecordIndex)
+
+			if s.Alias != nil {
+				// "export * as ns from 'path'"
+				p.namedImports[s.NamespaceRef] = js_ast.NamedImport{
+					AliasIsStar:       true,
+					AliasLoc:          s.Alias.Loc,
+					NamespaceRef:      ast.InvalidRef,
+					ImportRecordIndex: s.ImportRecordIndex,
+					IsExported:        true,
+				}
+				p.recordExport(s.Alias.Loc, s.Alias.OriginalName, s.NamespaceRef)
+
+				record.Flags |= ast.ContainsImportStar
+			} else {
+				// "export * from 'path'"
+				p.exportStarImportRecords = append(p.exportStarImportRecords, s.ImportRecordIndex)
+			}
+
+		case *js_ast.SExportFrom:
+			record := &p.importRecords[s.ImportRecordIndex]
+			p.importRecordsForCurrentPart = append(p.importRecordsForCurrentPart, s.ImportRecordIndex)
+
+			for _, item := range s.Items {
+				// Note that the imported alias is not item.Alias, which is the
+				// exported alias. This is somewhat confusing because each
+				// SExportFrom statement is basically SImport + SExportClause in one.
+				p.namedImports[item.Name.Ref] = js_ast.NamedImport{
+					Alias:             item.OriginalName,
+					AliasLoc:          item.Name.Loc,
+					NamespaceRef:      s.NamespaceRef,
+					ImportRecordIndex: s.ImportRecordIndex,
+					IsExported:        true,
+				}
+				p.recordExport(item.Name.Loc, item.Alias, item.Name.Ref)
+
+				if item.OriginalName == "default" {
+					record.Flags |= ast.ContainsDefaultAlias
+				} else if item.OriginalName == "__esModule" {
+					record.Flags |= ast.ContainsESModuleAlias
+				}
+			}
+
+			// Forbid non-default imports for JSON import assertions
+			if (record.Flags&ast.AssertTypeJSON) != 0 && p.options.mode == config.ModeBundle {
+				for _, item := range s.Items {
+					if item.OriginalName != "default" {
+						p.log.AddErrorWithNotes(&p.tracker, js_lexer.RangeOfIdentifier(p.source, item.Name.Loc),
+							fmt.Sprintf("Cannot use non-default import %q with a JSON import assertion", item.OriginalName),
+							p.notesForAssertTypeJSON(record, item.OriginalName))
+					}
+				}
+			}
+
+			// TypeScript always trims unused re-exports. This is important for
+			// correctness since some re-exports might be fake (only in the type
+			// system and used for type-only stuff).
+			if p.options.ts.Parse && len(s.Items) == 0 && (unusedImportFlags&config.TSUnusedImport_KeepStmt) == 0 {
+				continue
+			}
+		}
+
+		// Filter out statements we skipped over
+		stmts[stmtsEnd] = stmt
+		stmtsEnd++
+	}
+
+	result.stmts = stmts[:stmtsEnd]
+	return
+}
+
+func (p *parser) appendPart(parts []js_ast.Part, stmts []js_ast.Stmt) []js_ast.Part {
+	p.symbolUses = make(map[ast.Ref]js_ast.SymbolUse)
+	p.importSymbolPropertyUses = nil
+	p.symbolCallUses = nil
+	p.declaredSymbols = nil
+	p.importRecordsForCurrentPart = nil
+	p.scopesForCurrentPart = nil
+
+	part := js_ast.Part{
+		Stmts:      p.visitStmtsAndPrependTempRefs(stmts, prependTempRefsOpts{}),
+		SymbolUses: p.symbolUses,
+	}
+
+	// Sanity check
+	if p.currentScope != p.moduleScope {
+		panic("Internal error: Scope stack imbalance")
+	}
+
+	// Insert any relocated variable statements now
+	if len(p.relocatedTopLevelVars) > 0 {
+		alreadyDeclared := make(map[ast.Ref]bool)
+		for _, local := range p.relocatedTopLevelVars {
+			// Follow links because "var" declarations may be merged due to hoisting
+			for {
+				link := p.symbols[local.Ref.InnerIndex].Link
+				if link == ast.InvalidRef {
+					break
+				}
+				local.Ref = link
+			}
+
+			// Only declare a given relocated variable once
+			if !alreadyDeclared[local.Ref] {
+				alreadyDeclared[local.Ref] = true
+				part.Stmts = append(part.Stmts, js_ast.Stmt{Loc: local.Loc, Data: &js_ast.SLocal{
+					Decls: []js_ast.Decl{{
+						Binding: js_ast.Binding{Loc: local.Loc, Data: &js_ast.BIdentifier{Ref: local.Ref}},
+					}},
+				}})
+			}
+		}
+		p.relocatedTopLevelVars = nil
+	}
+
+	if len(part.Stmts) > 0 {
+		var flags js_ast.StmtsCanBeRemovedIfUnusedFlags
+		if p.options.mode == config.ModePassThrough {
+			// Exports are tracked separately, so export clauses can normally always
+			// be removed. Except we should keep them if we're not doing any format
+			// conversion because exports are not re-emitted in that case.
+			flags |= js_ast.KeepExportClauses
+		}
+		part.CanBeRemovedIfUnused = p.astHelpers.StmtsCanBeRemovedIfUnused(part.Stmts, flags)
+		part.DeclaredSymbols = p.declaredSymbols
+		part.ImportRecordIndices = p.importRecordsForCurrentPart
+		part.ImportSymbolPropertyUses = p.importSymbolPropertyUses
+		part.SymbolCallUses = p.symbolCallUses
+		part.Scopes = p.scopesForCurrentPart
+		parts = append(parts, part)
+	}
+	return parts
+}
+
+func newParser(log logger.Log, source logger.Source, lexer js_lexer.Lexer, options *Options) *parser {
+	if options.defines == nil {
+		defaultDefines := config.ProcessDefines(nil)
+		options.defines = &defaultDefines
+	}
+
+	p := &parser{
+		log:                log,
+		source:             source,
+		tracker:            logger.MakeLineColumnTracker(&source),
+		lexer:              lexer,
+		allowIn:            true,
+		options:            *options,
+		runtimeImports:     make(map[string]ast.LocRef),
+		promiseRef:         ast.InvalidRef,
+		regExpRef:          ast.InvalidRef,
+		afterArrowBodyLoc:  logger.Loc{Start: -1},
+		firstJSXElementLoc: logger.Loc{Start: -1},
+		importMetaRef:      ast.InvalidRef,
+		superCtorRef:       ast.InvalidRef,
+
+		// For lowering private methods
+		weakMapRef:     ast.InvalidRef,
+		weakSetRef:     ast.InvalidRef,
+		privateGetters: make(map[ast.Ref]ast.Ref),
+		privateSetters: make(map[ast.Ref]ast.Ref),
+
+		// These are for TypeScript
+		refToTSNamespaceMemberData: make(map[ast.Ref]js_ast.TSNamespaceMemberData),
+		emittedNamespaceVars:       make(map[ast.Ref]bool),
+		isExportedInsideNamespace:  make(map[ast.Ref]ast.Ref),
+		localTypeNames:             make(map[string]bool),
+
+		// These are for handling ES6 imports and exports
+		importItemsForNamespace: make(map[ast.Ref]namespaceImportItems),
+		isImportItem:            make(map[ast.Ref]bool),
+		namedImports:            make(map[ast.Ref]js_ast.NamedImport),
+		namedExports:            make(map[string]js_ast.NamedExport),
+
+		// For JSX runtime imports
+		jsxRuntimeImports: make(map[string]ast.LocRef),
+		jsxLegacyImports:  make(map[string]ast.LocRef),
+
+		suppressWarningsAboutWeirdCode: helpers.IsInsideNodeModules(source.KeyPath.Text),
+	}
+
+	if len(options.dropLabels) > 0 {
+		p.dropLabelsMap = make(map[string]struct{})
+		for _, name := range options.dropLabels {
+			p.dropLabelsMap[name] = struct{}{}
+		}
+	}
+
+	if !options.minifyWhitespace {
+		p.exprComments = make(map[logger.Loc][]string)
+	}
+
+	p.astHelpers = js_ast.MakeHelperContext(func(ref ast.Ref) bool {
+		return p.symbols[ref.InnerIndex].Kind == ast.SymbolUnbound
+	})
+
+	p.pushScopeForParsePass(js_ast.ScopeEntry, logger.Loc{Start: locModuleScope})
+
+	return p
+}
+
+var defaultJSXFactory = []string{"React", "createElement"}
+var defaultJSXFragment = []string{"React", "Fragment"}
+
+const defaultJSXImportSource = "react"
+
+func Parse(log logger.Log, source logger.Source, options Options) (result js_ast.AST, ok bool) {
+	ok = true
+	defer func() {
+		r := recover()
+		if _, isLexerPanic := r.(js_lexer.LexerPanic); isLexerPanic {
+			ok = false
+		} else if r != nil {
+			panic(r)
+		}
+	}()
+
+	// Default options for JSX elements
+	if len(options.jsx.Factory.Parts) == 0 {
+		options.jsx.Factory = config.DefineExpr{Parts: defaultJSXFactory}
+	}
+	if len(options.jsx.Fragment.Parts) == 0 && options.jsx.Fragment.Constant == nil {
+		options.jsx.Fragment = config.DefineExpr{Parts: defaultJSXFragment}
+	}
+	if len(options.jsx.ImportSource) == 0 {
+		options.jsx.ImportSource = defaultJSXImportSource
+	}
+
+	p := newParser(log, source, js_lexer.NewLexer(log, source, options.ts), &options)
+
+	// Consume a leading hashbang comment
+	hashbang := ""
+	if p.lexer.Token == js_lexer.THashbang {
+		hashbang = p.lexer.Identifier.String
+		p.lexer.Next()
+	}
+
+	// Allow top-level await
+	p.fnOrArrowDataParse.await = allowExpr
+	p.fnOrArrowDataParse.isTopLevel = true
+
+	// Parse the file in the first pass, but do not bind symbols
+	stmts := p.parseStmtsUpTo(js_lexer.TEndOfFile, parseStmtOpts{
+		isModuleScope:          true,
+		allowDirectivePrologue: true,
+	})
+	p.prepareForVisitPass()
+
+	// Insert a "use strict" directive if "alwaysStrict" is active
+	var directives []string
+	if tsAlwaysStrict := p.options.tsAlwaysStrict; tsAlwaysStrict != nil && tsAlwaysStrict.Value {
+		directives = append(directives, "use strict")
+	}
+
+	// Strip off all leading directives
+	{
+		totalCount := 0
+		keptCount := 0
+
+		for _, stmt := range stmts {
+			switch s := stmt.Data.(type) {
+			case *js_ast.SComment:
+				stmts[keptCount] = stmt
+				keptCount++
+				totalCount++
+				continue
+
+			case *js_ast.SDirective:
+				if p.isStrictMode() && s.LegacyOctalLoc.Start > 0 {
+					p.markStrictModeFeature(legacyOctalEscape, p.source.RangeOfLegacyOctalEscape(s.LegacyOctalLoc), "")
+				}
+				directive := helpers.UTF16ToString(s.Value)
+
+				// Remove duplicate directives
+				found := false
+				for _, existing := range directives {
+					if existing == directive {
+						found = true
+						break
+					}
+				}
+				if !found {
+					directives = append(directives, directive)
+				}
+
+				// Remove this directive from the statement list
+				totalCount++
+				continue
+			}
+
+			// Stop when the directive prologue ends
+			break
+		}
+
+		if keptCount < totalCount {
+			stmts = append(stmts[:keptCount], stmts[totalCount:]...)
+		}
+	}
+
+	// Add an empty part for the namespace export that we can fill in later
+	nsExportPart := js_ast.Part{
+		SymbolUses:           make(map[ast.Ref]js_ast.SymbolUse),
+		CanBeRemovedIfUnused: true,
+	}
+
+	var before = []js_ast.Part{nsExportPart}
+	var parts []js_ast.Part
+	var after []js_ast.Part
+
+	// Insert any injected import statements now that symbols have been declared
+	for _, file := range p.options.injectedFiles {
+		exportsNoConflict := make([]string, 0, len(file.Exports))
+		symbols := make(map[string]ast.LocRef)
+
+		if file.DefineName != "" {
+			ref := p.newSymbol(ast.SymbolOther, file.DefineName)
+			p.moduleScope.Generated = append(p.moduleScope.Generated, ref)
+			symbols["default"] = ast.LocRef{Ref: ref}
+			exportsNoConflict = append(exportsNoConflict, "default")
+			p.injectedDefineSymbols = append(p.injectedDefineSymbols, ref)
+		} else {
+		nextExport:
+			for _, export := range file.Exports {
+				// Skip injecting this symbol if it's already declared locally (i.e. it's not a reference to a global)
+				if _, ok := p.moduleScope.Members[export.Alias]; ok {
+					continue
+				}
+
+				parts := strings.Split(export.Alias, ".")
+
+				// The key must be a dot-separated identifier list
+				for _, part := range parts {
+					if !js_ast.IsIdentifier(part) {
+						continue nextExport
+					}
+				}
+
+				ref := p.newSymbol(ast.SymbolInjected, export.Alias)
+				symbols[export.Alias] = ast.LocRef{Ref: ref}
+				if len(parts) == 1 {
+					// Handle the identifier case by generating an injected symbol directly
+					p.moduleScope.Members[export.Alias] = js_ast.ScopeMember{Ref: ref}
+				} else {
+					// Handle the dot case using a map. This map is similar to the map
+					// "options.defines.DotDefines" but is kept separate instead of being
+					// implemented using the same mechanism because we allow you to use
+					// "define" to rewrite something to an injected symbol (i.e. we allow
+					// two levels of mappings). This was historically necessary to be able
+					// to map a dot name to an injected symbol because we previously didn't
+					// support dot names as injected symbols. But now dot names as injected
+					// symbols has been implemented, so supporting two levels of mappings
+					// is only for backward-compatibility.
+					if p.injectedDotNames == nil {
+						p.injectedDotNames = make(map[string][]injectedDotName)
+					}
+					tail := parts[len(parts)-1]
+					p.injectedDotNames[tail] = append(p.injectedDotNames[tail], injectedDotName{parts: parts, injectedDefineIndex: uint32(len(p.injectedDefineSymbols))})
+					p.injectedDefineSymbols = append(p.injectedDefineSymbols, ref)
+				}
+				exportsNoConflict = append(exportsNoConflict, export.Alias)
+				if p.injectedSymbolSources == nil {
+					p.injectedSymbolSources = make(map[ast.Ref]injectedSymbolSource)
+				}
+				p.injectedSymbolSources[ref] = injectedSymbolSource{
+					source: file.Source,
+					loc:    export.Loc,
+				}
+			}
+		}
+
+		if file.IsCopyLoader {
+			before, _ = p.generateImportStmt(file.Source.KeyPath.Text, logger.Range{}, exportsNoConflict, before, symbols, nil, &file.Source.Index)
+		} else {
+			before, _ = p.generateImportStmt(file.Source.KeyPath.Text, logger.Range{}, exportsNoConflict, before, symbols, &file.Source.Index, nil)
+		}
+	}
+
+	// When "using" declarations appear at the top level, we change all TDZ
+	// variables in the top-level scope into "var" so that they aren't harmed
+	// when they are moved into the try/catch statement that lowering will
+	// generate.
+	//
+	// This is necessary because exported function declarations must be hoisted
+	// outside of the try/catch statement because they can be evaluated before
+	// this module is evaluated due to ESM cross-file function hoisting. And
+	// these function bodies might reference anything else in this scope, which
+	// must still work when those things are moved inside a try/catch statement.
+	//
+	// Before:
+	//
+	//   using foo = get()
+	//   export function fn() {
+	//     return [foo, new Bar]
+	//   }
+	//   class Bar {}
+	//
+	// After ("fn" is hoisted, "Bar" is converted to "var"):
+	//
+	//   export function fn() {
+	//     return [foo, new Bar]
+	//   }
+	//   try {
+	//     var foo = get();
+	//     var Bar = class {};
+	//   } catch (_) {
+	//     ...
+	//   } finally {
+	//     ...
+	//   }
+	//
+	// This is also necessary because other code might be appended to the code
+	// that we're processing and expect to be able to access top-level variables.
+	p.willWrapModuleInTryCatchForUsing = p.shouldLowerUsingDeclarations(stmts)
+
+	// Bind symbols in a second pass over the AST. I started off doing this in a
+	// single pass, but it turns out it's pretty much impossible to do this
+	// correctly while handling arrow functions because of the grammar
+	// ambiguities.
+	//
+	// Note that top-level lowered "using" declarations disable tree-shaking
+	// because we only do tree-shaking on top-level statements and lowering
+	// a top-level "using" declaration moves all top-level statements into a
+	// nested scope.
+	if !p.options.treeShaking || p.willWrapModuleInTryCatchForUsing {
+		// When tree shaking is disabled, everything comes in a single part
+		parts = p.appendPart(parts, stmts)
+	} else {
+		var preprocessedEnums map[int][]js_ast.Part
+		if p.scopesInOrderForEnum != nil {
+			// Preprocess TypeScript enums to improve code generation. Otherwise
+			// uses of an enum before that enum has been declared won't be inlined:
+			//
+			//   console.log(Foo.FOO) // We want "FOO" to be inlined here
+			//   const enum Foo { FOO = 0 }
+			//
+			// The TypeScript compiler itself contains code with this pattern, so
+			// it's important to implement this optimization.
+			for i, stmt := range stmts {
+				if _, ok := stmt.Data.(*js_ast.SEnum); ok {
+					if preprocessedEnums == nil {
+						preprocessedEnums = make(map[int][]js_ast.Part)
+					}
+					oldScopesInOrder := p.scopesInOrder
+					p.scopesInOrder = p.scopesInOrderForEnum[stmt.Loc]
+					preprocessedEnums[i] = p.appendPart(nil, []js_ast.Stmt{stmt})
+					p.scopesInOrder = oldScopesInOrder
+				}
+			}
+		}
+
+		// When tree shaking is enabled, each top-level statement is potentially a separate part
+		for i, stmt := range stmts {
+			switch s := stmt.Data.(type) {
+			case *js_ast.SLocal:
+				// Split up top-level multi-declaration variable statements
+				for _, decl := range s.Decls {
+					clone := *s
+					clone.Decls = []js_ast.Decl{decl}
+					parts = p.appendPart(parts, []js_ast.Stmt{{Loc: stmt.Loc, Data: &clone}})
+				}
+
+			case *js_ast.SImport, *js_ast.SExportFrom, *js_ast.SExportStar:
+				if p.options.mode != config.ModePassThrough {
+					// Move imports (and import-like exports) to the top of the file to
+					// ensure that if they are converted to a require() call, the effects
+					// will take place before any other statements are evaluated.
+					before = p.appendPart(before, []js_ast.Stmt{stmt})
+				} else {
+					// If we aren't doing any format conversion, just keep these statements
+					// inline where they were. Exports are sorted so order doesn't matter:
+					// https://262.ecma-international.org/6.0/#sec-module-namespace-exotic-objects.
+					// However, this is likely an aesthetic issue that some people will
+					// complain about. In addition, there are code transformation tools
+					// such as TypeScript and Babel with bugs where the order of exports
+					// in the file is incorrectly preserved instead of sorted, so preserving
+					// the order of exports ourselves here may be preferable.
+					parts = p.appendPart(parts, []js_ast.Stmt{stmt})
+				}
+
+			case *js_ast.SExportEquals:
+				// TypeScript "export = value;" becomes "module.exports = value;". This
+				// must happen at the end after everything is parsed because TypeScript
+				// moves this statement to the end when it generates code.
+				after = p.appendPart(after, []js_ast.Stmt{stmt})
+
+			case *js_ast.SEnum:
+				parts = append(parts, preprocessedEnums[i]...)
+				p.scopesInOrder = p.scopesInOrder[len(p.scopesInOrderForEnum[stmt.Loc]):]
+
+			default:
+				parts = p.appendPart(parts, []js_ast.Stmt{stmt})
+			}
+		}
+	}
+
+	// Insert a variable for "import.meta" at the top of the file if it was used.
+	// We don't need to worry about "use strict" directives because this only
+	// happens when bundling, in which case we are flatting the module scopes of
+	// all modules together anyway so such directives are meaningless.
+	if p.importMetaRef != ast.InvalidRef {
+		importMetaStmt := js_ast.Stmt{Data: &js_ast.SLocal{
+			Kind: p.selectLocalKind(js_ast.LocalConst),
+			Decls: []js_ast.Decl{{
+				Binding:    js_ast.Binding{Data: &js_ast.BIdentifier{Ref: p.importMetaRef}},
+				ValueOrNil: js_ast.Expr{Data: &js_ast.EObject{}},
+			}},
+		}}
+		before = append(before, js_ast.Part{
+			Stmts:                []js_ast.Stmt{importMetaStmt},
+			SymbolUses:           make(map[ast.Ref]js_ast.SymbolUse),
+			DeclaredSymbols:      []js_ast.DeclaredSymbol{{Ref: p.importMetaRef, IsTopLevel: true}},
+			CanBeRemovedIfUnused: true,
+		})
+	}
+
+	// Pop the module scope to apply the "ContainsDirectEval" rules
+	p.popScope()
+
+	result = p.toAST(before, parts, after, hashbang, directives)
+	result.SourceMapComment = p.lexer.SourceMappingURL
+	return
+}
+
+func LazyExportAST(log logger.Log, source logger.Source, options Options, expr js_ast.Expr, apiCall string) js_ast.AST {
+	// Don't create a new lexer using js_lexer.NewLexer() here since that will
+	// actually attempt to parse the first token, which might cause a syntax
+	// error.
+	p := newParser(log, source, js_lexer.Lexer{}, &options)
+	p.prepareForVisitPass()
+
+	// Optionally call a runtime API function to transform the expression
+	if apiCall != "" {
+		p.symbolUses = make(map[ast.Ref]js_ast.SymbolUse)
+		expr = p.callRuntime(expr.Loc, apiCall, []js_ast.Expr{expr})
+	}
+
+	// Add an empty part for the namespace export that we can fill in later
+	nsExportPart := js_ast.Part{
+		SymbolUses:           make(map[ast.Ref]js_ast.SymbolUse),
+		CanBeRemovedIfUnused: true,
+	}
+
+	// Defer the actual code generation until linking
+	part := js_ast.Part{
+		Stmts:      []js_ast.Stmt{{Loc: expr.Loc, Data: &js_ast.SLazyExport{Value: expr}}},
+		SymbolUses: p.symbolUses,
+	}
+	p.symbolUses = nil
+
+	ast := p.toAST([]js_ast.Part{nsExportPart}, []js_ast.Part{part}, nil, "", nil)
+	ast.HasLazyExport = true
+	return ast
+}
+
+func GlobResolveAST(log logger.Log, source logger.Source, importRecords []ast.ImportRecord, object *js_ast.EObject, name string) js_ast.AST {
+	// Don't create a new lexer using js_lexer.NewLexer() here since that will
+	// actually attempt to parse the first token, which might cause a syntax
+	// error.
+	p := newParser(log, source, js_lexer.Lexer{}, &Options{})
+	p.prepareForVisitPass()
+
+	// Add an empty part for the namespace export that we can fill in later
+	nsExportPart := js_ast.Part{
+		SymbolUses:           make(map[ast.Ref]js_ast.SymbolUse),
+		CanBeRemovedIfUnused: true,
+	}
+
+	if len(p.importRecords) != 0 {
+		panic("Internal error")
+	}
+	p.importRecords = importRecords
+
+	importRecordIndices := make([]uint32, 0, len(importRecords))
+	for importRecordIndex := range importRecords {
+		importRecordIndices = append(importRecordIndices, uint32(importRecordIndex))
+	}
+
+	p.symbolUses = make(map[ast.Ref]js_ast.SymbolUse)
+	ref := p.newSymbol(ast.SymbolOther, name)
+	p.moduleScope.Generated = append(p.moduleScope.Generated, ref)
+
+	part := js_ast.Part{
+		Stmts: []js_ast.Stmt{{Data: &js_ast.SLocal{
+			IsExport: true,
+			Decls: []js_ast.Decl{{
+				Binding:    js_ast.Binding{Data: &js_ast.BIdentifier{Ref: ref}},
+				ValueOrNil: p.callRuntime(logger.Loc{}, "__glob", []js_ast.Expr{{Data: object}}),
+			}},
+		}}},
+		ImportRecordIndices: importRecordIndices,
+		SymbolUses:          p.symbolUses,
+	}
+	p.symbolUses = nil
+
+	p.esmExportKeyword.Len = 1
+	return p.toAST([]js_ast.Part{nsExportPart}, []js_ast.Part{part}, nil, "", nil)
+}
+
+func ParseDefineExprOrJSON(text string) (config.DefineExpr, js_ast.E) {
+	if text == "" {
+		return config.DefineExpr{}, nil
+	}
+
+	// Try a property chain
+	parts := strings.Split(text, ".")
+	for i, part := range parts {
+		if !js_ast.IsIdentifier(part) {
+			parts = nil
+			break
+		}
+
+		// Don't allow most keywords as the identifier
+		if i == 0 {
+			if token, ok := js_lexer.Keywords[part]; ok && token != js_lexer.TNull && token != js_lexer.TThis &&
+				(token != js_lexer.TImport || len(parts) < 2 || parts[1] != "meta") {
+				parts = nil
+				break
+			}
+		}
+	}
+	if parts != nil {
+		return config.DefineExpr{Parts: parts}, nil
+	}
+
+	// Try parsing a JSON value
+	log := logger.NewDeferLog(logger.DeferLogNoVerboseOrDebug, nil)
+	expr, ok := ParseJSON(log, logger.Source{Contents: text}, JSONOptions{})
+	if !ok {
+		return config.DefineExpr{}, nil
+	}
+
+	// Only primitive literals are inlined directly
+	switch expr.Data.(type) {
+	case *js_ast.ENull, *js_ast.EBoolean, *js_ast.EString, *js_ast.ENumber:
+		return config.DefineExpr{Constant: expr.Data}, nil
+	}
+
+	// If it's not a primitive, return the whole compound JSON value to be injected out-of-line
+	return config.DefineExpr{}, expr.Data
+}
+
+type whyESM uint8
+
+const (
+	whyESMUnknown whyESM = iota
+	whyESMExportKeyword
+	whyESMImportMeta
+	whyESMTopLevelAwait
+	whyESMFileMJS
+	whyESMFileMTS
+	whyESMTypeModulePackageJSON
+	whyESMImportStatement
+)
+
+// Say why this the current file is being considered an ES module
+func (p *parser) whyESModule() (whyESM, []logger.MsgData) {
+	because := "This file is considered to be an ECMAScript module because"
+	switch {
+	case p.esmExportKeyword.Len > 0:
+		return whyESMExportKeyword, []logger.MsgData{p.tracker.MsgData(p.esmExportKeyword,
+			because+" of the \"export\" keyword here:")}
+
+	case p.esmImportMeta.Len > 0:
+		return whyESMImportMeta, []logger.MsgData{p.tracker.MsgData(p.esmImportMeta,
+			because+" of the use of \"import.meta\" here:")}
+
+	case p.topLevelAwaitKeyword.Len > 0:
+		return whyESMTopLevelAwait, []logger.MsgData{p.tracker.MsgData(p.topLevelAwaitKeyword,
+			because+" of the top-level \"await\" keyword here:")}
+
+	case p.options.moduleTypeData.Type == js_ast.ModuleESM_MJS:
+		return whyESMFileMJS, []logger.MsgData{{Text: because + " the file name ends in \".mjs\"."}}
+
+	case p.options.moduleTypeData.Type == js_ast.ModuleESM_MTS:
+		return whyESMFileMTS, []logger.MsgData{{Text: because + " the file name ends in \".mts\"."}}
+
+	case p.options.moduleTypeData.Type == js_ast.ModuleESM_PackageJSON:
+		tracker := logger.MakeLineColumnTracker(p.options.moduleTypeData.Source)
+		return whyESMTypeModulePackageJSON, []logger.MsgData{tracker.MsgData(p.options.moduleTypeData.Range,
+			because+" the enclosing \"package.json\" file sets the type of this file to \"module\":")}
+
+	// This case must come last because some code cares about the "import"
+	// statement keyword and some doesn't, and we don't want to give code
+	// that doesn't care about the "import" statement the wrong error message.
+	case p.esmImportStatementKeyword.Len > 0:
+		return whyESMImportStatement, []logger.MsgData{p.tracker.MsgData(p.esmImportStatementKeyword,
+			because+" of the \"import\" keyword here:")}
+	}
+	return whyESMUnknown, nil
+}
+
+func (p *parser) prepareForVisitPass() {
+	p.pushScopeForVisitPass(js_ast.ScopeEntry, logger.Loc{Start: locModuleScope})
+	p.fnOrArrowDataVisit.isOutsideFnOrArrow = true
+	p.moduleScope = p.currentScope
+
+	// Force-enable strict mode if that's the way TypeScript is configured
+	if tsAlwaysStrict := p.options.tsAlwaysStrict; tsAlwaysStrict != nil && tsAlwaysStrict.Value {
+		p.currentScope.StrictMode = js_ast.ImplicitStrictModeTSAlwaysStrict
+	}
+
+	// Determine whether or not this file is ESM
+	p.isFileConsideredToHaveESMExports =
+		p.esmExportKeyword.Len > 0 ||
+			p.esmImportMeta.Len > 0 ||
+			p.topLevelAwaitKeyword.Len > 0 ||
+			p.options.moduleTypeData.Type.IsESM()
+	p.isFileConsideredESM =
+		p.isFileConsideredToHaveESMExports ||
+			p.esmImportStatementKeyword.Len > 0
+
+	// Legacy HTML comments are not allowed in ESM files
+	if p.isFileConsideredESM && p.lexer.LegacyHTMLCommentRange.Len > 0 {
+		_, notes := p.whyESModule()
+		p.log.AddErrorWithNotes(&p.tracker, p.lexer.LegacyHTMLCommentRange,
+			"Legacy HTML single-line comments are not allowed in ECMAScript modules", notes)
+	}
+
+	// ECMAScript modules are always interpreted as strict mode. This has to be
+	// done before "hoistSymbols" because strict mode can alter hoisting (!).
+	if p.isFileConsideredESM {
+		p.moduleScope.RecursiveSetStrictMode(js_ast.ImplicitStrictModeESM)
+	}
+
+	p.hoistSymbols(p.moduleScope)
+
+	if p.options.mode != config.ModePassThrough {
+		p.requireRef = p.declareCommonJSSymbol(ast.SymbolUnbound, "require")
+	} else {
+		p.requireRef = p.newSymbol(ast.SymbolUnbound, "require")
+	}
+
+	// CommonJS-style exports are only enabled if this isn't using ECMAScript-
+	// style exports. You can still use "require" in ESM, just not "module" or
+	// "exports". You can also still use "import" in CommonJS.
+	if p.options.mode != config.ModePassThrough && !p.isFileConsideredToHaveESMExports {
+		// CommonJS-style exports
+		p.exportsRef = p.declareCommonJSSymbol(ast.SymbolHoisted, "exports")
+		p.moduleRef = p.declareCommonJSSymbol(ast.SymbolHoisted, "module")
+	} else {
+		// ESM-style exports
+		p.exportsRef = p.newSymbol(ast.SymbolHoisted, "exports")
+		p.moduleRef = p.newSymbol(ast.SymbolHoisted, "module")
+	}
+
+	// Handle "@jsx" and "@jsxFrag" pragmas now that lexing is done
+	if p.options.jsx.Parse {
+		if jsxRuntime := p.lexer.JSXRuntimePragmaComment; jsxRuntime.Text != "" {
+			if jsxRuntime.Text == "automatic" {
+				p.options.jsx.AutomaticRuntime = true
+			} else if jsxRuntime.Text == "classic" {
+				p.options.jsx.AutomaticRuntime = false
+			} else {
+				p.log.AddIDWithNotes(logger.MsgID_JS_UnsupportedJSXComment, logger.Warning, &p.tracker, jsxRuntime.Range,
+					fmt.Sprintf("Invalid JSX runtime: %q", jsxRuntime.Text),
+					[]logger.MsgData{{Text: "The JSX runtime can only be set to either \"classic\" or \"automatic\"."}})
+			}
+		}
+
+		if jsxFactory := p.lexer.JSXFactoryPragmaComment; jsxFactory.Text != "" {
+			if p.options.jsx.AutomaticRuntime {
+				p.log.AddID(logger.MsgID_JS_UnsupportedJSXComment, logger.Warning, &p.tracker, jsxFactory.Range,
+					"The JSX factory cannot be set when using React's \"automatic\" JSX transform")
+			} else if expr, _ := ParseDefineExprOrJSON(jsxFactory.Text); len(expr.Parts) > 0 {
+				p.options.jsx.Factory = expr
+			} else {
+				p.log.AddID(logger.MsgID_JS_UnsupportedJSXComment, logger.Warning, &p.tracker, jsxFactory.Range,
+					fmt.Sprintf("Invalid JSX factory: %s", jsxFactory.Text))
+			}
+		}
+
+		if jsxFragment := p.lexer.JSXFragmentPragmaComment; jsxFragment.Text != "" {
+			if p.options.jsx.AutomaticRuntime {
+				p.log.AddID(logger.MsgID_JS_UnsupportedJSXComment, logger.Warning, &p.tracker, jsxFragment.Range,
+					"The JSX fragment cannot be set when using React's \"automatic\" JSX transform")
+			} else if expr, _ := ParseDefineExprOrJSON(jsxFragment.Text); len(expr.Parts) > 0 || expr.Constant != nil {
+				p.options.jsx.Fragment = expr
+			} else {
+				p.log.AddID(logger.MsgID_JS_UnsupportedJSXComment, logger.Warning, &p.tracker, jsxFragment.Range,
+					fmt.Sprintf("Invalid JSX fragment: %s", jsxFragment.Text))
+			}
+		}
+
+		if jsxImportSource := p.lexer.JSXImportSourcePragmaComment; jsxImportSource.Text != "" {
+			if !p.options.jsx.AutomaticRuntime {
+				p.log.AddIDWithNotes(logger.MsgID_JS_UnsupportedJSXComment, logger.Warning, &p.tracker, jsxImportSource.Range,
+					"The JSX import source cannot be set without also enabling React's \"automatic\" JSX transform",
+					[]logger.MsgData{{Text: "You can enable React's \"automatic\" JSX transform for this file by using a \"@jsxRuntime automatic\" comment."}})
+			} else {
+				p.options.jsx.ImportSource = jsxImportSource.Text
+			}
+		}
+	}
+
+	// Force-enable strict mode if the JSX "automatic" runtime is enabled and
+	// there is at least one JSX element. This is because the automatically-
+	// generated import statement turns the file into an ES module. This behavior
+	// matches TypeScript which also does this. See this PR for more information:
+	// https://github.com/microsoft/TypeScript/pull/39199
+	if p.currentScope.StrictMode == js_ast.SloppyMode && p.options.jsx.AutomaticRuntime && p.firstJSXElementLoc.Start != -1 {
+		p.currentScope.StrictMode = js_ast.ImplicitStrictModeJSXAutomaticRuntime
+	}
+}
+
+func (p *parser) declareCommonJSSymbol(kind ast.SymbolKind, name string) ast.Ref {
+	member, ok := p.moduleScope.Members[name]
+
+	// If the code declared this symbol using "var name", then this is actually
+	// not a collision. For example, node will let you do this:
+	//
+	//   var exports;
+	//   module.exports.foo = 123;
+	//   console.log(exports.foo);
+	//
+	// This works because node's implementation of CommonJS wraps the entire
+	// source file like this:
+	//
+	//   (function(require, exports, module, __filename, __dirname) {
+	//     var exports;
+	//     module.exports.foo = 123;
+	//     console.log(exports.foo);
+	//   })
+	//
+	// Both the "exports" argument and "var exports" are hoisted variables, so
+	// they don't collide.
+	if ok && p.symbols[member.Ref.InnerIndex].Kind == ast.SymbolHoisted &&
+		kind == ast.SymbolHoisted && !p.isFileConsideredToHaveESMExports {
+		return member.Ref
+	}
+
+	// Create a new symbol if we didn't merge with an existing one above
+	ref := p.newSymbol(kind, name)
+
+	// If the variable wasn't declared, declare it now. This means any references
+	// to this name will become bound to this symbol after this (since we haven't
+	// run the visit pass yet).
+	if !ok {
+		p.moduleScope.Members[name] = js_ast.ScopeMember{Ref: ref, Loc: logger.Loc{Start: -1}}
+		return ref
+	}
+
+	// If the variable was declared, then it shadows this symbol. The code in
+	// this module will be unable to reference this symbol. However, we must
+	// still add the symbol to the scope so it gets minified (automatically-
+	// generated code may still reference the symbol).
+	p.moduleScope.Generated = append(p.moduleScope.Generated, ref)
+	return ref
+}
+
+// Compute a character frequency histogram for everything that's not a bound
+// symbol. This is used to modify how minified names are generated for slightly
+// better gzip compression. Even though it's a very small win, we still do it
+// because it's simple to do and very cheap to compute.
+func (p *parser) computeCharacterFrequency() *ast.CharFreq {
+	if !p.options.minifyIdentifiers || p.source.Index == runtime.SourceIndex {
+		return nil
+	}
+
+	// Add everything in the file to the histogram
+	charFreq := &ast.CharFreq{}
+	charFreq.Scan(p.source.Contents, 1)
+
+	// Subtract out all comments
+	for _, commentRange := range p.lexer.AllComments {
+		charFreq.Scan(p.source.TextForRange(commentRange), -1)
+	}
+
+	// Subtract out all import paths
+	for _, record := range p.importRecords {
+		if !record.SourceIndex.IsValid() {
+			charFreq.Scan(record.Path.Text, -1)
+		}
+	}
+
+	// Subtract out all symbols that will be minified
+	var visit func(*js_ast.Scope)
+	visit = func(scope *js_ast.Scope) {
+		for _, member := range scope.Members {
+			symbol := &p.symbols[member.Ref.InnerIndex]
+			if symbol.SlotNamespace() != ast.SlotMustNotBeRenamed {
+				charFreq.Scan(symbol.OriginalName, -int32(symbol.UseCountEstimate))
+			}
+		}
+		if scope.Label.Ref != ast.InvalidRef {
+			symbol := &p.symbols[scope.Label.Ref.InnerIndex]
+			if symbol.SlotNamespace() != ast.SlotMustNotBeRenamed {
+				charFreq.Scan(symbol.OriginalName, -int32(symbol.UseCountEstimate)-1)
+			}
+		}
+		for _, child := range scope.Children {
+			visit(child)
+		}
+	}
+	visit(p.moduleScope)
+
+	// Subtract out all properties that will be mangled
+	for _, ref := range p.mangledProps {
+		symbol := &p.symbols[ref.InnerIndex]
+		charFreq.Scan(symbol.OriginalName, -int32(symbol.UseCountEstimate))
+	}
+
+	return charFreq
+}
+
+func (p *parser) generateImportStmt(
+	path string,
+	pathRange logger.Range,
+	imports []string,
+	parts []js_ast.Part,
+	symbols map[string]ast.LocRef,
+	sourceIndex *uint32,
+	copySourceIndex *uint32,
+) ([]js_ast.Part, uint32) {
+	if pathRange.Len == 0 {
+		isFirst := true
+		for _, it := range symbols {
+			if isFirst || it.Loc.Start < pathRange.Loc.Start {
+				pathRange.Loc = it.Loc
+			}
+			isFirst = false
+		}
+	}
+
+	namespaceRef := p.newSymbol(ast.SymbolOther, "import_"+js_ast.GenerateNonUniqueNameFromPath(path))
+	p.moduleScope.Generated = append(p.moduleScope.Generated, namespaceRef)
+	declaredSymbols := make([]js_ast.DeclaredSymbol, 1+len(imports))
+	clauseItems := make([]js_ast.ClauseItem, len(imports))
+	importRecordIndex := p.addImportRecord(ast.ImportStmt, pathRange, path, nil, 0)
+	if sourceIndex != nil {
+		p.importRecords[importRecordIndex].SourceIndex = ast.MakeIndex32(*sourceIndex)
+	}
+	if copySourceIndex != nil {
+		p.importRecords[importRecordIndex].CopySourceIndex = ast.MakeIndex32(*copySourceIndex)
+	}
+	declaredSymbols[0] = js_ast.DeclaredSymbol{Ref: namespaceRef, IsTopLevel: true}
+
+	// Create per-import information
+	for i, alias := range imports {
+		it := symbols[alias]
+		declaredSymbols[i+1] = js_ast.DeclaredSymbol{Ref: it.Ref, IsTopLevel: true}
+		clauseItems[i] = js_ast.ClauseItem{
+			Alias:    alias,
+			AliasLoc: it.Loc,
+			Name:     ast.LocRef{Loc: it.Loc, Ref: it.Ref},
+		}
+		p.isImportItem[it.Ref] = true
+		p.namedImports[it.Ref] = js_ast.NamedImport{
+			Alias:             alias,
+			AliasLoc:          it.Loc,
+			NamespaceRef:      namespaceRef,
+			ImportRecordIndex: importRecordIndex,
+		}
+	}
+
+	// Append a single import to the end of the file (ES6 imports are hoisted
+	// so we don't need to worry about where the import statement goes)
+	return append(parts, js_ast.Part{
+		DeclaredSymbols:     declaredSymbols,
+		ImportRecordIndices: []uint32{importRecordIndex},
+		Stmts: []js_ast.Stmt{{Loc: pathRange.Loc, Data: &js_ast.SImport{
+			NamespaceRef:      namespaceRef,
+			Items:             &clauseItems,
+			ImportRecordIndex: importRecordIndex,
+			IsSingleLine:      true,
+		}}},
+	}), importRecordIndex
+}
+
+// Sort the keys for determinism
+func sortedKeysOfMapStringLocRef(in map[string]ast.LocRef) []string {
+	keys := make([]string, 0, len(in))
+	for key := range in {
+		keys = append(keys, key)
+	}
+	sort.Strings(keys)
+	return keys
+}
+
+func (p *parser) toAST(before, parts, after []js_ast.Part, hashbang string, directives []string) js_ast.AST {
+	// Insert an import statement for any runtime imports we generated
+	if len(p.runtimeImports) > 0 && !p.options.omitRuntimeForTests {
+		keys := sortedKeysOfMapStringLocRef(p.runtimeImports)
+		sourceIndex := runtime.SourceIndex
+		before, _ = p.generateImportStmt("<runtime>", logger.Range{}, keys, before, p.runtimeImports, &sourceIndex, nil)
+	}
+
+	// Insert an import statement for any jsx runtime imports we generated
+	if len(p.jsxRuntimeImports) > 0 && !p.options.omitJSXRuntimeForTests {
+		keys := sortedKeysOfMapStringLocRef(p.jsxRuntimeImports)
+
+		// Determine the runtime source and whether it's prod or dev
+		path := p.options.jsx.ImportSource
+		if p.options.jsx.Development {
+			path = path + "/jsx-dev-runtime"
+		} else {
+			path = path + "/jsx-runtime"
+		}
+
+		before, _ = p.generateImportStmt(path, logger.Range{}, keys, before, p.jsxRuntimeImports, nil, nil)
+	}
+
+	// Insert an import statement for any legacy jsx imports we generated (i.e., createElement)
+	if len(p.jsxLegacyImports) > 0 && !p.options.omitJSXRuntimeForTests {
+		keys := sortedKeysOfMapStringLocRef(p.jsxLegacyImports)
+		path := p.options.jsx.ImportSource
+		before, _ = p.generateImportStmt(path, logger.Range{}, keys, before, p.jsxLegacyImports, nil, nil)
+	}
+
+	// Insert imports for each glob pattern
+	for _, glob := range p.globPatternImports {
+		symbols := map[string]ast.LocRef{glob.name: {Loc: glob.approximateRange.Loc, Ref: glob.ref}}
+		var importRecordIndex uint32
+		before, importRecordIndex = p.generateImportStmt(helpers.GlobPatternToString(glob.parts), glob.approximateRange, []string{glob.name}, before, symbols, nil, nil)
+		record := &p.importRecords[importRecordIndex]
+		record.AssertOrWith = glob.assertOrWith
+		record.GlobPattern = &ast.GlobPattern{
+			Parts:       glob.parts,
+			ExportAlias: glob.name,
+			Kind:        glob.kind,
+		}
+	}
+
+	// Generated imports are inserted before other code instead of appending them
+	// to the end of the file. Appending them should work fine because JavaScript
+	// import statements are "hoisted" to run before the importing file. However,
+	// some buggy JavaScript toolchains such as the TypeScript compiler convert
+	// ESM into CommonJS by replacing "import" statements inline without doing
+	// any hoisting, which is incorrect. See the following issue for more info:
+	// https://github.com/microsoft/TypeScript/issues/16166. Since JSX-related
+	// imports are present in the generated code when bundling is disabled, and
+	// could therefore be processed by these buggy tools, it's more robust to put
+	// them at the top even though it means potentially reallocating almost the
+	// entire array of parts.
+	if len(before) > 0 {
+		parts = append(before, parts...)
+	}
+	parts = append(parts, after...)
+
+	// Handle import paths after the whole file has been visited because we need
+	// symbol usage counts to be able to remove unused type-only imports in
+	// TypeScript code.
+	keptImportEquals := false
+	removedImportEquals := false
+	partsEnd := 0
+	for partIndex, part := range parts {
+		p.importRecordsForCurrentPart = nil
+		p.declaredSymbols = nil
+
+		result := p.scanForImportsAndExports(part.Stmts)
+		part.Stmts = result.stmts
+		keptImportEquals = keptImportEquals || result.keptImportEquals
+		removedImportEquals = removedImportEquals || result.removedImportEquals
+
+		part.ImportRecordIndices = append(part.ImportRecordIndices, p.importRecordsForCurrentPart...)
+		part.DeclaredSymbols = append(part.DeclaredSymbols, p.declaredSymbols...)
+
+		if len(part.Stmts) > 0 || uint32(partIndex) == js_ast.NSExportPartIndex {
+			if p.moduleScope.ContainsDirectEval && len(part.DeclaredSymbols) > 0 {
+				// If this file contains a direct call to "eval()", all parts that
+				// declare top-level symbols must be kept since the eval'd code may
+				// reference those symbols.
+				part.CanBeRemovedIfUnused = false
+			}
+			parts[partsEnd] = part
+			partsEnd++
+		}
+	}
+	parts = parts[:partsEnd]
+
+	// We need to iterate multiple times if an import-equals statement was
+	// removed and there are more import-equals statements that may be removed.
+	// In the example below, a/b/c should be kept but x/y/z should be removed
+	// (and removal requires multiple passes):
+	//
+	//   import a = foo.a
+	//   import b = a.b
+	//   import c = b.c
+	//
+	//   import x = foo.x
+	//   import y = x.y
+	//   import z = y.z
+	//
+	//   export let bar = c
+	//
+	// This is a smaller version of the general import/export scanning loop above.
+	// We only want to repeat the code that eliminates TypeScript import-equals
+	// statements, not the other code in the loop above.
+	for keptImportEquals && removedImportEquals {
+		keptImportEquals = false
+		removedImportEquals = false
+		partsEnd := 0
+		for partIndex, part := range parts {
+			result := p.scanForUnusedTSImportEquals(part.Stmts)
+			part.Stmts = result.stmts
+			keptImportEquals = keptImportEquals || result.keptImportEquals
+			removedImportEquals = removedImportEquals || result.removedImportEquals
+			if len(part.Stmts) > 0 || uint32(partIndex) == js_ast.NSExportPartIndex {
+				parts[partsEnd] = part
+				partsEnd++
+			}
+		}
+		parts = parts[:partsEnd]
+	}
+
+	// Do a second pass for exported items now that imported items are filled out
+	for _, part := range parts {
+		for _, stmt := range part.Stmts {
+			if s, ok := stmt.Data.(*js_ast.SExportClause); ok {
+				for _, item := range s.Items {
+					// Mark re-exported imports as such
+					if namedImport, ok := p.namedImports[item.Name.Ref]; ok {
+						namedImport.IsExported = true
+						p.namedImports[item.Name.Ref] = namedImport
+					}
+				}
+			}
+		}
+	}
+
+	// Analyze cross-part dependencies for tree shaking and code splitting
+	{
+		// Map locals to parts
+		p.topLevelSymbolToParts = make(map[ast.Ref][]uint32)
+		for partIndex, part := range parts {
+			for _, declared := range part.DeclaredSymbols {
+				if declared.IsTopLevel {
+					// If this symbol was merged, use the symbol at the end of the
+					// linked list in the map. This is the case for multiple "var"
+					// declarations with the same name, for example.
+					ref := declared.Ref
+					for p.symbols[ref.InnerIndex].Link != ast.InvalidRef {
+						ref = p.symbols[ref.InnerIndex].Link
+					}
+					p.topLevelSymbolToParts[ref] = append(
+						p.topLevelSymbolToParts[ref], uint32(partIndex))
+				}
+			}
+		}
+
+		// Pulling in the exports of this module always pulls in the export part
+		p.topLevelSymbolToParts[p.exportsRef] = append(p.topLevelSymbolToParts[p.exportsRef], js_ast.NSExportPartIndex)
+	}
+
+	// Make a wrapper symbol in case we need to be wrapped in a closure
+	wrapperRef := p.newSymbol(ast.SymbolOther, "require_"+p.source.IdentifierName)
+
+	// Assign slots to symbols in nested scopes. This is some precomputation for
+	// the symbol renaming pass that will happen later in the linker. It's done
+	// now in the parser because we want it to be done in parallel per file and
+	// we're already executing code in a dedicated goroutine for this file.
+	var nestedScopeSlotCounts ast.SlotCounts
+	if p.options.minifyIdentifiers {
+		nestedScopeSlotCounts = renamer.AssignNestedScopeSlots(p.moduleScope, p.symbols)
+	}
+
+	exportsKind := js_ast.ExportsNone
+	usesExportsRef := p.symbols[p.exportsRef.InnerIndex].UseCountEstimate > 0
+	usesModuleRef := p.symbols[p.moduleRef.InnerIndex].UseCountEstimate > 0
+
+	if p.esmExportKeyword.Len > 0 || p.esmImportMeta.Len > 0 || p.topLevelAwaitKeyword.Len > 0 {
+		exportsKind = js_ast.ExportsESM
+	} else if usesExportsRef || usesModuleRef || p.hasTopLevelReturn {
+		exportsKind = js_ast.ExportsCommonJS
+	} else {
+		// If this module has no exports, try to determine what kind of module it
+		// is by looking at node's "type" field in "package.json" and/or whether
+		// the file extension is ".mjs"/".mts" or ".cjs"/".cts".
+		switch {
+		case p.options.moduleTypeData.Type.IsCommonJS():
+			// ".cjs" or ".cts" or ("type: commonjs" and (".js" or ".jsx" or ".ts" or ".tsx"))
+			exportsKind = js_ast.ExportsCommonJS
+
+		case p.options.moduleTypeData.Type.IsESM():
+			// ".mjs" or ".mts" or ("type: module" and (".js" or ".jsx" or ".ts" or ".tsx"))
+			exportsKind = js_ast.ExportsESM
+
+		default:
+			// Treat unknown modules containing an import statement as ESM. Otherwise
+			// the bundler will treat this file as CommonJS if it's imported and ESM
+			// if it's not imported.
+			if p.esmImportStatementKeyword.Len > 0 {
+				exportsKind = js_ast.ExportsESM
+			}
+		}
+	}
+
+	return js_ast.AST{
+		Parts:                           parts,
+		ModuleTypeData:                  p.options.moduleTypeData,
+		ModuleScope:                     p.moduleScope,
+		CharFreq:                        p.computeCharacterFrequency(),
+		Symbols:                         p.symbols,
+		ExportsRef:                      p.exportsRef,
+		ModuleRef:                       p.moduleRef,
+		WrapperRef:                      wrapperRef,
+		Hashbang:                        hashbang,
+		Directives:                      directives,
+		NamedImports:                    p.namedImports,
+		NamedExports:                    p.namedExports,
+		TSEnums:                         p.tsEnums,
+		ConstValues:                     p.constValues,
+		ExprComments:                    p.exprComments,
+		NestedScopeSlotCounts:           nestedScopeSlotCounts,
+		TopLevelSymbolToPartsFromParser: p.topLevelSymbolToParts,
+		ExportStarImportRecords:         p.exportStarImportRecords,
+		ImportRecords:                   p.importRecords,
+		ApproximateLineCount:            int32(p.lexer.ApproximateNewlineCount) + 1,
+		MangledProps:                    p.mangledProps,
+		ReservedProps:                   p.reservedProps,
+		ManifestForYarnPnP:              p.manifestForYarnPnP,
+
+		// CommonJS features
+		UsesExportsRef: usesExportsRef,
+		UsesModuleRef:  usesModuleRef,
+		ExportsKind:    exportsKind,
+
+		// ES6 features
+		ExportKeyword:            p.esmExportKeyword,
+		TopLevelAwaitKeyword:     p.topLevelAwaitKeyword,
+		LiveTopLevelAwaitKeyword: p.liveTopLevelAwaitKeyword,
+	}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/js_parser/js_parser_lower.go b/source/vendor/github.com/evanw/esbuild/internal/js_parser/js_parser_lower.go
new file mode 100644
index 0000000..3991058
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/js_parser/js_parser_lower.go
@@ -0,0 +1,2131 @@
+// This file contains code for "lowering" syntax, which means converting it to
+// older JavaScript. For example, "a ** b" becomes a call to "Math.pow(a, b)"
+// when lowered. Which syntax is lowered is determined by the language target.
+
+package js_parser
+
+import (
+	"fmt"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/config"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+func (p *parser) markSyntaxFeature(feature compat.JSFeature, r logger.Range) (didGenerateError bool) {
+	didGenerateError = true
+
+	if !p.options.unsupportedJSFeatures.Has(feature) {
+		if feature == compat.TopLevelAwait && !p.options.outputFormat.KeepESMImportExportSyntax() {
+			p.log.AddError(&p.tracker, r, fmt.Sprintf(
+				"Top-level await is currently not supported with the %q output format", p.options.outputFormat.String()))
+			return
+		}
+
+		didGenerateError = false
+		return
+	}
+
+	var name string
+	where := config.PrettyPrintTargetEnvironment(p.options.originalTargetEnv, p.options.unsupportedJSFeatureOverridesMask)
+
+	switch feature {
+	case compat.DefaultArgument:
+		name = "default arguments"
+
+	case compat.RestArgument:
+		name = "rest arguments"
+
+	case compat.ArraySpread:
+		name = "array spread"
+
+	case compat.ForOf:
+		name = "for-of loops"
+
+	case compat.ObjectAccessors:
+		name = "object accessors"
+
+	case compat.ObjectExtensions:
+		name = "object literal extensions"
+
+	case compat.Destructuring:
+		name = "destructuring"
+
+	case compat.NewTarget:
+		name = "new.target"
+
+	case compat.ConstAndLet:
+		name = p.source.TextForRange(r)
+
+	case compat.Class:
+		name = "class syntax"
+
+	case compat.Generator:
+		name = "generator functions"
+
+	case compat.AsyncAwait:
+		name = "async functions"
+
+	case compat.AsyncGenerator:
+		name = "async generator functions"
+
+	case compat.ForAwait:
+		name = "for-await loops"
+
+	case compat.NestedRestBinding:
+		name = "non-identifier array rest patterns"
+
+	case compat.ImportAttributes:
+		p.log.AddError(&p.tracker, r, fmt.Sprintf(
+			"Using an arbitrary value as the second argument to \"import()\" is not possible in %s", where))
+		return
+
+	case compat.TopLevelAwait:
+		p.log.AddError(&p.tracker, r, fmt.Sprintf(
+			"Top-level await is not available in %s", where))
+		return
+
+	case compat.Bigint:
+		// Transforming these will never be supported
+		p.log.AddError(&p.tracker, r, fmt.Sprintf(
+			"Big integer literals are not available in %s", where))
+		return
+
+	case compat.ImportMeta:
+		// This can't be polyfilled
+		kind := logger.Warning
+		if p.suppressWarningsAboutWeirdCode || p.fnOrArrowDataVisit.tryBodyCount > 0 {
+			kind = logger.Debug
+		}
+		p.log.AddID(logger.MsgID_JS_EmptyImportMeta, kind, &p.tracker, r, fmt.Sprintf(
+			"\"import.meta\" is not available in %s and will be empty", where))
+		return
+
+	default:
+		p.log.AddError(&p.tracker, r, fmt.Sprintf(
+			"This feature is not available in %s", where))
+		return
+	}
+
+	p.log.AddError(&p.tracker, r, fmt.Sprintf(
+		"Transforming %s to %s is not supported yet", name, where))
+	return
+}
+
+func (p *parser) isStrictMode() bool {
+	return p.currentScope.StrictMode != js_ast.SloppyMode
+}
+
+func (p *parser) isStrictModeOutputFormat() bool {
+	return p.options.outputFormat == config.FormatESModule
+}
+
+type strictModeFeature uint8
+
+const (
+	withStatement strictModeFeature = iota
+	deleteBareName
+	forInVarInit
+	evalOrArguments
+	reservedWord
+	legacyOctalLiteral
+	legacyOctalEscape
+	ifElseFunctionStmt
+	labelFunctionStmt
+	duplicateLexicallyDeclaredNames
+)
+
+func (p *parser) markStrictModeFeature(feature strictModeFeature, r logger.Range, detail string) {
+	var text string
+	canBeTransformed := false
+
+	switch feature {
+	case withStatement:
+		text = "With statements"
+
+	case deleteBareName:
+		text = "Delete of a bare identifier"
+
+	case forInVarInit:
+		text = "Variable initializers inside for-in loops"
+		canBeTransformed = true
+
+	case evalOrArguments:
+		text = fmt.Sprintf("Declarations with the name %q", detail)
+
+	case reservedWord:
+		text = fmt.Sprintf("%q is a reserved word and", detail)
+
+	case legacyOctalLiteral:
+		text = "Legacy octal literals"
+
+	case legacyOctalEscape:
+		text = "Legacy octal escape sequences"
+
+	case ifElseFunctionStmt:
+		text = "Function declarations inside if statements"
+
+	case labelFunctionStmt:
+		text = "Function declarations inside labels"
+
+	case duplicateLexicallyDeclaredNames:
+		text = "Duplicate lexically-declared names"
+
+	default:
+		text = "This feature"
+	}
+
+	if p.isStrictMode() {
+		where, notes := p.whyStrictMode(p.currentScope)
+		p.log.AddErrorWithNotes(&p.tracker, r,
+			fmt.Sprintf("%s cannot be used %s", text, where), notes)
+	} else if !canBeTransformed && p.isStrictModeOutputFormat() {
+		p.log.AddError(&p.tracker, r,
+			fmt.Sprintf("%s cannot be used with the \"esm\" output format due to strict mode", text))
+	}
+}
+
+func (p *parser) whyStrictMode(scope *js_ast.Scope) (where string, notes []logger.MsgData) {
+	where = "in strict mode"
+
+	switch scope.StrictMode {
+	case js_ast.ImplicitStrictModeClass:
+		notes = []logger.MsgData{p.tracker.MsgData(p.enclosingClassKeyword,
+			"All code inside a class is implicitly in strict mode")}
+
+	case js_ast.ImplicitStrictModeTSAlwaysStrict:
+		tsAlwaysStrict := p.options.tsAlwaysStrict
+		t := logger.MakeLineColumnTracker(&tsAlwaysStrict.Source)
+		notes = []logger.MsgData{t.MsgData(tsAlwaysStrict.Range, fmt.Sprintf(
+			"TypeScript's %q setting was enabled here:", tsAlwaysStrict.Name))}
+
+	case js_ast.ImplicitStrictModeJSXAutomaticRuntime:
+		notes = []logger.MsgData{p.tracker.MsgData(logger.Range{Loc: p.firstJSXElementLoc, Len: 1},
+			"This file is implicitly in strict mode due to the JSX element here:"),
+			{Text: "When React's \"automatic\" JSX transform is enabled, using a JSX element automatically inserts " +
+				"an \"import\" statement at the top of the file for the corresponding the JSX helper function. " +
+				"This means the file is considered an ECMAScript module, and all ECMAScript modules use strict mode."}}
+
+	case js_ast.ExplicitStrictMode:
+		notes = []logger.MsgData{p.tracker.MsgData(p.source.RangeOfString(scope.UseStrictLoc),
+			"Strict mode is triggered by the \"use strict\" directive here:")}
+
+	case js_ast.ImplicitStrictModeESM:
+		_, notes = p.whyESModule()
+		where = "in an ECMAScript module"
+	}
+
+	return
+}
+
+func (p *parser) markAsyncFn(asyncRange logger.Range, isGenerator bool) (didGenerateError bool) {
+	// Lowered async functions are implemented in terms of generators. So if
+	// generators aren't supported, async functions aren't supported either.
+	// But if generators are supported, then async functions are unconditionally
+	// supported because we can use generators to implement them.
+	if !p.options.unsupportedJSFeatures.Has(compat.Generator) {
+		return false
+	}
+
+	feature := compat.AsyncAwait
+	if isGenerator {
+		feature = compat.AsyncGenerator
+	}
+	return p.markSyntaxFeature(feature, asyncRange)
+}
+
+func (p *parser) captureThis() ast.Ref {
+	if p.fnOnlyDataVisit.thisCaptureRef == nil {
+		ref := p.newSymbol(ast.SymbolHoisted, "_this")
+		p.fnOnlyDataVisit.thisCaptureRef = &ref
+	}
+
+	ref := *p.fnOnlyDataVisit.thisCaptureRef
+	p.recordUsage(ref)
+	return ref
+}
+
+func (p *parser) captureArguments() ast.Ref {
+	if p.fnOnlyDataVisit.argumentsCaptureRef == nil {
+		ref := p.newSymbol(ast.SymbolHoisted, "_arguments")
+		p.fnOnlyDataVisit.argumentsCaptureRef = &ref
+	}
+
+	ref := *p.fnOnlyDataVisit.argumentsCaptureRef
+	p.recordUsage(ref)
+	return ref
+}
+
+func (p *parser) lowerFunction(
+	isAsync *bool,
+	isGenerator *bool,
+	args *[]js_ast.Arg,
+	bodyLoc logger.Loc,
+	bodyBlock *js_ast.SBlock,
+	preferExpr *bool,
+	hasRestArg *bool,
+	isArrow bool,
+) {
+	// Lower object rest binding patterns in function arguments
+	if p.options.unsupportedJSFeatures.Has(compat.ObjectRestSpread) {
+		var prefixStmts []js_ast.Stmt
+
+		// Lower each argument individually instead of lowering all arguments
+		// together. There is a correctness tradeoff here around default values
+		// for function arguments, with no right answer.
+		//
+		// Lowering all arguments together will preserve the order of side effects
+		// for default values, but will mess up their scope:
+		//
+		//   // Side effect order: a(), b(), c()
+		//   function foo([{[a()]: w, ...x}, y = b()], z = c()) {}
+		//
+		//   // Side effect order is correct but scope is wrong
+		//   function foo(_a, _b) {
+		//     var [[{[a()]: w, ...x}, y = b()], z = c()] = [_a, _b]
+		//   }
+		//
+		// Lowering each argument individually will preserve the scope for default
+		// values that don't contain object rest binding patterns, but will mess up
+		// the side effect order:
+		//
+		//   // Side effect order: a(), b(), c()
+		//   function foo([{[a()]: w, ...x}, y = b()], z = c()) {}
+		//
+		//   // Side effect order is wrong but scope for c() is correct
+		//   function foo(_a, z = c()) {
+		//     var [{[a()]: w, ...x}, y = b()] = _a
+		//   }
+		//
+		// This transform chooses to lower each argument individually with the
+		// thinking that perhaps scope matters more in real-world code than side
+		// effect order.
+		for i, arg := range *args {
+			if bindingHasObjectRest(arg.Binding) {
+				ref := p.generateTempRef(tempRefNoDeclare, "")
+				target := js_ast.ConvertBindingToExpr(arg.Binding, nil)
+				init := js_ast.Expr{Loc: arg.Binding.Loc, Data: &js_ast.EIdentifier{Ref: ref}}
+				p.recordUsage(ref)
+
+				if decls, ok := p.lowerObjectRestToDecls(target, init, nil); ok {
+					// Replace the binding but leave the default value intact
+					(*args)[i].Binding.Data = &js_ast.BIdentifier{Ref: ref}
+
+					// Append a variable declaration to the function body
+					prefixStmts = append(prefixStmts, js_ast.Stmt{Loc: arg.Binding.Loc,
+						Data: &js_ast.SLocal{Kind: js_ast.LocalVar, Decls: decls}})
+				}
+			}
+		}
+
+		if len(prefixStmts) > 0 {
+			bodyBlock.Stmts = append(prefixStmts, bodyBlock.Stmts...)
+		}
+	}
+
+	// Lower async functions and async generator functions
+	if *isAsync && (p.options.unsupportedJSFeatures.Has(compat.AsyncAwait) || (isGenerator != nil && *isGenerator && p.options.unsupportedJSFeatures.Has(compat.AsyncGenerator))) {
+		// Use the shortened form if we're an arrow function
+		if preferExpr != nil {
+			*preferExpr = true
+		}
+
+		// Determine the value for "this"
+		thisValue, hasThisValue := p.valueForThis(
+			bodyLoc,
+			false, /* shouldWarn */
+			js_ast.AssignTargetNone,
+			false, /* isCallTarget */
+			false, /* isDeleteTarget */
+		)
+		if !hasThisValue {
+			thisValue = js_ast.Expr{Loc: bodyLoc, Data: js_ast.EThisShared}
+		}
+
+		// Move the code into a nested generator function
+		fn := js_ast.Fn{
+			IsGenerator: true,
+			Body:        js_ast.FnBody{Loc: bodyLoc, Block: *bodyBlock},
+		}
+		bodyBlock.Stmts = nil
+
+		// Errors thrown during argument evaluation must reject the
+		// resulting promise, which needs more complex code to handle
+		couldThrowErrors := false
+		for _, arg := range *args {
+			if _, ok := arg.Binding.Data.(*js_ast.BIdentifier); !ok ||
+				(arg.DefaultOrNil.Data != nil && couldPotentiallyThrow(arg.DefaultOrNil.Data)) {
+				couldThrowErrors = true
+				break
+			}
+		}
+
+		// Forward the arguments to the wrapper function
+		usesArgumentsRef := !isArrow && p.fnOnlyDataVisit.argumentsRef != nil &&
+			p.symbolUses[*p.fnOnlyDataVisit.argumentsRef].CountEstimate > 0
+		var forwardedArgs js_ast.Expr
+		if !couldThrowErrors && !usesArgumentsRef {
+			// Simple case: the arguments can stay on the outer function. It's
+			// worth separating out the simple case because it's the common case
+			// and it generates smaller code.
+			forwardedArgs = js_ast.Expr{Loc: bodyLoc, Data: js_ast.ENullShared}
+		} else {
+			// If code uses "arguments" then we must move the arguments to the inner
+			// function. This is because you can modify arguments by assigning to
+			// elements in the "arguments" object:
+			//
+			//   async function foo(x) {
+			//     arguments[0] = 1;
+			//     // "x" must be 1 here
+			//   }
+			//
+
+			// Complex case: the arguments must be moved to the inner function
+			fn.Args = *args
+			fn.HasRestArg = *hasRestArg
+			*args = nil
+			*hasRestArg = false
+
+			// Make sure to not change the value of the "length" property. This is
+			// done by generating dummy arguments for the outer function equal to
+			// the expected length of the function:
+			//
+			//   async function foo(a, b, c = d, ...e) {
+			//   }
+			//
+			// This turns into:
+			//
+			//   function foo(_0, _1) {
+			//     return __async(this, arguments, function* (a, b, c = d, ...e) {
+			//     });
+			//   }
+			//
+			// The "_0" and "_1" are dummy variables to ensure "foo.length" is 2.
+			for i, arg := range fn.Args {
+				if arg.DefaultOrNil.Data != nil || fn.HasRestArg && i+1 == len(fn.Args) {
+					// Arguments from here on don't add to the "length"
+					break
+				}
+
+				// Generate a dummy variable
+				argRef := p.newSymbol(ast.SymbolOther, fmt.Sprintf("_%d", i))
+				p.currentScope.Generated = append(p.currentScope.Generated, argRef)
+				*args = append(*args, js_ast.Arg{Binding: js_ast.Binding{Loc: arg.Binding.Loc, Data: &js_ast.BIdentifier{Ref: argRef}}})
+			}
+
+			// Forward all arguments from the outer function to the inner function
+			if !isArrow {
+				// Normal functions can just use "arguments" to forward everything
+				forwardedArgs = js_ast.Expr{Loc: bodyLoc, Data: &js_ast.EIdentifier{Ref: *p.fnOnlyDataVisit.argumentsRef}}
+			} else {
+				// Arrow functions can't use "arguments", so we need to forward
+				// the arguments manually.
+				//
+				// Note that if the arrow function references "arguments" in its body
+				// (even if it's inside another nested arrow function), that reference
+				// to "arguments" will have to be substituted with a captured variable.
+				// This is because we're changing the arrow function into a generator
+				// function, which introduces a variable named "arguments". This is
+				// handled separately during symbol resolution instead of being handled
+				// here so we don't need to re-traverse the arrow function body.
+
+				// If we need to forward more than the current number of arguments,
+				// add a rest argument to the set of forwarding variables. This is the
+				// case if the arrow function has rest or default arguments.
+				if len(*args) < len(fn.Args) {
+					argRef := p.newSymbol(ast.SymbolOther, fmt.Sprintf("_%d", len(*args)))
+					p.currentScope.Generated = append(p.currentScope.Generated, argRef)
+					*args = append(*args, js_ast.Arg{Binding: js_ast.Binding{Loc: bodyLoc, Data: &js_ast.BIdentifier{Ref: argRef}}})
+					*hasRestArg = true
+				}
+
+				// Forward all of the arguments
+				items := make([]js_ast.Expr, 0, len(*args))
+				for i, arg := range *args {
+					id := arg.Binding.Data.(*js_ast.BIdentifier)
+					item := js_ast.Expr{Loc: arg.Binding.Loc, Data: &js_ast.EIdentifier{Ref: id.Ref}}
+					if *hasRestArg && i+1 == len(*args) {
+						item.Data = &js_ast.ESpread{Value: item}
+					}
+					items = append(items, item)
+				}
+				forwardedArgs = js_ast.Expr{Loc: bodyLoc, Data: &js_ast.EArray{Items: items, IsSingleLine: true}}
+			}
+		}
+
+		var name string
+		if isGenerator != nil && *isGenerator {
+			// "async function* foo(a, b) { stmts }" => "function foo(a, b) { return __asyncGenerator(this, null, function* () { stmts }) }"
+			name = "__asyncGenerator"
+			*isGenerator = false
+		} else {
+			// "async function foo(a, b) { stmts }" => "function foo(a, b) { return __async(this, null, function* () { stmts }) }"
+			name = "__async"
+		}
+		*isAsync = false
+		callAsync := p.callRuntime(bodyLoc, name, []js_ast.Expr{
+			thisValue,
+			forwardedArgs,
+			{Loc: bodyLoc, Data: &js_ast.EFunction{Fn: fn}},
+		})
+		bodyBlock.Stmts = []js_ast.Stmt{{Loc: bodyLoc, Data: &js_ast.SReturn{ValueOrNil: callAsync}}}
+	}
+}
+
+func (p *parser) lowerOptionalChain(expr js_ast.Expr, in exprIn, childOut exprOut) (js_ast.Expr, exprOut) {
+	valueWhenUndefined := js_ast.Expr{Loc: expr.Loc, Data: js_ast.EUndefinedShared}
+	endsWithPropertyAccess := false
+	containsPrivateName := false
+	startsWithCall := false
+	originalExpr := expr
+	chain := []js_ast.Expr{}
+	loc := expr.Loc
+
+	// Step 1: Get an array of all expressions in the chain. We're traversing the
+	// chain from the outside in, so the array will be filled in "backwards".
+flatten:
+	for {
+		chain = append(chain, expr)
+
+		switch e := expr.Data.(type) {
+		case *js_ast.EDot:
+			expr = e.Target
+			if len(chain) == 1 {
+				endsWithPropertyAccess = true
+			}
+			if e.OptionalChain == js_ast.OptionalChainStart {
+				break flatten
+			}
+
+		case *js_ast.EIndex:
+			expr = e.Target
+			if len(chain) == 1 {
+				endsWithPropertyAccess = true
+			}
+
+			// If this is a private name that needs to be lowered, the entire chain
+			// itself will have to be lowered even if the language target supports
+			// optional chaining. This is because there's no way to use our shim
+			// function for private names with optional chaining syntax.
+			if private, ok := e.Index.Data.(*js_ast.EPrivateIdentifier); ok && p.privateSymbolNeedsToBeLowered(private) {
+				containsPrivateName = true
+			}
+
+			if e.OptionalChain == js_ast.OptionalChainStart {
+				break flatten
+			}
+
+		case *js_ast.ECall:
+			expr = e.Target
+			if e.OptionalChain == js_ast.OptionalChainStart {
+				startsWithCall = true
+				break flatten
+			}
+
+		case *js_ast.EUnary: // UnOpDelete
+			valueWhenUndefined = js_ast.Expr{Loc: loc, Data: &js_ast.EBoolean{Value: true}}
+			expr = e.Value
+
+		default:
+			panic("Internal error")
+		}
+	}
+
+	// Stop now if we can strip the whole chain as dead code. Since the chain is
+	// lazily evaluated, it's safe to just drop the code entirely.
+	if p.options.minifySyntax {
+		if isNullOrUndefined, sideEffects, ok := js_ast.ToNullOrUndefinedWithSideEffects(expr.Data); ok && isNullOrUndefined {
+			if sideEffects == js_ast.CouldHaveSideEffects {
+				return js_ast.JoinWithComma(p.astHelpers.SimplifyUnusedExpr(expr, p.options.unsupportedJSFeatures), valueWhenUndefined), exprOut{}
+			}
+			return valueWhenUndefined, exprOut{}
+		}
+	} else {
+		switch expr.Data.(type) {
+		case *js_ast.ENull, *js_ast.EUndefined:
+			return valueWhenUndefined, exprOut{}
+		}
+	}
+
+	// We need to lower this if this is an optional call off of a private name
+	// such as "foo.#bar?.()" because the value of "this" must be captured.
+	if _, _, private := p.extractPrivateIndex(expr); private != nil {
+		containsPrivateName = true
+	}
+
+	// Don't lower this if we don't need to. This check must be done here instead
+	// of earlier so we can do the dead code elimination above when the target is
+	// null or undefined.
+	if !p.options.unsupportedJSFeatures.Has(compat.OptionalChain) && !containsPrivateName {
+		return originalExpr, exprOut{}
+	}
+
+	// Step 2: Figure out if we need to capture the value for "this" for the
+	// initial ECall. This will be passed to ".call(this, ...args)" later.
+	var thisArg js_ast.Expr
+	var targetWrapFunc func(js_ast.Expr) js_ast.Expr
+	if startsWithCall {
+		if childOut.thisArgFunc != nil {
+			// The initial value is a nested optional chain that ended in a property
+			// access. The nested chain was processed first and has saved the
+			// appropriate value for "this". The callback here will return a
+			// reference to that saved location.
+			thisArg = childOut.thisArgFunc()
+		} else {
+			// The initial value is a normal expression. If it's a property access,
+			// strip the property off and save the target of the property access to
+			// be used as the value for "this".
+			switch e := expr.Data.(type) {
+			case *js_ast.EDot:
+				if _, ok := e.Target.Data.(*js_ast.ESuper); ok {
+					// Lower "super.prop" if necessary
+					if p.shouldLowerSuperPropertyAccess(e.Target) {
+						key := js_ast.Expr{Loc: e.NameLoc, Data: &js_ast.EString{Value: helpers.StringToUTF16(e.Name)}}
+						expr = p.lowerSuperPropertyGet(expr.Loc, key)
+					}
+
+					// Special-case "super.foo?.()" to avoid a syntax error. Without this,
+					// we would generate:
+					//
+					//   (_b = (_a = super).foo) == null ? void 0 : _b.call(_a)
+					//
+					// which is a syntax error. Now we generate this instead:
+					//
+					//   (_a = super.foo) == null ? void 0 : _a.call(this)
+					//
+					thisArg = js_ast.Expr{Loc: loc, Data: js_ast.EThisShared}
+				} else {
+					targetFunc, wrapFunc := p.captureValueWithPossibleSideEffects(loc, 2, e.Target, valueDefinitelyNotMutated)
+					expr = js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+						Target:  targetFunc(),
+						Name:    e.Name,
+						NameLoc: e.NameLoc,
+					}}
+					thisArg = targetFunc()
+					targetWrapFunc = wrapFunc
+				}
+
+			case *js_ast.EIndex:
+				if _, ok := e.Target.Data.(*js_ast.ESuper); ok {
+					// Lower "super[prop]" if necessary
+					if p.shouldLowerSuperPropertyAccess(e.Target) {
+						expr = p.lowerSuperPropertyGet(expr.Loc, e.Index)
+					}
+
+					// See the comment above about a similar special case for EDot
+					thisArg = js_ast.Expr{Loc: loc, Data: js_ast.EThisShared}
+				} else {
+					targetFunc, wrapFunc := p.captureValueWithPossibleSideEffects(loc, 2, e.Target, valueDefinitelyNotMutated)
+					targetWrapFunc = wrapFunc
+
+					// Capture the value of "this" if the target of the starting call
+					// expression is a private property access
+					if private, ok := e.Index.Data.(*js_ast.EPrivateIdentifier); ok && p.privateSymbolNeedsToBeLowered(private) {
+						// "foo().#bar?.()" must capture "foo()" for "this"
+						expr = p.lowerPrivateGet(targetFunc(), e.Index.Loc, private)
+						thisArg = targetFunc()
+						break
+					}
+
+					expr = js_ast.Expr{Loc: loc, Data: &js_ast.EIndex{
+						Target: targetFunc(),
+						Index:  e.Index,
+					}}
+					thisArg = targetFunc()
+				}
+			}
+		}
+	}
+
+	// Step 3: Figure out if we need to capture the starting value. We don't need
+	// to capture it if it doesn't have any side effects (e.g. it's just a bare
+	// identifier). Skipping the capture reduces code size and matches the output
+	// of the TypeScript compiler.
+	exprFunc, exprWrapFunc := p.captureValueWithPossibleSideEffects(loc, 2, expr, valueDefinitelyNotMutated)
+	expr = exprFunc()
+	result := exprFunc()
+
+	// Step 4: Wrap the starting value by each expression in the chain. We
+	// traverse the chain in reverse because we want to go from the inside out
+	// and the chain was built from the outside in.
+	var parentThisArgFunc func() js_ast.Expr
+	var parentThisArgWrapFunc func(js_ast.Expr) js_ast.Expr
+	var privateThisFunc func() js_ast.Expr
+	var privateThisWrapFunc func(js_ast.Expr) js_ast.Expr
+	for i := len(chain) - 1; i >= 0; i-- {
+		// Save a reference to the value of "this" for our parent ECall
+		if i == 0 && in.storeThisArgForParentOptionalChain && endsWithPropertyAccess {
+			parentThisArgFunc, parentThisArgWrapFunc = p.captureValueWithPossibleSideEffects(result.Loc, 2, result, valueDefinitelyNotMutated)
+			result = parentThisArgFunc()
+		}
+
+		switch e := chain[i].Data.(type) {
+		case *js_ast.EDot:
+			result = js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+				Target:  result,
+				Name:    e.Name,
+				NameLoc: e.NameLoc,
+			}}
+
+		case *js_ast.EIndex:
+			if private, ok := e.Index.Data.(*js_ast.EPrivateIdentifier); ok && p.privateSymbolNeedsToBeLowered(private) {
+				// If this is private name property access inside a call expression and
+				// the call expression is part of this chain, then the call expression
+				// is going to need a copy of the property access target as the value
+				// for "this" for the call. Example for this case: "foo.#bar?.()"
+				if i > 0 {
+					if _, ok := chain[i-1].Data.(*js_ast.ECall); ok {
+						privateThisFunc, privateThisWrapFunc = p.captureValueWithPossibleSideEffects(loc, 2, result, valueDefinitelyNotMutated)
+						result = privateThisFunc()
+					}
+				}
+
+				result = p.lowerPrivateGet(result, e.Index.Loc, private)
+				continue
+			}
+
+			result = js_ast.Expr{Loc: loc, Data: &js_ast.EIndex{
+				Target: result,
+				Index:  e.Index,
+			}}
+
+		case *js_ast.ECall:
+			// If this is the initial ECall in the chain and it's being called off of
+			// a property access, invoke the function using ".call(this, ...args)" to
+			// explicitly provide the value for "this".
+			if i == len(chain)-1 && thisArg.Data != nil {
+				result = js_ast.Expr{Loc: loc, Data: &js_ast.ECall{
+					Target: js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+						Target:  result,
+						Name:    "call",
+						NameLoc: loc,
+					}},
+					Args:                   append([]js_ast.Expr{thisArg}, e.Args...),
+					CanBeUnwrappedIfUnused: e.CanBeUnwrappedIfUnused,
+					IsMultiLine:            e.IsMultiLine,
+					Kind:                   js_ast.TargetWasOriginallyPropertyAccess,
+				}}
+				break
+			}
+
+			// If the target of this call expression is a private name property
+			// access that's also part of this chain, then we must use the copy of
+			// the property access target that was stashed away earlier as the value
+			// for "this" for the call. Example for this case: "foo.#bar?.()"
+			if privateThisFunc != nil {
+				result = privateThisWrapFunc(js_ast.Expr{Loc: loc, Data: &js_ast.ECall{
+					Target: js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+						Target:  result,
+						Name:    "call",
+						NameLoc: loc,
+					}},
+					Args:                   append([]js_ast.Expr{privateThisFunc()}, e.Args...),
+					CanBeUnwrappedIfUnused: e.CanBeUnwrappedIfUnused,
+					IsMultiLine:            e.IsMultiLine,
+					Kind:                   js_ast.TargetWasOriginallyPropertyAccess,
+				}})
+				privateThisFunc = nil
+				break
+			}
+
+			result = js_ast.Expr{Loc: loc, Data: &js_ast.ECall{
+				Target:                 result,
+				Args:                   e.Args,
+				CanBeUnwrappedIfUnused: e.CanBeUnwrappedIfUnused,
+				IsMultiLine:            e.IsMultiLine,
+				Kind:                   e.Kind,
+			}}
+
+		case *js_ast.EUnary:
+			result = js_ast.Expr{Loc: loc, Data: &js_ast.EUnary{
+				Op:    js_ast.UnOpDelete,
+				Value: result,
+
+				// If a delete of an optional chain takes place, it behaves as if the
+				// optional chain isn't there with regard to the "delete" semantics.
+				WasOriginallyDeleteOfIdentifierOrPropertyAccess: e.WasOriginallyDeleteOfIdentifierOrPropertyAccess,
+			}}
+
+		default:
+			panic("Internal error")
+		}
+	}
+
+	// Step 5: Wrap it all in a conditional that returns the chain or the default
+	// value if the initial value is null/undefined. The default value is usually
+	// "undefined" but is "true" if the chain ends in a "delete" operator.
+	// "x?.y" => "x == null ? void 0 : x.y"
+	// "x()?.y()" => "(_a = x()) == null ? void 0 : _a.y()"
+	result = js_ast.Expr{Loc: loc, Data: &js_ast.EIf{
+		Test: js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+			Op:    js_ast.BinOpLooseEq,
+			Left:  expr,
+			Right: js_ast.Expr{Loc: loc, Data: js_ast.ENullShared},
+		}},
+		Yes: valueWhenUndefined,
+		No:  result,
+	}}
+	if exprWrapFunc != nil {
+		result = exprWrapFunc(result)
+	}
+	if targetWrapFunc != nil {
+		result = targetWrapFunc(result)
+	}
+	if childOut.thisArgWrapFunc != nil {
+		result = childOut.thisArgWrapFunc(result)
+	}
+	return result, exprOut{
+		thisArgFunc:     parentThisArgFunc,
+		thisArgWrapFunc: parentThisArgWrapFunc,
+	}
+}
+
+func (p *parser) lowerParenthesizedOptionalChain(loc logger.Loc, e *js_ast.ECall, childOut exprOut) js_ast.Expr {
+	return childOut.thisArgWrapFunc(js_ast.Expr{Loc: loc, Data: &js_ast.ECall{
+		Target: js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+			Target:  e.Target,
+			Name:    "call",
+			NameLoc: loc,
+		}},
+		Args:        append(append(make([]js_ast.Expr, 0, len(e.Args)+1), childOut.thisArgFunc()), e.Args...),
+		IsMultiLine: e.IsMultiLine,
+		Kind:        js_ast.TargetWasOriginallyPropertyAccess,
+	}})
+}
+
+func (p *parser) lowerAssignmentOperator(value js_ast.Expr, callback func(js_ast.Expr, js_ast.Expr) js_ast.Expr) js_ast.Expr {
+	switch left := value.Data.(type) {
+	case *js_ast.EDot:
+		if left.OptionalChain == js_ast.OptionalChainNone {
+			referenceFunc, wrapFunc := p.captureValueWithPossibleSideEffects(value.Loc, 2, left.Target, valueDefinitelyNotMutated)
+			return wrapFunc(callback(
+				js_ast.Expr{Loc: value.Loc, Data: &js_ast.EDot{
+					Target:  referenceFunc(),
+					Name:    left.Name,
+					NameLoc: left.NameLoc,
+				}},
+				js_ast.Expr{Loc: value.Loc, Data: &js_ast.EDot{
+					Target:  referenceFunc(),
+					Name:    left.Name,
+					NameLoc: left.NameLoc,
+				}},
+			))
+		}
+
+	case *js_ast.EIndex:
+		if left.OptionalChain == js_ast.OptionalChainNone {
+			targetFunc, targetWrapFunc := p.captureValueWithPossibleSideEffects(value.Loc, 2, left.Target, valueDefinitelyNotMutated)
+			indexFunc, indexWrapFunc := p.captureValueWithPossibleSideEffects(value.Loc, 2, left.Index, valueDefinitelyNotMutated)
+			return targetWrapFunc(indexWrapFunc(callback(
+				js_ast.Expr{Loc: value.Loc, Data: &js_ast.EIndex{
+					Target: targetFunc(),
+					Index:  indexFunc(),
+				}},
+				js_ast.Expr{Loc: value.Loc, Data: &js_ast.EIndex{
+					Target: targetFunc(),
+					Index:  indexFunc(),
+				}},
+			)))
+		}
+
+	case *js_ast.EIdentifier:
+		return callback(
+			js_ast.Expr{Loc: value.Loc, Data: &js_ast.EIdentifier{Ref: left.Ref}},
+			value,
+		)
+	}
+
+	// We shouldn't get here with valid syntax? Just let this through for now
+	// since there's currently no assignment target validation. Garbage in,
+	// garbage out.
+	return value
+}
+
+func (p *parser) lowerExponentiationAssignmentOperator(loc logger.Loc, e *js_ast.EBinary) js_ast.Expr {
+	if target, privateLoc, private := p.extractPrivateIndex(e.Left); private != nil {
+		// "a.#b **= c" => "__privateSet(a, #b, __pow(__privateGet(a, #b), c))"
+		targetFunc, targetWrapFunc := p.captureValueWithPossibleSideEffects(loc, 2, target, valueDefinitelyNotMutated)
+		return targetWrapFunc(p.lowerPrivateSet(targetFunc(), privateLoc, private,
+			p.callRuntime(loc, "__pow", []js_ast.Expr{
+				p.lowerPrivateGet(targetFunc(), privateLoc, private),
+				e.Right,
+			})))
+	}
+
+	return p.lowerAssignmentOperator(e.Left, func(a js_ast.Expr, b js_ast.Expr) js_ast.Expr {
+		// "a **= b" => "a = __pow(a, b)"
+		return js_ast.Assign(a, p.callRuntime(loc, "__pow", []js_ast.Expr{b, e.Right}))
+	})
+}
+
+func (p *parser) lowerNullishCoalescingAssignmentOperator(loc logger.Loc, e *js_ast.EBinary) (js_ast.Expr, bool) {
+	if target, privateLoc, private := p.extractPrivateIndex(e.Left); private != nil {
+		if p.options.unsupportedJSFeatures.Has(compat.NullishCoalescing) {
+			// "a.#b ??= c" => "(_a = __privateGet(a, #b)) != null ? _a : __privateSet(a, #b, c)"
+			targetFunc, targetWrapFunc := p.captureValueWithPossibleSideEffects(loc, 2, target, valueDefinitelyNotMutated)
+			left := p.lowerPrivateGet(targetFunc(), privateLoc, private)
+			right := p.lowerPrivateSet(targetFunc(), privateLoc, private, e.Right)
+			return targetWrapFunc(p.lowerNullishCoalescing(loc, left, right)), true
+		}
+
+		// "a.#b ??= c" => "__privateGet(a, #b) ?? __privateSet(a, #b, c)"
+		targetFunc, targetWrapFunc := p.captureValueWithPossibleSideEffects(loc, 2, target, valueDefinitelyNotMutated)
+		return targetWrapFunc(js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+			Op:    js_ast.BinOpNullishCoalescing,
+			Left:  p.lowerPrivateGet(targetFunc(), privateLoc, private),
+			Right: p.lowerPrivateSet(targetFunc(), privateLoc, private, e.Right),
+		}}), true
+	}
+
+	if p.options.unsupportedJSFeatures.Has(compat.LogicalAssignment) {
+		return p.lowerAssignmentOperator(e.Left, func(a js_ast.Expr, b js_ast.Expr) js_ast.Expr {
+			if p.options.unsupportedJSFeatures.Has(compat.NullishCoalescing) {
+				// "a ??= b" => "(_a = a) != null ? _a : a = b"
+				return p.lowerNullishCoalescing(loc, a, js_ast.Assign(b, e.Right))
+			}
+
+			// "a ??= b" => "a ?? (a = b)"
+			return js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+				Op:    js_ast.BinOpNullishCoalescing,
+				Left:  a,
+				Right: js_ast.Assign(b, e.Right),
+			}}
+		}), true
+	}
+
+	return js_ast.Expr{}, false
+}
+
+func (p *parser) lowerLogicalAssignmentOperator(loc logger.Loc, e *js_ast.EBinary, op js_ast.OpCode) (js_ast.Expr, bool) {
+	if target, privateLoc, private := p.extractPrivateIndex(e.Left); private != nil {
+		// "a.#b &&= c" => "__privateGet(a, #b) && __privateSet(a, #b, c)"
+		// "a.#b ||= c" => "__privateGet(a, #b) || __privateSet(a, #b, c)"
+		targetFunc, targetWrapFunc := p.captureValueWithPossibleSideEffects(loc, 2, target, valueDefinitelyNotMutated)
+		return targetWrapFunc(js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+			Op:    op,
+			Left:  p.lowerPrivateGet(targetFunc(), privateLoc, private),
+			Right: p.lowerPrivateSet(targetFunc(), privateLoc, private, e.Right),
+		}}), true
+	}
+
+	if p.options.unsupportedJSFeatures.Has(compat.LogicalAssignment) {
+		return p.lowerAssignmentOperator(e.Left, func(a js_ast.Expr, b js_ast.Expr) js_ast.Expr {
+			// "a &&= b" => "a && (a = b)"
+			// "a ||= b" => "a || (a = b)"
+			return js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+				Op:    op,
+				Left:  a,
+				Right: js_ast.Assign(b, e.Right),
+			}}
+		}), true
+	}
+
+	return js_ast.Expr{}, false
+}
+
+func (p *parser) lowerNullishCoalescing(loc logger.Loc, left js_ast.Expr, right js_ast.Expr) js_ast.Expr {
+	// "x ?? y" => "x != null ? x : y"
+	// "x() ?? y()" => "_a = x(), _a != null ? _a : y"
+	leftFunc, wrapFunc := p.captureValueWithPossibleSideEffects(loc, 2, left, valueDefinitelyNotMutated)
+	return wrapFunc(js_ast.Expr{Loc: loc, Data: &js_ast.EIf{
+		Test: js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+			Op:    js_ast.BinOpLooseNe,
+			Left:  leftFunc(),
+			Right: js_ast.Expr{Loc: loc, Data: js_ast.ENullShared},
+		}},
+		Yes: leftFunc(),
+		No:  right,
+	}})
+}
+
+// Lower object spread for environments that don't support them. Non-spread
+// properties are grouped into object literals and then passed to the
+// "__spreadValues" and "__spreadProps" functions like this:
+//
+//	"{a, b, ...c, d, e}" => "__spreadProps(__spreadValues(__spreadProps({a, b}, c), {d, e})"
+//
+// If the object literal starts with a spread, then we pass an empty object
+// literal to "__spreadValues" to make sure we clone the object:
+//
+//	"{...a, b}" => "__spreadProps(__spreadValues({}, a), {b})"
+//
+// It's not immediately obvious why we don't compile everything to a single
+// call to a function that takes any number of arguments, since that would be
+// shorter. The reason is to preserve the order of side effects. Consider
+// this code:
+//
+//	let a = {
+//	  get x() {
+//	    b = {y: 2}
+//	    return 1
+//	  }
+//	}
+//	let b = {}
+//	let c = {...a, ...b}
+//
+// Converting the above code to "let c = __spreadFn({}, a, null, b)" means "c"
+// becomes "{x: 1}" which is incorrect. Converting the above code instead to
+// "let c = __spreadProps(__spreadProps({}, a), b)" means "c" becomes
+// "{x: 1, y: 2}" which is correct.
+func (p *parser) lowerObjectSpread(loc logger.Loc, e *js_ast.EObject) js_ast.Expr {
+	needsLowering := false
+
+	if p.options.unsupportedJSFeatures.Has(compat.ObjectRestSpread) {
+		for _, property := range e.Properties {
+			if property.Kind == js_ast.PropertySpread {
+				needsLowering = true
+				break
+			}
+		}
+	}
+
+	if !needsLowering {
+		return js_ast.Expr{Loc: loc, Data: e}
+	}
+
+	var result js_ast.Expr
+	properties := []js_ast.Property{}
+
+	for _, property := range e.Properties {
+		if property.Kind != js_ast.PropertySpread {
+			properties = append(properties, property)
+			continue
+		}
+
+		if len(properties) > 0 || result.Data == nil {
+			if result.Data == nil {
+				// "{a, ...b}" => "__spreadValues({a}, b)"
+				result = js_ast.Expr{Loc: loc, Data: &js_ast.EObject{
+					Properties:   properties,
+					IsSingleLine: e.IsSingleLine,
+				}}
+			} else {
+				// "{...a, b, ...c}" => "__spreadValues(__spreadProps(__spreadValues({}, a), {b}), c)"
+				result = p.callRuntime(loc, "__spreadProps",
+					[]js_ast.Expr{result, {Loc: loc, Data: &js_ast.EObject{
+						Properties:   properties,
+						IsSingleLine: e.IsSingleLine,
+					}}})
+			}
+			properties = []js_ast.Property{}
+		}
+
+		// "{a, ...b}" => "__spreadValues({a}, b)"
+		result = p.callRuntime(loc, "__spreadValues", []js_ast.Expr{result, property.ValueOrNil})
+	}
+
+	if len(properties) > 0 {
+		// "{...a, b}" => "__spreadProps(__spreadValues({}, a), {b})"
+		result = p.callRuntime(loc, "__spreadProps", []js_ast.Expr{result, {Loc: loc, Data: &js_ast.EObject{
+			Properties:    properties,
+			IsSingleLine:  e.IsSingleLine,
+			CloseBraceLoc: e.CloseBraceLoc,
+		}}})
+	}
+
+	return result
+}
+
+func (p *parser) maybeLowerAwait(loc logger.Loc, e *js_ast.EAwait) js_ast.Expr {
+	// "await x" turns into "yield __await(x)" when lowering async generator functions
+	if p.fnOrArrowDataVisit.isGenerator && (p.options.unsupportedJSFeatures.Has(compat.AsyncAwait) || p.options.unsupportedJSFeatures.Has(compat.AsyncGenerator)) {
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EYield{
+			ValueOrNil: js_ast.Expr{Loc: loc, Data: &js_ast.ENew{
+				Target: p.importFromRuntime(loc, "__await"),
+				Args:   []js_ast.Expr{e.Value},
+			}},
+		}}
+	}
+
+	// "await x" turns into "yield x" when lowering async functions
+	if p.options.unsupportedJSFeatures.Has(compat.AsyncAwait) {
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EYield{
+			ValueOrNil: e.Value,
+		}}
+	}
+
+	return js_ast.Expr{Loc: loc, Data: e}
+}
+
+func (p *parser) lowerForAwaitLoop(loc logger.Loc, loop *js_ast.SForOf, stmts []js_ast.Stmt) []js_ast.Stmt {
+	// This code:
+	//
+	//   for await (let x of y) z()
+	//
+	// is transformed into the following code:
+	//
+	//   try {
+	//     for (var iter = __forAwait(y), more, temp, error; more = !(temp = await iter.next()).done; more = false) {
+	//       let x = temp.value;
+	//       z();
+	//     }
+	//   } catch (temp) {
+	//     error = [temp]
+	//   } finally {
+	//     try {
+	//       more && (temp = iter.return) && (await temp.call(iter))
+	//     } finally {
+	//       if (error) throw error[0]
+	//     }
+	//   }
+	//
+	// except that "yield" is used instead of "await" if await is unsupported.
+	// This mostly follows TypeScript's implementation of the syntax transform.
+
+	iterRef := p.generateTempRef(tempRefNoDeclare, "iter")
+	moreRef := p.generateTempRef(tempRefNoDeclare, "more")
+	tempRef := p.generateTempRef(tempRefNoDeclare, "temp")
+	errorRef := p.generateTempRef(tempRefNoDeclare, "error")
+
+	switch init := loop.Init.Data.(type) {
+	case *js_ast.SLocal:
+		if len(init.Decls) == 1 {
+			init.Decls[0].ValueOrNil = js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+				Target:  js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: tempRef}},
+				NameLoc: loc,
+				Name:    "value",
+			}}
+		}
+	case *js_ast.SExpr:
+		init.Value.Data = &js_ast.EBinary{
+			Op:   js_ast.BinOpAssign,
+			Left: init.Value,
+			Right: js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+				Target:  js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: tempRef}},
+				NameLoc: loc,
+				Name:    "value",
+			}},
+		}
+	}
+
+	var body []js_ast.Stmt
+	var closeBraceLoc logger.Loc
+	body = append(body, loop.Init)
+
+	if block, ok := loop.Body.Data.(*js_ast.SBlock); ok {
+		body = append(body, block.Stmts...)
+		closeBraceLoc = block.CloseBraceLoc
+	} else {
+		body = append(body, loop.Body)
+	}
+
+	awaitIterNext := js_ast.Expr{Loc: loc, Data: &js_ast.ECall{
+		Target: js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+			Target:  js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: iterRef}},
+			NameLoc: loc,
+			Name:    "next",
+		}},
+		Kind: js_ast.TargetWasOriginallyPropertyAccess,
+	}}
+	awaitTempCallIter := js_ast.Expr{Loc: loc, Data: &js_ast.ECall{
+		Target: js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+			Target:  js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: tempRef}},
+			NameLoc: loc,
+			Name:    "call",
+		}},
+		Args: []js_ast.Expr{{Loc: loc, Data: &js_ast.EIdentifier{Ref: iterRef}}},
+		Kind: js_ast.TargetWasOriginallyPropertyAccess,
+	}}
+
+	// "await" expressions turn into "yield" expressions when lowering
+	awaitIterNext = p.maybeLowerAwait(awaitIterNext.Loc, &js_ast.EAwait{Value: awaitIterNext})
+	awaitTempCallIter = p.maybeLowerAwait(awaitTempCallIter.Loc, &js_ast.EAwait{Value: awaitTempCallIter})
+
+	return append(stmts, js_ast.Stmt{Loc: loc, Data: &js_ast.STry{
+		BlockLoc: loc,
+		Block: js_ast.SBlock{
+			Stmts: []js_ast.Stmt{{Loc: loc, Data: &js_ast.SFor{
+				InitOrNil: js_ast.Stmt{Loc: loc, Data: &js_ast.SLocal{Kind: js_ast.LocalVar, Decls: []js_ast.Decl{
+					{Binding: js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: iterRef}},
+						ValueOrNil: p.callRuntime(loc, "__forAwait", []js_ast.Expr{loop.Value})},
+					{Binding: js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: moreRef}}},
+					{Binding: js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: tempRef}}},
+					{Binding: js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: errorRef}}},
+				}}},
+				TestOrNil: js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+					Op:   js_ast.BinOpAssign,
+					Left: js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: moreRef}},
+					Right: js_ast.Expr{Loc: loc, Data: &js_ast.EUnary{
+						Op: js_ast.UnOpNot,
+						Value: js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+							Target: js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+								Op:    js_ast.BinOpAssign,
+								Left:  js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: tempRef}},
+								Right: awaitIterNext,
+							}},
+							NameLoc: loc,
+							Name:    "done",
+						}},
+					}},
+				}},
+				UpdateOrNil: js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+					Op:    js_ast.BinOpAssign,
+					Left:  js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: moreRef}},
+					Right: js_ast.Expr{Loc: loc, Data: &js_ast.EBoolean{Value: false}},
+				}},
+				Body: js_ast.Stmt{Loc: loop.Body.Loc, Data: &js_ast.SBlock{
+					Stmts:         body,
+					CloseBraceLoc: closeBraceLoc,
+				}},
+			}}},
+		},
+
+		Catch: &js_ast.Catch{
+			Loc: loc,
+			BindingOrNil: js_ast.Binding{
+				Loc:  loc,
+				Data: &js_ast.BIdentifier{Ref: tempRef},
+			},
+			Block: js_ast.SBlock{
+				Stmts: []js_ast.Stmt{{Loc: loc, Data: &js_ast.SExpr{Value: js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+					Op:   js_ast.BinOpAssign,
+					Left: js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: errorRef}},
+					Right: js_ast.Expr{Loc: loc, Data: &js_ast.EArray{
+						Items:        []js_ast.Expr{{Loc: loc, Data: &js_ast.EIdentifier{Ref: tempRef}}},
+						IsSingleLine: true,
+					}},
+				}}}}},
+			},
+		},
+
+		Finally: &js_ast.Finally{
+			Loc: loc,
+			Block: js_ast.SBlock{
+				Stmts: []js_ast.Stmt{{Loc: loc, Data: &js_ast.STry{
+					BlockLoc: loc,
+					Block: js_ast.SBlock{Stmts: []js_ast.Stmt{{Loc: loc, Data: &js_ast.SExpr{
+						Value: js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+							Op: js_ast.BinOpLogicalAnd,
+							Left: js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+								Op:   js_ast.BinOpLogicalAnd,
+								Left: js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: moreRef}},
+								Right: js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+									Op:   js_ast.BinOpAssign,
+									Left: js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: tempRef}},
+									Right: js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+										Target:  js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: iterRef}},
+										NameLoc: loc,
+										Name:    "return",
+									}},
+								}},
+							}},
+							Right: awaitTempCallIter,
+						}},
+					}}}},
+					Finally: &js_ast.Finally{
+						Loc: loc,
+						Block: js_ast.SBlock{Stmts: []js_ast.Stmt{{Loc: loc, Data: &js_ast.SIf{
+							Test: js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: errorRef}},
+							Yes: js_ast.Stmt{Loc: loc, Data: &js_ast.SThrow{Value: js_ast.Expr{Loc: loc, Data: &js_ast.EIndex{
+								Target: js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: errorRef}},
+								Index:  js_ast.Expr{Loc: loc, Data: &js_ast.ENumber{Value: 0}},
+							}}}},
+						}}}},
+					},
+				}}},
+			},
+		},
+	}})
+}
+
+func bindingHasObjectRest(binding js_ast.Binding) bool {
+	switch b := binding.Data.(type) {
+	case *js_ast.BArray:
+		for _, item := range b.Items {
+			if bindingHasObjectRest(item.Binding) {
+				return true
+			}
+		}
+	case *js_ast.BObject:
+		for _, property := range b.Properties {
+			if property.IsSpread || bindingHasObjectRest(property.Value) {
+				return true
+			}
+		}
+	}
+	return false
+}
+
+func exprHasObjectRest(expr js_ast.Expr) bool {
+	switch e := expr.Data.(type) {
+	case *js_ast.EBinary:
+		if e.Op == js_ast.BinOpAssign && exprHasObjectRest(e.Left) {
+			return true
+		}
+	case *js_ast.EArray:
+		for _, item := range e.Items {
+			if exprHasObjectRest(item) {
+				return true
+			}
+		}
+	case *js_ast.EObject:
+		for _, property := range e.Properties {
+			if property.Kind == js_ast.PropertySpread || exprHasObjectRest(property.ValueOrNil) {
+				return true
+			}
+		}
+	}
+	return false
+}
+
+func (p *parser) lowerObjectRestInDecls(decls []js_ast.Decl) []js_ast.Decl {
+	if !p.options.unsupportedJSFeatures.Has(compat.ObjectRestSpread) {
+		return decls
+	}
+
+	// Don't do any allocations if there are no object rest patterns. We want as
+	// little overhead as possible in the common case.
+	for i, decl := range decls {
+		if decl.ValueOrNil.Data != nil && bindingHasObjectRest(decl.Binding) {
+			clone := append([]js_ast.Decl{}, decls[:i]...)
+			for _, decl := range decls[i:] {
+				if decl.ValueOrNil.Data != nil {
+					target := js_ast.ConvertBindingToExpr(decl.Binding, nil)
+					if result, ok := p.lowerObjectRestToDecls(target, decl.ValueOrNil, clone); ok {
+						clone = result
+						continue
+					}
+				}
+				clone = append(clone, decl)
+			}
+
+			return clone
+		}
+	}
+
+	return decls
+}
+
+func (p *parser) lowerObjectRestInForLoopInit(init js_ast.Stmt, body *js_ast.Stmt) {
+	if !p.options.unsupportedJSFeatures.Has(compat.ObjectRestSpread) {
+		return
+	}
+
+	var bodyPrefixStmt js_ast.Stmt
+
+	switch s := init.Data.(type) {
+	case *js_ast.SExpr:
+		// "for ({...x} in y) {}"
+		// "for ({...x} of y) {}"
+		if exprHasObjectRest(s.Value) {
+			ref := p.generateTempRef(tempRefNeedsDeclare, "")
+			if expr, ok := p.lowerAssign(s.Value, js_ast.Expr{Loc: init.Loc, Data: &js_ast.EIdentifier{Ref: ref}}, objRestReturnValueIsUnused); ok {
+				p.recordUsage(ref)
+				s.Value.Data = &js_ast.EIdentifier{Ref: ref}
+				bodyPrefixStmt = js_ast.Stmt{Loc: expr.Loc, Data: &js_ast.SExpr{Value: expr}}
+			}
+		}
+
+	case *js_ast.SLocal:
+		// "for (let {...x} in y) {}"
+		// "for (let {...x} of y) {}"
+		if len(s.Decls) == 1 && bindingHasObjectRest(s.Decls[0].Binding) {
+			ref := p.generateTempRef(tempRefNoDeclare, "")
+			decl := js_ast.Decl{Binding: s.Decls[0].Binding, ValueOrNil: js_ast.Expr{Loc: init.Loc, Data: &js_ast.EIdentifier{Ref: ref}}}
+			p.recordUsage(ref)
+			decls := p.lowerObjectRestInDecls([]js_ast.Decl{decl})
+			s.Decls[0].Binding.Data = &js_ast.BIdentifier{Ref: ref}
+			bodyPrefixStmt = js_ast.Stmt{Loc: init.Loc, Data: &js_ast.SLocal{Kind: s.Kind, Decls: decls}}
+		}
+	}
+
+	if bodyPrefixStmt.Data != nil {
+		if block, ok := body.Data.(*js_ast.SBlock); ok {
+			// If there's already a block, insert at the front
+			stmts := make([]js_ast.Stmt, 0, 1+len(block.Stmts))
+			block.Stmts = append(append(stmts, bodyPrefixStmt), block.Stmts...)
+		} else {
+			// Otherwise, make a block and insert at the front
+			body.Data = &js_ast.SBlock{Stmts: []js_ast.Stmt{bodyPrefixStmt, *body}}
+		}
+	}
+}
+
+func (p *parser) lowerObjectRestInCatchBinding(catch *js_ast.Catch) {
+	if !p.options.unsupportedJSFeatures.Has(compat.ObjectRestSpread) {
+		return
+	}
+
+	if catch.BindingOrNil.Data != nil && bindingHasObjectRest(catch.BindingOrNil) {
+		ref := p.generateTempRef(tempRefNoDeclare, "")
+		decl := js_ast.Decl{Binding: catch.BindingOrNil, ValueOrNil: js_ast.Expr{Loc: catch.BindingOrNil.Loc, Data: &js_ast.EIdentifier{Ref: ref}}}
+		p.recordUsage(ref)
+		decls := p.lowerObjectRestInDecls([]js_ast.Decl{decl})
+		catch.BindingOrNil.Data = &js_ast.BIdentifier{Ref: ref}
+		stmts := make([]js_ast.Stmt, 0, 1+len(catch.Block.Stmts))
+		stmts = append(stmts, js_ast.Stmt{Loc: catch.BindingOrNil.Loc, Data: &js_ast.SLocal{Kind: js_ast.LocalLet, Decls: decls}})
+		catch.Block.Stmts = append(stmts, catch.Block.Stmts...)
+	}
+}
+
+type objRestMode uint8
+
+const (
+	objRestReturnValueIsUnused objRestMode = iota
+	objRestMustReturnInitExpr
+)
+
+func (p *parser) lowerAssign(rootExpr js_ast.Expr, rootInit js_ast.Expr, mode objRestMode) (js_ast.Expr, bool) {
+	rootExpr, didLower := p.lowerSuperPropertyOrPrivateInAssign(rootExpr)
+
+	var expr js_ast.Expr
+	assign := func(left js_ast.Expr, right js_ast.Expr) {
+		expr = js_ast.JoinWithComma(expr, js_ast.Assign(left, right))
+	}
+
+	if initWrapFunc, ok := p.lowerObjectRestHelper(rootExpr, rootInit, assign, tempRefNeedsDeclare, mode); ok {
+		if initWrapFunc != nil {
+			expr = initWrapFunc(expr)
+		}
+		return expr, true
+	}
+
+	if didLower {
+		return js_ast.Assign(rootExpr, rootInit), true
+	}
+
+	return js_ast.Expr{}, false
+}
+
+func (p *parser) lowerObjectRestToDecls(rootExpr js_ast.Expr, rootInit js_ast.Expr, decls []js_ast.Decl) ([]js_ast.Decl, bool) {
+	assign := func(left js_ast.Expr, right js_ast.Expr) {
+		binding, invalidLog := p.convertExprToBinding(left, invalidLog{})
+		if len(invalidLog.invalidTokens) > 0 {
+			panic("Internal error")
+		}
+		decls = append(decls, js_ast.Decl{Binding: binding, ValueOrNil: right})
+	}
+
+	if _, ok := p.lowerObjectRestHelper(rootExpr, rootInit, assign, tempRefNoDeclare, objRestReturnValueIsUnused); ok {
+		return decls, true
+	}
+
+	return nil, false
+}
+
+func (p *parser) lowerObjectRestHelper(
+	rootExpr js_ast.Expr,
+	rootInit js_ast.Expr,
+	assign func(js_ast.Expr, js_ast.Expr),
+	declare generateTempRefArg,
+	mode objRestMode,
+) (wrapFunc func(js_ast.Expr) js_ast.Expr, ok bool) {
+	if !p.options.unsupportedJSFeatures.Has(compat.ObjectRestSpread) {
+		return nil, false
+	}
+
+	// Check if this could possibly contain an object rest binding
+	switch rootExpr.Data.(type) {
+	case *js_ast.EArray, *js_ast.EObject:
+	default:
+		return nil, false
+	}
+
+	// Scan for object rest bindings and initialize rest binding containment
+	containsRestBinding := make(map[js_ast.E]bool)
+	var findRestBindings func(js_ast.Expr) bool
+	findRestBindings = func(expr js_ast.Expr) bool {
+		found := false
+		switch e := expr.Data.(type) {
+		case *js_ast.EBinary:
+			if e.Op == js_ast.BinOpAssign && findRestBindings(e.Left) {
+				found = true
+			}
+		case *js_ast.EArray:
+			for _, item := range e.Items {
+				if findRestBindings(item) {
+					found = true
+				}
+			}
+		case *js_ast.EObject:
+			for _, property := range e.Properties {
+				if property.Kind == js_ast.PropertySpread || findRestBindings(property.ValueOrNil) {
+					found = true
+				}
+			}
+		}
+		if found {
+			containsRestBinding[expr.Data] = true
+		}
+		return found
+	}
+	findRestBindings(rootExpr)
+	if len(containsRestBinding) == 0 {
+		return nil, false
+	}
+
+	// If there is at least one rest binding, lower the whole expression
+	var visit func(js_ast.Expr, js_ast.Expr, []func() js_ast.Expr)
+
+	captureIntoRef := func(expr js_ast.Expr) ast.Ref {
+		ref := p.generateTempRef(declare, "")
+		assign(js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EIdentifier{Ref: ref}}, expr)
+		p.recordUsage(ref)
+		return ref
+	}
+
+	lowerObjectRestPattern := func(
+		before []js_ast.Property,
+		binding js_ast.Expr,
+		init js_ast.Expr,
+		capturedKeys []func() js_ast.Expr,
+		isSingleLine bool,
+	) {
+		// If there are properties before this one, store the initializer in a
+		// temporary so we can reference it multiple times, then create a new
+		// destructuring assignment for these properties
+		if len(before) > 0 {
+			// "let {a, ...b} = c"
+			ref := captureIntoRef(init)
+			assign(js_ast.Expr{Loc: before[0].Key.Loc, Data: &js_ast.EObject{Properties: before, IsSingleLine: isSingleLine}},
+				js_ast.Expr{Loc: init.Loc, Data: &js_ast.EIdentifier{Ref: ref}})
+			init = js_ast.Expr{Loc: init.Loc, Data: &js_ast.EIdentifier{Ref: ref}}
+			p.recordUsage(ref)
+			p.recordUsage(ref)
+		}
+
+		// Call "__objRest" to clone the initializer without the keys for previous
+		// properties, then assign the result to the binding for the rest pattern
+		keysToExclude := make([]js_ast.Expr, len(capturedKeys))
+		for i, capturedKey := range capturedKeys {
+			keysToExclude[i] = capturedKey()
+		}
+		assign(binding, p.callRuntime(binding.Loc, "__objRest", []js_ast.Expr{init,
+			{Loc: binding.Loc, Data: &js_ast.EArray{Items: keysToExclude, IsSingleLine: isSingleLine}}}))
+	}
+
+	splitArrayPattern := func(
+		before []js_ast.Expr,
+		split js_ast.Expr,
+		after []js_ast.Expr,
+		init js_ast.Expr,
+		isSingleLine bool,
+	) {
+		// If this has a default value, skip the value to target the binding
+		binding := &split
+		if binary, ok := binding.Data.(*js_ast.EBinary); ok && binary.Op == js_ast.BinOpAssign {
+			binding = &binary.Left
+		}
+
+		// Swap the binding with a temporary
+		splitRef := p.generateTempRef(declare, "")
+		deferredBinding := *binding
+		binding.Data = &js_ast.EIdentifier{Ref: splitRef}
+		items := append(before, split)
+
+		// If there are any items left over, defer them until later too
+		var tailExpr js_ast.Expr
+		var tailInit js_ast.Expr
+		if len(after) > 0 {
+			tailRef := p.generateTempRef(declare, "")
+			loc := after[0].Loc
+			tailExpr = js_ast.Expr{Loc: loc, Data: &js_ast.EArray{Items: after, IsSingleLine: isSingleLine}}
+			tailInit = js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: tailRef}}
+			items = append(items, js_ast.Expr{Loc: loc, Data: &js_ast.ESpread{Value: js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: tailRef}}}})
+			p.recordUsage(tailRef)
+			p.recordUsage(tailRef)
+		}
+
+		// The original destructuring assignment must come first
+		assign(js_ast.Expr{Loc: split.Loc, Data: &js_ast.EArray{Items: items, IsSingleLine: isSingleLine}}, init)
+
+		// Then the deferred split is evaluated
+		visit(deferredBinding, js_ast.Expr{Loc: split.Loc, Data: &js_ast.EIdentifier{Ref: splitRef}}, nil)
+		p.recordUsage(splitRef)
+
+		// Then anything after the split
+		if len(after) > 0 {
+			visit(tailExpr, tailInit, nil)
+		}
+	}
+
+	splitObjectPattern := func(
+		upToSplit []js_ast.Property,
+		afterSplit []js_ast.Property,
+		init js_ast.Expr,
+		capturedKeys []func() js_ast.Expr,
+		isSingleLine bool,
+	) {
+		// If there are properties after the split, store the initializer in a
+		// temporary so we can reference it multiple times
+		var afterSplitInit js_ast.Expr
+		if len(afterSplit) > 0 {
+			ref := captureIntoRef(init)
+			init = js_ast.Expr{Loc: init.Loc, Data: &js_ast.EIdentifier{Ref: ref}}
+			afterSplitInit = js_ast.Expr{Loc: init.Loc, Data: &js_ast.EIdentifier{Ref: ref}}
+		}
+
+		split := &upToSplit[len(upToSplit)-1]
+		binding := &split.ValueOrNil
+
+		// Swap the binding with a temporary
+		splitRef := p.generateTempRef(declare, "")
+		deferredBinding := *binding
+		binding.Data = &js_ast.EIdentifier{Ref: splitRef}
+		p.recordUsage(splitRef)
+
+		// Use a destructuring assignment to unpack everything up to and including
+		// the split point
+		assign(js_ast.Expr{Loc: binding.Loc, Data: &js_ast.EObject{Properties: upToSplit, IsSingleLine: isSingleLine}}, init)
+
+		// Handle any nested rest binding patterns inside the split point
+		visit(deferredBinding, js_ast.Expr{Loc: binding.Loc, Data: &js_ast.EIdentifier{Ref: splitRef}}, nil)
+		p.recordUsage(splitRef)
+
+		// Then continue on to any properties after the split
+		if len(afterSplit) > 0 {
+			visit(js_ast.Expr{Loc: binding.Loc, Data: &js_ast.EObject{
+				Properties:   afterSplit,
+				IsSingleLine: isSingleLine,
+			}}, afterSplitInit, capturedKeys)
+		}
+	}
+
+	// This takes an expression representing a binding pattern as input and
+	// returns that binding pattern with any object rest patterns stripped out.
+	// The object rest patterns are lowered and appended to "exprChain" along
+	// with any child binding patterns that came after the binding pattern
+	// containing the object rest pattern.
+	//
+	// This transform must be very careful to preserve the exact evaluation
+	// order of all assignments, default values, and computed property keys.
+	//
+	// Unlike the Babel and TypeScript compilers, this transform does not
+	// lower binding patterns other than object rest patterns. For example,
+	// array spread patterns are preserved.
+	//
+	// Certain patterns such as "{a: {...a}, b: {...b}, ...c}" may need to be
+	// split multiple times. In this case the "capturedKeys" argument allows
+	// the visitor to pass on captured keys to the tail-recursive call that
+	// handles the properties after the split.
+	visit = func(expr js_ast.Expr, init js_ast.Expr, capturedKeys []func() js_ast.Expr) {
+		switch e := expr.Data.(type) {
+		case *js_ast.EArray:
+			// Split on the first binding with a nested rest binding pattern
+			for i, item := range e.Items {
+				// "let [a, {...b}, c] = d"
+				if containsRestBinding[item.Data] {
+					splitArrayPattern(e.Items[:i], item, append([]js_ast.Expr{}, e.Items[i+1:]...), init, e.IsSingleLine)
+					return
+				}
+			}
+
+		case *js_ast.EObject:
+			last := len(e.Properties) - 1
+			endsWithRestBinding := last >= 0 && e.Properties[last].Kind == js_ast.PropertySpread
+
+			// Split on the first binding with a nested rest binding pattern
+			for i := range e.Properties {
+				property := &e.Properties[i]
+
+				// "let {a, ...b} = c"
+				if property.Kind == js_ast.PropertySpread {
+					lowerObjectRestPattern(e.Properties[:i], property.ValueOrNil, init, capturedKeys, e.IsSingleLine)
+					return
+				}
+
+				// Save a copy of this key so the rest binding can exclude it
+				if endsWithRestBinding {
+					key, capturedKey := p.captureKeyForObjectRest(property.Key)
+					property.Key = key
+					capturedKeys = append(capturedKeys, capturedKey)
+				}
+
+				// "let {a: {...b}, c} = d"
+				if containsRestBinding[property.ValueOrNil.Data] {
+					splitObjectPattern(e.Properties[:i+1], e.Properties[i+1:], init, capturedKeys, e.IsSingleLine)
+					return
+				}
+			}
+		}
+
+		assign(expr, init)
+	}
+
+	// Capture and return the value of the initializer if this is an assignment
+	// expression and the return value is used:
+	//
+	//   // Input:
+	//   console.log({...x} = x);
+	//
+	//   // Output:
+	//   var _a;
+	//   console.log((x = __objRest(_a = x, []), _a));
+	//
+	// This isn't necessary if the return value is unused:
+	//
+	//   // Input:
+	//   ({...x} = x);
+	//
+	//   // Output:
+	//   x = __objRest(x, []);
+	//
+	if mode == objRestMustReturnInitExpr {
+		initFunc, initWrapFunc := p.captureValueWithPossibleSideEffects(rootInit.Loc, 2, rootInit, valueCouldBeMutated)
+		rootInit = initFunc()
+		wrapFunc = func(expr js_ast.Expr) js_ast.Expr {
+			return initWrapFunc(js_ast.JoinWithComma(expr, initFunc()))
+		}
+	}
+
+	visit(rootExpr, rootInit, nil)
+	return wrapFunc, true
+}
+
+// Save a copy of the key for the call to "__objRest" later on. Certain
+// expressions can be converted to keys more efficiently than others.
+func (p *parser) captureKeyForObjectRest(originalKey js_ast.Expr) (finalKey js_ast.Expr, capturedKey func() js_ast.Expr) {
+	loc := originalKey.Loc
+	finalKey = originalKey
+
+	switch k := originalKey.Data.(type) {
+	case *js_ast.EString:
+		capturedKey = func() js_ast.Expr { return js_ast.Expr{Loc: loc, Data: &js_ast.EString{Value: k.Value}} }
+
+	case *js_ast.ENumber:
+		// Emit it as the number plus a string (i.e. call toString() on it).
+		// It's important to do it this way instead of trying to print the
+		// float as a string because Go's floating-point printer doesn't
+		// behave exactly the same as JavaScript and if they are different,
+		// the generated code will be wrong.
+		capturedKey = func() js_ast.Expr {
+			return js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+				Op:    js_ast.BinOpAdd,
+				Left:  js_ast.Expr{Loc: loc, Data: &js_ast.ENumber{Value: k.Value}},
+				Right: js_ast.Expr{Loc: loc, Data: &js_ast.EString{}},
+			}}
+		}
+
+	case *js_ast.EIdentifier:
+		capturedKey = func() js_ast.Expr {
+			p.recordUsage(k.Ref)
+			return p.callRuntime(loc, "__restKey", []js_ast.Expr{{Loc: loc, Data: &js_ast.EIdentifier{Ref: k.Ref}}})
+		}
+
+	default:
+		// If it's an arbitrary expression, it probably has a side effect.
+		// Stash it in a temporary reference so we don't evaluate it twice.
+		tempRef := p.generateTempRef(tempRefNeedsDeclare, "")
+		finalKey = js_ast.Assign(js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: tempRef}}, originalKey)
+		capturedKey = func() js_ast.Expr {
+			p.recordUsage(tempRef)
+			return p.callRuntime(loc, "__restKey", []js_ast.Expr{{Loc: loc, Data: &js_ast.EIdentifier{Ref: tempRef}}})
+		}
+	}
+
+	return
+}
+
+func (p *parser) lowerTemplateLiteral(loc logger.Loc, e *js_ast.ETemplate, tagThisFunc func() js_ast.Expr, tagWrapFunc func(js_ast.Expr) js_ast.Expr) js_ast.Expr {
+	// If there is no tag, turn this into normal string concatenation
+	if e.TagOrNil.Data == nil {
+		var value js_ast.Expr
+
+		// Handle the head
+		value = js_ast.Expr{Loc: loc, Data: &js_ast.EString{
+			Value:          e.HeadCooked,
+			LegacyOctalLoc: e.LegacyOctalLoc,
+		}}
+
+		// Handle the tail. Each one is handled with a separate call to ".concat()"
+		// to handle various corner cases in the specification including:
+		//
+		//   * For objects, "toString" must be called instead of "valueOf"
+		//   * Side effects must happen inline instead of at the end
+		//   * Passing a "Symbol" instance should throw
+		//
+		for _, part := range e.Parts {
+			var args []js_ast.Expr
+			if len(part.TailCooked) > 0 {
+				args = []js_ast.Expr{part.Value, {Loc: part.TailLoc, Data: &js_ast.EString{Value: part.TailCooked}}}
+			} else {
+				args = []js_ast.Expr{part.Value}
+			}
+			value = js_ast.Expr{Loc: loc, Data: &js_ast.ECall{
+				Target: js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+					Target:  value,
+					Name:    "concat",
+					NameLoc: part.Value.Loc,
+				}},
+				Args: args,
+				Kind: js_ast.TargetWasOriginallyPropertyAccess,
+			}}
+		}
+
+		return value
+	}
+
+	// Otherwise, call the tag with the template object
+	needsRaw := false
+	cooked := []js_ast.Expr{}
+	raw := []js_ast.Expr{}
+	args := make([]js_ast.Expr, 0, 1+len(e.Parts))
+	args = append(args, js_ast.Expr{})
+
+	// Handle the head
+	if e.HeadCooked == nil {
+		cooked = append(cooked, js_ast.Expr{Loc: e.HeadLoc, Data: js_ast.EUndefinedShared})
+		needsRaw = true
+	} else {
+		cooked = append(cooked, js_ast.Expr{Loc: e.HeadLoc, Data: &js_ast.EString{Value: e.HeadCooked}})
+		if !helpers.UTF16EqualsString(e.HeadCooked, e.HeadRaw) {
+			needsRaw = true
+		}
+	}
+	raw = append(raw, js_ast.Expr{Loc: e.HeadLoc, Data: &js_ast.EString{Value: helpers.StringToUTF16(e.HeadRaw)}})
+
+	// Handle the tail
+	for _, part := range e.Parts {
+		args = append(args, part.Value)
+		if part.TailCooked == nil {
+			cooked = append(cooked, js_ast.Expr{Loc: part.TailLoc, Data: js_ast.EUndefinedShared})
+			needsRaw = true
+		} else {
+			cooked = append(cooked, js_ast.Expr{Loc: part.TailLoc, Data: &js_ast.EString{Value: part.TailCooked}})
+			if !helpers.UTF16EqualsString(part.TailCooked, part.TailRaw) {
+				needsRaw = true
+			}
+		}
+		raw = append(raw, js_ast.Expr{Loc: part.TailLoc, Data: &js_ast.EString{Value: helpers.StringToUTF16(part.TailRaw)}})
+	}
+
+	// Construct the template object
+	cookedArray := js_ast.Expr{Loc: e.HeadLoc, Data: &js_ast.EArray{Items: cooked, IsSingleLine: true}}
+	var arrays []js_ast.Expr
+	if needsRaw {
+		arrays = []js_ast.Expr{cookedArray, {Loc: e.HeadLoc, Data: &js_ast.EArray{Items: raw, IsSingleLine: true}}}
+	} else {
+		arrays = []js_ast.Expr{cookedArray}
+	}
+	templateObj := p.callRuntime(e.HeadLoc, "__template", arrays)
+
+	// Cache it in a temporary object (required by the specification)
+	tempRef := p.generateTopLevelTempRef()
+	p.recordUsage(tempRef)
+	p.recordUsage(tempRef)
+	args[0] = js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+		Op:   js_ast.BinOpLogicalOr,
+		Left: js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: tempRef}},
+		Right: js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+			Op:    js_ast.BinOpAssign,
+			Left:  js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: tempRef}},
+			Right: templateObj,
+		}},
+	}}
+
+	// If this optional chain was used as a template tag, then also forward the value for "this"
+	if tagThisFunc != nil {
+		return tagWrapFunc(js_ast.Expr{Loc: loc, Data: &js_ast.ECall{
+			Target: js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+				Target:  e.TagOrNil,
+				Name:    "call",
+				NameLoc: e.HeadLoc,
+			}},
+			Args: append([]js_ast.Expr{tagThisFunc()}, args...),
+			Kind: js_ast.TargetWasOriginallyPropertyAccess,
+		}})
+	}
+
+	// Call the tag function
+	kind := js_ast.NormalCall
+	if e.TagWasOriginallyPropertyAccess {
+		kind = js_ast.TargetWasOriginallyPropertyAccess
+	}
+	return js_ast.Expr{Loc: loc, Data: &js_ast.ECall{
+		Target: e.TagOrNil,
+		Args:   args,
+		Kind:   kind,
+	}}
+}
+
+func couldPotentiallyThrow(data js_ast.E) bool {
+	switch data.(type) {
+	case *js_ast.ENull, *js_ast.EUndefined, *js_ast.EBoolean, *js_ast.ENumber,
+		*js_ast.EBigInt, *js_ast.EString, *js_ast.EFunction, *js_ast.EArrow:
+		return false
+	}
+	return true
+}
+
+func (p *parser) maybeLowerSetBinOp(left js_ast.Expr, op js_ast.OpCode, right js_ast.Expr) js_ast.Expr {
+	if target, loc, private := p.extractPrivateIndex(left); private != nil {
+		return p.lowerPrivateSetBinOp(target, loc, private, op, right)
+	}
+	if property := p.extractSuperProperty(left); property.Data != nil {
+		return p.lowerSuperPropertySetBinOp(left.Loc, property, op, right)
+	}
+	return js_ast.Expr{}
+}
+
+func (p *parser) shouldLowerUsingDeclarations(stmts []js_ast.Stmt) bool {
+	for _, stmt := range stmts {
+		if local, ok := stmt.Data.(*js_ast.SLocal); ok &&
+			((local.Kind == js_ast.LocalUsing && p.options.unsupportedJSFeatures.Has(compat.Using)) ||
+				(local.Kind == js_ast.LocalAwaitUsing && (p.options.unsupportedJSFeatures.Has(compat.Using) ||
+					p.options.unsupportedJSFeatures.Has(compat.AsyncAwait) ||
+					(p.options.unsupportedJSFeatures.Has(compat.AsyncGenerator) && p.fnOrArrowDataVisit.isGenerator)))) {
+			return true
+		}
+	}
+	return false
+}
+
+type lowerUsingDeclarationContext struct {
+	firstUsingLoc logger.Loc
+	stackRef      ast.Ref
+	hasAwaitUsing bool
+}
+
+func (p *parser) lowerUsingDeclarationContext() lowerUsingDeclarationContext {
+	return lowerUsingDeclarationContext{
+		stackRef: p.newSymbol(ast.SymbolOther, "_stack"),
+	}
+}
+
+// If this returns "nil", then no lowering needed to be done
+func (ctx *lowerUsingDeclarationContext) scanStmts(p *parser, stmts []js_ast.Stmt) {
+	for _, stmt := range stmts {
+		if local, ok := stmt.Data.(*js_ast.SLocal); ok && local.Kind.IsUsing() {
+			// Wrap each "using" initializer in a call to the "__using" helper function
+			if ctx.firstUsingLoc.Start == 0 {
+				ctx.firstUsingLoc = stmt.Loc
+			}
+			if local.Kind == js_ast.LocalAwaitUsing {
+				ctx.hasAwaitUsing = true
+			}
+			for i, decl := range local.Decls {
+				if decl.ValueOrNil.Data != nil {
+					valueLoc := decl.ValueOrNil.Loc
+					p.recordUsage(ctx.stackRef)
+					args := []js_ast.Expr{
+						{Loc: valueLoc, Data: &js_ast.EIdentifier{Ref: ctx.stackRef}},
+						decl.ValueOrNil,
+					}
+					if local.Kind == js_ast.LocalAwaitUsing {
+						args = append(args, js_ast.Expr{Loc: valueLoc, Data: &js_ast.EBoolean{Value: true}})
+					}
+					local.Decls[i].ValueOrNil = p.callRuntime(valueLoc, "__using", args)
+				}
+			}
+			if p.willWrapModuleInTryCatchForUsing && p.currentScope.Parent == nil {
+				local.Kind = js_ast.LocalVar
+			} else {
+				local.Kind = p.selectLocalKind(js_ast.LocalConst)
+			}
+		}
+	}
+}
+
+func (ctx *lowerUsingDeclarationContext) finalize(p *parser, stmts []js_ast.Stmt, shouldHoistFunctions bool) []js_ast.Stmt {
+	var result []js_ast.Stmt
+	var exports []js_ast.ClauseItem
+	end := 0
+
+	// Filter out statements that can't go in a try/catch block
+	for _, stmt := range stmts {
+		switch s := stmt.Data.(type) {
+		// Note: We don't need to handle class declarations here because they
+		// should have been already converted into local "var" declarations
+		// before this point. It's done in "lowerClass" instead of here because
+		// "lowerClass" already does this sometimes for other reasons, and it's
+		// more straightforward to do it in one place because it's complicated.
+
+		case *js_ast.SDirective, *js_ast.SImport, *js_ast.SExportFrom, *js_ast.SExportStar:
+			// These can't go in a try/catch block
+			result = append(result, stmt)
+			continue
+
+		case *js_ast.SExportClause:
+			// Merge export clauses together
+			exports = append(exports, s.Items...)
+			continue
+
+		case *js_ast.SFunction:
+			if shouldHoistFunctions {
+				// Hoist function declarations for cross-file ESM references
+				result = append(result, stmt)
+				continue
+			}
+
+		case *js_ast.SExportDefault:
+			if _, ok := s.Value.Data.(*js_ast.SFunction); ok && shouldHoistFunctions {
+				// Hoist function declarations for cross-file ESM references
+				result = append(result, stmt)
+				continue
+			}
+
+		case *js_ast.SLocal:
+			// If any of these are exported, turn it into a "var" and add export clauses
+			if s.IsExport {
+				js_ast.ForEachIdentifierBindingInDecls(s.Decls, func(loc logger.Loc, b *js_ast.BIdentifier) {
+					exports = append(exports, js_ast.ClauseItem{
+						Alias:    p.symbols[b.Ref.InnerIndex].OriginalName,
+						AliasLoc: loc,
+						Name:     ast.LocRef{Loc: loc, Ref: b.Ref},
+					})
+					s.Kind = js_ast.LocalVar
+				})
+				s.IsExport = false
+			}
+		}
+
+		stmts[end] = stmt
+		end++
+	}
+	stmts = stmts[:end]
+
+	// Generate the variables we'll need
+	caughtRef := p.newSymbol(ast.SymbolOther, "_")
+	errorRef := p.newSymbol(ast.SymbolOther, "_error")
+	hasErrorRef := p.newSymbol(ast.SymbolOther, "_hasError")
+
+	// Generated variables are declared with "var", so hoist them up
+	scope := p.currentScope
+	for !scope.Kind.StopsHoisting() {
+		scope = scope.Parent
+	}
+	isTopLevel := scope == p.moduleScope
+	scope.Generated = append(scope.Generated, ctx.stackRef, caughtRef, errorRef, hasErrorRef)
+	p.declaredSymbols = append(p.declaredSymbols,
+		js_ast.DeclaredSymbol{IsTopLevel: isTopLevel, Ref: ctx.stackRef},
+		js_ast.DeclaredSymbol{IsTopLevel: isTopLevel, Ref: caughtRef},
+		js_ast.DeclaredSymbol{IsTopLevel: isTopLevel, Ref: errorRef},
+		js_ast.DeclaredSymbol{IsTopLevel: isTopLevel, Ref: hasErrorRef},
+	)
+
+	// Call the "__callDispose" helper function at the end of the scope
+	loc := ctx.firstUsingLoc
+	p.recordUsage(ctx.stackRef)
+	p.recordUsage(errorRef)
+	p.recordUsage(hasErrorRef)
+	callDispose := p.callRuntime(loc, "__callDispose", []js_ast.Expr{
+		{Loc: loc, Data: &js_ast.EIdentifier{Ref: ctx.stackRef}},
+		{Loc: loc, Data: &js_ast.EIdentifier{Ref: errorRef}},
+		{Loc: loc, Data: &js_ast.EIdentifier{Ref: hasErrorRef}},
+	})
+
+	// If there was an "await using", optionally await the returned promise
+	var finallyStmts []js_ast.Stmt
+	if ctx.hasAwaitUsing {
+		promiseRef := p.generateTempRef(tempRefNoDeclare, "_promise")
+		scope.Generated = append(scope.Generated, promiseRef)
+		p.declaredSymbols = append(p.declaredSymbols, js_ast.DeclaredSymbol{IsTopLevel: isTopLevel, Ref: promiseRef})
+
+		// "await" expressions turn into "yield" expressions when lowering
+		p.recordUsage(promiseRef)
+		awaitExpr := p.maybeLowerAwait(loc, &js_ast.EAwait{Value: js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: promiseRef}}})
+
+		p.recordUsage(promiseRef)
+		finallyStmts = []js_ast.Stmt{
+			{Loc: loc, Data: &js_ast.SLocal{Decls: []js_ast.Decl{{
+				Binding:    js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: promiseRef}},
+				ValueOrNil: callDispose,
+			}}}},
+
+			// The "await" must not happen if an error was thrown before the
+			// "await using", so we conditionally await here:
+			//
+			//   var promise = __callDispose(stack, error, hasError);
+			//   promise && await promise;
+			//
+			{Loc: loc, Data: &js_ast.SExpr{Value: js_ast.Expr{Loc: loc, Data: &js_ast.EBinary{
+				Op:    js_ast.BinOpLogicalAnd,
+				Left:  js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: promiseRef}},
+				Right: awaitExpr,
+			}}}},
+		}
+	} else {
+		finallyStmts = []js_ast.Stmt{{Loc: loc, Data: &js_ast.SExpr{Value: callDispose}}}
+	}
+
+	// Wrap everything in a try/catch/finally block
+	p.recordUsage(caughtRef)
+	result = append(result,
+		js_ast.Stmt{Loc: loc, Data: &js_ast.SLocal{
+			Decls: []js_ast.Decl{{
+				Binding:    js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: ctx.stackRef}},
+				ValueOrNil: js_ast.Expr{Loc: loc, Data: &js_ast.EArray{}},
+			}},
+		}},
+		js_ast.Stmt{Loc: loc, Data: &js_ast.STry{
+			Block: js_ast.SBlock{
+				Stmts: stmts,
+			},
+			BlockLoc: loc,
+			Catch: &js_ast.Catch{
+				Loc:          loc,
+				BindingOrNil: js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: caughtRef}},
+				Block: js_ast.SBlock{Stmts: []js_ast.Stmt{{Loc: loc, Data: &js_ast.SLocal{
+					Decls: []js_ast.Decl{{
+						Binding:    js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: errorRef}},
+						ValueOrNil: js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: caughtRef}},
+					}, {
+						Binding:    js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: hasErrorRef}},
+						ValueOrNil: js_ast.Expr{Loc: loc, Data: &js_ast.EBoolean{Value: true}},
+					}},
+				}}}},
+				BlockLoc: loc,
+			},
+			Finally: &js_ast.Finally{
+				Loc:   loc,
+				Block: js_ast.SBlock{Stmts: finallyStmts},
+			},
+		}},
+	)
+	if len(exports) > 0 {
+		result = append(result, js_ast.Stmt{Loc: loc, Data: &js_ast.SExportClause{Items: exports}})
+	}
+	return result
+}
+
+func (p *parser) lowerUsingDeclarationInForOf(loc logger.Loc, init *js_ast.SLocal, body *js_ast.Stmt) {
+	binding := init.Decls[0].Binding
+	id := binding.Data.(*js_ast.BIdentifier)
+	tempRef := p.generateTempRef(tempRefNoDeclare, "_"+p.symbols[id.Ref.InnerIndex].OriginalName)
+	block, ok := body.Data.(*js_ast.SBlock)
+	if !ok {
+		block = &js_ast.SBlock{}
+		if _, ok := body.Data.(*js_ast.SEmpty); !ok {
+			block.Stmts = append(block.Stmts, *body)
+		}
+		body.Data = block
+	}
+	blockStmts := make([]js_ast.Stmt, 0, 1+len(block.Stmts))
+	blockStmts = append(blockStmts, js_ast.Stmt{Loc: loc, Data: &js_ast.SLocal{
+		Kind: init.Kind,
+		Decls: []js_ast.Decl{{
+			Binding:    js_ast.Binding{Loc: binding.Loc, Data: &js_ast.BIdentifier{Ref: id.Ref}},
+			ValueOrNil: js_ast.Expr{Loc: binding.Loc, Data: &js_ast.EIdentifier{Ref: tempRef}},
+		}},
+	}})
+	blockStmts = append(blockStmts, block.Stmts...)
+	ctx := p.lowerUsingDeclarationContext()
+	ctx.scanStmts(p, blockStmts)
+	block.Stmts = ctx.finalize(p, blockStmts, p.willWrapModuleInTryCatchForUsing && p.currentScope.Parent == nil)
+	init.Kind = js_ast.LocalVar
+	id.Ref = tempRef
+}
+
+func (p *parser) maybeLowerUsingDeclarationsInSwitch(loc logger.Loc, s *js_ast.SSwitch) []js_ast.Stmt {
+	// Check for a "using" declaration in any case
+	shouldLower := false
+	for _, c := range s.Cases {
+		if p.shouldLowerUsingDeclarations(c.Body) {
+			shouldLower = true
+			break
+		}
+	}
+	if !shouldLower {
+		return nil
+	}
+
+	// If we find one, lower all cases together
+	ctx := p.lowerUsingDeclarationContext()
+	for _, c := range s.Cases {
+		ctx.scanStmts(p, c.Body)
+	}
+	return ctx.finalize(p, []js_ast.Stmt{{Loc: loc, Data: s}}, false)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/js_parser/js_parser_lower_class.go b/source/vendor/github.com/evanw/esbuild/internal/js_parser/js_parser_lower_class.go
new file mode 100644
index 0000000..0145836
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/js_parser/js_parser_lower_class.go
@@ -0,0 +1,2573 @@
+package js_parser
+
+import (
+	"fmt"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/config"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+func (p *parser) privateSymbolNeedsToBeLowered(private *js_ast.EPrivateIdentifier) bool {
+	symbol := &p.symbols[private.Ref.InnerIndex]
+	return p.options.unsupportedJSFeatures.Has(compat.SymbolFeature(symbol.Kind)) || symbol.Flags.Has(ast.PrivateSymbolMustBeLowered)
+}
+
+func (p *parser) lowerPrivateBrandCheck(target js_ast.Expr, loc logger.Loc, private *js_ast.EPrivateIdentifier) js_ast.Expr {
+	// "#field in this" => "__privateIn(#field, this)"
+	return p.callRuntime(loc, "__privateIn", []js_ast.Expr{
+		{Loc: loc, Data: &js_ast.EIdentifier{Ref: private.Ref}},
+		target,
+	})
+}
+
+func (p *parser) lowerPrivateGet(target js_ast.Expr, loc logger.Loc, private *js_ast.EPrivateIdentifier) js_ast.Expr {
+	switch p.symbols[private.Ref.InnerIndex].Kind {
+	case ast.SymbolPrivateMethod, ast.SymbolPrivateStaticMethod:
+		// "this.#method" => "__privateMethod(this, #method, method_fn)"
+		fnRef := p.privateGetters[private.Ref]
+		p.recordUsage(fnRef)
+		return p.callRuntime(target.Loc, "__privateMethod", []js_ast.Expr{
+			target,
+			{Loc: loc, Data: &js_ast.EIdentifier{Ref: private.Ref}},
+			{Loc: loc, Data: &js_ast.EIdentifier{Ref: fnRef}},
+		})
+
+	case ast.SymbolPrivateGet, ast.SymbolPrivateStaticGet,
+		ast.SymbolPrivateGetSetPair, ast.SymbolPrivateStaticGetSetPair:
+		// "this.#getter" => "__privateGet(this, #getter, getter_get)"
+		fnRef := p.privateGetters[private.Ref]
+		p.recordUsage(fnRef)
+		return p.callRuntime(target.Loc, "__privateGet", []js_ast.Expr{
+			target,
+			{Loc: loc, Data: &js_ast.EIdentifier{Ref: private.Ref}},
+			{Loc: loc, Data: &js_ast.EIdentifier{Ref: fnRef}},
+		})
+
+	default:
+		// "this.#field" => "__privateGet(this, #field)"
+		return p.callRuntime(target.Loc, "__privateGet", []js_ast.Expr{
+			target,
+			{Loc: loc, Data: &js_ast.EIdentifier{Ref: private.Ref}},
+		})
+	}
+}
+
+func (p *parser) lowerPrivateSet(
+	target js_ast.Expr,
+	loc logger.Loc,
+	private *js_ast.EPrivateIdentifier,
+	value js_ast.Expr,
+) js_ast.Expr {
+	switch p.symbols[private.Ref.InnerIndex].Kind {
+	case ast.SymbolPrivateSet, ast.SymbolPrivateStaticSet,
+		ast.SymbolPrivateGetSetPair, ast.SymbolPrivateStaticGetSetPair:
+		// "this.#setter = 123" => "__privateSet(this, #setter, 123, setter_set)"
+		fnRef := p.privateSetters[private.Ref]
+		p.recordUsage(fnRef)
+		return p.callRuntime(target.Loc, "__privateSet", []js_ast.Expr{
+			target,
+			{Loc: loc, Data: &js_ast.EIdentifier{Ref: private.Ref}},
+			value,
+			{Loc: loc, Data: &js_ast.EIdentifier{Ref: fnRef}},
+		})
+
+	default:
+		// "this.#field = 123" => "__privateSet(this, #field, 123)"
+		return p.callRuntime(target.Loc, "__privateSet", []js_ast.Expr{
+			target,
+			{Loc: loc, Data: &js_ast.EIdentifier{Ref: private.Ref}},
+			value,
+		})
+	}
+}
+
+func (p *parser) lowerPrivateSetUnOp(target js_ast.Expr, loc logger.Loc, private *js_ast.EPrivateIdentifier, op js_ast.OpCode) js_ast.Expr {
+	kind := p.symbols[private.Ref.InnerIndex].Kind
+
+	// Determine the setter, if any
+	var setter js_ast.Expr
+	switch kind {
+	case ast.SymbolPrivateSet, ast.SymbolPrivateStaticSet,
+		ast.SymbolPrivateGetSetPair, ast.SymbolPrivateStaticGetSetPair:
+		ref := p.privateSetters[private.Ref]
+		p.recordUsage(ref)
+		setter = js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: ref}}
+	}
+
+	// Determine the getter, if any
+	var getter js_ast.Expr
+	switch kind {
+	case ast.SymbolPrivateGet, ast.SymbolPrivateStaticGet,
+		ast.SymbolPrivateGetSetPair, ast.SymbolPrivateStaticGetSetPair:
+		ref := p.privateGetters[private.Ref]
+		p.recordUsage(ref)
+		getter = js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: ref}}
+	}
+
+	// Only include necessary arguments
+	args := []js_ast.Expr{
+		target,
+		{Loc: loc, Data: &js_ast.EIdentifier{Ref: private.Ref}},
+	}
+	if setter.Data != nil {
+		args = append(args, setter)
+	}
+	if getter.Data != nil {
+		if setter.Data == nil {
+			args = append(args, js_ast.Expr{Loc: loc, Data: js_ast.ENullShared})
+		}
+		args = append(args, getter)
+	}
+
+	// "target.#private++" => "__privateWrapper(target, #private, private_set, private_get)._++"
+	return js_ast.Expr{Loc: loc, Data: &js_ast.EUnary{
+		Op: op,
+		Value: js_ast.Expr{Loc: target.Loc, Data: &js_ast.EDot{
+			Target:  p.callRuntime(target.Loc, "__privateWrapper", args),
+			NameLoc: target.Loc,
+			Name:    "_",
+		}},
+	}}
+}
+
+func (p *parser) lowerPrivateSetBinOp(target js_ast.Expr, loc logger.Loc, private *js_ast.EPrivateIdentifier, op js_ast.OpCode, value js_ast.Expr) js_ast.Expr {
+	// "target.#private += 123" => "__privateSet(target, #private, __privateGet(target, #private) + 123)"
+	targetFunc, targetWrapFunc := p.captureValueWithPossibleSideEffects(target.Loc, 2, target, valueDefinitelyNotMutated)
+	return targetWrapFunc(p.lowerPrivateSet(targetFunc(), loc, private, js_ast.Expr{Loc: value.Loc, Data: &js_ast.EBinary{
+		Op:    op,
+		Left:  p.lowerPrivateGet(targetFunc(), loc, private),
+		Right: value,
+	}}))
+}
+
+// Returns valid data if target is an expression of the form "foo.#bar" and if
+// the language target is such that private members must be lowered
+func (p *parser) extractPrivateIndex(target js_ast.Expr) (js_ast.Expr, logger.Loc, *js_ast.EPrivateIdentifier) {
+	if index, ok := target.Data.(*js_ast.EIndex); ok {
+		if private, ok := index.Index.Data.(*js_ast.EPrivateIdentifier); ok && p.privateSymbolNeedsToBeLowered(private) {
+			return index.Target, index.Index.Loc, private
+		}
+	}
+	return js_ast.Expr{}, logger.Loc{}, nil
+}
+
+// Returns a valid property if target is an expression of the form "super.bar"
+// or "super[bar]" and if the situation is such that it must be lowered
+func (p *parser) extractSuperProperty(target js_ast.Expr) js_ast.Expr {
+	switch e := target.Data.(type) {
+	case *js_ast.EDot:
+		if p.shouldLowerSuperPropertyAccess(e.Target) {
+			return js_ast.Expr{Loc: e.NameLoc, Data: &js_ast.EString{Value: helpers.StringToUTF16(e.Name)}}
+		}
+	case *js_ast.EIndex:
+		if p.shouldLowerSuperPropertyAccess(e.Target) {
+			return e.Index
+		}
+	}
+	return js_ast.Expr{}
+}
+
+func (p *parser) lowerSuperPropertyOrPrivateInAssign(expr js_ast.Expr) (js_ast.Expr, bool) {
+	didLower := false
+
+	switch e := expr.Data.(type) {
+	case *js_ast.ESpread:
+		if value, ok := p.lowerSuperPropertyOrPrivateInAssign(e.Value); ok {
+			e.Value = value
+			didLower = true
+		}
+
+	case *js_ast.EDot:
+		// "[super.foo] = [bar]" => "[__superWrapper(this, 'foo')._] = [bar]"
+		if p.shouldLowerSuperPropertyAccess(e.Target) {
+			key := js_ast.Expr{Loc: e.NameLoc, Data: &js_ast.EString{Value: helpers.StringToUTF16(e.Name)}}
+			expr = p.callSuperPropertyWrapper(expr.Loc, key)
+			didLower = true
+		}
+
+	case *js_ast.EIndex:
+		// "[super[foo]] = [bar]" => "[__superWrapper(this, foo)._] = [bar]"
+		if p.shouldLowerSuperPropertyAccess(e.Target) {
+			expr = p.callSuperPropertyWrapper(expr.Loc, e.Index)
+			didLower = true
+			break
+		}
+
+		// "[a.#b] = [c]" => "[__privateWrapper(a, #b)._] = [c]"
+		if private, ok := e.Index.Data.(*js_ast.EPrivateIdentifier); ok && p.privateSymbolNeedsToBeLowered(private) {
+			var target js_ast.Expr
+
+			switch p.symbols[private.Ref.InnerIndex].Kind {
+			case ast.SymbolPrivateSet, ast.SymbolPrivateStaticSet,
+				ast.SymbolPrivateGetSetPair, ast.SymbolPrivateStaticGetSetPair:
+				// "this.#setter" => "__privateWrapper(this, #setter, setter_set)"
+				fnRef := p.privateSetters[private.Ref]
+				p.recordUsage(fnRef)
+				target = p.callRuntime(expr.Loc, "__privateWrapper", []js_ast.Expr{
+					e.Target,
+					{Loc: expr.Loc, Data: &js_ast.EIdentifier{Ref: private.Ref}},
+					{Loc: expr.Loc, Data: &js_ast.EIdentifier{Ref: fnRef}},
+				})
+
+			default:
+				// "this.#field" => "__privateWrapper(this, #field)"
+				target = p.callRuntime(expr.Loc, "__privateWrapper", []js_ast.Expr{
+					e.Target,
+					{Loc: expr.Loc, Data: &js_ast.EIdentifier{Ref: private.Ref}},
+				})
+			}
+
+			// "__privateWrapper(this, #field)" => "__privateWrapper(this, #field)._"
+			expr.Data = &js_ast.EDot{Target: target, Name: "_", NameLoc: expr.Loc}
+			didLower = true
+		}
+
+	case *js_ast.EArray:
+		for i, item := range e.Items {
+			if item, ok := p.lowerSuperPropertyOrPrivateInAssign(item); ok {
+				e.Items[i] = item
+				didLower = true
+			}
+		}
+
+	case *js_ast.EObject:
+		for i, property := range e.Properties {
+			if property.ValueOrNil.Data != nil {
+				if value, ok := p.lowerSuperPropertyOrPrivateInAssign(property.ValueOrNil); ok {
+					e.Properties[i].ValueOrNil = value
+					didLower = true
+				}
+			}
+		}
+	}
+
+	return expr, didLower
+}
+
+func (p *parser) shouldLowerSuperPropertyAccess(expr js_ast.Expr) bool {
+	if p.fnOrArrowDataVisit.shouldLowerSuperPropertyAccess {
+		_, isSuper := expr.Data.(*js_ast.ESuper)
+		return isSuper
+	}
+	return false
+}
+
+func (p *parser) callSuperPropertyWrapper(loc logger.Loc, key js_ast.Expr) js_ast.Expr {
+	ref := *p.fnOnlyDataVisit.innerClassNameRef
+	p.recordUsage(ref)
+	class := js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: ref}}
+	this := js_ast.Expr{Loc: loc, Data: js_ast.EThisShared}
+
+	// Handle "this" in lowered static class field initializers
+	if p.fnOnlyDataVisit.shouldReplaceThisWithInnerClassNameRef {
+		p.recordUsage(ref)
+		this.Data = &js_ast.EIdentifier{Ref: ref}
+	}
+
+	if !p.fnOnlyDataVisit.isInStaticClassContext {
+		// "super.foo" => "__superWrapper(Class.prototype, this, 'foo')._"
+		// "super[foo]" => "__superWrapper(Class.prototype, this, foo)._"
+		class.Data = &js_ast.EDot{Target: class, NameLoc: loc, Name: "prototype"}
+	}
+
+	return js_ast.Expr{Loc: loc, Data: &js_ast.EDot{Target: p.callRuntime(loc, "__superWrapper", []js_ast.Expr{
+		class,
+		this,
+		key,
+	}), Name: "_", NameLoc: loc}}
+}
+
+func (p *parser) lowerSuperPropertyGet(loc logger.Loc, key js_ast.Expr) js_ast.Expr {
+	ref := *p.fnOnlyDataVisit.innerClassNameRef
+	p.recordUsage(ref)
+	class := js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: ref}}
+	this := js_ast.Expr{Loc: loc, Data: js_ast.EThisShared}
+
+	// Handle "this" in lowered static class field initializers
+	if p.fnOnlyDataVisit.shouldReplaceThisWithInnerClassNameRef {
+		p.recordUsage(ref)
+		this.Data = &js_ast.EIdentifier{Ref: ref}
+	}
+
+	if !p.fnOnlyDataVisit.isInStaticClassContext {
+		// "super.foo" => "__superGet(Class.prototype, this, 'foo')"
+		// "super[foo]" => "__superGet(Class.prototype, this, foo)"
+		class.Data = &js_ast.EDot{Target: class, NameLoc: loc, Name: "prototype"}
+	}
+
+	return p.callRuntime(loc, "__superGet", []js_ast.Expr{
+		class,
+		this,
+		key,
+	})
+}
+
+func (p *parser) lowerSuperPropertySet(loc logger.Loc, key js_ast.Expr, value js_ast.Expr) js_ast.Expr {
+	// "super.foo = bar" => "__superSet(Class, this, 'foo', bar)"
+	// "super[foo] = bar" => "__superSet(Class, this, foo, bar)"
+	ref := *p.fnOnlyDataVisit.innerClassNameRef
+	p.recordUsage(ref)
+	class := js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: ref}}
+	this := js_ast.Expr{Loc: loc, Data: js_ast.EThisShared}
+
+	// Handle "this" in lowered static class field initializers
+	if p.fnOnlyDataVisit.shouldReplaceThisWithInnerClassNameRef {
+		p.recordUsage(ref)
+		this.Data = &js_ast.EIdentifier{Ref: ref}
+	}
+
+	if !p.fnOnlyDataVisit.isInStaticClassContext {
+		// "super.foo = bar" => "__superSet(Class.prototype, this, 'foo', bar)"
+		// "super[foo] = bar" => "__superSet(Class.prototype, this, foo, bar)"
+		class.Data = &js_ast.EDot{Target: class, NameLoc: loc, Name: "prototype"}
+	}
+
+	return p.callRuntime(loc, "__superSet", []js_ast.Expr{
+		class,
+		this,
+		key,
+		value,
+	})
+}
+
+func (p *parser) lowerSuperPropertySetBinOp(loc logger.Loc, property js_ast.Expr, op js_ast.OpCode, value js_ast.Expr) js_ast.Expr {
+	// "super.foo += bar" => "__superSet(Class, this, 'foo', __superGet(Class, this, 'foo') + bar)"
+	// "super[foo] += bar" => "__superSet(Class, this, foo, __superGet(Class, this, foo) + bar)"
+	// "super[foo()] += bar" => "__superSet(Class, this, _a = foo(), __superGet(Class, this, _a) + bar)"
+	targetFunc, targetWrapFunc := p.captureValueWithPossibleSideEffects(property.Loc, 2, property, valueDefinitelyNotMutated)
+	return targetWrapFunc(p.lowerSuperPropertySet(loc, targetFunc(), js_ast.Expr{Loc: value.Loc, Data: &js_ast.EBinary{
+		Op:    op,
+		Left:  p.lowerSuperPropertyGet(loc, targetFunc()),
+		Right: value,
+	}}))
+}
+
+func (p *parser) maybeLowerSuperPropertyGetInsideCall(call *js_ast.ECall) {
+	var key js_ast.Expr
+
+	switch e := call.Target.Data.(type) {
+	case *js_ast.EDot:
+		// Lower "super.prop" if necessary
+		if !p.shouldLowerSuperPropertyAccess(e.Target) {
+			return
+		}
+		key = js_ast.Expr{Loc: e.NameLoc, Data: &js_ast.EString{Value: helpers.StringToUTF16(e.Name)}}
+
+	case *js_ast.EIndex:
+		// Lower "super[prop]" if necessary
+		if !p.shouldLowerSuperPropertyAccess(e.Target) {
+			return
+		}
+		key = e.Index
+
+	default:
+		return
+	}
+
+	// "super.foo(a, b)" => "__superGet(Class, this, 'foo').call(this, a, b)"
+	call.Target.Data = &js_ast.EDot{
+		Target:  p.lowerSuperPropertyGet(call.Target.Loc, key),
+		NameLoc: key.Loc,
+		Name:    "call",
+	}
+	thisExpr := js_ast.Expr{Loc: call.Target.Loc, Data: js_ast.EThisShared}
+	call.Args = append([]js_ast.Expr{thisExpr}, call.Args...)
+}
+
+type classLoweringInfo struct {
+	lowerAllInstanceFields bool
+	lowerAllStaticFields   bool
+	shimSuperCtorCalls     bool
+}
+
+func (p *parser) computeClassLoweringInfo(class *js_ast.Class) (result classLoweringInfo) {
+	// Name keeping for classes is implemented with a static block. So we need to
+	// lower all static fields if static blocks are unsupported so that the name
+	// keeping comes first before other static initializers.
+	if p.options.keepNames && p.options.unsupportedJSFeatures.Has(compat.ClassStaticBlocks) {
+		result.lowerAllStaticFields = true
+	}
+
+	// TypeScript's "experimentalDecorators" feature replaces all references of
+	// the class name with the decorated class after class decorators have run.
+	// This cannot be done by only reassigning to the class symbol in JavaScript
+	// because it's shadowed by the class name within the class body. Instead,
+	// we need to hoist all code in static contexts out of the class body so
+	// that it's no longer shadowed:
+	//
+	//   const decorate = x => ({ x })
+	//   @decorate
+	//   class Foo {
+	//     static oldFoo = Foo
+	//     static newFoo = () => Foo
+	//   }
+	//   console.log('This must be false:', Foo.x.oldFoo === Foo.x.newFoo())
+	//
+	if p.options.ts.Parse && p.options.ts.Config.ExperimentalDecorators == config.True && len(class.Decorators) > 0 {
+		result.lowerAllStaticFields = true
+	}
+
+	// If something has decorators, just lower everything for now. It's possible
+	// that we could avoid lowering in certain cases, but doing so is very tricky
+	// due to the complexity of the decorator specification. The specification is
+	// also still evolving so trying to optimize it now is also potentially
+	// premature.
+	if class.ShouldLowerStandardDecorators {
+		for _, prop := range class.Properties {
+			if len(prop.Decorators) > 0 {
+				for _, prop := range class.Properties {
+					if private, ok := prop.Key.Data.(*js_ast.EPrivateIdentifier); ok {
+						p.symbols[private.Ref.InnerIndex].Flags |= ast.PrivateSymbolMustBeLowered
+					}
+				}
+				result.lowerAllStaticFields = true
+				result.lowerAllInstanceFields = true
+				break
+			}
+		}
+	}
+
+	// Conservatively lower fields of a given type (instance or static) when any
+	// member of that type needs to be lowered. This must be done to preserve
+	// evaluation order. For example:
+	//
+	//   class Foo {
+	//     #foo = 123
+	//     bar = this.#foo
+	//   }
+	//
+	// It would be bad if we transformed that into something like this:
+	//
+	//   var _foo;
+	//   class Foo {
+	//     constructor() {
+	//       _foo.set(this, 123);
+	//     }
+	//     bar = __privateGet(this, _foo);
+	//   }
+	//   _foo = new WeakMap();
+	//
+	// That evaluates "bar" then "foo" instead of "foo" then "bar" like the
+	// original code. We need to do this instead:
+	//
+	//   var _foo;
+	//   class Foo {
+	//     constructor() {
+	//       _foo.set(this, 123);
+	//       __publicField(this, "bar", __privateGet(this, _foo));
+	//     }
+	//   }
+	//   _foo = new WeakMap();
+	//
+	for _, prop := range class.Properties {
+		if prop.Kind == js_ast.PropertyClassStaticBlock {
+			if p.options.unsupportedJSFeatures.Has(compat.ClassStaticBlocks) {
+				result.lowerAllStaticFields = true
+			}
+			continue
+		}
+
+		if private, ok := prop.Key.Data.(*js_ast.EPrivateIdentifier); ok {
+			if prop.Flags.Has(js_ast.PropertyIsStatic) {
+				if p.privateSymbolNeedsToBeLowered(private) {
+					result.lowerAllStaticFields = true
+				}
+			} else {
+				if p.privateSymbolNeedsToBeLowered(private) {
+					result.lowerAllInstanceFields = true
+
+					// We can't transform this:
+					//
+					//   class Foo {
+					//     #foo = 123
+					//     static bar = new Foo().#foo
+					//   }
+					//
+					// into this:
+					//
+					//   var _foo;
+					//   const _Foo = class {
+					//     constructor() {
+					//       _foo.set(this, 123);
+					//     }
+					//     static bar = __privateGet(new _Foo(), _foo);
+					//   };
+					//   let Foo = _Foo;
+					//   _foo = new WeakMap();
+					//
+					// because "_Foo" won't be initialized in the initializer for "bar".
+					// So we currently lower all static fields in this case too. This
+					// isn't great and it would be good to find a way to avoid this.
+					// The inner class name symbol substitution mechanism should probably
+					// be rethought.
+					result.lowerAllStaticFields = true
+				}
+			}
+			continue
+		}
+
+		if prop.Kind == js_ast.PropertyAutoAccessor {
+			if prop.Flags.Has(js_ast.PropertyIsStatic) {
+				if p.options.unsupportedJSFeatures.Has(compat.ClassPrivateStaticField) {
+					result.lowerAllStaticFields = true
+				}
+			} else {
+				if p.options.unsupportedJSFeatures.Has(compat.ClassPrivateField) {
+					result.lowerAllInstanceFields = true
+					result.lowerAllStaticFields = true
+				}
+			}
+			continue
+		}
+
+		// This doesn't come before the private member check above because
+		// unsupported private methods must also trigger field lowering:
+		//
+		//   class Foo {
+		//     bar = this.#foo()
+		//     #foo() {}
+		//   }
+		//
+		// It would be bad if we transformed that to something like this:
+		//
+		//   var _foo, foo_fn;
+		//   class Foo {
+		//     constructor() {
+		//       _foo.add(this);
+		//     }
+		//     bar = __privateMethod(this, _foo, foo_fn).call(this);
+		//   }
+		//   _foo = new WeakSet();
+		//   foo_fn = function() {
+		//   };
+		//
+		// In that case the initializer of "bar" would fail to call "#foo" because
+		// it's only added to the instance in the body of the constructor.
+		if prop.Kind.IsMethodDefinition() {
+			// We need to shim "super()" inside the constructor if this is a derived
+			// class and the constructor has any parameter properties, since those
+			// use "this" and we can only access "this" after "super()" is called
+			if class.ExtendsOrNil.Data != nil {
+				if key, ok := prop.Key.Data.(*js_ast.EString); ok && helpers.UTF16EqualsString(key.Value, "constructor") {
+					if fn, ok := prop.ValueOrNil.Data.(*js_ast.EFunction); ok {
+						for _, arg := range fn.Fn.Args {
+							if arg.IsTypeScriptCtorField {
+								result.shimSuperCtorCalls = true
+								break
+							}
+						}
+					}
+				}
+			}
+			continue
+		}
+
+		if prop.Flags.Has(js_ast.PropertyIsStatic) {
+			// Static fields must be lowered if the target doesn't support them
+			if p.options.unsupportedJSFeatures.Has(compat.ClassStaticField) {
+				result.lowerAllStaticFields = true
+			}
+
+			// Convert static fields to assignment statements if the TypeScript
+			// setting for this is enabled. I don't think this matters for private
+			// fields because there's no way for this to call a setter in the base
+			// class, so this isn't done for private fields.
+			//
+			// If class static blocks are supported, then we can do this inline
+			// without needing to move the initializers outside of the class body.
+			// Otherwise, we need to lower all static class fields.
+			if p.options.ts.Parse && !class.UseDefineForClassFields && p.options.unsupportedJSFeatures.Has(compat.ClassStaticBlocks) {
+				result.lowerAllStaticFields = true
+			}
+		} else {
+			if p.options.ts.Parse && !class.UseDefineForClassFields {
+				// Convert instance fields to assignment statements if the TypeScript
+				// setting for this is enabled. I don't think this matters for private
+				// fields because there's no way for this to call a setter in the base
+				// class, so this isn't done for private fields.
+				if prop.InitializerOrNil.Data != nil {
+					// We can skip lowering all instance fields if all instance fields
+					// disappear completely when lowered. This happens when
+					// "useDefineForClassFields" is false and there is no initializer.
+					result.lowerAllInstanceFields = true
+				}
+			} else if p.options.unsupportedJSFeatures.Has(compat.ClassField) {
+				// Instance fields must be lowered if the target doesn't support them
+				result.lowerAllInstanceFields = true
+			}
+		}
+	}
+
+	// We need to shim "super()" inside the constructor if this is a derived
+	// class and there are any instance fields that need to be lowered, since
+	// those use "this" and we can only access "this" after "super()" is called
+	if result.lowerAllInstanceFields && class.ExtendsOrNil.Data != nil {
+		result.shimSuperCtorCalls = true
+	}
+
+	return
+}
+
+type classKind uint8
+
+const (
+	classKindExpr classKind = iota
+	classKindStmt
+	classKindExportStmt
+	classKindExportDefaultStmt
+)
+
+type lowerClassContext struct {
+	nameToKeep  string
+	kind        classKind
+	class       *js_ast.Class
+	classLoc    logger.Loc
+	classExpr   js_ast.Expr // Only for "kind == classKindExpr", may be replaced by "nameFunc()"
+	defaultName ast.LocRef
+
+	ctor                   *js_ast.EFunction
+	extendsRef             ast.Ref
+	parameterFields        []js_ast.Stmt
+	instanceMembers        []js_ast.Stmt
+	instancePrivateMethods []js_ast.Stmt
+	autoAccessorCount      int
+
+	// These expressions are generated after the class body, in this order
+	computedPropertyChain js_ast.Expr
+	privateMembers        []js_ast.Expr
+	staticMembers         []js_ast.Expr
+	staticPrivateMethods  []js_ast.Expr
+
+	// These contain calls to "__decorateClass" for TypeScript experimental decorators
+	instanceExperimentalDecorators []js_ast.Expr
+	staticExperimentalDecorators   []js_ast.Expr
+
+	// These are used for implementing JavaScript decorators
+	decoratorContextRef                          ast.Ref
+	decoratorClassDecorators                     js_ast.Expr
+	decoratorPropertyToInitializerMap            map[int]int
+	decoratorCallInstanceMethodExtraInitializers bool
+	decoratorCallStaticMethodExtraInitializers   bool
+	decoratorStaticNonFieldElements              []js_ast.Expr
+	decoratorInstanceNonFieldElements            []js_ast.Expr
+	decoratorStaticFieldElements                 []js_ast.Expr
+	decoratorInstanceFieldElements               []js_ast.Expr
+
+	// These are used by "lowerMethod"
+	privateInstanceMethodRef ast.Ref
+	privateStaticMethodRef   ast.Ref
+
+	// These are only for class expressions that need to be captured
+	nameFunc            func() js_ast.Expr
+	wrapFunc            func(js_ast.Expr) js_ast.Expr
+	didCaptureClassExpr bool
+}
+
+// Apply all relevant transforms to a class object (either a statement or an
+// expression) including:
+//
+//   - Transforming class fields for older environments
+//   - Transforming static blocks for older environments
+//   - Transforming TypeScript experimental decorators into JavaScript
+//   - Transforming TypeScript class fields into assignments for "useDefineForClassFields"
+//
+// Note that this doesn't transform any nested AST subtrees inside the class
+// body (e.g. the contents of initializers, methods, and static blocks). Those
+// have already been transformed by "visitClass" by this point. It's done that
+// way for performance so that we don't need to do another AST pass.
+func (p *parser) lowerClass(stmt js_ast.Stmt, expr js_ast.Expr, result visitClassResult, nameToKeep string) ([]js_ast.Stmt, js_ast.Expr) {
+	ctx := lowerClassContext{
+		nameToKeep:               nameToKeep,
+		extendsRef:               ast.InvalidRef,
+		decoratorContextRef:      ast.InvalidRef,
+		privateInstanceMethodRef: ast.InvalidRef,
+		privateStaticMethodRef:   ast.InvalidRef,
+	}
+
+	// Unpack the class from the statement or expression
+	if stmt.Data == nil {
+		e, _ := expr.Data.(*js_ast.EClass)
+		ctx.class = &e.Class
+		ctx.classExpr = expr
+		ctx.kind = classKindExpr
+		if ctx.class.Name != nil {
+			symbol := &p.symbols[ctx.class.Name.Ref.InnerIndex]
+			ctx.nameToKeep = symbol.OriginalName
+
+			// The inner class name inside the class expression should be the same as
+			// the class expression name itself
+			if result.innerClassNameRef != ast.InvalidRef {
+				p.mergeSymbols(result.innerClassNameRef, ctx.class.Name.Ref)
+			}
+
+			// Remove unused class names when minifying. Check this after we merge in
+			// the inner class name above since that will adjust the use count.
+			if p.options.minifySyntax && symbol.UseCountEstimate == 0 {
+				ctx.class.Name = nil
+			}
+		}
+	} else if s, ok := stmt.Data.(*js_ast.SClass); ok {
+		ctx.class = &s.Class
+		if ctx.class.Name != nil {
+			ctx.nameToKeep = p.symbols[ctx.class.Name.Ref.InnerIndex].OriginalName
+		}
+		if s.IsExport {
+			ctx.kind = classKindExportStmt
+		} else {
+			ctx.kind = classKindStmt
+		}
+	} else {
+		s, _ := stmt.Data.(*js_ast.SExportDefault)
+		s2, _ := s.Value.Data.(*js_ast.SClass)
+		ctx.class = &s2.Class
+		if ctx.class.Name != nil {
+			ctx.nameToKeep = p.symbols[ctx.class.Name.Ref.InnerIndex].OriginalName
+		}
+		ctx.defaultName = s.DefaultName
+		ctx.kind = classKindExportDefaultStmt
+	}
+	if stmt.Data == nil {
+		ctx.classLoc = expr.Loc
+	} else {
+		ctx.classLoc = stmt.Loc
+	}
+
+	classLoweringInfo := p.computeClassLoweringInfo(ctx.class)
+	ctx.enableNameCapture(p, result)
+	ctx.processProperties(p, classLoweringInfo, result)
+	ctx.insertInitializersIntoConstructor(p, classLoweringInfo, result)
+	return ctx.finishAndGenerateCode(p, result)
+}
+
+func (ctx *lowerClassContext) enableNameCapture(p *parser, result visitClassResult) {
+	// Class statements can be missing a name if they are in an
+	// "export default" statement:
+	//
+	//   export default class {
+	//     static foo = 123
+	//   }
+	//
+	ctx.nameFunc = func() js_ast.Expr {
+		if ctx.kind == classKindExpr {
+			// If this is a class expression, capture and store it. We have to
+			// do this even if it has a name since the name isn't exposed
+			// outside the class body.
+			classExpr := &js_ast.EClass{Class: *ctx.class}
+			ctx.class = &classExpr.Class
+			ctx.nameFunc, ctx.wrapFunc = p.captureValueWithPossibleSideEffects(ctx.classLoc, 2, js_ast.Expr{Loc: ctx.classLoc, Data: classExpr}, valueDefinitelyNotMutated)
+			ctx.classExpr = ctx.nameFunc()
+			ctx.didCaptureClassExpr = true
+			name := ctx.nameFunc()
+
+			// If we're storing the class expression in a variable, remove the class
+			// name and rewrite all references to the class name with references to
+			// the temporary variable holding the class expression. This ensures that
+			// references to the class expression by name in any expressions that end
+			// up being pulled outside of the class body still work. For example:
+			//
+			//   let Bar = class Foo {
+			//     static foo = 123
+			//     static bar = Foo.foo
+			//   }
+			//
+			// This might be converted into the following:
+			//
+			//   var _a;
+			//   let Bar = (_a = class {
+			//   }, _a.foo = 123, _a.bar = _a.foo, _a);
+			//
+			if ctx.class.Name != nil {
+				p.mergeSymbols(ctx.class.Name.Ref, name.Data.(*js_ast.EIdentifier).Ref)
+				ctx.class.Name = nil
+			}
+
+			return name
+		} else {
+			// If anything referenced the inner class name, then we should use that
+			// name for any automatically-generated initialization code, since it
+			// will come before the outer class name is initialized.
+			if result.innerClassNameRef != ast.InvalidRef {
+				p.recordUsage(result.innerClassNameRef)
+				return js_ast.Expr{Loc: ctx.class.Name.Loc, Data: &js_ast.EIdentifier{Ref: result.innerClassNameRef}}
+			}
+
+			// Otherwise we should just use the outer class name
+			if ctx.class.Name == nil {
+				if ctx.kind == classKindExportDefaultStmt {
+					ctx.class.Name = &ctx.defaultName
+				} else {
+					ctx.class.Name = &ast.LocRef{Loc: ctx.classLoc, Ref: p.generateTempRef(tempRefNoDeclare, "")}
+				}
+			}
+			p.recordUsage(ctx.class.Name.Ref)
+			return js_ast.Expr{Loc: ctx.class.Name.Loc, Data: &js_ast.EIdentifier{Ref: ctx.class.Name.Ref}}
+		}
+	}
+}
+
+// Handle lowering of instance and static fields. Move their initializers
+// from the class body to either the constructor (instance fields) or after
+// the class (static fields).
+//
+// If this returns true, the return property should be added to the class
+// body. Otherwise the property should be omitted from the class body.
+func (ctx *lowerClassContext) lowerField(
+	p *parser,
+	prop js_ast.Property,
+	private *js_ast.EPrivateIdentifier,
+	shouldOmitFieldInitializer bool,
+	staticFieldToBlockAssign bool,
+	initializerIndex int,
+) (js_ast.Property, ast.Ref, bool) {
+	mustLowerPrivate := private != nil && p.privateSymbolNeedsToBeLowered(private)
+	ref := ast.InvalidRef
+
+	// The TypeScript compiler doesn't follow the JavaScript spec for
+	// uninitialized fields. They are supposed to be set to undefined but the
+	// TypeScript compiler just omits them entirely.
+	if !shouldOmitFieldInitializer {
+		loc := prop.Loc
+
+		// Determine where to store the field
+		var target js_ast.Expr
+		if prop.Flags.Has(js_ast.PropertyIsStatic) && !staticFieldToBlockAssign {
+			target = ctx.nameFunc()
+		} else {
+			target = js_ast.Expr{Loc: loc, Data: js_ast.EThisShared}
+		}
+
+		// Generate the assignment initializer
+		var init js_ast.Expr
+		if prop.InitializerOrNil.Data != nil {
+			init = prop.InitializerOrNil
+		} else {
+			init = js_ast.Expr{Loc: loc, Data: js_ast.EUndefinedShared}
+		}
+
+		// Optionally call registered decorator initializers
+		if initializerIndex != -1 {
+			var value js_ast.Expr
+			if prop.Flags.Has(js_ast.PropertyIsStatic) {
+				value = ctx.nameFunc()
+			} else {
+				value = js_ast.Expr{Loc: loc, Data: js_ast.EThisShared}
+			}
+			args := []js_ast.Expr{
+				{Loc: loc, Data: &js_ast.EIdentifier{Ref: ctx.decoratorContextRef}},
+				{Loc: loc, Data: &js_ast.ENumber{Value: float64((4 + 2*initializerIndex) << 1)}},
+				value,
+			}
+			if _, ok := init.Data.(*js_ast.EUndefined); !ok {
+				args = append(args, init)
+			}
+			init = p.callRuntime(init.Loc, "__runInitializers", args)
+			p.recordUsage(ctx.decoratorContextRef)
+		}
+
+		// Generate the assignment target
+		var memberExpr js_ast.Expr
+		if mustLowerPrivate {
+			// Generate a new symbol for this private field
+			ref = p.generateTempRef(tempRefNeedsDeclare, "_"+p.symbols[private.Ref.InnerIndex].OriginalName[1:])
+			p.symbols[private.Ref.InnerIndex].Link = ref
+
+			// Initialize the private field to a new WeakMap
+			if p.weakMapRef == ast.InvalidRef {
+				p.weakMapRef = p.newSymbol(ast.SymbolUnbound, "WeakMap")
+				p.moduleScope.Generated = append(p.moduleScope.Generated, p.weakMapRef)
+			}
+			ctx.privateMembers = append(ctx.privateMembers, js_ast.Assign(
+				js_ast.Expr{Loc: prop.Key.Loc, Data: &js_ast.EIdentifier{Ref: ref}},
+				js_ast.Expr{Loc: prop.Key.Loc, Data: &js_ast.ENew{Target: js_ast.Expr{Loc: prop.Key.Loc, Data: &js_ast.EIdentifier{Ref: p.weakMapRef}}}},
+			))
+			p.recordUsage(ref)
+
+			// Add every newly-constructed instance into this map
+			key := js_ast.Expr{Loc: prop.Key.Loc, Data: &js_ast.EIdentifier{Ref: ref}}
+			args := []js_ast.Expr{target, key}
+			if _, ok := init.Data.(*js_ast.EUndefined); !ok {
+				args = append(args, init)
+			}
+			memberExpr = p.callRuntime(loc, "__privateAdd", args)
+			p.recordUsage(ref)
+		} else if private == nil && ctx.class.UseDefineForClassFields {
+			args := []js_ast.Expr{target, prop.Key}
+			if _, ok := init.Data.(*js_ast.EUndefined); !ok {
+				args = append(args, init)
+			}
+			memberExpr = js_ast.Expr{Loc: loc, Data: &js_ast.ECall{
+				Target: p.importFromRuntime(loc, "__publicField"),
+				Args:   args,
+			}}
+		} else {
+			if key, ok := prop.Key.Data.(*js_ast.EString); ok && !prop.Flags.Has(js_ast.PropertyIsComputed) && !prop.Flags.Has(js_ast.PropertyPreferQuotedKey) {
+				target = js_ast.Expr{Loc: loc, Data: &js_ast.EDot{
+					Target:  target,
+					Name:    helpers.UTF16ToString(key.Value),
+					NameLoc: prop.Key.Loc,
+				}}
+			} else {
+				target = js_ast.Expr{Loc: loc, Data: &js_ast.EIndex{
+					Target: target,
+					Index:  prop.Key,
+				}}
+			}
+
+			memberExpr = js_ast.Assign(target, init)
+		}
+
+		// Run extra initializers
+		if initializerIndex != -1 {
+			var value js_ast.Expr
+			if prop.Flags.Has(js_ast.PropertyIsStatic) {
+				value = ctx.nameFunc()
+			} else {
+				value = js_ast.Expr{Loc: loc, Data: js_ast.EThisShared}
+			}
+			memberExpr = js_ast.JoinWithComma(memberExpr, p.callRuntime(loc, "__runInitializers", []js_ast.Expr{
+				{Loc: loc, Data: &js_ast.EIdentifier{Ref: ctx.decoratorContextRef}},
+				{Loc: loc, Data: &js_ast.ENumber{Value: float64(((5 + 2*initializerIndex) << 1) | 1)}},
+				value,
+			}))
+			p.recordUsage(ctx.decoratorContextRef)
+		}
+
+		if prop.Flags.Has(js_ast.PropertyIsStatic) {
+			// Move this property to an assignment after the class ends
+			if staticFieldToBlockAssign {
+				// Use inline assignment in a static block instead of lowering
+				return js_ast.Property{
+					Loc:  loc,
+					Kind: js_ast.PropertyClassStaticBlock,
+					ClassStaticBlock: &js_ast.ClassStaticBlock{
+						Loc: loc,
+						Block: js_ast.SBlock{Stmts: []js_ast.Stmt{
+							{Loc: loc, Data: &js_ast.SExpr{Value: memberExpr}}},
+						},
+					},
+				}, ref, true
+			} else {
+				// Move this property to an assignment after the class ends
+				ctx.staticMembers = append(ctx.staticMembers, memberExpr)
+			}
+		} else {
+			// Move this property to an assignment inside the class constructor
+			ctx.instanceMembers = append(ctx.instanceMembers, js_ast.Stmt{Loc: loc, Data: &js_ast.SExpr{Value: memberExpr}})
+		}
+	}
+
+	if private == nil || mustLowerPrivate {
+		// Remove the field from the class body
+		return js_ast.Property{}, ref, false
+	}
+
+	// Keep the private field but remove the initializer
+	prop.InitializerOrNil = js_ast.Expr{}
+	return prop, ref, true
+}
+
+func (ctx *lowerClassContext) lowerPrivateMethod(p *parser, prop js_ast.Property, private *js_ast.EPrivateIdentifier) {
+	// All private methods can share the same WeakSet
+	var ref *ast.Ref
+	if prop.Flags.Has(js_ast.PropertyIsStatic) {
+		ref = &ctx.privateStaticMethodRef
+	} else {
+		ref = &ctx.privateInstanceMethodRef
+	}
+	if *ref == ast.InvalidRef {
+		// Generate a new symbol to store the WeakSet
+		var name string
+		if prop.Flags.Has(js_ast.PropertyIsStatic) {
+			name = "_static"
+		} else {
+			name = "_instances"
+		}
+		if ctx.nameToKeep != "" {
+			name = fmt.Sprintf("_%s%s", ctx.nameToKeep, name)
+		}
+		*ref = p.generateTempRef(tempRefNeedsDeclare, name)
+
+		// Generate the initializer
+		if p.weakSetRef == ast.InvalidRef {
+			p.weakSetRef = p.newSymbol(ast.SymbolUnbound, "WeakSet")
+			p.moduleScope.Generated = append(p.moduleScope.Generated, p.weakSetRef)
+		}
+		ctx.privateMembers = append(ctx.privateMembers, js_ast.Assign(
+			js_ast.Expr{Loc: ctx.classLoc, Data: &js_ast.EIdentifier{Ref: *ref}},
+			js_ast.Expr{Loc: ctx.classLoc, Data: &js_ast.ENew{Target: js_ast.Expr{Loc: ctx.classLoc, Data: &js_ast.EIdentifier{Ref: p.weakSetRef}}}},
+		))
+		p.recordUsage(*ref)
+		p.recordUsage(p.weakSetRef)
+
+		// Determine what to store in the WeakSet
+		var target js_ast.Expr
+		if prop.Flags.Has(js_ast.PropertyIsStatic) {
+			target = ctx.nameFunc()
+		} else {
+			target = js_ast.Expr{Loc: ctx.classLoc, Data: js_ast.EThisShared}
+		}
+
+		// Add every newly-constructed instance into this set
+		methodExpr := p.callRuntime(ctx.classLoc, "__privateAdd", []js_ast.Expr{
+			target,
+			{Loc: ctx.classLoc, Data: &js_ast.EIdentifier{Ref: *ref}},
+		})
+		p.recordUsage(*ref)
+
+		// Make sure that adding to the map happens before any field
+		// initializers to handle cases like this:
+		//
+		//   class A {
+		//     pub = this.#priv;
+		//     #priv() {}
+		//   }
+		//
+		if prop.Flags.Has(js_ast.PropertyIsStatic) {
+			// Move this property to an assignment after the class ends
+			ctx.staticPrivateMethods = append(ctx.staticPrivateMethods, methodExpr)
+		} else {
+			// Move this property to an assignment inside the class constructor
+			ctx.instancePrivateMethods = append(ctx.instancePrivateMethods, js_ast.Stmt{Loc: ctx.classLoc, Data: &js_ast.SExpr{Value: methodExpr}})
+		}
+	}
+	p.symbols[private.Ref.InnerIndex].Link = *ref
+}
+
+// If this returns true, the method property should be dropped as it has
+// already been accounted for elsewhere (e.g. a lowered private method).
+func (ctx *lowerClassContext) lowerMethod(p *parser, prop js_ast.Property, private *js_ast.EPrivateIdentifier) bool {
+	if private != nil && p.privateSymbolNeedsToBeLowered(private) {
+		ctx.lowerPrivateMethod(p, prop, private)
+
+		// Move the method definition outside the class body
+		methodRef := p.generateTempRef(tempRefNeedsDeclare, "_")
+		if prop.Kind == js_ast.PropertySetter {
+			p.symbols[methodRef.InnerIndex].Link = p.privateSetters[private.Ref]
+		} else {
+			p.symbols[methodRef.InnerIndex].Link = p.privateGetters[private.Ref]
+		}
+		p.recordUsage(methodRef)
+		ctx.privateMembers = append(ctx.privateMembers, js_ast.Assign(
+			js_ast.Expr{Loc: prop.Key.Loc, Data: &js_ast.EIdentifier{Ref: methodRef}},
+			prop.ValueOrNil,
+		))
+		return true
+	}
+
+	if key, ok := prop.Key.Data.(*js_ast.EString); ok && helpers.UTF16EqualsString(key.Value, "constructor") {
+		if fn, ok := prop.ValueOrNil.Data.(*js_ast.EFunction); ok {
+			// Remember where the constructor is for later
+			ctx.ctor = fn
+
+			// Initialize TypeScript constructor parameter fields
+			if p.options.ts.Parse {
+				for _, arg := range ctx.ctor.Fn.Args {
+					if arg.IsTypeScriptCtorField {
+						if id, ok := arg.Binding.Data.(*js_ast.BIdentifier); ok {
+							ctx.parameterFields = append(ctx.parameterFields, js_ast.AssignStmt(
+								js_ast.Expr{Loc: arg.Binding.Loc, Data: p.dotOrMangledPropVisit(
+									js_ast.Expr{Loc: arg.Binding.Loc, Data: js_ast.EThisShared},
+									p.symbols[id.Ref.InnerIndex].OriginalName,
+									arg.Binding.Loc,
+								)},
+								js_ast.Expr{Loc: arg.Binding.Loc, Data: &js_ast.EIdentifier{Ref: id.Ref}},
+							))
+						}
+					}
+				}
+			}
+		}
+	}
+
+	return false
+}
+
+type propertyAnalysis struct {
+	private                         *js_ast.EPrivateIdentifier
+	propExperimentalDecorators      []js_ast.Decorator
+	propDecorators                  []js_ast.Decorator
+	mustLowerField                  bool
+	needsValueOfKey                 bool
+	rewriteAutoAccessorToGetSet     bool
+	shouldOmitFieldInitializer      bool
+	staticFieldToBlockAssign        bool
+	isComputedPropertyCopiedOrMoved bool
+}
+
+func (ctx *lowerClassContext) analyzeProperty(p *parser, prop js_ast.Property, classLoweringInfo classLoweringInfo) (analysis propertyAnalysis) {
+	// The TypeScript class field transform requires removing fields without
+	// initializers. If the field is removed, then we only need the key for
+	// its side effects and we don't need a temporary reference for the key.
+	// However, the TypeScript compiler doesn't remove the field when doing
+	// strict class field initialization, so we shouldn't either.
+	analysis.private, _ = prop.Key.Data.(*js_ast.EPrivateIdentifier)
+	mustLowerPrivate := analysis.private != nil && p.privateSymbolNeedsToBeLowered(analysis.private)
+	analysis.shouldOmitFieldInitializer = p.options.ts.Parse && !prop.Kind.IsMethodDefinition() && prop.InitializerOrNil.Data == nil &&
+		!ctx.class.UseDefineForClassFields && !mustLowerPrivate && !ctx.class.ShouldLowerStandardDecorators
+
+	// Class fields must be lowered if the environment doesn't support them
+	if !prop.Kind.IsMethodDefinition() {
+		if prop.Flags.Has(js_ast.PropertyIsStatic) {
+			analysis.mustLowerField = classLoweringInfo.lowerAllStaticFields
+		} else if prop.Kind == js_ast.PropertyField && p.options.ts.Parse && !ctx.class.UseDefineForClassFields && analysis.private == nil {
+			// Lower non-private instance fields (not accessors) if TypeScript's
+			// "useDefineForClassFields" setting is disabled. When all such fields
+			// have no initializers, we avoid setting the "lowerAllInstanceFields"
+			// flag as an optimization because we can just remove all class field
+			// declarations in that case without messing with the constructor. But
+			// we must set the "mustLowerField" flag here to cause this class field
+			// declaration to still be removed.
+			analysis.mustLowerField = true
+		} else {
+			analysis.mustLowerField = classLoweringInfo.lowerAllInstanceFields
+		}
+	}
+
+	// If the field uses the TypeScript "declare" or "abstract" keyword, just
+	// omit it entirely. However, we must still keep any side-effects in the
+	// computed value and/or in the decorators.
+	if prop.Kind == js_ast.PropertyDeclareOrAbstract && prop.ValueOrNil.Data == nil {
+		analysis.mustLowerField = true
+		analysis.shouldOmitFieldInitializer = true
+	}
+
+	// For convenience, split decorators off into separate fields based on how
+	// they will end up being lowered (if they are even being lowered at all)
+	if p.options.ts.Parse && p.options.ts.Config.ExperimentalDecorators == config.True {
+		analysis.propExperimentalDecorators = prop.Decorators
+	} else if ctx.class.ShouldLowerStandardDecorators {
+		analysis.propDecorators = prop.Decorators
+	}
+
+	// Note: Auto-accessors use a different transform when they are decorated.
+	// This transform trades off worse run-time performance for better code size.
+	analysis.rewriteAutoAccessorToGetSet = len(analysis.propDecorators) == 0 && prop.Kind == js_ast.PropertyAutoAccessor &&
+		(p.options.unsupportedJSFeatures.Has(compat.Decorators) || analysis.mustLowerField)
+
+	// Transform non-lowered static fields that use assign semantics into an
+	// assignment in an inline static block instead of lowering them. This lets
+	// us avoid having to unnecessarily lower static private fields when
+	// "useDefineForClassFields" is disabled.
+	analysis.staticFieldToBlockAssign = prop.Kind == js_ast.PropertyField && !analysis.mustLowerField && !ctx.class.UseDefineForClassFields &&
+		prop.Flags.Has(js_ast.PropertyIsStatic) && analysis.private == nil
+
+	// Computed properties can't be copied or moved because they have side effects
+	// and we don't want to evaluate their side effects twice or change their
+	// evaluation order. We'll need to store them in temporary variables to keep
+	// their side effects in place when we reference them elsewhere.
+	analysis.needsValueOfKey = true
+	if prop.Flags.Has(js_ast.PropertyIsComputed) &&
+		(len(analysis.propExperimentalDecorators) > 0 ||
+			len(analysis.propDecorators) > 0 ||
+			analysis.mustLowerField ||
+			analysis.staticFieldToBlockAssign ||
+			analysis.rewriteAutoAccessorToGetSet) {
+		analysis.isComputedPropertyCopiedOrMoved = true
+
+		// Determine if we don't actually need the value of the key (only the side
+		// effects). In that case we don't need a temporary variable.
+		if len(analysis.propExperimentalDecorators) == 0 &&
+			len(analysis.propDecorators) == 0 &&
+			!analysis.rewriteAutoAccessorToGetSet &&
+			analysis.shouldOmitFieldInitializer {
+			analysis.needsValueOfKey = false
+		}
+	}
+	return
+}
+
+func (p *parser) propertyNameHint(key js_ast.Expr) string {
+	switch k := key.Data.(type) {
+	case *js_ast.EString:
+		return helpers.UTF16ToString(k.Value)
+	case *js_ast.EIdentifier:
+		return p.symbols[k.Ref.InnerIndex].OriginalName
+	case *js_ast.EPrivateIdentifier:
+		return p.symbols[k.Ref.InnerIndex].OriginalName[1:]
+	default:
+		return ""
+	}
+}
+
+func (ctx *lowerClassContext) hoistComputedProperties(p *parser, classLoweringInfo classLoweringInfo) (
+	propertyKeyTempRefs map[int]ast.Ref, decoratorTempRefs map[int]ast.Ref) {
+	var nextComputedPropertyKey *js_ast.Expr
+
+	// Computed property keys must be evaluated in a specific order for their
+	// side effects. This order must be preserved even when we have to move a
+	// class element around. For example, this can happen when using class fields
+	// with computed property keys and targeting environments without class field
+	// support. For example:
+	//
+	//   class Foo {
+	//     [a()]() {}
+	//     static [b()] = null;
+	//     [c()]() {}
+	//   }
+	//
+	// If we need to lower the static field because static fields aren't supported,
+	// we still need to ensure that "b()" is called before "a()" and after "c()".
+	// That looks something like this:
+	//
+	//   var _a;
+	//   class Foo {
+	//     [a()]() {}
+	//     [(_a = b(), c())]() {}
+	//   }
+	//   __publicField(Foo, _a, null);
+	//
+	// Iterate in reverse so that any initializers are "pushed up" before the
+	// class body if there's nowhere else to put them. They can't be "pushed
+	// down" into a static block in the class body (the logical place to put
+	// them that's next in the evaluation order) because these expressions
+	// may contain "await" and static blocks do not allow "await".
+	for propIndex := len(ctx.class.Properties) - 1; propIndex >= 0; propIndex-- {
+		prop := &ctx.class.Properties[propIndex]
+		analysis := ctx.analyzeProperty(p, *prop, classLoweringInfo)
+
+		// Evaluate the decorator expressions inline before computed property keys
+		var decorators js_ast.Expr
+		if len(analysis.propDecorators) > 0 {
+			name := p.propertyNameHint(prop.Key)
+			if name != "" {
+				name = "_" + name
+			}
+			name += "_dec"
+			ref := p.generateTempRef(tempRefNeedsDeclare, name)
+			values := make([]js_ast.Expr, len(analysis.propDecorators))
+			for i, decorator := range analysis.propDecorators {
+				values[i] = decorator.Value
+			}
+			atLoc := analysis.propDecorators[0].AtLoc
+			decorators = js_ast.Assign(
+				js_ast.Expr{Loc: atLoc, Data: &js_ast.EIdentifier{Ref: ref}},
+				js_ast.Expr{Loc: atLoc, Data: &js_ast.EArray{Items: values, IsSingleLine: true}})
+			p.recordUsage(ref)
+			if decoratorTempRefs == nil {
+				decoratorTempRefs = make(map[int]ast.Ref)
+			}
+			decoratorTempRefs[propIndex] = ref
+		}
+
+		// Skip property keys that we know are side-effect free
+		switch prop.Key.Data.(type) {
+		case *js_ast.EString, *js_ast.ENameOfSymbol, *js_ast.ENumber, *js_ast.EPrivateIdentifier:
+			// Figure out where to stick the decorator side effects to preserve their order
+			if nextComputedPropertyKey != nil {
+				// Insert it before everything that comes after it
+				*nextComputedPropertyKey = js_ast.JoinWithComma(decorators, *nextComputedPropertyKey)
+			} else {
+				// Insert it after the first thing that comes before it
+				ctx.computedPropertyChain = js_ast.JoinWithComma(decorators, ctx.computedPropertyChain)
+			}
+			continue
+
+		default:
+			// Otherwise, evaluate the decorators right before the property key
+			if decorators.Data != nil {
+				prop.Key = js_ast.JoinWithComma(decorators, prop.Key)
+				prop.Flags |= js_ast.PropertyIsComputed
+			}
+		}
+
+		// If this key is referenced elsewhere, make sure to still preserve
+		// its side effects in the property's original location
+		if analysis.isComputedPropertyCopiedOrMoved {
+			// If this property is being duplicated instead of moved or removed, then
+			// we still need the assignment to the temporary so that we can reference
+			// it in multiple places, but we don't have to hoist the assignment to an
+			// earlier property (since this property is still there). In that case
+			// we can reduce generated code size by avoiding the hoist. One example
+			// of this case is a decorator on a class element with a computed
+			// property key:
+			//
+			//   class Foo {
+			//     @dec [a()]() {}
+			//   }
+			//
+			// We want to do this:
+			//
+			//   var _a;
+			//   class Foo {
+			//     [_a = a()]() {}
+			//   }
+			//   __decorateClass([dec], Foo.prototype, _a, 1);
+			//
+			// instead of this:
+			//
+			//   var _a;
+			//   _a = a();
+			//   class Foo {
+			//     [_a]() {}
+			//   }
+			//   __decorateClass([dec], Foo.prototype, _a, 1);
+			//
+			// So only do the hoist if this property is being moved or removed.
+			if !analysis.rewriteAutoAccessorToGetSet && (analysis.mustLowerField || analysis.staticFieldToBlockAssign) {
+				inlineKey := prop.Key
+
+				if !analysis.needsValueOfKey {
+					// In certain cases, we only need to evaluate a property key for its
+					// side effects but we don't actually need the value of the key itself.
+					// For example, a TypeScript class field without an initializer is
+					// omitted when TypeScript's "useDefineForClassFields" setting is false.
+				} else {
+					// Store the key in a temporary so we can refer to it later
+					ref := p.generateTempRef(tempRefNeedsDeclare, "")
+					inlineKey = js_ast.Assign(js_ast.Expr{Loc: prop.Key.Loc, Data: &js_ast.EIdentifier{Ref: ref}}, prop.Key)
+					p.recordUsage(ref)
+
+					// Replace this property key with a reference to the temporary. We
+					// don't need to store the temporary in the "propertyKeyTempRefs"
+					// map because all references will refer to the temporary, not just
+					// some of them.
+					prop.Key = js_ast.Expr{Loc: prop.Key.Loc, Data: &js_ast.EIdentifier{Ref: ref}}
+					p.recordUsage(ref)
+				}
+
+				// Figure out where to stick this property's side effect to preserve its order
+				if nextComputedPropertyKey != nil {
+					// Insert it before everything that comes after it
+					*nextComputedPropertyKey = js_ast.JoinWithComma(inlineKey, *nextComputedPropertyKey)
+				} else {
+					// Insert it after the first thing that comes before it
+					ctx.computedPropertyChain = js_ast.JoinWithComma(inlineKey, ctx.computedPropertyChain)
+				}
+				continue
+			}
+
+			// Otherwise, we keep the side effects in place (as described above) but
+			// just store the key in a temporary so we can refer to it later.
+			ref := p.generateTempRef(tempRefNeedsDeclare, "")
+			prop.Key = js_ast.Assign(js_ast.Expr{Loc: prop.Key.Loc, Data: &js_ast.EIdentifier{Ref: ref}}, prop.Key)
+			p.recordUsage(ref)
+
+			// Use this temporary when creating duplicate references to this key
+			if propertyKeyTempRefs == nil {
+				propertyKeyTempRefs = make(map[int]ast.Ref)
+			}
+			propertyKeyTempRefs[propIndex] = ref
+
+			// Deliberately continue to fall through to the "computed" case below:
+		}
+
+		// Otherwise, this computed property could be a good location to evaluate
+		// something that comes before it. Remember this location for later.
+		if prop.Flags.Has(js_ast.PropertyIsComputed) {
+			// If any side effects after this were hoisted here, then inline them now.
+			// We don't want to reorder any side effects.
+			if ctx.computedPropertyChain.Data != nil {
+				ref, ok := propertyKeyTempRefs[propIndex]
+				if !ok {
+					ref = p.generateTempRef(tempRefNeedsDeclare, "")
+					prop.Key = js_ast.Assign(js_ast.Expr{Loc: prop.Key.Loc, Data: &js_ast.EIdentifier{Ref: ref}}, prop.Key)
+					p.recordUsage(ref)
+				}
+				prop.Key = js_ast.JoinWithComma(
+					js_ast.JoinWithComma(prop.Key, ctx.computedPropertyChain),
+					js_ast.Expr{Loc: prop.Key.Loc, Data: &js_ast.EIdentifier{Ref: ref}})
+				p.recordUsage(ref)
+				ctx.computedPropertyChain = js_ast.Expr{}
+			}
+
+			// Remember this location for later
+			nextComputedPropertyKey = &prop.Key
+		}
+	}
+
+	// If any side effects in the class body were hoisted up to the "extends"
+	// clause, then inline them before the "extends" clause is evaluated. We
+	// don't want to reorder any side effects. For example:
+	//
+	//   class Foo extends a() {
+	//     static [b()]
+	//   }
+	//
+	// We want to do this:
+	//
+	//   var _a, _b;
+	//   class Foo extends (_b = a(), _a = b(), _b) {
+	//   }
+	//   __publicField(Foo, _a);
+	//
+	// instead of this:
+	//
+	//   var _a;
+	//   _a = b();
+	//   class Foo extends a() {
+	//   }
+	//   __publicField(Foo, _a);
+	//
+	if ctx.computedPropertyChain.Data != nil && ctx.class.ExtendsOrNil.Data != nil {
+		ctx.extendsRef = p.generateTempRef(tempRefNeedsDeclare, "")
+		ctx.class.ExtendsOrNil = js_ast.JoinWithComma(js_ast.JoinWithComma(
+			js_ast.Assign(js_ast.Expr{Loc: ctx.class.ExtendsOrNil.Loc, Data: &js_ast.EIdentifier{Ref: ctx.extendsRef}}, ctx.class.ExtendsOrNil),
+			ctx.computedPropertyChain),
+			js_ast.Expr{Loc: ctx.class.ExtendsOrNil.Loc, Data: &js_ast.EIdentifier{Ref: ctx.extendsRef}})
+		p.recordUsage(ctx.extendsRef)
+		p.recordUsage(ctx.extendsRef)
+		ctx.computedPropertyChain = js_ast.Expr{}
+	}
+	return
+}
+
+// This corresponds to the initialization order in the specification:
+//
+//  27. For each element e of staticElements, do
+//     a. If e is a ClassElementDefinition Record and e.[[Kind]] is not field, then
+//
+//  28. For each element e of instanceElements, do
+//     a. If e.[[Kind]] is not field, then
+//
+//  29. For each element e of staticElements, do
+//     a. If e.[[Kind]] is field, then
+//
+//  30. For each element e of instanceElements, do
+//     a. If e.[[Kind]] is field, then
+func fieldOrAccessorOrder(kind js_ast.PropertyKind, flags js_ast.PropertyFlags) (int, bool) {
+	if kind == js_ast.PropertyAutoAccessor {
+		if flags.Has(js_ast.PropertyIsStatic) {
+			return 0, true
+		} else {
+			return 1, true
+		}
+	} else if kind == js_ast.PropertyField {
+		if flags.Has(js_ast.PropertyIsStatic) {
+			return 2, true
+		} else {
+			return 3, true
+		}
+	}
+	return 0, false
+}
+
+func (ctx *lowerClassContext) processProperties(p *parser, classLoweringInfo classLoweringInfo, result visitClassResult) {
+	properties := make([]js_ast.Property, 0, len(ctx.class.Properties))
+	propertyKeyTempRefs, decoratorTempRefs := ctx.hoistComputedProperties(p, classLoweringInfo)
+
+	// Save the initializer index for each field and accessor element
+	if ctx.class.ShouldLowerStandardDecorators {
+		var counts [4]int
+
+		// Count how many initializers there are in each section
+		for _, prop := range ctx.class.Properties {
+			if len(prop.Decorators) > 0 {
+				if i, ok := fieldOrAccessorOrder(prop.Kind, prop.Flags); ok {
+					counts[i]++
+				} else if prop.Flags.Has(js_ast.PropertyIsStatic) {
+					ctx.decoratorCallStaticMethodExtraInitializers = true
+				} else {
+					ctx.decoratorCallInstanceMethodExtraInitializers = true
+				}
+			}
+		}
+
+		// Give each on an index for the order it will be initialized in
+		if counts[0] > 0 || counts[1] > 0 || counts[2] > 0 || counts[3] > 0 {
+			indices := [4]int{0, counts[0], counts[0] + counts[1], counts[0] + counts[1] + counts[2]}
+			ctx.decoratorPropertyToInitializerMap = make(map[int]int)
+
+			for propIndex, prop := range ctx.class.Properties {
+				if len(prop.Decorators) > 0 {
+					if i, ok := fieldOrAccessorOrder(prop.Kind, prop.Flags); ok {
+						ctx.decoratorPropertyToInitializerMap[propIndex] = indices[i]
+						indices[i]++
+					}
+				}
+			}
+		}
+	}
+
+	// Evaluate the decorator expressions inline
+	if ctx.class.ShouldLowerStandardDecorators && len(ctx.class.Decorators) > 0 {
+		name := ctx.nameToKeep
+		if name == "" {
+			name = "class"
+		}
+		decoratorsRef := p.generateTempRef(tempRefNeedsDeclare, fmt.Sprintf("_%s_decorators", name))
+		values := make([]js_ast.Expr, len(ctx.class.Decorators))
+		for i, decorator := range ctx.class.Decorators {
+			values[i] = decorator.Value
+		}
+		atLoc := ctx.class.Decorators[0].AtLoc
+		ctx.computedPropertyChain = js_ast.JoinWithComma(js_ast.Assign(
+			js_ast.Expr{Loc: atLoc, Data: &js_ast.EIdentifier{Ref: decoratorsRef}},
+			js_ast.Expr{Loc: atLoc, Data: &js_ast.EArray{Items: values, IsSingleLine: true}},
+		), ctx.computedPropertyChain)
+		p.recordUsage(decoratorsRef)
+		ctx.decoratorClassDecorators = js_ast.Expr{Loc: atLoc, Data: &js_ast.EIdentifier{Ref: decoratorsRef}}
+		p.recordUsage(decoratorsRef)
+		ctx.class.Decorators = nil
+	}
+
+	for propIndex, prop := range ctx.class.Properties {
+		if prop.Kind == js_ast.PropertyClassStaticBlock {
+			// Drop empty class blocks when minifying
+			if p.options.minifySyntax && len(prop.ClassStaticBlock.Block.Stmts) == 0 {
+				continue
+			}
+
+			// Lower this block if needed
+			if classLoweringInfo.lowerAllStaticFields {
+				ctx.lowerStaticBlock(p, prop.Loc, *prop.ClassStaticBlock)
+				continue
+			}
+
+			// Otherwise, keep this property
+			properties = append(properties, prop)
+			continue
+		}
+
+		// Merge parameter decorators with method decorators
+		if p.options.ts.Parse && prop.Kind.IsMethodDefinition() {
+			if fn, ok := prop.ValueOrNil.Data.(*js_ast.EFunction); ok {
+				isConstructor := false
+				if key, ok := prop.Key.Data.(*js_ast.EString); ok {
+					isConstructor = helpers.UTF16EqualsString(key.Value, "constructor")
+				}
+				args := fn.Fn.Args
+				for i, arg := range args {
+					for _, decorator := range arg.Decorators {
+						// Generate a call to "__decorateParam()" for this parameter decorator
+						var decorators *[]js_ast.Decorator = &prop.Decorators
+						if isConstructor {
+							decorators = &ctx.class.Decorators
+						}
+						*decorators = append(*decorators, js_ast.Decorator{
+							Value: p.callRuntime(decorator.Value.Loc, "__decorateParam", []js_ast.Expr{
+								{Loc: decorator.Value.Loc, Data: &js_ast.ENumber{Value: float64(i)}},
+								decorator.Value,
+							}),
+							AtLoc: decorator.AtLoc,
+						})
+						args[i].Decorators = nil
+					}
+				}
+			}
+		}
+
+		analysis := ctx.analyzeProperty(p, prop, classLoweringInfo)
+
+		// When the property key needs to be referenced multiple times, subsequent
+		// references may need to reference a temporary variable instead of copying
+		// the whole property key expression (since we only want to evaluate side
+		// effects once).
+		keyExprNoSideEffects := prop.Key
+		if ref, ok := propertyKeyTempRefs[propIndex]; ok {
+			keyExprNoSideEffects.Data = &js_ast.EIdentifier{Ref: ref}
+		}
+
+		// Handle TypeScript experimental decorators
+		if len(analysis.propExperimentalDecorators) > 0 {
+			prop.Decorators = nil
+
+			// Generate a single call to "__decorateClass()" for this property
+			loc := prop.Key.Loc
+
+			// This code tells "__decorateClass()" if the descriptor should be undefined
+			descriptorKind := float64(1)
+			if prop.Kind == js_ast.PropertyField || prop.Kind == js_ast.PropertyDeclareOrAbstract {
+				descriptorKind = 2
+			}
+
+			// Instance properties use the prototype, static properties use the class
+			var target js_ast.Expr
+			if prop.Flags.Has(js_ast.PropertyIsStatic) {
+				target = ctx.nameFunc()
+			} else {
+				target = js_ast.Expr{Loc: loc, Data: &js_ast.EDot{Target: ctx.nameFunc(), Name: "prototype", NameLoc: loc}}
+			}
+
+			values := make([]js_ast.Expr, len(analysis.propExperimentalDecorators))
+			for i, decorator := range analysis.propExperimentalDecorators {
+				values[i] = decorator.Value
+			}
+			decorator := p.callRuntime(loc, "__decorateClass", []js_ast.Expr{
+				{Loc: loc, Data: &js_ast.EArray{Items: values}},
+				target,
+				cloneKeyForLowerClass(keyExprNoSideEffects),
+				{Loc: loc, Data: &js_ast.ENumber{Value: descriptorKind}},
+			})
+
+			// Static decorators are grouped after instance decorators
+			if prop.Flags.Has(js_ast.PropertyIsStatic) {
+				ctx.staticExperimentalDecorators = append(ctx.staticExperimentalDecorators, decorator)
+			} else {
+				ctx.instanceExperimentalDecorators = append(ctx.instanceExperimentalDecorators, decorator)
+			}
+		}
+
+		// Handle JavaScript decorators
+		initializerIndex := -1
+		if len(analysis.propDecorators) > 0 {
+			prop.Decorators = nil
+			loc := prop.Loc
+			keyLoc := prop.Key.Loc
+			atLoc := analysis.propDecorators[0].AtLoc
+
+			// Encode information about this property using bit flags
+			var flags int
+			switch prop.Kind {
+			case js_ast.PropertyMethod:
+				flags = 1
+			case js_ast.PropertyGetter:
+				flags = 2
+			case js_ast.PropertySetter:
+				flags = 3
+			case js_ast.PropertyAutoAccessor:
+				flags = 4
+			case js_ast.PropertyField:
+				flags = 5
+			}
+			if flags >= 4 {
+				initializerIndex = ctx.decoratorPropertyToInitializerMap[propIndex]
+			}
+			if prop.Flags.Has(js_ast.PropertyIsStatic) {
+				flags |= 8
+			}
+			if analysis.private != nil {
+				flags |= 16
+			}
+
+			// Start the arguments for the call to "__decorateElement"
+			var key js_ast.Expr
+			decoratorsRef := decoratorTempRefs[propIndex]
+			if ctx.decoratorContextRef == ast.InvalidRef {
+				ctx.decoratorContextRef = p.generateTempRef(tempRefNeedsDeclare, "_init")
+			}
+			if analysis.private != nil {
+				key = js_ast.Expr{Loc: loc, Data: &js_ast.EString{Value: helpers.StringToUTF16(p.symbols[analysis.private.Ref.InnerIndex].OriginalName)}}
+			} else {
+				key = cloneKeyForLowerClass(keyExprNoSideEffects)
+			}
+			args := []js_ast.Expr{
+				{Loc: loc, Data: &js_ast.EIdentifier{Ref: ctx.decoratorContextRef}},
+				{Loc: loc, Data: &js_ast.ENumber{Value: float64(flags)}},
+				key,
+				{Loc: atLoc, Data: &js_ast.EIdentifier{Ref: decoratorsRef}},
+			}
+			p.recordUsage(ctx.decoratorContextRef)
+			p.recordUsage(decoratorsRef)
+
+			// Append any optional additional arguments
+			privateFnRef := ast.InvalidRef
+			if analysis.private != nil {
+				// Add the "target" argument (the weak set)
+				args = append(args, js_ast.Expr{Loc: keyLoc, Data: &js_ast.EIdentifier{Ref: analysis.private.Ref}})
+				p.recordUsage(analysis.private.Ref)
+
+				// Add the "extra" argument (the function)
+				switch prop.Kind {
+				case js_ast.PropertyMethod:
+					privateFnRef = p.privateGetters[analysis.private.Ref]
+				case js_ast.PropertyGetter:
+					privateFnRef = p.privateGetters[analysis.private.Ref]
+				case js_ast.PropertySetter:
+					privateFnRef = p.privateSetters[analysis.private.Ref]
+				}
+				if privateFnRef != ast.InvalidRef {
+					args = append(args, js_ast.Expr{Loc: keyLoc, Data: &js_ast.EIdentifier{Ref: privateFnRef}})
+					p.recordUsage(privateFnRef)
+				}
+			} else {
+				// Add the "target" argument (the class object)
+				args = append(args, ctx.nameFunc())
+			}
+
+			// Auto-accessors will generate a private field for storage. Lower this
+			// field, which will generate a WeakMap instance, and then pass the
+			// WeakMap instance into the decorator helper so the lowered getter and
+			// setter can use it.
+			if prop.Kind == js_ast.PropertyAutoAccessor {
+				var kind ast.SymbolKind
+				if prop.Flags.Has(js_ast.PropertyIsStatic) {
+					kind = ast.SymbolPrivateStaticField
+				} else {
+					kind = ast.SymbolPrivateField
+				}
+				ref := p.newSymbol(kind, "#"+p.propertyNameHint(prop.Key))
+				p.symbols[ref.InnerIndex].Flags |= ast.PrivateSymbolMustBeLowered
+				_, autoAccessorWeakMapRef, _ := ctx.lowerField(p, prop, &js_ast.EPrivateIdentifier{Ref: ref}, false, false, initializerIndex)
+				args = append(args, js_ast.Expr{Loc: keyLoc, Data: &js_ast.EIdentifier{Ref: autoAccessorWeakMapRef}})
+				p.recordUsage(autoAccessorWeakMapRef)
+			}
+
+			// Assign the result
+			element := p.callRuntime(loc, "__decorateElement", args)
+			if privateFnRef != ast.InvalidRef {
+				element = js_ast.Assign(js_ast.Expr{Loc: keyLoc, Data: &js_ast.EIdentifier{Ref: privateFnRef}}, element)
+				p.recordUsage(privateFnRef)
+			} else if prop.Kind == js_ast.PropertyAutoAccessor && analysis.private != nil {
+				ref := p.generateTempRef(tempRefNeedsDeclare, "")
+				privateGetFnRef := p.generateTempRef(tempRefNeedsDeclare, "_")
+				privateSetFnRef := p.generateTempRef(tempRefNeedsDeclare, "_")
+				p.symbols[privateGetFnRef.InnerIndex].Link = p.privateGetters[analysis.private.Ref]
+				p.symbols[privateSetFnRef.InnerIndex].Link = p.privateSetters[analysis.private.Ref]
+
+				// Unpack the "get" and "set" properties from the returned property descriptor
+				element = js_ast.JoinWithComma(js_ast.JoinWithComma(
+					js_ast.Assign(
+						js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: ref}},
+						element),
+					js_ast.Assign(
+						js_ast.Expr{Loc: keyLoc, Data: &js_ast.EIdentifier{Ref: privateGetFnRef}},
+						js_ast.Expr{Loc: loc, Data: &js_ast.EDot{Target: js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: ref}}, Name: "get", NameLoc: loc}})),
+					js_ast.Assign(
+						js_ast.Expr{Loc: keyLoc, Data: &js_ast.EIdentifier{Ref: privateSetFnRef}},
+						js_ast.Expr{Loc: loc, Data: &js_ast.EDot{Target: js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: ref}}, Name: "set", NameLoc: loc}}))
+				p.recordUsage(ref)
+				p.recordUsage(privateGetFnRef)
+				p.recordUsage(ref)
+				p.recordUsage(privateSetFnRef)
+				p.recordUsage(ref)
+			}
+
+			// Put the call to the decorators in the right place
+			if prop.Kind == js_ast.PropertyField {
+				// Field
+				if prop.Flags.Has(js_ast.PropertyIsStatic) {
+					ctx.decoratorStaticFieldElements = append(ctx.decoratorStaticFieldElements, element)
+				} else {
+					ctx.decoratorInstanceFieldElements = append(ctx.decoratorInstanceFieldElements, element)
+				}
+			} else {
+				// Non-field
+				if prop.Flags.Has(js_ast.PropertyIsStatic) {
+					ctx.decoratorStaticNonFieldElements = append(ctx.decoratorStaticNonFieldElements, element)
+				} else {
+					ctx.decoratorInstanceNonFieldElements = append(ctx.decoratorInstanceNonFieldElements, element)
+				}
+			}
+
+			// Omit decorated auto-accessors as they will be now generated at run-time instead
+			if prop.Kind == js_ast.PropertyAutoAccessor {
+				if analysis.private != nil {
+					ctx.lowerPrivateMethod(p, prop, analysis.private)
+				}
+				continue
+			}
+		}
+
+		// Generate get/set methods for auto-accessors
+		if analysis.rewriteAutoAccessorToGetSet {
+			properties = ctx.rewriteAutoAccessorToGetSet(p, prop, properties, keyExprNoSideEffects, analysis.mustLowerField, analysis.private, result)
+			continue
+		}
+
+		// Lower fields
+		if (!prop.Kind.IsMethodDefinition() && analysis.mustLowerField) || analysis.staticFieldToBlockAssign {
+			var keep bool
+			prop, _, keep = ctx.lowerField(p, prop, analysis.private, analysis.shouldOmitFieldInitializer, analysis.staticFieldToBlockAssign, initializerIndex)
+			if !keep {
+				continue
+			}
+		}
+
+		// Lower methods
+		if prop.Kind.IsMethodDefinition() && ctx.lowerMethod(p, prop, analysis.private) {
+			continue
+		}
+
+		// Keep this property
+		properties = append(properties, prop)
+	}
+
+	// Finish the filtering operation
+	ctx.class.Properties = properties
+}
+
+func (ctx *lowerClassContext) lowerStaticBlock(p *parser, loc logger.Loc, block js_ast.ClassStaticBlock) {
+	isAllExprs := []js_ast.Expr{}
+
+	// Are all statements in the block expression statements?
+loop:
+	for _, stmt := range block.Block.Stmts {
+		switch s := stmt.Data.(type) {
+		case *js_ast.SEmpty:
+			// Omit stray semicolons completely
+		case *js_ast.SExpr:
+			isAllExprs = append(isAllExprs, s.Value)
+		default:
+			isAllExprs = nil
+			break loop
+		}
+	}
+
+	if isAllExprs != nil {
+		// I think it should be safe to inline the static block IIFE here
+		// since all uses of "this" should have already been replaced by now.
+		ctx.staticMembers = append(ctx.staticMembers, isAllExprs...)
+	} else {
+		// But if there is a non-expression statement, fall back to using an
+		// IIFE since we may be in an expression context and can't use a block.
+		ctx.staticMembers = append(ctx.staticMembers, js_ast.Expr{Loc: loc, Data: &js_ast.ECall{
+			Target: js_ast.Expr{Loc: loc, Data: &js_ast.EArrow{Body: js_ast.FnBody{
+				Loc:   block.Loc,
+				Block: block.Block,
+			}}},
+			CanBeUnwrappedIfUnused: p.astHelpers.StmtsCanBeRemovedIfUnused(block.Block.Stmts, 0),
+		}})
+	}
+}
+
+func (ctx *lowerClassContext) rewriteAutoAccessorToGetSet(
+	p *parser,
+	prop js_ast.Property,
+	properties []js_ast.Property,
+	keyExprNoSideEffects js_ast.Expr,
+	mustLowerField bool,
+	private *js_ast.EPrivateIdentifier,
+	result visitClassResult,
+) []js_ast.Property {
+	var storageKind ast.SymbolKind
+	if prop.Flags.Has(js_ast.PropertyIsStatic) {
+		storageKind = ast.SymbolPrivateStaticField
+	} else {
+		storageKind = ast.SymbolPrivateField
+	}
+
+	// Generate the name of the private field to use for storage
+	var storageName string
+	switch k := keyExprNoSideEffects.Data.(type) {
+	case *js_ast.EString:
+		storageName = "#" + helpers.UTF16ToString(k.Value)
+	case *js_ast.EPrivateIdentifier:
+		storageName = "#_" + p.symbols[k.Ref.InnerIndex].OriginalName[1:]
+	default:
+		storageName = "#" + ast.DefaultNameMinifierJS.NumberToMinifiedName(ctx.autoAccessorCount)
+		ctx.autoAccessorCount++
+	}
+
+	// Generate the symbols we need
+	storageRef := p.newSymbol(storageKind, storageName)
+	argRef := p.newSymbol(ast.SymbolOther, "_")
+	result.bodyScope.Generated = append(result.bodyScope.Generated, storageRef)
+	result.bodyScope.Children = append(result.bodyScope.Children, &js_ast.Scope{Kind: js_ast.ScopeFunctionBody, Generated: []ast.Ref{argRef}})
+
+	// Replace this accessor with other properties
+	loc := keyExprNoSideEffects.Loc
+	storagePrivate := &js_ast.EPrivateIdentifier{Ref: storageRef}
+	if mustLowerField {
+		// Forward the accessor's lowering status on to the storage field. If we
+		// don't do this, then we risk having the underlying private symbol
+		// behaving differently than if it were authored manually (e.g. being
+		// placed outside of the class body, which is a syntax error).
+		p.symbols[storageRef.InnerIndex].Flags |= ast.PrivateSymbolMustBeLowered
+	}
+	storageNeedsToBeLowered := p.privateSymbolNeedsToBeLowered(storagePrivate)
+	storageProp := js_ast.Property{
+		Loc:              prop.Loc,
+		Kind:             js_ast.PropertyField,
+		Flags:            prop.Flags & js_ast.PropertyIsStatic,
+		Key:              js_ast.Expr{Loc: loc, Data: storagePrivate},
+		InitializerOrNil: prop.InitializerOrNil,
+	}
+	if !mustLowerField {
+		properties = append(properties, storageProp)
+	} else if prop, _, ok := ctx.lowerField(p, storageProp, storagePrivate, false, false, -1); ok {
+		properties = append(properties, prop)
+	}
+
+	// Getter
+	var getExpr js_ast.Expr
+	if storageNeedsToBeLowered {
+		getExpr = p.lowerPrivateGet(js_ast.Expr{Loc: loc, Data: js_ast.EThisShared}, loc, storagePrivate)
+	} else {
+		p.recordUsage(storageRef)
+		getExpr = js_ast.Expr{Loc: loc, Data: &js_ast.EIndex{
+			Target: js_ast.Expr{Loc: loc, Data: js_ast.EThisShared},
+			Index:  js_ast.Expr{Loc: loc, Data: &js_ast.EPrivateIdentifier{Ref: storageRef}},
+		}}
+	}
+	getterProp := js_ast.Property{
+		Loc:   prop.Loc,
+		Kind:  js_ast.PropertyGetter,
+		Flags: prop.Flags,
+		Key:   prop.Key,
+		ValueOrNil: js_ast.Expr{Loc: loc, Data: &js_ast.EFunction{
+			Fn: js_ast.Fn{
+				Body: js_ast.FnBody{
+					Loc: loc,
+					Block: js_ast.SBlock{
+						Stmts: []js_ast.Stmt{
+							{Loc: loc, Data: &js_ast.SReturn{ValueOrNil: getExpr}},
+						},
+					},
+				},
+			},
+		}},
+	}
+	if !ctx.lowerMethod(p, getterProp, private) {
+		properties = append(properties, getterProp)
+	}
+
+	// Setter
+	var setExpr js_ast.Expr
+	if storageNeedsToBeLowered {
+		setExpr = p.lowerPrivateSet(js_ast.Expr{Loc: loc, Data: js_ast.EThisShared}, loc, storagePrivate,
+			js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: argRef}})
+	} else {
+		p.recordUsage(storageRef)
+		p.recordUsage(argRef)
+		setExpr = js_ast.Assign(
+			js_ast.Expr{Loc: loc, Data: &js_ast.EIndex{
+				Target: js_ast.Expr{Loc: loc, Data: js_ast.EThisShared},
+				Index:  js_ast.Expr{Loc: loc, Data: &js_ast.EPrivateIdentifier{Ref: storageRef}},
+			}},
+			js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: argRef}},
+		)
+	}
+	setterProp := js_ast.Property{
+		Loc:   prop.Loc,
+		Kind:  js_ast.PropertySetter,
+		Flags: prop.Flags,
+		Key:   cloneKeyForLowerClass(keyExprNoSideEffects),
+		ValueOrNil: js_ast.Expr{Loc: loc, Data: &js_ast.EFunction{
+			Fn: js_ast.Fn{
+				Args: []js_ast.Arg{
+					{Binding: js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: argRef}}},
+				},
+				Body: js_ast.FnBody{
+					Loc: loc,
+					Block: js_ast.SBlock{
+						Stmts: []js_ast.Stmt{
+							{Loc: loc, Data: &js_ast.SExpr{Value: setExpr}},
+						},
+					},
+				},
+			},
+		}},
+	}
+	if !ctx.lowerMethod(p, setterProp, private) {
+		properties = append(properties, setterProp)
+	}
+	return properties
+}
+
+func (ctx *lowerClassContext) insertInitializersIntoConstructor(p *parser, classLoweringInfo classLoweringInfo, result visitClassResult) {
+	if len(ctx.parameterFields) == 0 &&
+		!ctx.decoratorCallInstanceMethodExtraInitializers &&
+		len(ctx.instancePrivateMethods) == 0 &&
+		len(ctx.instanceMembers) == 0 &&
+		(ctx.ctor == nil || result.superCtorRef == ast.InvalidRef) {
+		// No need to generate a constructor
+		return
+	}
+
+	// Create a constructor if one doesn't already exist
+	if ctx.ctor == nil {
+		ctx.ctor = &js_ast.EFunction{Fn: js_ast.Fn{Body: js_ast.FnBody{Loc: ctx.classLoc}}}
+
+		// Append it to the list to reuse existing allocation space
+		ctx.class.Properties = append(ctx.class.Properties, js_ast.Property{
+			Kind:       js_ast.PropertyMethod,
+			Loc:        ctx.classLoc,
+			Key:        js_ast.Expr{Loc: ctx.classLoc, Data: &js_ast.EString{Value: helpers.StringToUTF16("constructor")}},
+			ValueOrNil: js_ast.Expr{Loc: ctx.classLoc, Data: ctx.ctor},
+		})
+
+		// Make sure the constructor has a super() call if needed
+		if ctx.class.ExtendsOrNil.Data != nil {
+			target := js_ast.Expr{Loc: ctx.classLoc, Data: js_ast.ESuperShared}
+			if classLoweringInfo.shimSuperCtorCalls {
+				p.recordUsage(result.superCtorRef)
+				target.Data = &js_ast.EIdentifier{Ref: result.superCtorRef}
+			}
+			argumentsRef := p.newSymbol(ast.SymbolUnbound, "arguments")
+			p.currentScope.Generated = append(p.currentScope.Generated, argumentsRef)
+			ctx.ctor.Fn.Body.Block.Stmts = append(ctx.ctor.Fn.Body.Block.Stmts, js_ast.Stmt{Loc: ctx.classLoc, Data: &js_ast.SExpr{Value: js_ast.Expr{Loc: ctx.classLoc, Data: &js_ast.ECall{
+				Target: target,
+				Args:   []js_ast.Expr{{Loc: ctx.classLoc, Data: &js_ast.ESpread{Value: js_ast.Expr{Loc: ctx.classLoc, Data: &js_ast.EIdentifier{Ref: argumentsRef}}}}},
+			}}}})
+		}
+	}
+
+	// Run instanceMethodExtraInitializers if needed
+	var decoratorInstanceMethodExtraInitializers js_ast.Expr
+	if ctx.decoratorCallInstanceMethodExtraInitializers {
+		decoratorInstanceMethodExtraInitializers = p.callRuntime(ctx.classLoc, "__runInitializers", []js_ast.Expr{
+			{Loc: ctx.classLoc, Data: &js_ast.EIdentifier{Ref: ctx.decoratorContextRef}},
+			{Loc: ctx.classLoc, Data: &js_ast.ENumber{Value: (2 << 1) | 1}},
+			{Loc: ctx.classLoc, Data: js_ast.EThisShared},
+		})
+		p.recordUsage(ctx.decoratorContextRef)
+	}
+
+	// Make sure the instance field initializers come after "super()" since
+	// they need "this" to ba available
+	generatedStmts := make([]js_ast.Stmt, 0,
+		len(ctx.parameterFields)+
+			len(ctx.instancePrivateMethods)+
+			len(ctx.instanceMembers))
+	generatedStmts = append(generatedStmts, ctx.parameterFields...)
+	if decoratorInstanceMethodExtraInitializers.Data != nil {
+		generatedStmts = append(generatedStmts, js_ast.Stmt{Loc: decoratorInstanceMethodExtraInitializers.Loc, Data: &js_ast.SExpr{Value: decoratorInstanceMethodExtraInitializers}})
+	}
+	generatedStmts = append(generatedStmts, ctx.instancePrivateMethods...)
+	generatedStmts = append(generatedStmts, ctx.instanceMembers...)
+	p.insertStmtsAfterSuperCall(&ctx.ctor.Fn.Body, generatedStmts, result.superCtorRef)
+
+	// Sort the constructor first to match the TypeScript compiler's output
+	for i := 0; i < len(ctx.class.Properties); i++ {
+		if ctx.class.Properties[i].ValueOrNil.Data == ctx.ctor {
+			ctorProp := ctx.class.Properties[i]
+			for j := i; j > 0; j-- {
+				ctx.class.Properties[j] = ctx.class.Properties[j-1]
+			}
+			ctx.class.Properties[0] = ctorProp
+			break
+		}
+	}
+}
+
+func (ctx *lowerClassContext) finishAndGenerateCode(p *parser, result visitClassResult) ([]js_ast.Stmt, js_ast.Expr) {
+	// When bundling is enabled, we convert top-level class statements to
+	// expressions:
+	//
+	//   // Before
+	//   class Foo {
+	//     static foo = () => Foo
+	//   }
+	//   Foo = wrap(Foo)
+	//
+	//   // After
+	//   var _Foo = class _Foo {
+	//     static foo = () => _Foo;
+	//   };
+	//   var Foo = _Foo;
+	//   Foo = wrap(Foo);
+	//
+	// One reason to do this is that esbuild's bundler sometimes needs to lazily-
+	// evaluate a module. For example, a module may end up being both the target
+	// of a dynamic "import()" call and a static "import" statement. Lazy module
+	// evaluation is done by wrapping the top-level module code in a closure. To
+	// avoid a performance hit for static "import" statements, esbuild stores
+	// top-level exported symbols outside of the closure and references them
+	// directly instead of indirectly.
+	//
+	// Another reason to do this is that multiple JavaScript VMs have had and
+	// continue to have performance issues with TDZ (i.e. "temporal dead zone")
+	// checks. These checks validate that a let, or const, or class symbol isn't
+	// used before it's initialized. Here are two issues with well-known VMs:
+	//
+	//   * V8: https://bugs.chromium.org/p/v8/issues/detail?id=13723 (10% slowdown)
+	//   * JavaScriptCore: https://bugs.webkit.org/show_bug.cgi?id=199866 (1,000% slowdown!)
+	//
+	// JavaScriptCore had a severe performance issue as their TDZ implementation
+	// had time complexity that was quadratic in the number of variables needing
+	// TDZ checks in the same scope (with the top-level scope typically being the
+	// worst offender). V8 has ongoing issues with TDZ checks being present
+	// throughout the code their JIT generates even when they have already been
+	// checked earlier in the same function or when the function in question has
+	// already been run (so the checks have already happened).
+	//
+	// Due to esbuild's parallel architecture, we both a) need to transform class
+	// statements to variables during parsing and b) don't yet know whether this
+	// module will need to be lazily-evaluated or not in the parser. So we always
+	// do this just in case it's needed.
+	mustConvertStmtToExpr := ctx.kind != classKindExpr && p.currentScope.Parent == nil && (p.options.mode == config.ModeBundle || p.willWrapModuleInTryCatchForUsing)
+
+	// Check to see if we have lowered decorators on the class itself
+	var classDecorators js_ast.Expr
+	var classExperimentalDecorators []js_ast.Decorator
+	if p.options.ts.Parse && p.options.ts.Config.ExperimentalDecorators == config.True {
+		classExperimentalDecorators = ctx.class.Decorators
+		ctx.class.Decorators = nil
+	} else if ctx.class.ShouldLowerStandardDecorators {
+		classDecorators = ctx.decoratorClassDecorators
+	}
+
+	var decorateClassExpr js_ast.Expr
+	if classDecorators.Data != nil {
+		// Handle JavaScript decorators on the class itself
+		if ctx.decoratorContextRef == ast.InvalidRef {
+			ctx.decoratorContextRef = p.generateTempRef(tempRefNeedsDeclare, "_init")
+		}
+		decorateClassExpr = p.callRuntime(ctx.classLoc, "__decorateElement", []js_ast.Expr{
+			{Loc: ctx.classLoc, Data: &js_ast.EIdentifier{Ref: ctx.decoratorContextRef}},
+			{Loc: ctx.classLoc, Data: &js_ast.ENumber{Value: 0}},
+			{Loc: ctx.classLoc, Data: &js_ast.EString{Value: helpers.StringToUTF16(ctx.nameToKeep)}},
+			classDecorators,
+			ctx.nameFunc(),
+		})
+		p.recordUsage(ctx.decoratorContextRef)
+		decorateClassExpr = js_ast.Assign(ctx.nameFunc(), decorateClassExpr)
+	} else if ctx.decoratorContextRef != ast.InvalidRef {
+		// Decorator metadata is present if there are any decorators on the class at all
+		decorateClassExpr = p.callRuntime(ctx.classLoc, "__decoratorMetadata", []js_ast.Expr{
+			{Loc: ctx.classLoc, Data: &js_ast.EIdentifier{Ref: ctx.decoratorContextRef}},
+			ctx.nameFunc(),
+		})
+	}
+
+	// If this is true, we have removed some code from the class body that could
+	// potentially contain an expression that captures the inner class name.
+	// In this case we must explicitly store the class to a separate inner class
+	// name binding to avoid incorrect behavior if the class is later re-assigned,
+	// since the removed code will no longer be in the class body scope.
+	hasPotentialInnerClassNameEscape := result.innerClassNameRef != ast.InvalidRef &&
+		(ctx.computedPropertyChain.Data != nil ||
+			len(ctx.privateMembers) > 0 ||
+			len(ctx.staticPrivateMethods) > 0 ||
+			len(ctx.staticMembers) > 0 ||
+
+			// TypeScript experimental decorators
+			len(ctx.instanceExperimentalDecorators) > 0 ||
+			len(ctx.staticExperimentalDecorators) > 0 ||
+			len(classExperimentalDecorators) > 0 ||
+
+			// JavaScript decorators
+			ctx.decoratorContextRef != ast.InvalidRef)
+
+	// If we need to represent the class as an expression (even if it's a
+	// statement), then generate another symbol to use as the class name
+	nameForClassDecorators := ast.LocRef{Ref: ast.InvalidRef}
+	if len(classExperimentalDecorators) > 0 || hasPotentialInnerClassNameEscape || mustConvertStmtToExpr {
+		if ctx.kind == classKindExpr {
+			// For expressions, the inner and outer class names are the same
+			name := ctx.nameFunc()
+			nameForClassDecorators = ast.LocRef{Loc: name.Loc, Ref: name.Data.(*js_ast.EIdentifier).Ref}
+		} else {
+			// For statements we need to use the outer class name, not the inner one
+			if ctx.class.Name != nil {
+				nameForClassDecorators = *ctx.class.Name
+			} else if ctx.kind == classKindExportDefaultStmt {
+				nameForClassDecorators = ctx.defaultName
+			} else {
+				nameForClassDecorators = ast.LocRef{Loc: ctx.classLoc, Ref: p.generateTempRef(tempRefNoDeclare, "")}
+			}
+			p.recordUsage(nameForClassDecorators.Ref)
+		}
+	}
+
+	var prefixExprs []js_ast.Expr
+	var suffixExprs []js_ast.Expr
+
+	// If there are JavaScript decorators, start by allocating a context object
+	if ctx.decoratorContextRef != ast.InvalidRef {
+		base := js_ast.Expr{Loc: ctx.classLoc, Data: js_ast.ENullShared}
+		if ctx.class.ExtendsOrNil.Data != nil {
+			if ctx.extendsRef == ast.InvalidRef {
+				ctx.extendsRef = p.generateTempRef(tempRefNeedsDeclare, "")
+				ctx.class.ExtendsOrNil = js_ast.Assign(js_ast.Expr{Loc: ctx.class.ExtendsOrNil.Loc, Data: &js_ast.EIdentifier{Ref: ctx.extendsRef}}, ctx.class.ExtendsOrNil)
+				p.recordUsage(ctx.extendsRef)
+			}
+			base.Data = &js_ast.EIdentifier{Ref: ctx.extendsRef}
+		}
+		suffixExprs = append(suffixExprs, js_ast.Assign(
+			js_ast.Expr{Loc: ctx.classLoc, Data: &js_ast.EIdentifier{Ref: ctx.decoratorContextRef}},
+			p.callRuntime(ctx.classLoc, "__decoratorStart", []js_ast.Expr{base}),
+		))
+		p.recordUsage(ctx.decoratorContextRef)
+	}
+
+	// Any of the computed property chain that we hoisted out of the class
+	// body needs to come before the class expression.
+	if ctx.computedPropertyChain.Data != nil {
+		prefixExprs = append(prefixExprs, ctx.computedPropertyChain)
+	}
+
+	// WeakSets and WeakMaps
+	suffixExprs = append(suffixExprs, ctx.privateMembers...)
+
+	// Evaluate JavaScript decorators here
+	suffixExprs = append(suffixExprs, ctx.decoratorStaticNonFieldElements...)
+	suffixExprs = append(suffixExprs, ctx.decoratorInstanceNonFieldElements...)
+	suffixExprs = append(suffixExprs, ctx.decoratorStaticFieldElements...)
+	suffixExprs = append(suffixExprs, ctx.decoratorInstanceFieldElements...)
+
+	// Lowered initializers for static methods (including getters and setters)
+	suffixExprs = append(suffixExprs, ctx.staticPrivateMethods...)
+
+	// Run JavaScript class decorators at the end of class initialization
+	if decorateClassExpr.Data != nil {
+		suffixExprs = append(suffixExprs, decorateClassExpr)
+	}
+
+	// For each element initializer of staticMethodExtraInitializers
+	if ctx.decoratorCallStaticMethodExtraInitializers {
+		suffixExprs = append(suffixExprs, p.callRuntime(ctx.classLoc, "__runInitializers", []js_ast.Expr{
+			{Loc: ctx.classLoc, Data: &js_ast.EIdentifier{Ref: ctx.decoratorContextRef}},
+			{Loc: ctx.classLoc, Data: &js_ast.ENumber{Value: (1 << 1) | 1}},
+			ctx.nameFunc(),
+		}))
+		p.recordUsage(ctx.decoratorContextRef)
+	}
+
+	// Lowered initializers for static fields, static accessors, and static blocks
+	suffixExprs = append(suffixExprs, ctx.staticMembers...)
+
+	// The official TypeScript compiler adds generated code after the class body
+	// in this exact order. Matching this order is important for correctness.
+	suffixExprs = append(suffixExprs, ctx.instanceExperimentalDecorators...)
+	suffixExprs = append(suffixExprs, ctx.staticExperimentalDecorators...)
+
+	// For each element initializer of classExtraInitializers
+	if classDecorators.Data != nil {
+		suffixExprs = append(suffixExprs, p.callRuntime(ctx.classLoc, "__runInitializers", []js_ast.Expr{
+			{Loc: ctx.classLoc, Data: &js_ast.EIdentifier{Ref: ctx.decoratorContextRef}},
+			{Loc: ctx.classLoc, Data: &js_ast.ENumber{Value: (0 << 1) | 1}},
+			ctx.nameFunc(),
+		}))
+		p.recordUsage(ctx.decoratorContextRef)
+	}
+
+	// Run TypeScript experimental class decorators at the end of class initialization
+	if len(classExperimentalDecorators) > 0 {
+		values := make([]js_ast.Expr, len(classExperimentalDecorators))
+		for i, decorator := range classExperimentalDecorators {
+			values[i] = decorator.Value
+		}
+		suffixExprs = append(suffixExprs, js_ast.Assign(
+			js_ast.Expr{Loc: nameForClassDecorators.Loc, Data: &js_ast.EIdentifier{Ref: nameForClassDecorators.Ref}},
+			p.callRuntime(ctx.classLoc, "__decorateClass", []js_ast.Expr{
+				{Loc: ctx.classLoc, Data: &js_ast.EArray{Items: values}},
+				{Loc: nameForClassDecorators.Loc, Data: &js_ast.EIdentifier{Ref: nameForClassDecorators.Ref}},
+			}),
+		))
+		p.recordUsage(nameForClassDecorators.Ref)
+		p.recordUsage(nameForClassDecorators.Ref)
+	}
+
+	// Our caller expects us to return the same form that was originally given to
+	// us. If the class was originally an expression, then return an expression.
+	if ctx.kind == classKindExpr {
+		// Calling "nameFunc" will replace "classExpr", so make sure to do that first
+		// before joining "classExpr" with any other expressions
+		var nameToJoin js_ast.Expr
+		if ctx.didCaptureClassExpr || len(suffixExprs) > 0 {
+			nameToJoin = ctx.nameFunc()
+		}
+
+		// Insert expressions on either side of the class as appropriate
+		ctx.classExpr = js_ast.JoinWithComma(js_ast.JoinAllWithComma(prefixExprs), ctx.classExpr)
+		ctx.classExpr = js_ast.JoinWithComma(ctx.classExpr, js_ast.JoinAllWithComma(suffixExprs))
+
+		// Finally join "classExpr" with the variable that holds the class object
+		ctx.classExpr = js_ast.JoinWithComma(ctx.classExpr, nameToJoin)
+		if ctx.wrapFunc != nil {
+			ctx.classExpr = ctx.wrapFunc(ctx.classExpr)
+		}
+		return nil, ctx.classExpr
+	}
+
+	// Otherwise, the class was originally a statement. Return an array of
+	// statements instead.
+	var stmts []js_ast.Stmt
+	var outerClassNameDecl js_ast.Stmt
+
+	// Insert expressions before the class as appropriate
+	for _, expr := range prefixExprs {
+		stmts = append(stmts, js_ast.Stmt{Loc: expr.Loc, Data: &js_ast.SExpr{Value: expr}})
+	}
+
+	// Handle converting a class statement to a class expression
+	if nameForClassDecorators.Ref != ast.InvalidRef {
+		classExpr := js_ast.EClass{Class: *ctx.class}
+		ctx.class = &classExpr.Class
+		init := js_ast.Expr{Loc: ctx.classLoc, Data: &classExpr}
+
+		// If the inner class name was referenced, then set the name of the class
+		// that we will end up printing to the inner class name. Otherwise if the
+		// inner class name was unused, we can just leave it blank.
+		if result.innerClassNameRef != ast.InvalidRef {
+			// "class Foo { x = Foo }" => "const Foo = class _Foo { x = _Foo }"
+			ctx.class.Name.Ref = result.innerClassNameRef
+		} else {
+			// "class Foo {}" => "const Foo = class {}"
+			ctx.class.Name = nil
+		}
+
+		// Generate the class initialization statement
+		if len(classExperimentalDecorators) > 0 {
+			// If there are class decorators, then we actually need to mutate the
+			// immutable "const" binding that shadows everything in the class body.
+			// The official TypeScript compiler does this by rewriting all class name
+			// references in the class body to another temporary variable. This is
+			// basically what we're doing here.
+			p.recordUsage(nameForClassDecorators.Ref)
+			stmts = append(stmts, js_ast.Stmt{Loc: ctx.classLoc, Data: &js_ast.SLocal{
+				Kind:     p.selectLocalKind(js_ast.LocalLet),
+				IsExport: ctx.kind == classKindExportStmt,
+				Decls: []js_ast.Decl{{
+					Binding:    js_ast.Binding{Loc: nameForClassDecorators.Loc, Data: &js_ast.BIdentifier{Ref: nameForClassDecorators.Ref}},
+					ValueOrNil: init,
+				}},
+			}})
+			if ctx.class.Name != nil {
+				p.mergeSymbols(ctx.class.Name.Ref, nameForClassDecorators.Ref)
+				ctx.class.Name = nil
+			}
+		} else if hasPotentialInnerClassNameEscape {
+			// If the inner class name was used, then we explicitly generate a binding
+			// for it. That means the mutable outer class name is separate, and is
+			// initialized after all static member initializers have finished.
+			captureRef := p.newSymbol(ast.SymbolOther, p.symbols[result.innerClassNameRef.InnerIndex].OriginalName)
+			p.currentScope.Generated = append(p.currentScope.Generated, captureRef)
+			p.recordDeclaredSymbol(captureRef)
+			p.mergeSymbols(result.innerClassNameRef, captureRef)
+			kind := js_ast.LocalConst
+			if classDecorators.Data != nil {
+				// Class decorators need to be able to potentially mutate this binding
+				kind = js_ast.LocalLet
+			}
+			stmts = append(stmts, js_ast.Stmt{Loc: ctx.classLoc, Data: &js_ast.SLocal{
+				Kind: p.selectLocalKind(kind),
+				Decls: []js_ast.Decl{{
+					Binding:    js_ast.Binding{Loc: nameForClassDecorators.Loc, Data: &js_ast.BIdentifier{Ref: captureRef}},
+					ValueOrNil: init,
+				}},
+			}})
+			p.recordUsage(nameForClassDecorators.Ref)
+			p.recordUsage(captureRef)
+			outerClassNameDecl = js_ast.Stmt{Loc: ctx.classLoc, Data: &js_ast.SLocal{
+				Kind:     p.selectLocalKind(js_ast.LocalLet),
+				IsExport: ctx.kind == classKindExportStmt,
+				Decls: []js_ast.Decl{{
+					Binding:    js_ast.Binding{Loc: nameForClassDecorators.Loc, Data: &js_ast.BIdentifier{Ref: nameForClassDecorators.Ref}},
+					ValueOrNil: js_ast.Expr{Loc: ctx.classLoc, Data: &js_ast.EIdentifier{Ref: captureRef}},
+				}},
+			}}
+		} else {
+			// Otherwise, the inner class name isn't needed and we can just
+			// use a single variable declaration for the outer class name.
+			p.recordUsage(nameForClassDecorators.Ref)
+			stmts = append(stmts, js_ast.Stmt{Loc: ctx.classLoc, Data: &js_ast.SLocal{
+				Kind:     p.selectLocalKind(js_ast.LocalLet),
+				IsExport: ctx.kind == classKindExportStmt,
+				Decls: []js_ast.Decl{{
+					Binding:    js_ast.Binding{Loc: nameForClassDecorators.Loc, Data: &js_ast.BIdentifier{Ref: nameForClassDecorators.Ref}},
+					ValueOrNil: init,
+				}},
+			}})
+		}
+	} else {
+		// Generate the specific kind of class statement that was passed in to us
+		switch ctx.kind {
+		case classKindStmt:
+			stmts = append(stmts, js_ast.Stmt{Loc: ctx.classLoc, Data: &js_ast.SClass{Class: *ctx.class}})
+		case classKindExportStmt:
+			stmts = append(stmts, js_ast.Stmt{Loc: ctx.classLoc, Data: &js_ast.SClass{Class: *ctx.class, IsExport: true}})
+		case classKindExportDefaultStmt:
+			stmts = append(stmts, js_ast.Stmt{Loc: ctx.classLoc, Data: &js_ast.SExportDefault{
+				DefaultName: ctx.defaultName,
+				Value:       js_ast.Stmt{Loc: ctx.classLoc, Data: &js_ast.SClass{Class: *ctx.class}},
+			}})
+		}
+
+		// The inner class name inside the class statement should be the same as
+		// the class statement name itself
+		if ctx.class.Name != nil && result.innerClassNameRef != ast.InvalidRef {
+			// If the class body contains a direct eval call, then the inner class
+			// name will be marked as "MustNotBeRenamed" (because we have already
+			// popped the class body scope) but the outer class name won't be marked
+			// as "MustNotBeRenamed" yet (because we haven't yet popped the containing
+			// scope). Propagate this flag now before we merge these symbols so we
+			// don't end up accidentally renaming the outer class name to the inner
+			// class name.
+			if p.currentScope.ContainsDirectEval {
+				p.symbols[ctx.class.Name.Ref.InnerIndex].Flags |= (p.symbols[result.innerClassNameRef.InnerIndex].Flags & ast.MustNotBeRenamed)
+			}
+			p.mergeSymbols(result.innerClassNameRef, ctx.class.Name.Ref)
+		}
+	}
+
+	// Insert expressions after the class as appropriate
+	for _, expr := range suffixExprs {
+		stmts = append(stmts, js_ast.Stmt{Loc: expr.Loc, Data: &js_ast.SExpr{Value: expr}})
+	}
+
+	// This must come after the class body initializers have finished
+	if outerClassNameDecl.Data != nil {
+		stmts = append(stmts, outerClassNameDecl)
+	}
+
+	if nameForClassDecorators.Ref != ast.InvalidRef && ctx.kind == classKindExportDefaultStmt {
+		// "export default class x {}" => "class x {} export {x as default}"
+		stmts = append(stmts, js_ast.Stmt{Loc: ctx.classLoc, Data: &js_ast.SExportClause{
+			Items: []js_ast.ClauseItem{{Alias: "default", Name: ctx.defaultName}},
+		}})
+	}
+	return stmts, js_ast.Expr{}
+}
+
+func cloneKeyForLowerClass(key js_ast.Expr) js_ast.Expr {
+	switch k := key.Data.(type) {
+	case *js_ast.ENumber:
+		clone := *k
+		key.Data = &clone
+	case *js_ast.EString:
+		clone := *k
+		key.Data = &clone
+	case *js_ast.EIdentifier:
+		clone := *k
+		key.Data = &clone
+	case *js_ast.ENameOfSymbol:
+		clone := *k
+		key.Data = &clone
+	case *js_ast.EPrivateIdentifier:
+		clone := *k
+		key.Data = &clone
+	default:
+		panic("Internal error")
+	}
+	return key
+}
+
+// Replace "super()" calls with our shim so that we can guarantee
+// that instance field initialization doesn't happen before "super()"
+// is called, since at that point "this" isn't available.
+func (p *parser) insertStmtsAfterSuperCall(body *js_ast.FnBody, stmtsToInsert []js_ast.Stmt, superCtorRef ast.Ref) {
+	// If this class has no base class, then there's no "super()" call to handle
+	if superCtorRef == ast.InvalidRef || p.symbols[superCtorRef.InnerIndex].UseCountEstimate == 0 {
+		body.Block.Stmts = append(stmtsToInsert, body.Block.Stmts...)
+		return
+	}
+
+	// It's likely that there's only one "super()" call, and that it's a
+	// top-level expression in the constructor function body. If so, we
+	// can generate tighter code for this common case.
+	if p.symbols[superCtorRef.InnerIndex].UseCountEstimate == 1 {
+		for i, stmt := range body.Block.Stmts {
+			var before js_ast.Expr
+			var callLoc logger.Loc
+			var callData *js_ast.ECall
+			var after js_ast.Stmt
+
+			switch s := stmt.Data.(type) {
+			case *js_ast.SExpr:
+				if b, loc, c, a := findFirstTopLevelSuperCall(s.Value, superCtorRef); c != nil {
+					before, callLoc, callData = b, loc, c
+					if a.Data != nil {
+						s.Value = a
+						after = js_ast.Stmt{Loc: a.Loc, Data: s}
+					}
+				}
+
+			case *js_ast.SReturn:
+				if s.ValueOrNil.Data != nil {
+					if b, loc, c, a := findFirstTopLevelSuperCall(s.ValueOrNil, superCtorRef); c != nil && a.Data != nil {
+						before, callLoc, callData = b, loc, c
+						s.ValueOrNil = a
+						after = js_ast.Stmt{Loc: a.Loc, Data: s}
+					}
+				}
+
+			case *js_ast.SThrow:
+				if b, loc, c, a := findFirstTopLevelSuperCall(s.Value, superCtorRef); c != nil && a.Data != nil {
+					before, callLoc, callData = b, loc, c
+					s.Value = a
+					after = js_ast.Stmt{Loc: a.Loc, Data: s}
+				}
+
+			case *js_ast.SIf:
+				if b, loc, c, a := findFirstTopLevelSuperCall(s.Test, superCtorRef); c != nil && a.Data != nil {
+					before, callLoc, callData = b, loc, c
+					s.Test = a
+					after = js_ast.Stmt{Loc: a.Loc, Data: s}
+				}
+
+			case *js_ast.SSwitch:
+				if b, loc, c, a := findFirstTopLevelSuperCall(s.Test, superCtorRef); c != nil && a.Data != nil {
+					before, callLoc, callData = b, loc, c
+					s.Test = a
+					after = js_ast.Stmt{Loc: a.Loc, Data: s}
+				}
+
+			case *js_ast.SFor:
+				if expr, ok := s.InitOrNil.Data.(*js_ast.SExpr); ok {
+					if b, loc, c, a := findFirstTopLevelSuperCall(expr.Value, superCtorRef); c != nil {
+						before, callLoc, callData = b, loc, c
+						if a.Data != nil {
+							expr.Value = a
+						} else {
+							s.InitOrNil.Data = nil
+						}
+						after = js_ast.Stmt{Loc: a.Loc, Data: s}
+					}
+				}
+			}
+
+			if callData != nil {
+				// Revert "__super()" back to "super()"
+				callData.Target.Data = js_ast.ESuperShared
+				p.ignoreUsage(superCtorRef)
+
+				// Inject "stmtsToInsert" after "super()"
+				stmtsBefore := body.Block.Stmts[:i]
+				stmtsAfter := body.Block.Stmts[i+1:]
+				stmts := append([]js_ast.Stmt{}, stmtsBefore...)
+				if before.Data != nil {
+					stmts = append(stmts, js_ast.Stmt{Loc: before.Loc, Data: &js_ast.SExpr{Value: before}})
+				}
+				stmts = append(stmts, js_ast.Stmt{Loc: callLoc, Data: &js_ast.SExpr{Value: js_ast.Expr{Loc: callLoc, Data: callData}}})
+				stmts = append(stmts, stmtsToInsert...)
+				if after.Data != nil {
+					stmts = append(stmts, after)
+				}
+				stmts = append(stmts, stmtsAfter...)
+				body.Block.Stmts = stmts
+				return
+			}
+		}
+	}
+
+	// Otherwise, inject a generated "__super" helper function at the top of the
+	// constructor that looks like this:
+	//
+	//   var __super = (...args) => {
+	//     super(...args);
+	//     ...stmtsToInsert...
+	//     return this;
+	//   };
+	//
+	argsRef := p.newSymbol(ast.SymbolOther, "args")
+	p.currentScope.Generated = append(p.currentScope.Generated, argsRef)
+	p.recordUsage(argsRef)
+	superCall := js_ast.Expr{Loc: body.Loc, Data: &js_ast.ECall{
+		Target: js_ast.Expr{Loc: body.Loc, Data: js_ast.ESuperShared},
+		Args:   []js_ast.Expr{{Loc: body.Loc, Data: &js_ast.ESpread{Value: js_ast.Expr{Loc: body.Loc, Data: &js_ast.EIdentifier{Ref: argsRef}}}}},
+	}}
+	stmtsToInsert = append(append(
+		[]js_ast.Stmt{{Loc: body.Loc, Data: &js_ast.SExpr{Value: superCall}}},
+		stmtsToInsert...),
+		js_ast.Stmt{Loc: body.Loc, Data: &js_ast.SReturn{ValueOrNil: js_ast.Expr{Loc: body.Loc, Data: js_ast.EThisShared}}},
+	)
+	if p.options.minifySyntax {
+		stmtsToInsert = p.mangleStmts(stmtsToInsert, stmtsFnBody)
+	}
+	body.Block.Stmts = append([]js_ast.Stmt{{Loc: body.Loc, Data: &js_ast.SLocal{Decls: []js_ast.Decl{{
+		Binding: js_ast.Binding{Loc: body.Loc, Data: &js_ast.BIdentifier{Ref: superCtorRef}}, ValueOrNil: js_ast.Expr{Loc: body.Loc, Data: &js_ast.EArrow{
+			HasRestArg: true,
+			PreferExpr: true,
+			Args:       []js_ast.Arg{{Binding: js_ast.Binding{Loc: body.Loc, Data: &js_ast.BIdentifier{Ref: argsRef}}}},
+			Body:       js_ast.FnBody{Loc: body.Loc, Block: js_ast.SBlock{Stmts: stmtsToInsert}},
+		}},
+	}}}}}, body.Block.Stmts...)
+}
+
+func findFirstTopLevelSuperCall(expr js_ast.Expr, superCtorRef ast.Ref) (js_ast.Expr, logger.Loc, *js_ast.ECall, js_ast.Expr) {
+	if call, ok := expr.Data.(*js_ast.ECall); ok {
+		if target, ok := call.Target.Data.(*js_ast.EIdentifier); ok && target.Ref == superCtorRef {
+			call.Target.Data = js_ast.ESuperShared
+			return js_ast.Expr{}, expr.Loc, call, js_ast.Expr{}
+		}
+	}
+
+	// Also search down comma operator chains for a super call
+	if comma, ok := expr.Data.(*js_ast.EBinary); ok && comma.Op == js_ast.BinOpComma {
+		if before, loc, call, after := findFirstTopLevelSuperCall(comma.Left, superCtorRef); call != nil {
+			return before, loc, call, js_ast.JoinWithComma(after, comma.Right)
+		}
+
+		if before, loc, call, after := findFirstTopLevelSuperCall(comma.Right, superCtorRef); call != nil {
+			return js_ast.JoinWithComma(comma.Left, before), loc, call, after
+		}
+	}
+
+	return js_ast.Expr{}, logger.Loc{}, nil, js_ast.Expr{}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/js_parser/json_parser.go b/source/vendor/github.com/evanw/esbuild/internal/js_parser/json_parser.go
new file mode 100644
index 0000000..64062ca
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/js_parser/json_parser.go
@@ -0,0 +1,238 @@
+package js_parser
+
+import (
+	"fmt"
+
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/js_lexer"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type jsonParser struct {
+	log                            logger.Log
+	source                         logger.Source
+	tracker                        logger.LineColumnTracker
+	lexer                          js_lexer.Lexer
+	options                        JSONOptions
+	suppressWarningsAboutWeirdCode bool
+}
+
+func (p *jsonParser) parseMaybeTrailingComma(closeToken js_lexer.T) bool {
+	commaRange := p.lexer.Range()
+	p.lexer.Expect(js_lexer.TComma)
+
+	if p.lexer.Token == closeToken {
+		if p.options.Flavor == js_lexer.JSON {
+			p.log.AddError(&p.tracker, commaRange, "JSON does not support trailing commas")
+		}
+		return false
+	}
+
+	return true
+}
+
+func (p *jsonParser) parseExpr() js_ast.Expr {
+	loc := p.lexer.Loc()
+
+	switch p.lexer.Token {
+	case js_lexer.TFalse:
+		p.lexer.Next()
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EBoolean{Value: false}}
+
+	case js_lexer.TTrue:
+		p.lexer.Next()
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EBoolean{Value: true}}
+
+	case js_lexer.TNull:
+		p.lexer.Next()
+		return js_ast.Expr{Loc: loc, Data: js_ast.ENullShared}
+
+	case js_lexer.TStringLiteral:
+		value := p.lexer.StringLiteral()
+		p.lexer.Next()
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EString{Value: value}}
+
+	case js_lexer.TNumericLiteral:
+		value := p.lexer.Number
+		p.lexer.Next()
+		return js_ast.Expr{Loc: loc, Data: &js_ast.ENumber{Value: value}}
+
+	case js_lexer.TMinus:
+		p.lexer.Next()
+		value := p.lexer.Number
+		p.lexer.Expect(js_lexer.TNumericLiteral)
+		return js_ast.Expr{Loc: loc, Data: &js_ast.ENumber{Value: -value}}
+
+	case js_lexer.TOpenBracket:
+		p.lexer.Next()
+		isSingleLine := !p.lexer.HasNewlineBefore
+		items := []js_ast.Expr{}
+
+		for p.lexer.Token != js_lexer.TCloseBracket {
+			if len(items) > 0 {
+				if p.lexer.HasNewlineBefore {
+					isSingleLine = false
+				}
+				if !p.parseMaybeTrailingComma(js_lexer.TCloseBracket) {
+					break
+				}
+				if p.lexer.HasNewlineBefore {
+					isSingleLine = false
+				}
+			}
+
+			item := p.parseExpr()
+			items = append(items, item)
+		}
+
+		if p.lexer.HasNewlineBefore {
+			isSingleLine = false
+		}
+		closeBracketLoc := p.lexer.Loc()
+		p.lexer.Expect(js_lexer.TCloseBracket)
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EArray{
+			Items:           items,
+			IsSingleLine:    isSingleLine,
+			CloseBracketLoc: closeBracketLoc,
+		}}
+
+	case js_lexer.TOpenBrace:
+		p.lexer.Next()
+		isSingleLine := !p.lexer.HasNewlineBefore
+		properties := []js_ast.Property{}
+		duplicates := make(map[string]logger.Range)
+
+		for p.lexer.Token != js_lexer.TCloseBrace {
+			if len(properties) > 0 {
+				if p.lexer.HasNewlineBefore {
+					isSingleLine = false
+				}
+				if !p.parseMaybeTrailingComma(js_lexer.TCloseBrace) {
+					break
+				}
+				if p.lexer.HasNewlineBefore {
+					isSingleLine = false
+				}
+			}
+
+			keyString := p.lexer.StringLiteral()
+			keyRange := p.lexer.Range()
+			key := js_ast.Expr{Loc: keyRange.Loc, Data: &js_ast.EString{Value: keyString}}
+			p.lexer.Expect(js_lexer.TStringLiteral)
+
+			// Warn about duplicate keys
+			if !p.suppressWarningsAboutWeirdCode {
+				keyText := helpers.UTF16ToString(keyString)
+				if prevRange, ok := duplicates[keyText]; ok {
+					p.log.AddIDWithNotes(logger.MsgID_JS_DuplicateObjectKey, logger.Warning, &p.tracker, keyRange,
+						fmt.Sprintf("Duplicate key %q in object literal", keyText),
+						[]logger.MsgData{p.tracker.MsgData(prevRange, fmt.Sprintf("The original key %q is here:", keyText))})
+				} else {
+					duplicates[keyText] = keyRange
+				}
+			}
+
+			p.lexer.Expect(js_lexer.TColon)
+			value := p.parseExpr()
+
+			property := js_ast.Property{
+				Kind:       js_ast.PropertyField,
+				Loc:        keyRange.Loc,
+				Key:        key,
+				ValueOrNil: value,
+			}
+
+			// The key "__proto__" must not be a string literal in JavaScript because
+			// that actually modifies the prototype of the object. This can be
+			// avoided by using a computed property key instead of a string literal.
+			if helpers.UTF16EqualsString(keyString, "__proto__") && !p.options.UnsupportedJSFeatures.Has(compat.ObjectExtensions) {
+				property.Flags |= js_ast.PropertyIsComputed
+			}
+
+			properties = append(properties, property)
+		}
+
+		if p.lexer.HasNewlineBefore {
+			isSingleLine = false
+		}
+		closeBraceLoc := p.lexer.Loc()
+		p.lexer.Expect(js_lexer.TCloseBrace)
+		return js_ast.Expr{Loc: loc, Data: &js_ast.EObject{
+			Properties:    properties,
+			IsSingleLine:  isSingleLine,
+			CloseBraceLoc: closeBraceLoc,
+		}}
+
+	default:
+		p.lexer.Unexpected()
+		return js_ast.Expr{}
+	}
+}
+
+type JSONOptions struct {
+	UnsupportedJSFeatures compat.JSFeature
+	Flavor                js_lexer.JSONFlavor
+	ErrorSuffix           string
+}
+
+func ParseJSON(log logger.Log, source logger.Source, options JSONOptions) (result js_ast.Expr, ok bool) {
+	ok = true
+	defer func() {
+		r := recover()
+		if _, isLexerPanic := r.(js_lexer.LexerPanic); isLexerPanic {
+			ok = false
+		} else if r != nil {
+			panic(r)
+		}
+	}()
+
+	if options.ErrorSuffix == "" {
+		options.ErrorSuffix = " in JSON"
+	}
+
+	p := &jsonParser{
+		log:                            log,
+		source:                         source,
+		tracker:                        logger.MakeLineColumnTracker(&source),
+		options:                        options,
+		lexer:                          js_lexer.NewLexerJSON(log, source, options.Flavor, options.ErrorSuffix),
+		suppressWarningsAboutWeirdCode: helpers.IsInsideNodeModules(source.KeyPath.Text),
+	}
+
+	result = p.parseExpr()
+	p.lexer.Expect(js_lexer.TEndOfFile)
+	return
+}
+
+func isValidJSON(value js_ast.Expr) bool {
+	switch e := value.Data.(type) {
+	case *js_ast.ENull, *js_ast.EBoolean, *js_ast.EString, *js_ast.ENumber:
+		return true
+
+	case *js_ast.EArray:
+		for _, item := range e.Items {
+			if !isValidJSON(item) {
+				return false
+			}
+		}
+		return true
+
+	case *js_ast.EObject:
+		for _, property := range e.Properties {
+			if property.Kind != js_ast.PropertyField || property.Flags.Has(js_ast.PropertyIsComputed) {
+				return false
+			}
+			if _, ok := property.Key.Data.(*js_ast.EString); !ok {
+				return false
+			}
+			if !isValidJSON(property.ValueOrNil) {
+				return false
+			}
+		}
+		return true
+	}
+
+	return false
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/js_parser/sourcemap_parser.go b/source/vendor/github.com/evanw/esbuild/internal/js_parser/sourcemap_parser.go
new file mode 100644
index 0000000..c83d767
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/js_parser/sourcemap_parser.go
@@ -0,0 +1,277 @@
+package js_parser
+
+import (
+	"fmt"
+	"sort"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/logger"
+	"github.com/evanw/esbuild/internal/sourcemap"
+)
+
+// Specification: https://sourcemaps.info/spec.html
+func ParseSourceMap(log logger.Log, source logger.Source) *sourcemap.SourceMap {
+	expr, ok := ParseJSON(log, source, JSONOptions{ErrorSuffix: " in source map"})
+	if !ok {
+		return nil
+	}
+
+	obj, ok := expr.Data.(*js_ast.EObject)
+	tracker := logger.MakeLineColumnTracker(&source)
+	if !ok {
+		log.AddError(&tracker, logger.Range{Loc: expr.Loc}, "Invalid source map")
+		return nil
+	}
+
+	var sources []string
+	var sourcesContent []sourcemap.SourceContent
+	var names []string
+	var mappingsRaw []uint16
+	var mappingsStart int32
+	hasVersion := false
+
+	for _, prop := range obj.Properties {
+		keyRange := source.RangeOfString(prop.Key.Loc)
+
+		switch helpers.UTF16ToString(prop.Key.Data.(*js_ast.EString).Value) {
+		case "sections":
+			log.AddID(logger.MsgID_SourceMap_SectionsInSourceMap, logger.Warning, &tracker, keyRange, "Source maps with \"sections\" are not supported")
+			return nil
+
+		case "version":
+			if value, ok := prop.ValueOrNil.Data.(*js_ast.ENumber); ok && value.Value == 3 {
+				hasVersion = true
+			}
+
+		case "mappings":
+			if value, ok := prop.ValueOrNil.Data.(*js_ast.EString); ok {
+				mappingsRaw = value.Value
+				mappingsStart = prop.ValueOrNil.Loc.Start + 1
+			}
+
+		case "sources":
+			if value, ok := prop.ValueOrNil.Data.(*js_ast.EArray); ok {
+				sources = []string{}
+				for _, item := range value.Items {
+					if element, ok := item.Data.(*js_ast.EString); ok {
+						sources = append(sources, helpers.UTF16ToString(element.Value))
+					} else {
+						sources = append(sources, "")
+					}
+				}
+			}
+
+		case "sourcesContent":
+			if value, ok := prop.ValueOrNil.Data.(*js_ast.EArray); ok {
+				sourcesContent = []sourcemap.SourceContent{}
+				for _, item := range value.Items {
+					if element, ok := item.Data.(*js_ast.EString); ok {
+						sourcesContent = append(sourcesContent, sourcemap.SourceContent{
+							Value:  element.Value,
+							Quoted: source.TextForRange(source.RangeOfString(item.Loc)),
+						})
+					} else {
+						sourcesContent = append(sourcesContent, sourcemap.SourceContent{})
+					}
+				}
+			}
+
+		case "names":
+			if value, ok := prop.ValueOrNil.Data.(*js_ast.EArray); ok {
+				names = []string{}
+				for _, item := range value.Items {
+					if element, ok := item.Data.(*js_ast.EString); ok {
+						names = append(names, helpers.UTF16ToString(element.Value))
+					} else {
+						names = append(names, "")
+					}
+				}
+			}
+		}
+	}
+
+	// Silently fail if the version was missing or incorrect
+	if !hasVersion {
+		return nil
+	}
+
+	// Silently fail if the source map is pointless (i.e. empty)
+	if len(sources) == 0 || len(mappingsRaw) == 0 {
+		return nil
+	}
+
+	var mappings mappingArray
+	mappingsLen := len(mappingsRaw)
+	sourcesLen := len(sources)
+	namesLen := len(names)
+	var generatedLine int32
+	var generatedColumn int32
+	var sourceIndex int32
+	var originalLine int32
+	var originalColumn int32
+	var originalName int32
+	current := 0
+	errorText := ""
+	errorLen := 0
+	needSort := false
+
+	// Parse the mappings
+	for current < mappingsLen {
+		// Handle a line break
+		if mappingsRaw[current] == ';' {
+			generatedLine++
+			generatedColumn = 0
+			current++
+			continue
+		}
+
+		// Read the generated column
+		generatedColumnDelta, i, ok := sourcemap.DecodeVLQUTF16(mappingsRaw[current:])
+		if !ok {
+			errorText = "Missing generated column"
+			errorLen = i
+			break
+		}
+		if generatedColumnDelta < 0 {
+			// This would mess up binary search
+			needSort = true
+		}
+		generatedColumn += generatedColumnDelta
+		if generatedColumn < 0 {
+			errorText = fmt.Sprintf("Invalid generated column value: %d", generatedColumn)
+			errorLen = i
+			break
+		}
+		current += i
+
+		// According to the specification, it's valid for a mapping to have 1,
+		// 4, or 5 variable-length fields. Having one field means there's no
+		// original location information, which is pretty useless. Just ignore
+		// those entries.
+		if current == mappingsLen {
+			break
+		}
+		switch mappingsRaw[current] {
+		case ',':
+			current++
+			continue
+		case ';':
+			continue
+		}
+
+		// Read the original source
+		sourceIndexDelta, i, ok := sourcemap.DecodeVLQUTF16(mappingsRaw[current:])
+		if !ok {
+			errorText = "Missing source index"
+			errorLen = i
+			break
+		}
+		sourceIndex += sourceIndexDelta
+		if sourceIndex < 0 || sourceIndex >= int32(sourcesLen) {
+			errorText = fmt.Sprintf("Invalid source index value: %d", sourceIndex)
+			errorLen = i
+			break
+		}
+		current += i
+
+		// Read the original line
+		originalLineDelta, i, ok := sourcemap.DecodeVLQUTF16(mappingsRaw[current:])
+		if !ok {
+			errorText = "Missing original line"
+			errorLen = i
+			break
+		}
+		originalLine += originalLineDelta
+		if originalLine < 0 {
+			errorText = fmt.Sprintf("Invalid original line value: %d", originalLine)
+			errorLen = i
+			break
+		}
+		current += i
+
+		// Read the original column
+		originalColumnDelta, i, ok := sourcemap.DecodeVLQUTF16(mappingsRaw[current:])
+		if !ok {
+			errorText = "Missing original column"
+			errorLen = i
+			break
+		}
+		originalColumn += originalColumnDelta
+		if originalColumn < 0 {
+			errorText = fmt.Sprintf("Invalid original column value: %d", originalColumn)
+			errorLen = i
+			break
+		}
+		current += i
+
+		// Read the original name
+		var optionalName ast.Index32
+		if originalNameDelta, i, ok := sourcemap.DecodeVLQUTF16(mappingsRaw[current:]); ok {
+			originalName += originalNameDelta
+			if originalName < 0 || originalName >= int32(namesLen) {
+				errorText = fmt.Sprintf("Invalid name index value: %d", originalName)
+				errorLen = i
+				break
+			}
+			optionalName = ast.MakeIndex32(uint32(originalName))
+			current += i
+		}
+
+		// Handle the next character
+		if current < mappingsLen {
+			if c := mappingsRaw[current]; c == ',' {
+				current++
+			} else if c != ';' {
+				errorText = fmt.Sprintf("Invalid character after mapping: %q",
+					helpers.UTF16ToString(mappingsRaw[current:current+1]))
+				errorLen = 1
+				break
+			}
+		}
+
+		mappings = append(mappings, sourcemap.Mapping{
+			GeneratedLine:   generatedLine,
+			GeneratedColumn: generatedColumn,
+			SourceIndex:     sourceIndex,
+			OriginalLine:    originalLine,
+			OriginalColumn:  originalColumn,
+			OriginalName:    optionalName,
+		})
+	}
+
+	if errorText != "" {
+		r := logger.Range{Loc: logger.Loc{Start: mappingsStart + int32(current)}, Len: int32(errorLen)}
+		log.AddID(logger.MsgID_SourceMap_InvalidSourceMappings, logger.Warning, &tracker, r,
+			fmt.Sprintf("Bad \"mappings\" data in source map at character %d: %s", current, errorText))
+		return nil
+	}
+
+	if needSort {
+		// If we get here, some mappings are out of order. Lines can't be out of
+		// order by construction but columns can. This is a pretty rare situation
+		// because almost all source map generators always write out mappings in
+		// order as they write the output instead of scrambling the order.
+		sort.Stable(mappings)
+	}
+
+	return &sourcemap.SourceMap{
+		Sources:        sources,
+		SourcesContent: sourcesContent,
+		Mappings:       mappings,
+		Names:          names,
+	}
+}
+
+// This type is just so we can use Go's native sort function
+type mappingArray []sourcemap.Mapping
+
+func (a mappingArray) Len() int          { return len(a) }
+func (a mappingArray) Swap(i int, j int) { a[i], a[j] = a[j], a[i] }
+
+func (a mappingArray) Less(i int, j int) bool {
+	ai := a[i]
+	aj := a[j]
+	return ai.GeneratedLine < aj.GeneratedLine || (ai.GeneratedLine == aj.GeneratedLine && ai.GeneratedColumn <= aj.GeneratedColumn)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/js_parser/ts_parser.go b/source/vendor/github.com/evanw/esbuild/internal/js_parser/ts_parser.go
new file mode 100644
index 0000000..6338fa0
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/js_parser/ts_parser.go
@@ -0,0 +1,1999 @@
+// This file contains code for parsing TypeScript syntax. The parser just skips
+// over type expressions as if they are whitespace and doesn't bother generating
+// an AST because nothing uses type information.
+
+package js_parser
+
+import (
+	"fmt"
+	"strings"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/js_lexer"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+func (p *parser) skipTypeScriptBinding() {
+	switch p.lexer.Token {
+	case js_lexer.TIdentifier, js_lexer.TThis:
+		p.lexer.Next()
+
+	case js_lexer.TOpenBracket:
+		p.lexer.Next()
+
+		// "[, , a]"
+		for p.lexer.Token == js_lexer.TComma {
+			p.lexer.Next()
+		}
+
+		// "[a, b]"
+		for p.lexer.Token != js_lexer.TCloseBracket {
+			// "[...a]"
+			if p.lexer.Token == js_lexer.TDotDotDot {
+				p.lexer.Next()
+			}
+
+			p.skipTypeScriptBinding()
+			if p.lexer.Token != js_lexer.TComma {
+				break
+			}
+			p.lexer.Next()
+		}
+
+		p.lexer.Expect(js_lexer.TCloseBracket)
+
+	case js_lexer.TOpenBrace:
+		p.lexer.Next()
+
+		for p.lexer.Token != js_lexer.TCloseBrace {
+			foundIdentifier := false
+
+			switch p.lexer.Token {
+			case js_lexer.TDotDotDot:
+				p.lexer.Next()
+
+				if p.lexer.Token != js_lexer.TIdentifier {
+					p.lexer.Unexpected()
+				}
+
+				// "{...x}"
+				foundIdentifier = true
+				p.lexer.Next()
+
+			case js_lexer.TIdentifier:
+				// "{x}"
+				// "{x: y}"
+				foundIdentifier = true
+				p.lexer.Next()
+
+				// "{1: y}"
+				// "{'x': y}"
+			case js_lexer.TStringLiteral, js_lexer.TNumericLiteral:
+				p.lexer.Next()
+
+			default:
+				if p.lexer.IsIdentifierOrKeyword() {
+					// "{if: x}"
+					p.lexer.Next()
+				} else {
+					p.lexer.Unexpected()
+				}
+			}
+
+			if p.lexer.Token == js_lexer.TColon || !foundIdentifier {
+				p.lexer.Expect(js_lexer.TColon)
+				p.skipTypeScriptBinding()
+			}
+
+			if p.lexer.Token != js_lexer.TComma {
+				break
+			}
+			p.lexer.Next()
+		}
+
+		p.lexer.Expect(js_lexer.TCloseBrace)
+
+	default:
+		p.lexer.Unexpected()
+	}
+}
+
+func (p *parser) skipTypeScriptFnArgs() {
+	p.lexer.Expect(js_lexer.TOpenParen)
+
+	for p.lexer.Token != js_lexer.TCloseParen {
+		// "(...a)"
+		if p.lexer.Token == js_lexer.TDotDotDot {
+			p.lexer.Next()
+		}
+
+		p.skipTypeScriptBinding()
+
+		// "(a?)"
+		if p.lexer.Token == js_lexer.TQuestion {
+			p.lexer.Next()
+		}
+
+		// "(a: any)"
+		if p.lexer.Token == js_lexer.TColon {
+			p.lexer.Next()
+			p.skipTypeScriptType(js_ast.LLowest)
+		}
+
+		// "(a, b)"
+		if p.lexer.Token != js_lexer.TComma {
+			break
+		}
+		p.lexer.Next()
+	}
+
+	p.lexer.Expect(js_lexer.TCloseParen)
+}
+
+// This is a spot where the TypeScript grammar is highly ambiguous. Here are
+// some cases that are valid:
+//
+//	let x = (y: any): (() => {}) => { };
+//	let x = (y: any): () => {} => { };
+//	let x = (y: any): (y) => {} => { };
+//	let x = (y: any): (y[]) => {};
+//	let x = (y: any): (a | b) => {};
+//
+// Here are some cases that aren't valid:
+//
+//	let x = (y: any): (y) => {};
+//	let x = (y: any): (y) => {return 0};
+//	let x = (y: any): asserts y is (y) => {};
+func (p *parser) skipTypeScriptParenOrFnType() {
+	if p.trySkipTypeScriptArrowArgsWithBacktracking() {
+		p.skipTypeScriptReturnType()
+	} else {
+		p.lexer.Expect(js_lexer.TOpenParen)
+		p.skipTypeScriptType(js_ast.LLowest)
+		p.lexer.Expect(js_lexer.TCloseParen)
+	}
+}
+
+func (p *parser) skipTypeScriptReturnType() {
+	p.skipTypeScriptTypeWithFlags(js_ast.LLowest, isReturnTypeFlag)
+}
+
+func (p *parser) skipTypeScriptType(level js_ast.L) {
+	p.skipTypeScriptTypeWithFlags(level, 0)
+}
+
+type skipTypeFlags uint8
+
+const (
+	isReturnTypeFlag skipTypeFlags = 1 << iota
+	isIndexSignatureFlag
+	allowTupleLabelsFlag
+	disallowConditionalTypesFlag
+)
+
+func (flags skipTypeFlags) has(flag skipTypeFlags) bool {
+	return (flags & flag) != 0
+}
+
+type tsTypeIdentifierKind uint8
+
+const (
+	tsTypeIdentifierNormal tsTypeIdentifierKind = iota
+	tsTypeIdentifierUnique
+	tsTypeIdentifierAbstract
+	tsTypeIdentifierAsserts
+	tsTypeIdentifierPrefix
+	tsTypeIdentifierPrimitive
+	tsTypeIdentifierInfer
+)
+
+// Use a map to improve lookup speed
+var tsTypeIdentifierMap = map[string]tsTypeIdentifierKind{
+	"unique":   tsTypeIdentifierUnique,
+	"abstract": tsTypeIdentifierAbstract,
+	"asserts":  tsTypeIdentifierAsserts,
+
+	"keyof":    tsTypeIdentifierPrefix,
+	"readonly": tsTypeIdentifierPrefix,
+
+	"any":       tsTypeIdentifierPrimitive,
+	"never":     tsTypeIdentifierPrimitive,
+	"unknown":   tsTypeIdentifierPrimitive,
+	"undefined": tsTypeIdentifierPrimitive,
+	"object":    tsTypeIdentifierPrimitive,
+	"number":    tsTypeIdentifierPrimitive,
+	"string":    tsTypeIdentifierPrimitive,
+	"boolean":   tsTypeIdentifierPrimitive,
+	"bigint":    tsTypeIdentifierPrimitive,
+	"symbol":    tsTypeIdentifierPrimitive,
+
+	"infer": tsTypeIdentifierInfer,
+}
+
+func (p *parser) skipTypeScriptTypeWithFlags(level js_ast.L, flags skipTypeFlags) {
+loop:
+	for {
+		switch p.lexer.Token {
+		case js_lexer.TNumericLiteral, js_lexer.TBigIntegerLiteral, js_lexer.TStringLiteral,
+			js_lexer.TNoSubstitutionTemplateLiteral, js_lexer.TTrue, js_lexer.TFalse,
+			js_lexer.TNull, js_lexer.TVoid:
+			p.lexer.Next()
+
+		case js_lexer.TConst:
+			r := p.lexer.Range()
+			p.lexer.Next()
+
+			// "[const: number]"
+			if flags.has(allowTupleLabelsFlag) && p.lexer.Token == js_lexer.TColon {
+				p.log.AddError(&p.tracker, r, "Unexpected \"const\"")
+			}
+
+		case js_lexer.TThis:
+			p.lexer.Next()
+
+			// "function check(): this is boolean"
+			if p.lexer.IsContextualKeyword("is") && !p.lexer.HasNewlineBefore {
+				p.lexer.Next()
+				p.skipTypeScriptType(js_ast.LLowest)
+				return
+			}
+
+		case js_lexer.TMinus:
+			// "-123"
+			// "-123n"
+			p.lexer.Next()
+			if p.lexer.Token == js_lexer.TBigIntegerLiteral {
+				p.lexer.Next()
+			} else {
+				p.lexer.Expect(js_lexer.TNumericLiteral)
+			}
+
+		case js_lexer.TAmpersand:
+		case js_lexer.TBar:
+			// Support things like "type Foo = | A | B" and "type Foo = & A & B"
+			p.lexer.Next()
+			continue
+
+		case js_lexer.TImport:
+			// "import('fs')"
+			p.lexer.Next()
+
+			// "[import: number]"
+			if flags.has(allowTupleLabelsFlag) && p.lexer.Token == js_lexer.TColon {
+				return
+			}
+
+			p.lexer.Expect(js_lexer.TOpenParen)
+			p.lexer.Expect(js_lexer.TStringLiteral)
+
+			// "import('./foo.json', { assert: { type: 'json' } })"
+			if p.lexer.Token == js_lexer.TComma {
+				p.lexer.Next()
+				p.skipTypeScriptObjectType()
+
+				// "import('./foo.json', { assert: { type: 'json' } }, )"
+				if p.lexer.Token == js_lexer.TComma {
+					p.lexer.Next()
+				}
+			}
+
+			p.lexer.Expect(js_lexer.TCloseParen)
+
+		case js_lexer.TNew:
+			// "new () => Foo"
+			// "new <T>() => Foo<T>"
+			p.lexer.Next()
+
+			// "[new: number]"
+			if flags.has(allowTupleLabelsFlag) && p.lexer.Token == js_lexer.TColon {
+				return
+			}
+
+			p.skipTypeScriptTypeParameters(allowConstModifier)
+			p.skipTypeScriptParenOrFnType()
+
+		case js_lexer.TLessThan:
+			// "<T>() => Foo<T>"
+			p.skipTypeScriptTypeParameters(allowConstModifier)
+			p.skipTypeScriptParenOrFnType()
+
+		case js_lexer.TOpenParen:
+			// "(number | string)"
+			p.skipTypeScriptParenOrFnType()
+
+		case js_lexer.TIdentifier:
+			kind := tsTypeIdentifierMap[p.lexer.Identifier.String]
+			checkTypeParameters := true
+
+			switch kind {
+			case tsTypeIdentifierPrefix:
+				p.lexer.Next()
+
+				// Valid:
+				//   "[keyof: string]"
+				//   "{[keyof: string]: number}"
+				//   "{[keyof in string]: number}"
+				//
+				// Invalid:
+				//   "A extends B ? keyof : string"
+				//
+				if (p.lexer.Token != js_lexer.TColon && p.lexer.Token != js_lexer.TIn) || (!flags.has(isIndexSignatureFlag) && !flags.has(allowTupleLabelsFlag)) {
+					p.skipTypeScriptType(js_ast.LPrefix)
+				}
+				break loop
+
+			case tsTypeIdentifierInfer:
+				p.lexer.Next()
+
+				// "type Foo = Bar extends [infer T] ? T : null"
+				// "type Foo = Bar extends [infer T extends string] ? T : null"
+				// "type Foo = Bar extends [infer T extends string ? infer T : never] ? T : null"
+				// "type Foo = { [infer in Bar]: number }"
+				if (p.lexer.Token != js_lexer.TColon && p.lexer.Token != js_lexer.TIn) || (!flags.has(isIndexSignatureFlag) && !flags.has(allowTupleLabelsFlag)) {
+					p.lexer.Expect(js_lexer.TIdentifier)
+					if p.lexer.Token == js_lexer.TExtends {
+						p.trySkipTypeScriptConstraintOfInferTypeWithBacktracking(flags)
+					}
+				}
+				break loop
+
+			case tsTypeIdentifierUnique:
+				p.lexer.Next()
+
+				// "let foo: unique symbol"
+				if p.lexer.IsContextualKeyword("symbol") {
+					p.lexer.Next()
+					break loop
+				}
+
+			case tsTypeIdentifierAbstract:
+				p.lexer.Next()
+
+				// "let foo: abstract new () => {}" added in TypeScript 4.2
+				if p.lexer.Token == js_lexer.TNew {
+					continue
+				}
+
+			case tsTypeIdentifierAsserts:
+				p.lexer.Next()
+
+				// "function assert(x: boolean): asserts x"
+				// "function assert(x: boolean): asserts x is boolean"
+				if flags.has(isReturnTypeFlag) && !p.lexer.HasNewlineBefore && (p.lexer.Token == js_lexer.TIdentifier || p.lexer.Token == js_lexer.TThis) {
+					p.lexer.Next()
+				}
+
+			case tsTypeIdentifierPrimitive:
+				p.lexer.Next()
+				checkTypeParameters = false
+
+			default:
+				p.lexer.Next()
+			}
+
+			// "function assert(x: any): x is boolean"
+			if p.lexer.IsContextualKeyword("is") && !p.lexer.HasNewlineBefore {
+				p.lexer.Next()
+				p.skipTypeScriptType(js_ast.LLowest)
+				return
+			}
+
+			// "let foo: any \n <number>foo" must not become a single type
+			if checkTypeParameters && !p.lexer.HasNewlineBefore {
+				p.skipTypeScriptTypeArguments(skipTypeScriptTypeArgumentsOpts{})
+			}
+
+		case js_lexer.TTypeof:
+			p.lexer.Next()
+
+			// "[typeof: number]"
+			if flags.has(allowTupleLabelsFlag) && p.lexer.Token == js_lexer.TColon {
+				return
+			}
+
+			if p.lexer.Token == js_lexer.TImport {
+				// "typeof import('fs')"
+				continue
+			} else {
+				// "typeof x"
+				if !p.lexer.IsIdentifierOrKeyword() {
+					p.lexer.Expected(js_lexer.TIdentifier)
+				}
+				p.lexer.Next()
+
+				// "typeof x.y"
+				// "typeof x.#y"
+				for p.lexer.Token == js_lexer.TDot {
+					p.lexer.Next()
+					if !p.lexer.IsIdentifierOrKeyword() && p.lexer.Token != js_lexer.TPrivateIdentifier {
+						p.lexer.Expected(js_lexer.TIdentifier)
+					}
+					p.lexer.Next()
+				}
+
+				if !p.lexer.HasNewlineBefore {
+					p.skipTypeScriptTypeArguments(skipTypeScriptTypeArgumentsOpts{})
+				}
+			}
+
+		case js_lexer.TOpenBracket:
+			// "[number, string]"
+			// "[first: number, second: string]"
+			p.lexer.Next()
+			for p.lexer.Token != js_lexer.TCloseBracket {
+				if p.lexer.Token == js_lexer.TDotDotDot {
+					p.lexer.Next()
+				}
+				p.skipTypeScriptTypeWithFlags(js_ast.LLowest, allowTupleLabelsFlag)
+				if p.lexer.Token == js_lexer.TQuestion {
+					p.lexer.Next()
+				}
+				if p.lexer.Token == js_lexer.TColon {
+					p.lexer.Next()
+					p.skipTypeScriptType(js_ast.LLowest)
+				}
+				if p.lexer.Token != js_lexer.TComma {
+					break
+				}
+				p.lexer.Next()
+			}
+			p.lexer.Expect(js_lexer.TCloseBracket)
+
+		case js_lexer.TOpenBrace:
+			p.skipTypeScriptObjectType()
+
+		case js_lexer.TTemplateHead:
+			// "`${'a' | 'b'}-${'c' | 'd'}`"
+			for {
+				p.lexer.Next()
+				p.skipTypeScriptType(js_ast.LLowest)
+				p.lexer.RescanCloseBraceAsTemplateToken()
+				if p.lexer.Token == js_lexer.TTemplateTail {
+					p.lexer.Next()
+					break
+				}
+			}
+
+		default:
+			// "[function: number]"
+			if flags.has(allowTupleLabelsFlag) && p.lexer.IsIdentifierOrKeyword() {
+				if p.lexer.Token != js_lexer.TFunction {
+					p.log.AddError(&p.tracker, p.lexer.Range(), fmt.Sprintf("Unexpected %q", p.lexer.Raw()))
+				}
+				p.lexer.Next()
+				if p.lexer.Token != js_lexer.TColon {
+					p.lexer.Expect(js_lexer.TColon)
+				}
+				return
+			}
+
+			p.lexer.Unexpected()
+		}
+		break
+	}
+
+	for {
+		switch p.lexer.Token {
+		case js_lexer.TBar:
+			if level >= js_ast.LBitwiseOr {
+				return
+			}
+			p.lexer.Next()
+			p.skipTypeScriptTypeWithFlags(js_ast.LBitwiseOr, flags)
+
+		case js_lexer.TAmpersand:
+			if level >= js_ast.LBitwiseAnd {
+				return
+			}
+			p.lexer.Next()
+			p.skipTypeScriptTypeWithFlags(js_ast.LBitwiseAnd, flags)
+
+		case js_lexer.TExclamation:
+			// A postfix "!" is allowed in JSDoc types in TypeScript, which are only
+			// present in comments. While it's not valid in a non-comment position,
+			// it's still parsed and turned into a soft error by the TypeScript
+			// compiler. It turns out parsing this is important for correctness for
+			// "as" casts because the "!" token must still be consumed.
+			if p.lexer.HasNewlineBefore {
+				return
+			}
+			p.lexer.Next()
+
+		case js_lexer.TDot:
+			p.lexer.Next()
+			if !p.lexer.IsIdentifierOrKeyword() {
+				p.lexer.Expect(js_lexer.TIdentifier)
+			}
+			p.lexer.Next()
+
+			// "{ <A extends B>(): c.d \n <E extends F>(): g.h }" must not become a single type
+			if !p.lexer.HasNewlineBefore {
+				p.skipTypeScriptTypeArguments(skipTypeScriptTypeArgumentsOpts{})
+			}
+
+		case js_lexer.TOpenBracket:
+			// "{ ['x']: string \n ['y']: string }" must not become a single type
+			if p.lexer.HasNewlineBefore {
+				return
+			}
+			p.lexer.Next()
+			if p.lexer.Token != js_lexer.TCloseBracket {
+				p.skipTypeScriptType(js_ast.LLowest)
+			}
+			p.lexer.Expect(js_lexer.TCloseBracket)
+
+		case js_lexer.TExtends:
+			// "{ x: number \n extends: boolean }" must not become a single type
+			if p.lexer.HasNewlineBefore || flags.has(disallowConditionalTypesFlag) {
+				return
+			}
+			p.lexer.Next()
+
+			// The type following "extends" is not permitted to be another conditional type
+			p.skipTypeScriptTypeWithFlags(js_ast.LLowest, disallowConditionalTypesFlag)
+			p.lexer.Expect(js_lexer.TQuestion)
+			p.skipTypeScriptType(js_ast.LLowest)
+			p.lexer.Expect(js_lexer.TColon)
+			p.skipTypeScriptType(js_ast.LLowest)
+
+		default:
+			return
+		}
+	}
+}
+
+func (p *parser) skipTypeScriptObjectType() {
+	p.lexer.Expect(js_lexer.TOpenBrace)
+
+	for p.lexer.Token != js_lexer.TCloseBrace {
+		// "{ -readonly [K in keyof T]: T[K] }"
+		// "{ +readonly [K in keyof T]: T[K] }"
+		if p.lexer.Token == js_lexer.TPlus || p.lexer.Token == js_lexer.TMinus {
+			p.lexer.Next()
+		}
+
+		// Skip over modifiers and the property identifier
+		foundKey := false
+		for p.lexer.IsIdentifierOrKeyword() ||
+			p.lexer.Token == js_lexer.TStringLiteral ||
+			p.lexer.Token == js_lexer.TNumericLiteral {
+			p.lexer.Next()
+			foundKey = true
+		}
+
+		if p.lexer.Token == js_lexer.TOpenBracket {
+			// Index signature or computed property
+			p.lexer.Next()
+			p.skipTypeScriptTypeWithFlags(js_ast.LLowest, isIndexSignatureFlag)
+
+			// "{ [key: string]: number }"
+			// "{ readonly [K in keyof T]: T[K] }"
+			if p.lexer.Token == js_lexer.TColon {
+				p.lexer.Next()
+				p.skipTypeScriptType(js_ast.LLowest)
+			} else if p.lexer.Token == js_lexer.TIn {
+				p.lexer.Next()
+				p.skipTypeScriptType(js_ast.LLowest)
+				if p.lexer.IsContextualKeyword("as") {
+					// "{ [K in keyof T as `get-${K}`]: T[K] }"
+					p.lexer.Next()
+					p.skipTypeScriptType(js_ast.LLowest)
+				}
+			}
+
+			p.lexer.Expect(js_lexer.TCloseBracket)
+
+			// "{ [K in keyof T]+?: T[K] }"
+			// "{ [K in keyof T]-?: T[K] }"
+			if p.lexer.Token == js_lexer.TPlus || p.lexer.Token == js_lexer.TMinus {
+				p.lexer.Next()
+			}
+
+			foundKey = true
+		}
+
+		// "?" indicates an optional property
+		// "!" indicates an initialization assertion
+		if foundKey && (p.lexer.Token == js_lexer.TQuestion || p.lexer.Token == js_lexer.TExclamation) {
+			p.lexer.Next()
+		}
+
+		// Type parameters come right after the optional mark
+		p.skipTypeScriptTypeParameters(allowConstModifier)
+
+		switch p.lexer.Token {
+		case js_lexer.TColon:
+			// Regular property
+			if !foundKey {
+				p.lexer.Expect(js_lexer.TIdentifier)
+			}
+			p.lexer.Next()
+			p.skipTypeScriptType(js_ast.LLowest)
+
+		case js_lexer.TOpenParen:
+			// Method signature
+			p.skipTypeScriptFnArgs()
+			if p.lexer.Token == js_lexer.TColon {
+				p.lexer.Next()
+				p.skipTypeScriptReturnType()
+			}
+
+		default:
+			if !foundKey {
+				p.lexer.Unexpected()
+			}
+		}
+
+		switch p.lexer.Token {
+		case js_lexer.TCloseBrace:
+
+		case js_lexer.TComma, js_lexer.TSemicolon:
+			p.lexer.Next()
+
+		default:
+			if !p.lexer.HasNewlineBefore {
+				p.lexer.Unexpected()
+			}
+		}
+	}
+
+	p.lexer.Expect(js_lexer.TCloseBrace)
+}
+
+type typeParameterFlags uint8
+
+const (
+	// TypeScript 4.7
+	allowInOutVarianceAnnotations typeParameterFlags = 1 << iota
+
+	// TypeScript 5.0
+	allowConstModifier
+
+	// Allow "<>" without any type parameters
+	allowEmptyTypeParameters
+)
+
+type skipTypeScriptTypeParametersResult uint8
+
+const (
+	didNotSkipAnything skipTypeScriptTypeParametersResult = iota
+	couldBeTypeCast
+	definitelyTypeParameters
+)
+
+// This is the type parameter declarations that go with other symbol
+// declarations (class, function, type, etc.)
+func (p *parser) skipTypeScriptTypeParameters(flags typeParameterFlags) skipTypeScriptTypeParametersResult {
+	if p.lexer.Token != js_lexer.TLessThan {
+		return didNotSkipAnything
+	}
+
+	p.lexer.Next()
+	result := couldBeTypeCast
+
+	if (flags&allowEmptyTypeParameters) != 0 && p.lexer.Token == js_lexer.TGreaterThan {
+		p.lexer.Next()
+		return definitelyTypeParameters
+	}
+
+	for {
+		hasIn := false
+		hasOut := false
+		expectIdentifier := true
+		invalidModifierRange := logger.Range{}
+
+		// Scan over a sequence of "in" and "out" modifiers (a.k.a. optional
+		// variance annotations) as well as "const" modifiers
+		for {
+			if p.lexer.Token == js_lexer.TConst {
+				if invalidModifierRange.Len == 0 && (flags&allowConstModifier) == 0 {
+					// Valid:
+					//   "class Foo<const T> {}"
+					// Invalid:
+					//   "interface Foo<const T> {}"
+					invalidModifierRange = p.lexer.Range()
+				}
+				result = definitelyTypeParameters
+				p.lexer.Next()
+				expectIdentifier = true
+				continue
+			}
+
+			if p.lexer.Token == js_lexer.TIn {
+				if invalidModifierRange.Len == 0 && ((flags&allowInOutVarianceAnnotations) == 0 || hasIn || hasOut) {
+					// Valid:
+					//   "type Foo<in T> = T"
+					// Invalid:
+					//   "type Foo<in in T> = T"
+					//   "type Foo<out in T> = T"
+					invalidModifierRange = p.lexer.Range()
+				}
+				p.lexer.Next()
+				hasIn = true
+				expectIdentifier = true
+				continue
+			}
+
+			if p.lexer.IsContextualKeyword("out") {
+				r := p.lexer.Range()
+				if invalidModifierRange.Len == 0 && (flags&allowInOutVarianceAnnotations) == 0 {
+					invalidModifierRange = r
+				}
+				p.lexer.Next()
+				if invalidModifierRange.Len == 0 && hasOut && (p.lexer.Token == js_lexer.TIn || p.lexer.Token == js_lexer.TIdentifier) {
+					// Valid:
+					//   "type Foo<out T> = T"
+					//   "type Foo<out out> = T"
+					//   "type Foo<out out, T> = T"
+					//   "type Foo<out out = T> = T"
+					//   "type Foo<out out extends T> = T"
+					// Invalid:
+					//   "type Foo<out out in T> = T"
+					//   "type Foo<out out T> = T"
+					invalidModifierRange = r
+				}
+				hasOut = true
+				expectIdentifier = false
+				continue
+			}
+
+			break
+		}
+
+		// Only report an error for the first invalid modifier
+		if invalidModifierRange.Len > 0 {
+			p.log.AddError(&p.tracker, invalidModifierRange, fmt.Sprintf(
+				"The modifier %q is not valid here:", p.source.TextForRange(invalidModifierRange)))
+		}
+
+		// expectIdentifier => Mandatory identifier (e.g. after "type Foo <in ___")
+		// !expectIdentifier => Optional identifier (e.g. after "type Foo <out ___" since "out" may be the identifier)
+		if expectIdentifier || p.lexer.Token == js_lexer.TIdentifier {
+			p.lexer.Expect(js_lexer.TIdentifier)
+		}
+
+		// "class Foo<T extends number> {}"
+		if p.lexer.Token == js_lexer.TExtends {
+			result = definitelyTypeParameters
+			p.lexer.Next()
+			p.skipTypeScriptType(js_ast.LLowest)
+		}
+
+		// "class Foo<T = void> {}"
+		if p.lexer.Token == js_lexer.TEquals {
+			result = definitelyTypeParameters
+			p.lexer.Next()
+			p.skipTypeScriptType(js_ast.LLowest)
+		}
+
+		if p.lexer.Token != js_lexer.TComma {
+			break
+		}
+		p.lexer.Next()
+		if p.lexer.Token == js_lexer.TGreaterThan {
+			result = definitelyTypeParameters
+			break
+		}
+	}
+
+	p.lexer.ExpectGreaterThan(false /* isInsideJSXElement */)
+	return result
+}
+
+type skipTypeScriptTypeArgumentsOpts struct {
+	isInsideJSXElement               bool
+	isParseTypeArgumentsInExpression bool
+}
+
+func (p *parser) skipTypeScriptTypeArguments(opts skipTypeScriptTypeArgumentsOpts) bool {
+	switch p.lexer.Token {
+	case js_lexer.TLessThan, js_lexer.TLessThanEquals,
+		js_lexer.TLessThanLessThan, js_lexer.TLessThanLessThanEquals:
+	default:
+		return false
+	}
+
+	p.lexer.ExpectLessThan(false /* isInsideJSXElement */)
+
+	for {
+		p.skipTypeScriptType(js_ast.LLowest)
+		if p.lexer.Token != js_lexer.TComma {
+			break
+		}
+		p.lexer.Next()
+	}
+
+	// This type argument list must end with a ">"
+	if !opts.isParseTypeArgumentsInExpression {
+		// Normally TypeScript allows any token starting with ">". For example,
+		// "Array<Array<number>>()" is a type argument list even though there's a
+		// ">>" token, because ">>" starts with ">".
+		p.lexer.ExpectGreaterThan(opts.isInsideJSXElement)
+	} else {
+		// However, if we're emulating the TypeScript compiler's function called
+		// "parseTypeArgumentsInExpression" function, then we must only allow the
+		// ">" token itself. For example, "x < y >= z" is not a type argument list.
+		//
+		// This doesn't detect ">>" in "Array<Array<number>>()" because the inner
+		// type argument list isn't a call to "parseTypeArgumentsInExpression"
+		// because it's within a type context, not an expression context. So the
+		// token that we see here is ">" in that case because the first ">" has
+		// already been stripped off of the ">>" by the inner call.
+		if opts.isInsideJSXElement {
+			p.lexer.ExpectInsideJSXElement(js_lexer.TGreaterThan)
+		} else {
+			p.lexer.Expect(js_lexer.TGreaterThan)
+		}
+	}
+	return true
+}
+
+func (p *parser) trySkipTypeArgumentsInExpressionWithBacktracking() bool {
+	oldLexer := p.lexer
+	p.lexer.IsLogDisabled = true
+
+	// Implement backtracking by restoring the lexer's memory to its original state
+	defer func() {
+		r := recover()
+		if _, isLexerPanic := r.(js_lexer.LexerPanic); isLexerPanic {
+			p.lexer = oldLexer
+		} else if r != nil {
+			panic(r)
+		}
+	}()
+
+	if p.skipTypeScriptTypeArguments(skipTypeScriptTypeArgumentsOpts{isParseTypeArgumentsInExpression: true}) {
+		// Check the token after the type argument list and backtrack if it's invalid
+		if !p.tsCanFollowTypeArgumentsInExpression() {
+			p.lexer.Unexpected()
+		}
+	}
+
+	// Restore the log disabled flag. Note that we can't just set it back to false
+	// because it may have been true to start with.
+	p.lexer.IsLogDisabled = oldLexer.IsLogDisabled
+	return true
+}
+
+func (p *parser) trySkipTypeScriptTypeParametersThenOpenParenWithBacktracking() skipTypeScriptTypeParametersResult {
+	oldLexer := p.lexer
+	p.lexer.IsLogDisabled = true
+
+	// Implement backtracking by restoring the lexer's memory to its original state
+	defer func() {
+		r := recover()
+		if _, isLexerPanic := r.(js_lexer.LexerPanic); isLexerPanic {
+			p.lexer = oldLexer
+		} else if r != nil {
+			panic(r)
+		}
+	}()
+
+	result := p.skipTypeScriptTypeParameters(allowConstModifier)
+	if p.lexer.Token != js_lexer.TOpenParen {
+		p.lexer.Unexpected()
+	}
+
+	// Restore the log disabled flag. Note that we can't just set it back to false
+	// because it may have been true to start with.
+	p.lexer.IsLogDisabled = oldLexer.IsLogDisabled
+	return result
+}
+
+func (p *parser) trySkipTypeScriptArrowReturnTypeWithBacktracking() bool {
+	oldLexer := p.lexer
+	p.lexer.IsLogDisabled = true
+
+	// Implement backtracking by restoring the lexer's memory to its original state
+	defer func() {
+		r := recover()
+		if _, isLexerPanic := r.(js_lexer.LexerPanic); isLexerPanic {
+			p.lexer = oldLexer
+		} else if r != nil {
+			panic(r)
+		}
+	}()
+
+	p.lexer.Expect(js_lexer.TColon)
+	p.skipTypeScriptReturnType()
+
+	// Check the token after this and backtrack if it's the wrong one
+	if p.lexer.Token != js_lexer.TEqualsGreaterThan {
+		p.lexer.Unexpected()
+	}
+
+	// Restore the log disabled flag. Note that we can't just set it back to false
+	// because it may have been true to start with.
+	p.lexer.IsLogDisabled = oldLexer.IsLogDisabled
+	return true
+}
+
+func (p *parser) trySkipTypeScriptArrowArgsWithBacktracking() bool {
+	oldLexer := p.lexer
+	p.lexer.IsLogDisabled = true
+
+	// Implement backtracking by restoring the lexer's memory to its original state
+	defer func() {
+		r := recover()
+		if _, isLexerPanic := r.(js_lexer.LexerPanic); isLexerPanic {
+			p.lexer = oldLexer
+		} else if r != nil {
+			panic(r)
+		}
+	}()
+
+	p.skipTypeScriptFnArgs()
+	p.lexer.Expect(js_lexer.TEqualsGreaterThan)
+
+	// Restore the log disabled flag. Note that we can't just set it back to false
+	// because it may have been true to start with.
+	p.lexer.IsLogDisabled = oldLexer.IsLogDisabled
+	return true
+}
+
+func (p *parser) trySkipTypeScriptConstraintOfInferTypeWithBacktracking(flags skipTypeFlags) bool {
+	oldLexer := p.lexer
+	p.lexer.IsLogDisabled = true
+
+	// Implement backtracking by restoring the lexer's memory to its original state
+	defer func() {
+		r := recover()
+		if _, isLexerPanic := r.(js_lexer.LexerPanic); isLexerPanic {
+			p.lexer = oldLexer
+		} else if r != nil {
+			panic(r)
+		}
+	}()
+
+	p.lexer.Expect(js_lexer.TExtends)
+	p.skipTypeScriptTypeWithFlags(js_ast.LPrefix, disallowConditionalTypesFlag)
+	if !flags.has(disallowConditionalTypesFlag) && p.lexer.Token == js_lexer.TQuestion {
+		p.lexer.Unexpected()
+	}
+
+	// Restore the log disabled flag. Note that we can't just set it back to false
+	// because it may have been true to start with.
+	p.lexer.IsLogDisabled = oldLexer.IsLogDisabled
+	return true
+}
+
+// Returns true if the current less-than token is considered to be an arrow
+// function under TypeScript's rules for files containing JSX syntax
+func (p *parser) isTSArrowFnJSX() (isTSArrowFn bool) {
+	oldLexer := p.lexer
+	p.lexer.Next()
+
+	// Look ahead to see if this should be an arrow function instead
+	if p.lexer.Token == js_lexer.TConst {
+		p.lexer.Next()
+	}
+	if p.lexer.Token == js_lexer.TIdentifier {
+		p.lexer.Next()
+		if p.lexer.Token == js_lexer.TComma || p.lexer.Token == js_lexer.TEquals {
+			isTSArrowFn = true
+		} else if p.lexer.Token == js_lexer.TExtends {
+			p.lexer.Next()
+			isTSArrowFn = p.lexer.Token != js_lexer.TEquals && p.lexer.Token != js_lexer.TGreaterThan && p.lexer.Token != js_lexer.TSlash
+		}
+	}
+
+	// Restore the lexer
+	p.lexer = oldLexer
+	return
+}
+
+// This function is taken from the official TypeScript compiler source code:
+// https://github.com/microsoft/TypeScript/blob/master/src/compiler/parser.ts
+//
+// This function is pretty inefficient as written, and could be collapsed into
+// a single switch statement. But that would make it harder to keep this in
+// sync with the TypeScript compiler's source code, so we keep doing it the
+// slow way.
+func (p *parser) tsCanFollowTypeArgumentsInExpression() bool {
+	switch p.lexer.Token {
+	case
+		// These tokens can follow a type argument list in a call expression.
+		js_lexer.TOpenParen,                     // foo<x>(
+		js_lexer.TNoSubstitutionTemplateLiteral, // foo<T> `...`
+		js_lexer.TTemplateHead:                  // foo<T> `...${100}...`
+		return true
+
+	// A type argument list followed by `<` never makes sense, and a type argument list followed
+	// by `>` is ambiguous with a (re-scanned) `>>` operator, so we disqualify both. Also, in
+	// this context, `+` and `-` are unary operators, not binary operators.
+	case js_lexer.TLessThan,
+		js_lexer.TGreaterThan,
+		js_lexer.TPlus,
+		js_lexer.TMinus,
+		// TypeScript always sees "TGreaterThan" instead of these tokens since
+		// their scanner works a little differently than our lexer. So since
+		// "TGreaterThan" is forbidden above, we also forbid these too.
+		js_lexer.TGreaterThanEquals,
+		js_lexer.TGreaterThanGreaterThan,
+		js_lexer.TGreaterThanGreaterThanEquals,
+		js_lexer.TGreaterThanGreaterThanGreaterThan,
+		js_lexer.TGreaterThanGreaterThanGreaterThanEquals:
+		return false
+	}
+
+	// We favor the type argument list interpretation when it is immediately followed by
+	// a line break, a binary operator, or something that can't start an expression.
+	return p.lexer.HasNewlineBefore || p.tsIsBinaryOperator() || !p.tsIsStartOfExpression()
+}
+
+// This function is taken from the official TypeScript compiler source code:
+// https://github.com/microsoft/TypeScript/blob/master/src/compiler/parser.ts
+func (p *parser) tsIsBinaryOperator() bool {
+	switch p.lexer.Token {
+	case js_lexer.TIn:
+		return p.allowIn
+
+	case
+		js_lexer.TQuestionQuestion,
+		js_lexer.TBarBar,
+		js_lexer.TAmpersandAmpersand,
+		js_lexer.TBar,
+		js_lexer.TCaret,
+		js_lexer.TAmpersand,
+		js_lexer.TEqualsEquals,
+		js_lexer.TExclamationEquals,
+		js_lexer.TEqualsEqualsEquals,
+		js_lexer.TExclamationEqualsEquals,
+		js_lexer.TLessThan,
+		js_lexer.TGreaterThan,
+		js_lexer.TLessThanEquals,
+		js_lexer.TGreaterThanEquals,
+		js_lexer.TInstanceof,
+		js_lexer.TLessThanLessThan,
+		js_lexer.TGreaterThanGreaterThan,
+		js_lexer.TGreaterThanGreaterThanGreaterThan,
+		js_lexer.TPlus,
+		js_lexer.TMinus,
+		js_lexer.TAsterisk,
+		js_lexer.TSlash,
+		js_lexer.TPercent,
+		js_lexer.TAsteriskAsterisk:
+		return true
+
+	case js_lexer.TIdentifier:
+		if p.lexer.IsContextualKeyword("as") || p.lexer.IsContextualKeyword("satisfies") {
+			return true
+		}
+	}
+
+	return false
+}
+
+// This function is taken from the official TypeScript compiler source code:
+// https://github.com/microsoft/TypeScript/blob/master/src/compiler/parser.ts
+func (p *parser) tsIsStartOfExpression() bool {
+	if p.tsIsStartOfLeftHandSideExpression() {
+		return true
+	}
+
+	switch p.lexer.Token {
+	case
+		js_lexer.TPlus,
+		js_lexer.TMinus,
+		js_lexer.TTilde,
+		js_lexer.TExclamation,
+		js_lexer.TDelete,
+		js_lexer.TTypeof,
+		js_lexer.TVoid,
+		js_lexer.TPlusPlus,
+		js_lexer.TMinusMinus,
+		js_lexer.TLessThan,
+		js_lexer.TPrivateIdentifier,
+		js_lexer.TAt:
+		return true
+
+	default:
+		if p.lexer.Token == js_lexer.TIdentifier && (p.lexer.Identifier.String == "await" || p.lexer.Identifier.String == "yield") {
+			// Yield/await always starts an expression.  Either it is an identifier (in which case
+			// it is definitely an expression).  Or it's a keyword (either because we're in
+			// a generator or async function, or in strict mode (or both)) and it started a yield or await expression.
+			return true
+		}
+
+		// Error tolerance.  If we see the start of some binary operator, we consider
+		// that the start of an expression.  That way we'll parse out a missing identifier,
+		// give a good message about an identifier being missing, and then consume the
+		// rest of the binary expression.
+		if p.tsIsBinaryOperator() {
+			return true
+		}
+
+		return p.tsIsIdentifier()
+	}
+}
+
+// This function is taken from the official TypeScript compiler source code:
+// https://github.com/microsoft/TypeScript/blob/master/src/compiler/parser.ts
+func (p *parser) tsIsStartOfLeftHandSideExpression() bool {
+	switch p.lexer.Token {
+	case
+		js_lexer.TThis,
+		js_lexer.TSuper,
+		js_lexer.TNull,
+		js_lexer.TTrue,
+		js_lexer.TFalse,
+		js_lexer.TNumericLiteral,
+		js_lexer.TBigIntegerLiteral,
+		js_lexer.TStringLiteral,
+		js_lexer.TNoSubstitutionTemplateLiteral,
+		js_lexer.TTemplateHead,
+		js_lexer.TOpenParen,
+		js_lexer.TOpenBracket,
+		js_lexer.TOpenBrace,
+		js_lexer.TFunction,
+		js_lexer.TClass,
+		js_lexer.TNew,
+		js_lexer.TSlash,
+		js_lexer.TSlashEquals,
+		js_lexer.TIdentifier:
+		return true
+
+	case js_lexer.TImport:
+		return p.tsLookAheadNextTokenIsOpenParenOrLessThanOrDot()
+
+	default:
+		return p.tsIsIdentifier()
+	}
+}
+
+// This function is taken from the official TypeScript compiler source code:
+// https://github.com/microsoft/TypeScript/blob/master/src/compiler/parser.ts
+func (p *parser) tsLookAheadNextTokenIsOpenParenOrLessThanOrDot() (result bool) {
+	oldLexer := p.lexer
+	p.lexer.Next()
+
+	result = p.lexer.Token == js_lexer.TOpenParen ||
+		p.lexer.Token == js_lexer.TLessThan ||
+		p.lexer.Token == js_lexer.TDot
+
+	// Restore the lexer
+	p.lexer = oldLexer
+	return
+}
+
+// This function is taken from the official TypeScript compiler source code:
+// https://github.com/microsoft/TypeScript/blob/master/src/compiler/parser.ts
+func (p *parser) tsIsIdentifier() bool {
+	if p.lexer.Token == js_lexer.TIdentifier {
+		// If we have a 'yield' keyword, and we're in the [yield] context, then 'yield' is
+		// considered a keyword and is not an identifier.
+		if p.fnOrArrowDataParse.yield != allowIdent && p.lexer.Identifier.String == "yield" {
+			return false
+		}
+
+		// If we have a 'await' keyword, and we're in the [Await] context, then 'await' is
+		// considered a keyword and is not an identifier.
+		if p.fnOrArrowDataParse.await != allowIdent && p.lexer.Identifier.String == "await" {
+			return false
+		}
+
+		return true
+	}
+
+	return false
+}
+
+func (p *parser) skipTypeScriptInterfaceStmt(opts parseStmtOpts) {
+	name := p.lexer.Identifier.String
+	p.lexer.Expect(js_lexer.TIdentifier)
+
+	if opts.isModuleScope {
+		p.localTypeNames[name] = true
+	}
+
+	p.skipTypeScriptTypeParameters(allowInOutVarianceAnnotations | allowEmptyTypeParameters)
+
+	if p.lexer.Token == js_lexer.TExtends {
+		p.lexer.Next()
+		for {
+			p.skipTypeScriptType(js_ast.LLowest)
+			if p.lexer.Token != js_lexer.TComma {
+				break
+			}
+			p.lexer.Next()
+		}
+	}
+
+	if p.lexer.IsContextualKeyword("implements") {
+		p.lexer.Next()
+		for {
+			p.skipTypeScriptType(js_ast.LLowest)
+			if p.lexer.Token != js_lexer.TComma {
+				break
+			}
+			p.lexer.Next()
+		}
+	}
+
+	p.skipTypeScriptObjectType()
+}
+
+func (p *parser) skipTypeScriptTypeStmt(opts parseStmtOpts) {
+	if opts.isExport {
+		switch p.lexer.Token {
+		case js_lexer.TOpenBrace:
+			// "export type {foo}"
+			// "export type {foo} from 'bar'"
+			p.parseExportClause()
+			if p.lexer.IsContextualKeyword("from") {
+				p.lexer.Next()
+				p.parsePath()
+			}
+			p.lexer.ExpectOrInsertSemicolon()
+			return
+
+		// This is invalid TypeScript, and is rejected by the TypeScript compiler:
+		//
+		//   example.ts:1:1 - error TS1383: Only named exports may use 'export type'.
+		//
+		//   1 export type * from './types'
+		//     ~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+		//
+		// However, people may not know this and then blame esbuild for it not
+		// working. So we parse it anyway and then discard it (since we always
+		// discard all types). People who do this should be running the TypeScript
+		// type checker when using TypeScript, which will then report this error.
+		case js_lexer.TAsterisk:
+			// "export type * from 'path'"
+			p.lexer.Next()
+			if p.lexer.IsContextualKeyword("as") {
+				// "export type * as ns from 'path'"
+				p.lexer.Next()
+				p.parseClauseAlias("export")
+				p.lexer.Next()
+			}
+			p.lexer.ExpectContextualKeyword("from")
+			p.parsePath()
+			p.lexer.ExpectOrInsertSemicolon()
+			return
+		}
+	}
+
+	name := p.lexer.Identifier.String
+	p.lexer.Expect(js_lexer.TIdentifier)
+
+	if opts.isModuleScope {
+		p.localTypeNames[name] = true
+	}
+
+	p.skipTypeScriptTypeParameters(allowInOutVarianceAnnotations | allowEmptyTypeParameters)
+	p.lexer.Expect(js_lexer.TEquals)
+	p.skipTypeScriptType(js_ast.LLowest)
+	p.lexer.ExpectOrInsertSemicolon()
+}
+
+func (p *parser) parseTypeScriptEnumStmt(loc logger.Loc, opts parseStmtOpts) js_ast.Stmt {
+	p.lexer.Expect(js_lexer.TEnum)
+	nameLoc := p.lexer.Loc()
+	nameText := p.lexer.Identifier.String
+	p.lexer.Expect(js_lexer.TIdentifier)
+	name := ast.LocRef{Loc: nameLoc, Ref: ast.InvalidRef}
+
+	// Generate the namespace object
+	exportedMembers := p.getOrCreateExportedNamespaceMembers(nameText, opts.isExport)
+	tsNamespace := &js_ast.TSNamespaceScope{
+		ExportedMembers: exportedMembers,
+		ArgRef:          ast.InvalidRef,
+		IsEnumScope:     true,
+	}
+	enumMemberData := &js_ast.TSNamespaceMemberNamespace{
+		ExportedMembers: exportedMembers,
+	}
+
+	// Declare the enum and create the scope
+	scopeIndex := len(p.scopesInOrder)
+	if !opts.isTypeScriptDeclare {
+		name.Ref = p.declareSymbol(ast.SymbolTSEnum, nameLoc, nameText)
+		p.pushScopeForParsePass(js_ast.ScopeEntry, loc)
+		p.currentScope.TSNamespace = tsNamespace
+		p.refToTSNamespaceMemberData[name.Ref] = enumMemberData
+	}
+
+	p.lexer.Expect(js_lexer.TOpenBrace)
+	values := []js_ast.EnumValue{}
+
+	oldFnOrArrowData := p.fnOrArrowDataParse
+	p.fnOrArrowDataParse = fnOrArrowDataParse{
+		isThisDisallowed: true,
+		needsAsyncLoc:    logger.Loc{Start: -1},
+	}
+
+	// Parse the body
+	for p.lexer.Token != js_lexer.TCloseBrace {
+		nameRange := p.lexer.Range()
+		value := js_ast.EnumValue{
+			Loc: nameRange.Loc,
+			Ref: ast.InvalidRef,
+		}
+
+		// Parse the name
+		var nameText string
+		if p.lexer.Token == js_lexer.TStringLiteral {
+			value.Name = p.lexer.StringLiteral()
+			nameText = helpers.UTF16ToString(value.Name)
+		} else if p.lexer.IsIdentifierOrKeyword() {
+			nameText = p.lexer.Identifier.String
+			value.Name = helpers.StringToUTF16(nameText)
+		} else {
+			p.lexer.Expect(js_lexer.TIdentifier)
+		}
+		p.lexer.Next()
+
+		// Identifiers can be referenced by other values
+		if !opts.isTypeScriptDeclare && js_ast.IsIdentifierUTF16(value.Name) {
+			value.Ref = p.declareSymbol(ast.SymbolOther, value.Loc, helpers.UTF16ToString(value.Name))
+		}
+
+		// Parse the initializer
+		if p.lexer.Token == js_lexer.TEquals {
+			p.lexer.Next()
+			value.ValueOrNil = p.parseExpr(js_ast.LComma)
+		}
+
+		values = append(values, value)
+
+		// Add this enum value as a member of the enum's namespace
+		exportedMembers[nameText] = js_ast.TSNamespaceMember{
+			Loc:         value.Loc,
+			Data:        &js_ast.TSNamespaceMemberProperty{},
+			IsEnumValue: true,
+		}
+
+		if p.lexer.Token != js_lexer.TComma && p.lexer.Token != js_lexer.TSemicolon {
+			if p.lexer.IsIdentifierOrKeyword() || p.lexer.Token == js_lexer.TStringLiteral {
+				var errorLoc logger.Loc
+				var errorText string
+
+				if value.ValueOrNil.Data == nil {
+					errorLoc = logger.Loc{Start: nameRange.End()}
+					errorText = fmt.Sprintf("Expected \",\" after %q in enum", nameText)
+				} else {
+					var nextName string
+					if p.lexer.Token == js_lexer.TStringLiteral {
+						nextName = helpers.UTF16ToString(p.lexer.StringLiteral())
+					} else {
+						nextName = p.lexer.Identifier.String
+					}
+					errorLoc = p.lexer.Loc()
+					errorText = fmt.Sprintf("Expected \",\" before %q in enum", nextName)
+				}
+
+				data := p.tracker.MsgData(logger.Range{Loc: errorLoc}, errorText)
+				data.Location.Suggestion = ","
+				p.log.AddMsg(logger.Msg{Kind: logger.Error, Data: data})
+				panic(js_lexer.LexerPanic{})
+			}
+			break
+		}
+		p.lexer.Next()
+	}
+
+	p.fnOrArrowDataParse = oldFnOrArrowData
+
+	if !opts.isTypeScriptDeclare {
+		// Avoid a collision with the enum closure argument variable if the
+		// enum exports a symbol with the same name as the enum itself:
+		//
+		//   enum foo {
+		//     foo = 123,
+		//     bar = foo,
+		//   }
+		//
+		// TypeScript generates the following code in this case:
+		//
+		//   var foo;
+		//   (function (foo) {
+		//     foo[foo["foo"] = 123] = "foo";
+		//     foo[foo["bar"] = 123] = "bar";
+		//   })(foo || (foo = {}));
+		//
+		// Whereas in this case:
+		//
+		//   enum foo {
+		//     bar = foo as any,
+		//   }
+		//
+		// TypeScript generates the following code:
+		//
+		//   var foo;
+		//   (function (foo) {
+		//     foo[foo["bar"] = foo] = "bar";
+		//   })(foo || (foo = {}));
+		//
+		if _, ok := p.currentScope.Members[nameText]; ok {
+			// Add a "_" to make tests easier to read, since non-bundler tests don't
+			// run the renamer. For external-facing things the renamer will avoid
+			// collisions automatically so this isn't important for correctness.
+			tsNamespace.ArgRef = p.newSymbol(ast.SymbolHoisted, "_"+nameText)
+			p.currentScope.Generated = append(p.currentScope.Generated, tsNamespace.ArgRef)
+		} else {
+			tsNamespace.ArgRef = p.declareSymbol(ast.SymbolHoisted, nameLoc, nameText)
+		}
+		p.refToTSNamespaceMemberData[tsNamespace.ArgRef] = enumMemberData
+
+		p.popScope()
+	}
+
+	p.lexer.Expect(js_lexer.TCloseBrace)
+
+	if opts.isTypeScriptDeclare {
+		if opts.isNamespaceScope && opts.isExport {
+			p.hasNonLocalExportDeclareInsideNamespace = true
+		}
+
+		return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptShared}
+	}
+
+	// Save these for when we do out-of-order enum visiting
+	if p.scopesInOrderForEnum == nil {
+		p.scopesInOrderForEnum = make(map[logger.Loc][]scopeOrder)
+	}
+
+	// Make a copy of "scopesInOrder" instead of a slice since the original
+	// array may be flattened in the future by "popAndFlattenScope"
+	p.scopesInOrderForEnum[loc] = append([]scopeOrder{}, p.scopesInOrder[scopeIndex:]...)
+
+	return js_ast.Stmt{Loc: loc, Data: &js_ast.SEnum{
+		Name:     name,
+		Arg:      tsNamespace.ArgRef,
+		Values:   values,
+		IsExport: opts.isExport,
+	}}
+}
+
+// This assumes the caller has already parsed the "import" token
+func (p *parser) parseTypeScriptImportEqualsStmt(loc logger.Loc, opts parseStmtOpts, defaultNameLoc logger.Loc, defaultName string) js_ast.Stmt {
+	p.lexer.Expect(js_lexer.TEquals)
+
+	kind := p.selectLocalKind(js_ast.LocalConst)
+	name := p.lexer.Identifier
+	value := js_ast.Expr{Loc: p.lexer.Loc(), Data: &js_ast.EIdentifier{Ref: p.storeNameInRef(name)}}
+	p.lexer.Expect(js_lexer.TIdentifier)
+
+	if name.String == "require" && p.lexer.Token == js_lexer.TOpenParen {
+		// "import ns = require('x')"
+		p.lexer.Next()
+		path := js_ast.Expr{Loc: p.lexer.Loc(), Data: &js_ast.EString{Value: p.lexer.StringLiteral()}}
+		p.lexer.Expect(js_lexer.TStringLiteral)
+		p.lexer.Expect(js_lexer.TCloseParen)
+		value.Data = &js_ast.ECall{
+			Target: value,
+			Args:   []js_ast.Expr{path},
+		}
+	} else {
+		// "import Foo = Bar"
+		// "import Foo = Bar.Baz"
+		for p.lexer.Token == js_lexer.TDot {
+			p.lexer.Next()
+			value.Data = &js_ast.EDot{
+				Target:               value,
+				Name:                 p.lexer.Identifier.String,
+				NameLoc:              p.lexer.Loc(),
+				CanBeRemovedIfUnused: true,
+			}
+			p.lexer.Expect(js_lexer.TIdentifier)
+		}
+	}
+
+	p.lexer.ExpectOrInsertSemicolon()
+
+	if opts.isTypeScriptDeclare {
+		// "import type foo = require('bar');"
+		// "import type foo = bar.baz;"
+		return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptShared}
+	}
+
+	ref := p.declareSymbol(ast.SymbolConst, defaultNameLoc, defaultName)
+	decls := []js_ast.Decl{{
+		Binding:    js_ast.Binding{Loc: defaultNameLoc, Data: &js_ast.BIdentifier{Ref: ref}},
+		ValueOrNil: value,
+	}}
+
+	return js_ast.Stmt{Loc: loc, Data: &js_ast.SLocal{
+		Kind:              kind,
+		Decls:             decls,
+		IsExport:          opts.isExport,
+		WasTSImportEquals: true,
+	}}
+}
+
+// Generate a TypeScript namespace object for this namespace's scope. If this
+// namespace is another block that is to be merged with an existing namespace,
+// use that earlier namespace's object instead.
+func (p *parser) getOrCreateExportedNamespaceMembers(name string, isExport bool) js_ast.TSNamespaceMembers {
+	// Merge with a sibling namespace from the same scope
+	if existingMember, ok := p.currentScope.Members[name]; ok {
+		if memberData, ok := p.refToTSNamespaceMemberData[existingMember.Ref]; ok {
+			if nsMemberData, ok := memberData.(*js_ast.TSNamespaceMemberNamespace); ok {
+				return nsMemberData.ExportedMembers
+			}
+		}
+	}
+
+	// Merge with a sibling namespace from a different scope
+	if isExport {
+		if parentNamespace := p.currentScope.TSNamespace; parentNamespace != nil {
+			if existing, ok := parentNamespace.ExportedMembers[name]; ok {
+				if existing, ok := existing.Data.(*js_ast.TSNamespaceMemberNamespace); ok {
+					return existing.ExportedMembers
+				}
+			}
+		}
+	}
+
+	// Otherwise, generate a new namespace object
+	return make(js_ast.TSNamespaceMembers)
+}
+
+func (p *parser) parseTypeScriptNamespaceStmt(loc logger.Loc, opts parseStmtOpts) js_ast.Stmt {
+	// "namespace Foo {}"
+	nameLoc := p.lexer.Loc()
+	nameText := p.lexer.Identifier.String
+	p.lexer.Next()
+
+	// Generate the namespace object
+	exportedMembers := p.getOrCreateExportedNamespaceMembers(nameText, opts.isExport)
+	tsNamespace := &js_ast.TSNamespaceScope{
+		ExportedMembers: exportedMembers,
+		ArgRef:          ast.InvalidRef,
+	}
+	nsMemberData := &js_ast.TSNamespaceMemberNamespace{
+		ExportedMembers: exportedMembers,
+	}
+
+	// Declare the namespace and create the scope
+	name := ast.LocRef{Loc: nameLoc, Ref: ast.InvalidRef}
+	scopeIndex := p.pushScopeForParsePass(js_ast.ScopeEntry, loc)
+	p.currentScope.TSNamespace = tsNamespace
+
+	oldHasNonLocalExportDeclareInsideNamespace := p.hasNonLocalExportDeclareInsideNamespace
+	oldFnOrArrowData := p.fnOrArrowDataParse
+	p.hasNonLocalExportDeclareInsideNamespace = false
+	p.fnOrArrowDataParse = fnOrArrowDataParse{
+		isThisDisallowed:   true,
+		isReturnDisallowed: true,
+		needsAsyncLoc:      logger.Loc{Start: -1},
+	}
+
+	// Parse the statements inside the namespace
+	var stmts []js_ast.Stmt
+	if p.lexer.Token == js_lexer.TDot {
+		dotLoc := p.lexer.Loc()
+		p.lexer.Next()
+		stmts = []js_ast.Stmt{p.parseTypeScriptNamespaceStmt(dotLoc, parseStmtOpts{
+			isExport:            true,
+			isNamespaceScope:    true,
+			isTypeScriptDeclare: opts.isTypeScriptDeclare,
+		})}
+	} else if opts.isTypeScriptDeclare && p.lexer.Token != js_lexer.TOpenBrace {
+		p.lexer.ExpectOrInsertSemicolon()
+	} else {
+		p.lexer.Expect(js_lexer.TOpenBrace)
+		stmts = p.parseStmtsUpTo(js_lexer.TCloseBrace, parseStmtOpts{
+			isNamespaceScope:    true,
+			isTypeScriptDeclare: opts.isTypeScriptDeclare,
+		})
+		p.lexer.Next()
+	}
+
+	hasNonLocalExportDeclareInsideNamespace := p.hasNonLocalExportDeclareInsideNamespace
+	p.hasNonLocalExportDeclareInsideNamespace = oldHasNonLocalExportDeclareInsideNamespace
+	p.fnOrArrowDataParse = oldFnOrArrowData
+
+	// Add any exported members from this namespace's body as members of the
+	// associated namespace object.
+	for _, stmt := range stmts {
+		switch s := stmt.Data.(type) {
+		case *js_ast.SFunction:
+			if s.IsExport {
+				name := p.symbols[s.Fn.Name.Ref.InnerIndex].OriginalName
+				member := js_ast.TSNamespaceMember{
+					Loc:  s.Fn.Name.Loc,
+					Data: &js_ast.TSNamespaceMemberProperty{},
+				}
+				exportedMembers[name] = member
+				p.refToTSNamespaceMemberData[s.Fn.Name.Ref] = member.Data
+			}
+
+		case *js_ast.SClass:
+			if s.IsExport {
+				name := p.symbols[s.Class.Name.Ref.InnerIndex].OriginalName
+				member := js_ast.TSNamespaceMember{
+					Loc:  s.Class.Name.Loc,
+					Data: &js_ast.TSNamespaceMemberProperty{},
+				}
+				exportedMembers[name] = member
+				p.refToTSNamespaceMemberData[s.Class.Name.Ref] = member.Data
+			}
+
+		case *js_ast.SNamespace:
+			if s.IsExport {
+				if memberData, ok := p.refToTSNamespaceMemberData[s.Name.Ref]; ok {
+					if nsMemberData, ok := memberData.(*js_ast.TSNamespaceMemberNamespace); ok {
+						member := js_ast.TSNamespaceMember{
+							Loc: s.Name.Loc,
+							Data: &js_ast.TSNamespaceMemberNamespace{
+								ExportedMembers: nsMemberData.ExportedMembers,
+							},
+						}
+						exportedMembers[p.symbols[s.Name.Ref.InnerIndex].OriginalName] = member
+						p.refToTSNamespaceMemberData[s.Name.Ref] = member.Data
+					}
+				}
+			}
+
+		case *js_ast.SEnum:
+			if s.IsExport {
+				if memberData, ok := p.refToTSNamespaceMemberData[s.Name.Ref]; ok {
+					if nsMemberData, ok := memberData.(*js_ast.TSNamespaceMemberNamespace); ok {
+						member := js_ast.TSNamespaceMember{
+							Loc: s.Name.Loc,
+							Data: &js_ast.TSNamespaceMemberNamespace{
+								ExportedMembers: nsMemberData.ExportedMembers,
+							},
+						}
+						exportedMembers[p.symbols[s.Name.Ref.InnerIndex].OriginalName] = member
+						p.refToTSNamespaceMemberData[s.Name.Ref] = member.Data
+					}
+				}
+			}
+
+		case *js_ast.SLocal:
+			if s.IsExport {
+				js_ast.ForEachIdentifierBindingInDecls(s.Decls, func(loc logger.Loc, b *js_ast.BIdentifier) {
+					name := p.symbols[b.Ref.InnerIndex].OriginalName
+					member := js_ast.TSNamespaceMember{
+						Loc:  loc,
+						Data: &js_ast.TSNamespaceMemberProperty{},
+					}
+					exportedMembers[name] = member
+					p.refToTSNamespaceMemberData[b.Ref] = member.Data
+				})
+			}
+		}
+	}
+
+	// Import assignments may be only used in type expressions, not value
+	// expressions. If this is the case, the TypeScript compiler removes
+	// them entirely from the output. That can cause the namespace itself
+	// to be considered empty and thus be removed.
+	importEqualsCount := 0
+	for _, stmt := range stmts {
+		if local, ok := stmt.Data.(*js_ast.SLocal); ok && local.WasTSImportEquals && !local.IsExport {
+			importEqualsCount++
+		}
+	}
+
+	// TypeScript omits namespaces without values. These namespaces
+	// are only allowed to be used in type expressions. They are
+	// allowed to be exported, but can also only be used in type
+	// expressions when imported. So we shouldn't count them as a
+	// real export either.
+	//
+	// TypeScript also strangely counts namespaces containing only
+	// "export declare" statements as non-empty even though "declare"
+	// statements are only type annotations. We cannot omit the namespace
+	// in that case. See https://github.com/evanw/esbuild/issues/1158.
+	if (len(stmts) == importEqualsCount && !hasNonLocalExportDeclareInsideNamespace) || opts.isTypeScriptDeclare {
+		p.popAndDiscardScope(scopeIndex)
+		if opts.isModuleScope {
+			p.localTypeNames[nameText] = true
+		}
+		return js_ast.Stmt{Loc: loc, Data: js_ast.STypeScriptShared}
+	}
+
+	if !opts.isTypeScriptDeclare {
+		// Avoid a collision with the namespace closure argument variable if the
+		// namespace exports a symbol with the same name as the namespace itself:
+		//
+		//   namespace foo {
+		//     export let foo = 123
+		//     console.log(foo)
+		//   }
+		//
+		// TypeScript generates the following code in this case:
+		//
+		//   var foo;
+		//   (function (foo_1) {
+		//     foo_1.foo = 123;
+		//     console.log(foo_1.foo);
+		//   })(foo || (foo = {}));
+		//
+		if _, ok := p.currentScope.Members[nameText]; ok {
+			// Add a "_" to make tests easier to read, since non-bundler tests don't
+			// run the renamer. For external-facing things the renamer will avoid
+			// collisions automatically so this isn't important for correctness.
+			tsNamespace.ArgRef = p.newSymbol(ast.SymbolHoisted, "_"+nameText)
+			p.currentScope.Generated = append(p.currentScope.Generated, tsNamespace.ArgRef)
+		} else {
+			tsNamespace.ArgRef = p.declareSymbol(ast.SymbolHoisted, nameLoc, nameText)
+		}
+		p.refToTSNamespaceMemberData[tsNamespace.ArgRef] = nsMemberData
+	}
+
+	p.popScope()
+	if !opts.isTypeScriptDeclare {
+		name.Ref = p.declareSymbol(ast.SymbolTSNamespace, nameLoc, nameText)
+		p.refToTSNamespaceMemberData[name.Ref] = nsMemberData
+	}
+	return js_ast.Stmt{Loc: loc, Data: &js_ast.SNamespace{
+		Name:     name,
+		Arg:      tsNamespace.ArgRef,
+		Stmts:    stmts,
+		IsExport: opts.isExport,
+	}}
+}
+
+func (p *parser) generateClosureForTypeScriptNamespaceOrEnum(
+	stmts []js_ast.Stmt, stmtLoc logger.Loc, isExport bool, nameLoc logger.Loc,
+	nameRef ast.Ref, argRef ast.Ref, stmtsInsideClosure []js_ast.Stmt,
+) []js_ast.Stmt {
+	// Follow the link chain in case symbols were merged
+	symbol := p.symbols[nameRef.InnerIndex]
+	for symbol.Link != ast.InvalidRef {
+		nameRef = symbol.Link
+		symbol = p.symbols[nameRef.InnerIndex]
+	}
+
+	// Make sure to only emit a variable once for a given namespace, since there
+	// can be multiple namespace blocks for the same namespace
+	if (symbol.Kind == ast.SymbolTSNamespace || symbol.Kind == ast.SymbolTSEnum) && !p.emittedNamespaceVars[nameRef] {
+		decls := []js_ast.Decl{{Binding: js_ast.Binding{Loc: nameLoc, Data: &js_ast.BIdentifier{Ref: nameRef}}}}
+		p.emittedNamespaceVars[nameRef] = true
+		if p.currentScope == p.moduleScope {
+			// Top-level namespace: "var"
+			stmts = append(stmts, js_ast.Stmt{Loc: stmtLoc, Data: &js_ast.SLocal{
+				Kind:     js_ast.LocalVar,
+				Decls:    decls,
+				IsExport: isExport,
+			}})
+		} else {
+			// Nested namespace: "let"
+			stmts = append(stmts, js_ast.Stmt{Loc: stmtLoc, Data: &js_ast.SLocal{
+				Kind:  js_ast.LocalLet,
+				Decls: decls,
+			}})
+		}
+	}
+
+	var argExpr js_ast.Expr
+	if p.options.minifySyntax && !p.options.unsupportedJSFeatures.Has(compat.LogicalAssignment) {
+		// If the "||=" operator is supported, our minified output can be slightly smaller
+		if isExport && p.enclosingNamespaceArgRef != nil {
+			// "name = (enclosing.name ||= {})"
+			argExpr = js_ast.Assign(
+				js_ast.Expr{Loc: nameLoc, Data: &js_ast.EIdentifier{Ref: nameRef}},
+				js_ast.Expr{Loc: nameLoc, Data: &js_ast.EBinary{
+					Op: js_ast.BinOpLogicalOrAssign,
+					Left: js_ast.Expr{Loc: nameLoc, Data: p.dotOrMangledPropVisit(
+						js_ast.Expr{Loc: nameLoc, Data: &js_ast.EIdentifier{Ref: *p.enclosingNamespaceArgRef}},
+						p.symbols[nameRef.InnerIndex].OriginalName,
+						nameLoc,
+					)},
+					Right: js_ast.Expr{Loc: nameLoc, Data: &js_ast.EObject{}},
+				}},
+			)
+			p.recordUsage(*p.enclosingNamespaceArgRef)
+			p.recordUsage(nameRef)
+		} else {
+			// "name ||= {}"
+			argExpr = js_ast.Expr{Loc: nameLoc, Data: &js_ast.EBinary{
+				Op:    js_ast.BinOpLogicalOrAssign,
+				Left:  js_ast.Expr{Loc: nameLoc, Data: &js_ast.EIdentifier{Ref: nameRef}},
+				Right: js_ast.Expr{Loc: nameLoc, Data: &js_ast.EObject{}},
+			}}
+			p.recordUsage(nameRef)
+		}
+	} else {
+		if isExport && p.enclosingNamespaceArgRef != nil {
+			// "name = enclosing.name || (enclosing.name = {})"
+			name := p.symbols[nameRef.InnerIndex].OriginalName
+			argExpr = js_ast.Assign(
+				js_ast.Expr{Loc: nameLoc, Data: &js_ast.EIdentifier{Ref: nameRef}},
+				js_ast.Expr{Loc: nameLoc, Data: &js_ast.EBinary{
+					Op: js_ast.BinOpLogicalOr,
+					Left: js_ast.Expr{Loc: nameLoc, Data: p.dotOrMangledPropVisit(
+						js_ast.Expr{Loc: nameLoc, Data: &js_ast.EIdentifier{Ref: *p.enclosingNamespaceArgRef}},
+						name,
+						nameLoc,
+					)},
+					Right: js_ast.Assign(
+						js_ast.Expr{Loc: nameLoc, Data: p.dotOrMangledPropVisit(
+							js_ast.Expr{Loc: nameLoc, Data: &js_ast.EIdentifier{Ref: *p.enclosingNamespaceArgRef}},
+							name,
+							nameLoc,
+						)},
+						js_ast.Expr{Loc: nameLoc, Data: &js_ast.EObject{}},
+					),
+				}},
+			)
+			p.recordUsage(*p.enclosingNamespaceArgRef)
+			p.recordUsage(*p.enclosingNamespaceArgRef)
+			p.recordUsage(nameRef)
+		} else {
+			// "name || (name = {})"
+			argExpr = js_ast.Expr{Loc: nameLoc, Data: &js_ast.EBinary{
+				Op:   js_ast.BinOpLogicalOr,
+				Left: js_ast.Expr{Loc: nameLoc, Data: &js_ast.EIdentifier{Ref: nameRef}},
+				Right: js_ast.Assign(
+					js_ast.Expr{Loc: nameLoc, Data: &js_ast.EIdentifier{Ref: nameRef}},
+					js_ast.Expr{Loc: nameLoc, Data: &js_ast.EObject{}},
+				),
+			}}
+			p.recordUsage(nameRef)
+			p.recordUsage(nameRef)
+		}
+	}
+
+	// Try to use an arrow function if possible for compactness
+	var targetExpr js_ast.Expr
+	args := []js_ast.Arg{{Binding: js_ast.Binding{Loc: nameLoc, Data: &js_ast.BIdentifier{Ref: argRef}}}}
+	if p.options.unsupportedJSFeatures.Has(compat.Arrow) {
+		targetExpr = js_ast.Expr{Loc: stmtLoc, Data: &js_ast.EFunction{Fn: js_ast.Fn{
+			Args: args,
+			Body: js_ast.FnBody{Loc: stmtLoc, Block: js_ast.SBlock{Stmts: stmtsInsideClosure}},
+		}}}
+	} else {
+		// "(() => { foo() })()" => "(() => foo())()"
+		if p.options.minifySyntax && len(stmtsInsideClosure) == 1 {
+			if expr, ok := stmtsInsideClosure[0].Data.(*js_ast.SExpr); ok {
+				stmtsInsideClosure[0].Data = &js_ast.SReturn{ValueOrNil: expr.Value}
+			}
+		}
+		targetExpr = js_ast.Expr{Loc: stmtLoc, Data: &js_ast.EArrow{
+			Args:       args,
+			Body:       js_ast.FnBody{Loc: stmtLoc, Block: js_ast.SBlock{Stmts: stmtsInsideClosure}},
+			PreferExpr: true,
+		}}
+	}
+
+	// Call the closure with the name object
+	stmts = append(stmts, js_ast.Stmt{Loc: stmtLoc, Data: &js_ast.SExpr{Value: js_ast.Expr{Loc: stmtLoc, Data: &js_ast.ECall{
+		Target: targetExpr,
+		Args:   []js_ast.Expr{argExpr},
+	}}}})
+
+	return stmts
+}
+
+func (p *parser) generateClosureForTypeScriptEnum(
+	stmts []js_ast.Stmt, stmtLoc logger.Loc, isExport bool, nameLoc logger.Loc,
+	nameRef ast.Ref, argRef ast.Ref, exprsInsideClosure []js_ast.Expr,
+	allValuesArePure bool,
+) []js_ast.Stmt {
+	// Bail back to the namespace code for enums that aren't at the top level.
+	// Doing this for nested enums is problematic for two reasons. First of all
+	// enums inside of namespaces must be property accesses off the namespace
+	// object instead of variable declarations. Also we'd need to use "let"
+	// instead of "var" which doesn't allow sibling declarations to be merged.
+	if p.currentScope != p.moduleScope {
+		stmtsInsideClosure := []js_ast.Stmt{}
+		if len(exprsInsideClosure) > 0 {
+			if p.options.minifySyntax {
+				// "a; b; c;" => "a, b, c;"
+				joined := js_ast.JoinAllWithComma(exprsInsideClosure)
+				stmtsInsideClosure = append(stmtsInsideClosure, js_ast.Stmt{Loc: joined.Loc, Data: &js_ast.SExpr{Value: joined}})
+			} else {
+				for _, expr := range exprsInsideClosure {
+					stmtsInsideClosure = append(stmtsInsideClosure, js_ast.Stmt{Loc: expr.Loc, Data: &js_ast.SExpr{Value: expr}})
+				}
+			}
+		}
+		return p.generateClosureForTypeScriptNamespaceOrEnum(
+			stmts, stmtLoc, isExport, nameLoc, nameRef, argRef, stmtsInsideClosure)
+	}
+
+	// This uses an output format for enums that's different but equivalent to
+	// what TypeScript uses. Here is TypeScript's output:
+	//
+	//   var x;
+	//   (function (x) {
+	//     x[x["y"] = 1] = "y";
+	//   })(x || (x = {}));
+	//
+	// And here's our output:
+	//
+	//   var x = /* @__PURE__ */ ((x) => {
+	//     x[x["y"] = 1] = "y";
+	//     return x;
+	//   })(x || {});
+	//
+	// One benefit is that the minified output is smaller:
+	//
+	//   // Old output minified
+	//   var x;(function(n){n[n.y=1]="y"})(x||(x={}));
+	//
+	//   // New output minified
+	//   var x=(r=>(r[r.y=1]="y",r))(x||{});
+	//
+	// Another benefit is that the @__PURE__ annotation means it automatically
+	// works with tree-shaking, even with more advanced features such as sibling
+	// enum declarations and enum/namespace merges. Ideally all uses of the enum
+	// are just direct references to enum members (and are therefore inlined as
+	// long as the enum value is a constant) and the enum definition itself is
+	// unused and can be removed as dead code.
+
+	// Follow the link chain in case symbols were merged
+	symbol := p.symbols[nameRef.InnerIndex]
+	for symbol.Link != ast.InvalidRef {
+		nameRef = symbol.Link
+		symbol = p.symbols[nameRef.InnerIndex]
+	}
+
+	// Generate the body of the closure, including a return statement at the end
+	stmtsInsideClosure := []js_ast.Stmt{}
+	argExpr := js_ast.Expr{Loc: nameLoc, Data: &js_ast.EIdentifier{Ref: argRef}}
+	if p.options.minifySyntax {
+		// "a; b; return c;" => "return a, b, c;"
+		joined := js_ast.JoinAllWithComma(exprsInsideClosure)
+		joined = js_ast.JoinWithComma(joined, argExpr)
+		stmtsInsideClosure = append(stmtsInsideClosure, js_ast.Stmt{Loc: joined.Loc, Data: &js_ast.SReturn{ValueOrNil: joined}})
+	} else {
+		for _, expr := range exprsInsideClosure {
+			stmtsInsideClosure = append(stmtsInsideClosure, js_ast.Stmt{Loc: expr.Loc, Data: &js_ast.SExpr{Value: expr}})
+		}
+		stmtsInsideClosure = append(stmtsInsideClosure, js_ast.Stmt{Loc: argExpr.Loc, Data: &js_ast.SReturn{ValueOrNil: argExpr}})
+	}
+
+	// Try to use an arrow function if possible for compactness
+	var targetExpr js_ast.Expr
+	args := []js_ast.Arg{{Binding: js_ast.Binding{Loc: nameLoc, Data: &js_ast.BIdentifier{Ref: argRef}}}}
+	if p.options.unsupportedJSFeatures.Has(compat.Arrow) {
+		targetExpr = js_ast.Expr{Loc: stmtLoc, Data: &js_ast.EFunction{Fn: js_ast.Fn{
+			Args: args,
+			Body: js_ast.FnBody{Loc: stmtLoc, Block: js_ast.SBlock{Stmts: stmtsInsideClosure}},
+		}}}
+	} else {
+		targetExpr = js_ast.Expr{Loc: stmtLoc, Data: &js_ast.EArrow{
+			Args:       args,
+			Body:       js_ast.FnBody{Loc: stmtLoc, Block: js_ast.SBlock{Stmts: stmtsInsideClosure}},
+			PreferExpr: p.options.minifySyntax,
+		}}
+	}
+
+	// Call the closure with the name object and store it to the variable
+	decls := []js_ast.Decl{{
+		Binding: js_ast.Binding{Loc: nameLoc, Data: &js_ast.BIdentifier{Ref: nameRef}},
+		ValueOrNil: js_ast.Expr{Loc: stmtLoc, Data: &js_ast.ECall{
+			Target: targetExpr,
+			Args: []js_ast.Expr{{Loc: nameLoc, Data: &js_ast.EBinary{
+				Op:    js_ast.BinOpLogicalOr,
+				Left:  js_ast.Expr{Loc: nameLoc, Data: &js_ast.EIdentifier{Ref: nameRef}},
+				Right: js_ast.Expr{Loc: nameLoc, Data: &js_ast.EObject{}},
+			}}},
+			CanBeUnwrappedIfUnused: allValuesArePure,
+		}},
+	}}
+	p.recordUsage(nameRef)
+
+	// Use a "var" statement since this is a top-level enum, but only use "export" once
+	stmts = append(stmts, js_ast.Stmt{Loc: stmtLoc, Data: &js_ast.SLocal{
+		Kind:     js_ast.LocalVar,
+		Decls:    decls,
+		IsExport: isExport && !p.emittedNamespaceVars[nameRef],
+	}})
+	p.emittedNamespaceVars[nameRef] = true
+
+	return stmts
+}
+
+func (p *parser) wrapInlinedEnum(value js_ast.Expr, comment string) js_ast.Expr {
+	if strings.Contains(comment, "*/") {
+		// Don't wrap with a comment
+		return value
+	}
+
+	// Wrap with a comment
+	return js_ast.Expr{Loc: value.Loc, Data: &js_ast.EInlinedEnum{
+		Value:   value,
+		Comment: comment,
+	}}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/js_printer/js_printer.go b/source/vendor/github.com/evanw/esbuild/internal/js_printer/js_printer.go
new file mode 100644
index 0000000..3c5cab0
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/js_printer/js_printer.go
@@ -0,0 +1,4924 @@
+package js_printer
+
+import (
+	"bytes"
+	"fmt"
+	"math"
+	"strconv"
+	"strings"
+	"unicode/utf8"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/config"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/logger"
+	"github.com/evanw/esbuild/internal/renamer"
+	"github.com/evanw/esbuild/internal/sourcemap"
+)
+
+var positiveInfinity = math.Inf(1)
+var negativeInfinity = math.Inf(-1)
+
+const hexChars = "0123456789ABCDEF"
+const firstASCII = 0x20
+const lastASCII = 0x7E
+const firstHighSurrogate = 0xD800
+const lastHighSurrogate = 0xDBFF
+const firstLowSurrogate = 0xDC00
+const lastLowSurrogate = 0xDFFF
+
+func QuoteIdentifier(js []byte, name string, unsupportedFeatures compat.JSFeature) []byte {
+	isASCII := false
+	asciiStart := 0
+	for i, c := range name {
+		if c >= firstASCII && c <= lastASCII {
+			// Fast path: a run of ASCII characters
+			if !isASCII {
+				isASCII = true
+				asciiStart = i
+			}
+		} else {
+			// Slow path: escape non-ACSII characters
+			if isASCII {
+				js = append(js, name[asciiStart:i]...)
+				isASCII = false
+			}
+			if c <= 0xFFFF {
+				js = append(js, '\\', 'u', hexChars[c>>12], hexChars[(c>>8)&15], hexChars[(c>>4)&15], hexChars[c&15])
+			} else if !unsupportedFeatures.Has(compat.UnicodeEscapes) {
+				js = append(js, fmt.Sprintf("\\u{%X}", c)...)
+			} else {
+				panic("Internal error: Cannot encode identifier: Unicode escapes are unsupported")
+			}
+		}
+	}
+	if isASCII {
+		// Print one final run of ASCII characters
+		js = append(js, name[asciiStart:]...)
+	}
+	return js
+}
+
+func (p *printer) printUnquotedUTF16(text []uint16, quote rune, flags printQuotedFlags) {
+	temp := make([]byte, utf8.UTFMax)
+	js := p.js
+	i := 0
+	n := len(text)
+
+	// Only compute the line length if necessary
+	var startLineLength int
+	wrapLongLines := false
+	if p.options.LineLimit > 0 && (flags&printQuotedNoWrap) == 0 {
+		startLineLength = p.currentLineLength()
+		if startLineLength > p.options.LineLimit {
+			startLineLength = p.options.LineLimit
+		}
+		wrapLongLines = true
+	}
+
+	for i < n {
+		// Wrap long lines that are over the limit using escaped newlines
+		if wrapLongLines && startLineLength+i >= p.options.LineLimit {
+			js = append(js, "\\\n"...)
+			startLineLength -= p.options.LineLimit
+		}
+
+		c := text[i]
+		i++
+
+		switch c {
+		// Special-case the null character since it may mess with code written in C
+		// that treats null characters as the end of the string.
+		case '\x00':
+			// We don't want "\x001" to be written as "\01"
+			if i < n && text[i] >= '0' && text[i] <= '9' {
+				js = append(js, "\\x00"...)
+			} else {
+				js = append(js, "\\0"...)
+			}
+
+		// Special-case the bell character since it may cause dumping this file to
+		// the terminal to make a sound, which is undesirable. Note that we can't
+		// use an octal literal to print this shorter since octal literals are not
+		// allowed in strict mode (or in template strings).
+		case '\x07':
+			js = append(js, "\\x07"...)
+
+		case '\b':
+			js = append(js, "\\b"...)
+
+		case '\f':
+			js = append(js, "\\f"...)
+
+		case '\n':
+			if quote == '`' {
+				startLineLength = -i // Printing a real newline resets the line length
+				js = append(js, '\n')
+			} else {
+				js = append(js, "\\n"...)
+			}
+
+		case '\r':
+			js = append(js, "\\r"...)
+
+		case '\v':
+			js = append(js, "\\v"...)
+
+		case '\x1B':
+			js = append(js, "\\x1B"...)
+
+		case '\\':
+			js = append(js, "\\\\"...)
+
+		case '/':
+			// Avoid generating the sequence "</script" in JS code
+			if !p.options.UnsupportedFeatures.Has(compat.InlineScript) && i >= 2 && text[i-2] == '<' && i+6 <= len(text) {
+				script := "script"
+				matches := true
+				for j := 0; j < 6; j++ {
+					a := text[i+j]
+					b := uint16(script[j])
+					if a >= 'A' && a <= 'Z' {
+						a += 'a' - 'A'
+					}
+					if a != b {
+						matches = false
+						break
+					}
+				}
+				if matches {
+					js = append(js, '\\')
+				}
+			}
+			js = append(js, '/')
+
+		case '\'':
+			if quote == '\'' {
+				js = append(js, '\\')
+			}
+			js = append(js, '\'')
+
+		case '"':
+			if quote == '"' {
+				js = append(js, '\\')
+			}
+			js = append(js, '"')
+
+		case '`':
+			if quote == '`' {
+				js = append(js, '\\')
+			}
+			js = append(js, '`')
+
+		case '$':
+			if quote == '`' && i < n && text[i] == '{' {
+				js = append(js, '\\')
+			}
+			js = append(js, '$')
+
+		case '\u2028':
+			js = append(js, "\\u2028"...)
+
+		case '\u2029':
+			js = append(js, "\\u2029"...)
+
+		case '\uFEFF':
+			js = append(js, "\\uFEFF"...)
+
+		default:
+			switch {
+			// Common case: just append a single byte
+			case c <= lastASCII:
+				js = append(js, byte(c))
+
+			// Is this a high surrogate?
+			case c >= firstHighSurrogate && c <= lastHighSurrogate:
+				// Is there a next character?
+				if i < n {
+					c2 := text[i]
+
+					// Is it a low surrogate?
+					if c2 >= firstLowSurrogate && c2 <= lastLowSurrogate {
+						r := (rune(c) << 10) + rune(c2) + (0x10000 - (firstHighSurrogate << 10) - firstLowSurrogate)
+						i++
+
+						// Escape this character if UTF-8 isn't allowed
+						if p.options.ASCIIOnly {
+							if !p.options.UnsupportedFeatures.Has(compat.UnicodeEscapes) {
+								js = append(js, fmt.Sprintf("\\u{%X}", r)...)
+							} else {
+								js = append(js,
+									'\\', 'u', hexChars[c>>12], hexChars[(c>>8)&15], hexChars[(c>>4)&15], hexChars[c&15],
+									'\\', 'u', hexChars[c2>>12], hexChars[(c2>>8)&15], hexChars[(c2>>4)&15], hexChars[c2&15],
+								)
+							}
+							continue
+						}
+
+						// Otherwise, encode to UTF-8
+						width := utf8.EncodeRune(temp, r)
+						js = append(js, temp[:width]...)
+						continue
+					}
+				}
+
+				// Write an unpaired high surrogate
+				js = append(js, '\\', 'u', hexChars[c>>12], hexChars[(c>>8)&15], hexChars[(c>>4)&15], hexChars[c&15])
+
+			// Is this an unpaired low surrogate or four-digit hex escape?
+			case (c >= firstLowSurrogate && c <= lastLowSurrogate) || (p.options.ASCIIOnly && c > 0xFF):
+				js = append(js, '\\', 'u', hexChars[c>>12], hexChars[(c>>8)&15], hexChars[(c>>4)&15], hexChars[c&15])
+
+			// Can this be a two-digit hex escape?
+			case p.options.ASCIIOnly:
+				js = append(js, '\\', 'x', hexChars[c>>4], hexChars[c&15])
+
+			// Otherwise, just encode to UTF-8
+			default:
+				width := utf8.EncodeRune(temp, rune(c))
+				js = append(js, temp[:width]...)
+			}
+		}
+	}
+
+	p.js = js
+}
+
+// JSX tag syntax doesn't support character escapes so non-ASCII identifiers
+// must be printed as UTF-8 even when the charset is set to ASCII.
+func (p *printer) printJSXTag(tagOrNil js_ast.Expr) {
+	switch e := tagOrNil.Data.(type) {
+	case *js_ast.EString:
+		p.addSourceMapping(tagOrNil.Loc)
+		p.print(helpers.UTF16ToString(e.Value))
+
+	case *js_ast.EIdentifier:
+		name := p.renamer.NameForSymbol(e.Ref)
+		p.addSourceMappingForName(tagOrNil.Loc, name, e.Ref)
+		p.print(name)
+
+	case *js_ast.EDot:
+		p.printJSXTag(e.Target)
+		p.print(".")
+		p.addSourceMapping(e.NameLoc)
+		p.print(e.Name)
+
+	default:
+		if tagOrNil.Data != nil {
+			p.printExpr(tagOrNil, js_ast.LLowest, 0)
+		}
+	}
+}
+
+type printer struct {
+	symbols                ast.SymbolMap
+	astHelpers             js_ast.HelperContext
+	renamer                renamer.Renamer
+	importRecords          []ast.ImportRecord
+	callTarget             js_ast.E
+	exprComments           map[logger.Loc][]string
+	printedExprComments    map[logger.Loc]bool
+	hasLegalComment        map[string]struct{}
+	extractedLegalComments []string
+	js                     []byte
+	jsonMetadataImports    []string
+	binaryExprStack        []binaryExprVisitor
+	options                Options
+	builder                sourcemap.ChunkBuilder
+	printNextIndentAsSpace bool
+
+	stmtStart          int
+	exportDefaultStart int
+	arrowExprStart     int
+	forOfInitStart     int
+
+	withNesting          int
+	prevOpEnd            int
+	needSpaceBeforeDot   int
+	prevRegExpEnd        int
+	noLeadingNewlineHere int
+	oldLineStart         int
+	oldLineEnd           int
+	intToBytesBuffer     [64]byte
+	needsSemicolon       bool
+	wasLazyExport        bool
+	prevOp               js_ast.OpCode
+	moduleType           js_ast.ModuleType
+}
+
+func (p *printer) print(text string) {
+	p.js = append(p.js, text...)
+}
+
+// This is the same as "print(string(bytes))" without any unnecessary temporary
+// allocations
+func (p *printer) printBytes(bytes []byte) {
+	p.js = append(p.js, bytes...)
+}
+
+type printQuotedFlags uint8
+
+const (
+	printQuotedAllowBacktick printQuotedFlags = 1 << iota
+	printQuotedNoWrap
+)
+
+func (p *printer) printQuotedUTF8(text string, flags printQuotedFlags) {
+	p.printQuotedUTF16(helpers.StringToUTF16(text), flags)
+}
+
+func (p *printer) addSourceMapping(loc logger.Loc) {
+	if p.options.AddSourceMappings {
+		p.builder.AddSourceMapping(loc, "", p.js)
+	}
+}
+
+func (p *printer) addSourceMappingForName(loc logger.Loc, name string, ref ast.Ref) {
+	if p.options.AddSourceMappings {
+		if originalName := p.symbols.Get(ast.FollowSymbols(p.symbols, ref)).OriginalName; originalName != name {
+			p.builder.AddSourceMapping(loc, originalName, p.js)
+		} else {
+			p.builder.AddSourceMapping(loc, "", p.js)
+		}
+	}
+}
+
+func (p *printer) printIndent() {
+	if p.options.MinifyWhitespace {
+		return
+	}
+
+	if p.printNextIndentAsSpace {
+		p.print(" ")
+		p.printNextIndentAsSpace = false
+		return
+	}
+
+	indent := p.options.Indent
+	if p.options.LineLimit > 0 && indent*2 >= p.options.LineLimit {
+		indent = p.options.LineLimit / 2
+	}
+	for i := 0; i < indent; i++ {
+		p.print("  ")
+	}
+}
+
+func (p *printer) mangledPropName(ref ast.Ref) string {
+	ref = ast.FollowSymbols(p.symbols, ref)
+	if name, ok := p.options.MangledProps[ref]; ok {
+		return name
+	}
+	return p.renamer.NameForSymbol(ref)
+}
+
+func (p *printer) tryToGetImportedEnumValue(target js_ast.Expr, name string) (js_ast.TSEnumValue, bool) {
+	if id, ok := target.Data.(*js_ast.EImportIdentifier); ok {
+		ref := ast.FollowSymbols(p.symbols, id.Ref)
+		if symbol := p.symbols.Get(ref); symbol.Kind == ast.SymbolTSEnum {
+			if enum, ok := p.options.TSEnums[ref]; ok {
+				value, ok := enum[name]
+				return value, ok
+			}
+		}
+	}
+	return js_ast.TSEnumValue{}, false
+}
+
+func (p *printer) tryToGetImportedEnumValueUTF16(target js_ast.Expr, name []uint16) (js_ast.TSEnumValue, string, bool) {
+	if id, ok := target.Data.(*js_ast.EImportIdentifier); ok {
+		ref := ast.FollowSymbols(p.symbols, id.Ref)
+		if symbol := p.symbols.Get(ref); symbol.Kind == ast.SymbolTSEnum {
+			if enum, ok := p.options.TSEnums[ref]; ok {
+				name := helpers.UTF16ToString(name)
+				value, ok := enum[name]
+				return value, name, ok
+			}
+		}
+	}
+	return js_ast.TSEnumValue{}, "", false
+}
+
+func (p *printer) printClauseAlias(loc logger.Loc, alias string) {
+	if js_ast.IsIdentifier(alias) {
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(loc)
+		p.printIdentifier(alias)
+	} else {
+		p.addSourceMapping(loc)
+		p.printQuotedUTF8(alias, 0)
+	}
+}
+
+// Note: The functions below check whether something can be printed as an
+// identifier or if it needs to be quoted (e.g. "x.y" vs. "x['y']") using the
+// ES5 identifier validity test to maximize cross-platform portability. Even
+// though newer JavaScript environments can handle more Unicode characters,
+// there isn't a published document that says which Unicode versions are
+// supported by which browsers. Even if a character is considered valid in the
+// latest version of Unicode, we don't know if the browser we're targeting
+// contains an older version of Unicode or not. So for safety, we quote
+// anything that isn't guaranteed to be compatible with ES5, the oldest
+// JavaScript language target that we support.
+
+func CanEscapeIdentifier(name string, UnsupportedFeatures compat.JSFeature, asciiOnly bool) bool {
+	return js_ast.IsIdentifierES5AndESNext(name) && (!asciiOnly ||
+		!UnsupportedFeatures.Has(compat.UnicodeEscapes) ||
+		!helpers.ContainsNonBMPCodePoint(name))
+}
+
+func (p *printer) canPrintIdentifier(name string) bool {
+	return js_ast.IsIdentifierES5AndESNext(name) && (!p.options.ASCIIOnly ||
+		!p.options.UnsupportedFeatures.Has(compat.UnicodeEscapes) ||
+		!helpers.ContainsNonBMPCodePoint(name))
+}
+
+func (p *printer) canPrintIdentifierUTF16(name []uint16) bool {
+	return js_ast.IsIdentifierES5AndESNextUTF16(name) && (!p.options.ASCIIOnly ||
+		!p.options.UnsupportedFeatures.Has(compat.UnicodeEscapes) ||
+		!helpers.ContainsNonBMPCodePointUTF16(name))
+}
+
+func (p *printer) printIdentifier(name string) {
+	if p.options.ASCIIOnly {
+		p.js = QuoteIdentifier(p.js, name, p.options.UnsupportedFeatures)
+	} else {
+		p.print(name)
+	}
+}
+
+// This is the same as "printIdentifier(StringToUTF16(bytes))" without any
+// unnecessary temporary allocations
+func (p *printer) printIdentifierUTF16(name []uint16) {
+	var temp [utf8.UTFMax]byte
+	n := len(name)
+
+	for i := 0; i < n; i++ {
+		c := rune(name[i])
+
+		if c >= firstHighSurrogate && c <= lastHighSurrogate && i+1 < n {
+			if c2 := rune(name[i+1]); c2 >= firstLowSurrogate && c2 <= lastLowSurrogate {
+				c = (c << 10) + c2 + (0x10000 - (firstHighSurrogate << 10) - firstLowSurrogate)
+				i++
+			}
+		}
+
+		if p.options.ASCIIOnly && c > lastASCII {
+			if c <= 0xFFFF {
+				p.js = append(p.js, '\\', 'u', hexChars[c>>12], hexChars[(c>>8)&15], hexChars[(c>>4)&15], hexChars[c&15])
+			} else if !p.options.UnsupportedFeatures.Has(compat.UnicodeEscapes) {
+				p.js = append(p.js, fmt.Sprintf("\\u{%X}", c)...)
+			} else {
+				panic("Internal error: Cannot encode identifier: Unicode escapes are unsupported")
+			}
+			continue
+		}
+
+		width := utf8.EncodeRune(temp[:], c)
+		p.js = append(p.js, temp[:width]...)
+	}
+}
+
+func (p *printer) printNumber(value float64, level js_ast.L) {
+	absValue := math.Abs(value)
+
+	if value != value {
+		p.printSpaceBeforeIdentifier()
+		if p.withNesting != 0 {
+			// "with (x) NaN" really means "x.NaN" so avoid identifiers when "with" is present
+			wrap := level >= js_ast.LMultiply
+			if wrap {
+				p.print("(")
+			}
+			if p.options.MinifyWhitespace {
+				p.print("0/0")
+			} else {
+				p.print("0 / 0")
+			}
+			if wrap {
+				p.print(")")
+			}
+		} else {
+			p.print("NaN")
+		}
+	} else if value == positiveInfinity || value == negativeInfinity {
+		// "with (x) Infinity" really means "x.Infinity" so avoid identifiers when "with" is present
+		wrap := ((p.options.MinifySyntax || p.withNesting != 0) && level >= js_ast.LMultiply) ||
+			(value == negativeInfinity && level >= js_ast.LPrefix)
+		if wrap {
+			p.print("(")
+		}
+		if value == negativeInfinity {
+			p.printSpaceBeforeOperator(js_ast.UnOpNeg)
+			p.print("-")
+		} else {
+			p.printSpaceBeforeIdentifier()
+		}
+		if !p.options.MinifySyntax && p.withNesting == 0 {
+			p.print("Infinity")
+		} else if p.options.MinifyWhitespace {
+			p.print("1/0")
+		} else {
+			p.print("1 / 0")
+		}
+		if wrap {
+			p.print(")")
+		}
+	} else {
+		if !math.Signbit(value) {
+			p.printSpaceBeforeIdentifier()
+			p.printNonNegativeFloat(absValue)
+		} else if level >= js_ast.LPrefix {
+			// Expressions such as "(-1).toString" need to wrap negative numbers.
+			// Instead of testing for "value < 0" we test for "signbit(value)" and
+			// "!isNaN(value)" because we need this to be true for "-0" and "-0 < 0"
+			// is false.
+			p.print("(-")
+			p.printNonNegativeFloat(absValue)
+			p.print(")")
+		} else {
+			p.printSpaceBeforeOperator(js_ast.UnOpNeg)
+			p.print("-")
+			p.printNonNegativeFloat(absValue)
+		}
+	}
+}
+
+func (p *printer) willPrintExprCommentsAtLoc(loc logger.Loc) bool {
+	return !p.options.MinifyWhitespace && p.exprComments[loc] != nil && !p.printedExprComments[loc]
+}
+
+func (p *printer) willPrintExprCommentsForAnyOf(exprs []js_ast.Expr) bool {
+	for _, expr := range exprs {
+		if p.willPrintExprCommentsAtLoc(expr.Loc) {
+			return true
+		}
+	}
+	return false
+}
+
+func (p *printer) printBinding(binding js_ast.Binding) {
+	switch b := binding.Data.(type) {
+	case *js_ast.BMissing:
+		p.addSourceMapping(binding.Loc)
+
+	case *js_ast.BIdentifier:
+		name := p.renamer.NameForSymbol(b.Ref)
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMappingForName(binding.Loc, name, b.Ref)
+		p.printIdentifier(name)
+
+	case *js_ast.BArray:
+		isMultiLine := (len(b.Items) > 0 && !b.IsSingleLine) || p.willPrintExprCommentsAtLoc(b.CloseBracketLoc)
+		if !p.options.MinifyWhitespace && !isMultiLine {
+			for _, item := range b.Items {
+				if p.willPrintExprCommentsAtLoc(item.Loc) {
+					isMultiLine = true
+					break
+				}
+			}
+		}
+		p.addSourceMapping(binding.Loc)
+		p.print("[")
+		if len(b.Items) > 0 || isMultiLine {
+			if isMultiLine {
+				p.options.Indent++
+			}
+
+			for i, item := range b.Items {
+				if i != 0 {
+					p.print(",")
+				}
+				if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit() {
+					if isMultiLine {
+						p.printNewline()
+						p.printIndent()
+					} else if i != 0 {
+						p.printSpace()
+					}
+				}
+				p.printExprCommentsAtLoc(item.Loc)
+				if b.HasSpread && i+1 == len(b.Items) {
+					p.addSourceMapping(item.Loc)
+					p.print("...")
+					p.printExprCommentsAtLoc(item.Binding.Loc)
+				}
+				p.printBinding(item.Binding)
+
+				if item.DefaultValueOrNil.Data != nil {
+					p.printSpace()
+					p.print("=")
+					p.printSpace()
+					p.printExprWithoutLeadingNewline(item.DefaultValueOrNil, js_ast.LComma, 0)
+				}
+
+				// Make sure there's a comma after trailing missing items
+				if _, ok := item.Binding.Data.(*js_ast.BMissing); ok && i == len(b.Items)-1 {
+					p.print(",")
+				}
+			}
+
+			if isMultiLine {
+				p.printNewline()
+				p.printExprCommentsAfterCloseTokenAtLoc(b.CloseBracketLoc)
+				p.options.Indent--
+				p.printIndent()
+			}
+		}
+		p.addSourceMapping(b.CloseBracketLoc)
+		p.print("]")
+
+	case *js_ast.BObject:
+		isMultiLine := (len(b.Properties) > 0 && !b.IsSingleLine) || p.willPrintExprCommentsAtLoc(b.CloseBraceLoc)
+		if !p.options.MinifyWhitespace && !isMultiLine {
+			for _, property := range b.Properties {
+				if p.willPrintExprCommentsAtLoc(property.Loc) {
+					isMultiLine = true
+					break
+				}
+			}
+		}
+		p.addSourceMapping(binding.Loc)
+		p.print("{")
+		if len(b.Properties) > 0 || isMultiLine {
+			if isMultiLine {
+				p.options.Indent++
+			}
+
+			for i, property := range b.Properties {
+				if i != 0 {
+					p.print(",")
+				}
+				if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit() {
+					if isMultiLine {
+						p.printNewline()
+						p.printIndent()
+					} else {
+						p.printSpace()
+					}
+				}
+
+				p.printExprCommentsAtLoc(property.Loc)
+
+				if property.IsSpread {
+					p.addSourceMapping(property.Loc)
+					p.print("...")
+					p.printExprCommentsAtLoc(property.Value.Loc)
+				} else {
+					if property.IsComputed {
+						p.addSourceMapping(property.Loc)
+						isMultiLine := p.willPrintExprCommentsAtLoc(property.Key.Loc) || p.willPrintExprCommentsAtLoc(property.CloseBracketLoc)
+						p.print("[")
+						if isMultiLine {
+							p.printNewline()
+							p.options.Indent++
+							p.printIndent()
+						}
+						p.printExpr(property.Key, js_ast.LComma, 0)
+						if isMultiLine {
+							p.printNewline()
+							p.printExprCommentsAfterCloseTokenAtLoc(property.CloseBracketLoc)
+							p.options.Indent--
+							p.printIndent()
+						}
+						if property.CloseBracketLoc.Start > property.Loc.Start {
+							p.addSourceMapping(property.CloseBracketLoc)
+						}
+						p.print("]:")
+						p.printSpace()
+						p.printBinding(property.Value)
+
+						if property.DefaultValueOrNil.Data != nil {
+							p.printSpace()
+							p.print("=")
+							p.printSpace()
+							p.printExprWithoutLeadingNewline(property.DefaultValueOrNil, js_ast.LComma, 0)
+						}
+						continue
+					}
+
+					if str, ok := property.Key.Data.(*js_ast.EString); ok && !property.PreferQuotedKey && p.canPrintIdentifierUTF16(str.Value) {
+						// Use a shorthand property if the names are the same
+						if id, ok := property.Value.Data.(*js_ast.BIdentifier); ok &&
+							!p.willPrintExprCommentsAtLoc(property.Value.Loc) &&
+							helpers.UTF16EqualsString(str.Value, p.renamer.NameForSymbol(id.Ref)) {
+							if p.options.AddSourceMappings {
+								p.addSourceMappingForName(property.Key.Loc, helpers.UTF16ToString(str.Value), id.Ref)
+							}
+							p.printIdentifierUTF16(str.Value)
+							if property.DefaultValueOrNil.Data != nil {
+								p.printSpace()
+								p.print("=")
+								p.printSpace()
+								p.printExprWithoutLeadingNewline(property.DefaultValueOrNil, js_ast.LComma, 0)
+							}
+							continue
+						}
+
+						p.addSourceMapping(property.Key.Loc)
+						p.printIdentifierUTF16(str.Value)
+					} else if mangled, ok := property.Key.Data.(*js_ast.ENameOfSymbol); ok {
+						if name := p.mangledPropName(mangled.Ref); p.canPrintIdentifier(name) {
+							p.addSourceMappingForName(property.Key.Loc, name, mangled.Ref)
+							p.printIdentifier(name)
+
+							// Use a shorthand property if the names are the same
+							if id, ok := property.Value.Data.(*js_ast.BIdentifier); ok &&
+								!p.willPrintExprCommentsAtLoc(property.Value.Loc) &&
+								name == p.renamer.NameForSymbol(id.Ref) {
+								if property.DefaultValueOrNil.Data != nil {
+									p.printSpace()
+									p.print("=")
+									p.printSpace()
+									p.printExprWithoutLeadingNewline(property.DefaultValueOrNil, js_ast.LComma, 0)
+								}
+								continue
+							}
+						} else {
+							p.addSourceMapping(property.Key.Loc)
+							p.printQuotedUTF8(name, 0)
+						}
+					} else {
+						p.printExpr(property.Key, js_ast.LLowest, 0)
+					}
+
+					p.print(":")
+					p.printSpace()
+				}
+				p.printBinding(property.Value)
+
+				if property.DefaultValueOrNil.Data != nil {
+					p.printSpace()
+					p.print("=")
+					p.printSpace()
+					p.printExprWithoutLeadingNewline(property.DefaultValueOrNil, js_ast.LComma, 0)
+				}
+			}
+
+			if isMultiLine {
+				p.printNewline()
+				p.printExprCommentsAfterCloseTokenAtLoc(b.CloseBraceLoc)
+				p.options.Indent--
+				p.printIndent()
+			} else {
+				// This block is only reached if len(b.Properties) > 0
+				p.printSpace()
+			}
+		}
+		p.addSourceMapping(b.CloseBraceLoc)
+		p.print("}")
+
+	default:
+		panic(fmt.Sprintf("Unexpected binding of type %T", binding.Data))
+	}
+}
+
+func (p *printer) printSpace() {
+	if !p.options.MinifyWhitespace {
+		p.print(" ")
+	}
+}
+
+func (p *printer) printNewline() {
+	if !p.options.MinifyWhitespace {
+		p.print("\n")
+	}
+}
+
+func (p *printer) currentLineLength() int {
+	js := p.js
+	n := len(js)
+	stop := p.oldLineEnd
+
+	// Update "oldLineStart" to the start of the current line
+	for i := n; i > stop; i-- {
+		if c := js[i-1]; c == '\r' || c == '\n' {
+			p.oldLineStart = i
+			break
+		}
+	}
+
+	p.oldLineEnd = n
+	return n - p.oldLineStart
+}
+
+func (p *printer) printNewlinePastLineLimit() bool {
+	if p.currentLineLength() < p.options.LineLimit {
+		return false
+	}
+	p.print("\n")
+	p.printIndent()
+	return true
+}
+
+func (p *printer) printSpaceBeforeOperator(next js_ast.OpCode) {
+	if p.prevOpEnd == len(p.js) {
+		prev := p.prevOp
+
+		// "+ + y" => "+ +y"
+		// "+ ++ y" => "+ ++y"
+		// "x + + y" => "x+ +y"
+		// "x ++ + y" => "x+++y"
+		// "x + ++ y" => "x+ ++y"
+		// "-- >" => "-- >"
+		// "< ! --" => "<! --"
+		if ((prev == js_ast.BinOpAdd || prev == js_ast.UnOpPos) && (next == js_ast.BinOpAdd || next == js_ast.UnOpPos || next == js_ast.UnOpPreInc)) ||
+			((prev == js_ast.BinOpSub || prev == js_ast.UnOpNeg) && (next == js_ast.BinOpSub || next == js_ast.UnOpNeg || next == js_ast.UnOpPreDec)) ||
+			(prev == js_ast.UnOpPostDec && next == js_ast.BinOpGt) ||
+			(prev == js_ast.UnOpNot && next == js_ast.UnOpPreDec && len(p.js) > 1 && p.js[len(p.js)-2] == '<') {
+			p.print(" ")
+		}
+	}
+}
+
+func (p *printer) printSemicolonAfterStatement() {
+	if !p.options.MinifyWhitespace {
+		p.print(";\n")
+	} else {
+		p.needsSemicolon = true
+	}
+}
+
+func (p *printer) printSemicolonIfNeeded() {
+	if p.needsSemicolon {
+		p.print(";")
+		p.needsSemicolon = false
+	}
+}
+
+func (p *printer) printSpaceBeforeIdentifier() {
+	if c, _ := utf8.DecodeLastRune(p.js); js_ast.IsIdentifierContinue(c) || p.prevRegExpEnd == len(p.js) {
+		p.print(" ")
+	}
+}
+
+type fnArgsOpts struct {
+	openParenLoc              logger.Loc
+	addMappingForOpenParenLoc bool
+	hasRestArg                bool
+	isArrow                   bool
+}
+
+func (p *printer) printFnArgs(args []js_ast.Arg, opts fnArgsOpts) {
+	wrap := true
+
+	// Minify "(a) => {}" as "a=>{}"
+	if p.options.MinifyWhitespace && !opts.hasRestArg && opts.isArrow && len(args) == 1 {
+		if _, ok := args[0].Binding.Data.(*js_ast.BIdentifier); ok && args[0].DefaultOrNil.Data == nil {
+			wrap = false
+		}
+	}
+
+	if wrap {
+		if opts.addMappingForOpenParenLoc {
+			p.addSourceMapping(opts.openParenLoc)
+		}
+		p.print("(")
+	}
+
+	for i, arg := range args {
+		if i != 0 {
+			p.print(",")
+			p.printSpace()
+		}
+		p.printDecorators(arg.Decorators, printSpaceAfterDecorator)
+		if opts.hasRestArg && i+1 == len(args) {
+			p.print("...")
+		}
+		p.printBinding(arg.Binding)
+
+		if arg.DefaultOrNil.Data != nil {
+			p.printSpace()
+			p.print("=")
+			p.printSpace()
+			p.printExprWithoutLeadingNewline(arg.DefaultOrNil, js_ast.LComma, 0)
+		}
+	}
+
+	if wrap {
+		p.print(")")
+	}
+}
+
+func (p *printer) printFn(fn js_ast.Fn) {
+	p.printFnArgs(fn.Args, fnArgsOpts{hasRestArg: fn.HasRestArg})
+	p.printSpace()
+	p.printBlock(fn.Body.Loc, fn.Body.Block)
+}
+
+type printAfterDecorator uint8
+
+const (
+	printNewlineAfterDecorator printAfterDecorator = iota
+	printSpaceAfterDecorator
+)
+
+func (p *printer) printDecorators(decorators []js_ast.Decorator, defaultMode printAfterDecorator) (omitIndentAfter bool) {
+	oldMode := defaultMode
+
+	for _, decorator := range decorators {
+		wrap := false
+		wasCallTarget := false
+		expr := decorator.Value
+		mode := defaultMode
+		if decorator.OmitNewlineAfter {
+			mode = printSpaceAfterDecorator
+		}
+
+	outer:
+		for {
+			isCallTarget := wasCallTarget
+			wasCallTarget = false
+
+			switch e := expr.Data.(type) {
+			case *js_ast.EIdentifier:
+				// "@foo"
+				break outer
+
+			case *js_ast.ECall:
+				// "@foo()"
+				expr = e.Target
+				wasCallTarget = true
+				continue
+
+			case *js_ast.EDot:
+				// "@foo.bar"
+				if p.canPrintIdentifier(e.Name) {
+					expr = e.Target
+					continue
+				}
+
+				// "@foo.\u30FF" => "@(foo['\u30FF'])"
+				break
+
+			case *js_ast.EIndex:
+				if _, ok := e.Index.Data.(*js_ast.EPrivateIdentifier); ok {
+					// "@foo.#bar"
+					expr = e.Target
+					continue
+				}
+
+				// "@(foo[bar])"
+				break
+
+			case *js_ast.EImportIdentifier:
+				ref := ast.FollowSymbols(p.symbols, e.Ref)
+				symbol := p.symbols.Get(ref)
+
+				if symbol.ImportItemStatus == ast.ImportItemMissing {
+					// "@(void 0)"
+					break
+				}
+
+				if symbol.NamespaceAlias != nil && isCallTarget && e.WasOriginallyIdentifier {
+					// "@((0, import_ns.fn)())"
+					break
+				}
+
+				if value := p.options.ConstValues[ref]; value.Kind != js_ast.ConstValueNone {
+					// "@(<inlined constant>)"
+					break
+				}
+
+				// "@foo"
+				// "@import_ns.fn"
+				break outer
+
+			default:
+				// "@(foo + bar)"
+				// "@(() => {})"
+				break
+			}
+
+			wrap = true
+			break outer
+		}
+
+		p.addSourceMapping(decorator.AtLoc)
+		if oldMode == printNewlineAfterDecorator {
+			p.printIndent()
+		}
+
+		p.print("@")
+		if wrap {
+			p.print("(")
+		}
+		p.printExpr(decorator.Value, js_ast.LLowest, 0)
+		if wrap {
+			p.print(")")
+		}
+
+		switch mode {
+		case printNewlineAfterDecorator:
+			p.printNewline()
+
+		case printSpaceAfterDecorator:
+			p.printSpace()
+		}
+		oldMode = mode
+	}
+
+	omitIndentAfter = oldMode == printSpaceAfterDecorator
+	return
+}
+
+func (p *printer) printClass(class js_ast.Class) {
+	if class.ExtendsOrNil.Data != nil {
+		p.print(" extends")
+		p.printSpace()
+		p.printExpr(class.ExtendsOrNil, js_ast.LNew-1, 0)
+	}
+	p.printSpace()
+
+	p.addSourceMapping(class.BodyLoc)
+	p.print("{")
+	p.printNewline()
+	p.options.Indent++
+
+	for _, item := range class.Properties {
+		p.printSemicolonIfNeeded()
+		omitIndent := p.printDecorators(item.Decorators, printNewlineAfterDecorator)
+		if !omitIndent {
+			p.printIndent()
+		}
+
+		if item.Kind == js_ast.PropertyClassStaticBlock {
+			p.addSourceMapping(item.Loc)
+			p.print("static")
+			p.printSpace()
+			p.printBlock(item.ClassStaticBlock.Loc, item.ClassStaticBlock.Block)
+			p.printNewline()
+			continue
+		}
+
+		p.printProperty(item)
+
+		// Need semicolons after class fields
+		if item.ValueOrNil.Data == nil {
+			p.printSemicolonAfterStatement()
+		} else {
+			p.printNewline()
+		}
+	}
+
+	p.needsSemicolon = false
+	p.printExprCommentsAfterCloseTokenAtLoc(class.CloseBraceLoc)
+	p.options.Indent--
+	p.printIndent()
+	if class.CloseBraceLoc.Start > class.BodyLoc.Start {
+		p.addSourceMapping(class.CloseBraceLoc)
+	}
+	p.print("}")
+}
+
+func (p *printer) printProperty(property js_ast.Property) {
+	p.printExprCommentsAtLoc(property.Loc)
+
+	if property.Kind == js_ast.PropertySpread {
+		p.addSourceMapping(property.Loc)
+		p.print("...")
+		p.printExpr(property.ValueOrNil, js_ast.LComma, 0)
+		return
+	}
+
+	// Handle key syntax compression for cross-module constant inlining of enums
+	if p.options.MinifySyntax && property.Flags.Has(js_ast.PropertyIsComputed) {
+		if dot, ok := property.Key.Data.(*js_ast.EDot); ok {
+			if value, ok := p.tryToGetImportedEnumValue(dot.Target, dot.Name); ok {
+				if value.String != nil {
+					property.Key.Data = &js_ast.EString{Value: value.String}
+
+					// Problematic key names must stay computed for correctness
+					if !helpers.UTF16EqualsString(value.String, "__proto__") &&
+						!helpers.UTF16EqualsString(value.String, "constructor") &&
+						!helpers.UTF16EqualsString(value.String, "prototype") {
+						property.Flags &= ^js_ast.PropertyIsComputed
+					}
+				} else {
+					property.Key.Data = &js_ast.ENumber{Value: value.Number}
+					property.Flags &= ^js_ast.PropertyIsComputed
+				}
+			}
+		}
+	}
+
+	if property.Flags.Has(js_ast.PropertyIsStatic) {
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(property.Loc)
+		p.print("static")
+		p.printSpace()
+	}
+
+	switch property.Kind {
+	case js_ast.PropertyGetter:
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(property.Loc)
+		p.print("get")
+		p.printSpace()
+
+	case js_ast.PropertySetter:
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(property.Loc)
+		p.print("set")
+		p.printSpace()
+
+	case js_ast.PropertyAutoAccessor:
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(property.Loc)
+		p.print("accessor")
+		p.printSpace()
+	}
+
+	if fn, ok := property.ValueOrNil.Data.(*js_ast.EFunction); property.Kind.IsMethodDefinition() && ok {
+		if fn.Fn.IsAsync {
+			p.printSpaceBeforeIdentifier()
+			p.addSourceMapping(property.Loc)
+			p.print("async")
+			p.printSpace()
+		}
+		if fn.Fn.IsGenerator {
+			p.addSourceMapping(property.Loc)
+			p.print("*")
+		}
+	}
+
+	isComputed := property.Flags.Has(js_ast.PropertyIsComputed)
+
+	// Automatically print numbers that would cause a syntax error as computed properties
+	if !isComputed {
+		if key, ok := property.Key.Data.(*js_ast.ENumber); ok {
+			if math.Signbit(key.Value) || (key.Value == positiveInfinity && p.options.MinifySyntax) {
+				// "{ -1: 0 }" must be printed as "{ [-1]: 0 }"
+				// "{ 1/0: 0 }" must be printed as "{ [1/0]: 0 }"
+				isComputed = true
+			}
+		}
+	}
+
+	if isComputed {
+		p.addSourceMapping(property.Loc)
+		isMultiLine := p.willPrintExprCommentsAtLoc(property.Key.Loc) || p.willPrintExprCommentsAtLoc(property.CloseBracketLoc)
+		p.print("[")
+		if isMultiLine {
+			p.printNewline()
+			p.options.Indent++
+			p.printIndent()
+		}
+		p.printExpr(property.Key, js_ast.LComma, 0)
+		if isMultiLine {
+			p.printNewline()
+			p.printExprCommentsAfterCloseTokenAtLoc(property.CloseBracketLoc)
+			p.options.Indent--
+			p.printIndent()
+		}
+		if property.CloseBracketLoc.Start > property.Loc.Start {
+			p.addSourceMapping(property.CloseBracketLoc)
+		}
+		p.print("]")
+
+		if property.ValueOrNil.Data != nil {
+			if fn, ok := property.ValueOrNil.Data.(*js_ast.EFunction); property.Kind.IsMethodDefinition() && ok {
+				p.printFn(fn.Fn)
+				return
+			}
+
+			p.print(":")
+			p.printSpace()
+			p.printExprWithoutLeadingNewline(property.ValueOrNil, js_ast.LComma, 0)
+		}
+
+		if property.InitializerOrNil.Data != nil {
+			p.printSpace()
+			p.print("=")
+			p.printSpace()
+			p.printExprWithoutLeadingNewline(property.InitializerOrNil, js_ast.LComma, 0)
+		}
+		return
+	}
+
+	switch key := property.Key.Data.(type) {
+	case *js_ast.EPrivateIdentifier:
+		name := p.renamer.NameForSymbol(key.Ref)
+		p.addSourceMappingForName(property.Key.Loc, name, key.Ref)
+		p.printIdentifier(name)
+
+	case *js_ast.ENameOfSymbol:
+		if name := p.mangledPropName(key.Ref); p.canPrintIdentifier(name) {
+			p.printSpaceBeforeIdentifier()
+			p.addSourceMappingForName(property.Key.Loc, name, key.Ref)
+			p.printIdentifier(name)
+
+			// Use a shorthand property if the names are the same
+			if !p.options.UnsupportedFeatures.Has(compat.ObjectExtensions) && property.ValueOrNil.Data != nil && !p.willPrintExprCommentsAtLoc(property.ValueOrNil.Loc) {
+				switch e := property.ValueOrNil.Data.(type) {
+				case *js_ast.EIdentifier:
+					if name == p.renamer.NameForSymbol(e.Ref) {
+						if property.InitializerOrNil.Data != nil {
+							p.printSpace()
+							p.print("=")
+							p.printSpace()
+							p.printExprWithoutLeadingNewline(property.InitializerOrNil, js_ast.LComma, 0)
+						}
+						return
+					}
+
+				case *js_ast.EImportIdentifier:
+					// Make sure we're not using a property access instead of an identifier
+					ref := ast.FollowSymbols(p.symbols, e.Ref)
+					if symbol := p.symbols.Get(ref); symbol.NamespaceAlias == nil && name == p.renamer.NameForSymbol(ref) &&
+						p.options.ConstValues[ref].Kind == js_ast.ConstValueNone {
+						if property.InitializerOrNil.Data != nil {
+							p.printSpace()
+							p.print("=")
+							p.printSpace()
+							p.printExprWithoutLeadingNewline(property.InitializerOrNil, js_ast.LComma, 0)
+						}
+						return
+					}
+				}
+			}
+		} else {
+			p.addSourceMapping(property.Key.Loc)
+			p.printQuotedUTF8(name, 0)
+		}
+
+	case *js_ast.EString:
+		if !property.Flags.Has(js_ast.PropertyPreferQuotedKey) && p.canPrintIdentifierUTF16(key.Value) {
+			p.printSpaceBeforeIdentifier()
+
+			// Use a shorthand property if the names are the same
+			if !p.options.UnsupportedFeatures.Has(compat.ObjectExtensions) && property.ValueOrNil.Data != nil && !p.willPrintExprCommentsAtLoc(property.ValueOrNil.Loc) {
+				switch e := property.ValueOrNil.Data.(type) {
+				case *js_ast.EIdentifier:
+					if canUseShorthandProperty(key.Value, p.renamer.NameForSymbol(e.Ref), property.Flags) {
+						if p.options.AddSourceMappings {
+							p.addSourceMappingForName(property.Key.Loc, helpers.UTF16ToString(key.Value), e.Ref)
+						}
+						p.printIdentifierUTF16(key.Value)
+						if property.InitializerOrNil.Data != nil {
+							p.printSpace()
+							p.print("=")
+							p.printSpace()
+							p.printExprWithoutLeadingNewline(property.InitializerOrNil, js_ast.LComma, 0)
+						}
+						return
+					}
+
+				case *js_ast.EImportIdentifier:
+					// Make sure we're not using a property access instead of an identifier
+					ref := ast.FollowSymbols(p.symbols, e.Ref)
+					if symbol := p.symbols.Get(ref); symbol.NamespaceAlias == nil && canUseShorthandProperty(key.Value, p.renamer.NameForSymbol(ref), property.Flags) &&
+						p.options.ConstValues[ref].Kind == js_ast.ConstValueNone {
+						if p.options.AddSourceMappings {
+							p.addSourceMappingForName(property.Key.Loc, helpers.UTF16ToString(key.Value), ref)
+						}
+						p.printIdentifierUTF16(key.Value)
+						if property.InitializerOrNil.Data != nil {
+							p.printSpace()
+							p.print("=")
+							p.printSpace()
+							p.printExprWithoutLeadingNewline(property.InitializerOrNil, js_ast.LComma, 0)
+						}
+						return
+					}
+				}
+			}
+
+			// The JavaScript specification special-cases the property identifier
+			// "__proto__" with a colon after it to set the prototype of the object.
+			// If we keep the identifier but add a colon then we'll cause a behavior
+			// change because the prototype will now be set. Avoid using an identifier
+			// by using a computed property with a string instead. For more info see:
+			// https://tc39.es/ecma262/#sec-runtime-semantics-propertydefinitionevaluation
+			if property.Flags.Has(js_ast.PropertyWasShorthand) && !p.options.UnsupportedFeatures.Has(compat.ObjectExtensions) &&
+				helpers.UTF16EqualsString(key.Value, "__proto__") {
+				p.print("[")
+				p.addSourceMapping(property.Key.Loc)
+				p.printQuotedUTF16(key.Value, 0)
+				p.print("]")
+				break
+			}
+
+			p.addSourceMapping(property.Key.Loc)
+			p.printIdentifierUTF16(key.Value)
+		} else {
+			p.addSourceMapping(property.Key.Loc)
+			p.printQuotedUTF16(key.Value, 0)
+		}
+
+	default:
+		p.printExpr(property.Key, js_ast.LLowest, 0)
+	}
+
+	if fn, ok := property.ValueOrNil.Data.(*js_ast.EFunction); property.Kind.IsMethodDefinition() && ok {
+		p.printFn(fn.Fn)
+		return
+	}
+
+	if property.ValueOrNil.Data != nil {
+		p.print(":")
+		p.printSpace()
+		p.printExprWithoutLeadingNewline(property.ValueOrNil, js_ast.LComma, 0)
+	}
+
+	if property.InitializerOrNil.Data != nil {
+		p.printSpace()
+		p.print("=")
+		p.printSpace()
+		p.printExprWithoutLeadingNewline(property.InitializerOrNil, js_ast.LComma, 0)
+	}
+}
+
+func canUseShorthandProperty(key []uint16, name string, flags js_ast.PropertyFlags) bool {
+	// The JavaScript specification special-cases the property identifier
+	// "__proto__" with a colon after it to set the prototype of the object. If
+	// we remove the colon then we'll cause a behavior change because the
+	// prototype will no longer be set, but we also don't want to add a colon
+	// if it was omitted. Always use a shorthand property if the property is not
+	// "__proto__", otherwise try to preserve the original shorthand status. See:
+	// https://tc39.es/ecma262/#sec-runtime-semantics-propertydefinitionevaluation
+	if !helpers.UTF16EqualsString(key, name) {
+		return false
+	}
+	return helpers.UTF16EqualsString(key, name) && (name != "__proto__" || flags.Has(js_ast.PropertyWasShorthand))
+}
+
+func (p *printer) printQuotedUTF16(data []uint16, flags printQuotedFlags) {
+	if p.options.UnsupportedFeatures.Has(compat.TemplateLiteral) {
+		flags &= ^printQuotedAllowBacktick
+	}
+
+	singleCost := 0
+	doubleCost := 0
+	backtickCost := 0
+
+	for i, c := range data {
+		switch c {
+		case '\n':
+			if p.options.MinifySyntax {
+				// The backslash for the newline costs an extra character for old-style
+				// string literals when compared to a template literal
+				backtickCost--
+			}
+		case '\'':
+			singleCost++
+		case '"':
+			doubleCost++
+		case '`':
+			backtickCost++
+		case '$':
+			// "${" sequences need to be escaped in template literals
+			if i+1 < len(data) && data[i+1] == '{' {
+				backtickCost++
+			}
+		}
+	}
+
+	c := "\""
+	if doubleCost > singleCost {
+		c = "'"
+		if singleCost > backtickCost && (flags&printQuotedAllowBacktick) != 0 {
+			c = "`"
+		}
+	} else if doubleCost > backtickCost && (flags&printQuotedAllowBacktick) != 0 {
+		c = "`"
+	}
+
+	p.print(c)
+	p.printUnquotedUTF16(data, rune(c[0]), flags)
+	p.print(c)
+}
+
+func (p *printer) printRequireOrImportExpr(importRecordIndex uint32, level js_ast.L, flags printExprFlags, closeParenLoc logger.Loc) {
+	record := &p.importRecords[importRecordIndex]
+
+	if level >= js_ast.LNew || (flags&forbidCall) != 0 {
+		p.print("(")
+		defer p.print(")")
+		level = js_ast.LLowest
+	}
+
+	if !record.SourceIndex.IsValid() {
+		// External "require()"
+		if record.Kind != ast.ImportDynamic {
+			// Wrap this with a call to "__toESM()" if this is a CommonJS file
+			wrapWithToESM := record.Flags.Has(ast.WrapWithToESM)
+			if wrapWithToESM {
+				p.printSpaceBeforeIdentifier()
+				p.printIdentifier(p.renamer.NameForSymbol(p.options.ToESMRef))
+				p.print("(")
+			}
+
+			// Potentially substitute our own "__require" stub for "require"
+			p.printSpaceBeforeIdentifier()
+			if record.Flags.Has(ast.CallRuntimeRequire) {
+				p.printIdentifier(p.renamer.NameForSymbol(p.options.RuntimeRequireRef))
+			} else {
+				p.print("require")
+			}
+
+			isMultiLine := p.willPrintExprCommentsAtLoc(record.Range.Loc) || p.willPrintExprCommentsAtLoc(closeParenLoc)
+			p.print("(")
+			if isMultiLine {
+				p.printNewline()
+				p.options.Indent++
+				p.printIndent()
+			}
+			p.printExprCommentsAtLoc(record.Range.Loc)
+			p.printPath(importRecordIndex, ast.ImportRequire)
+			if isMultiLine {
+				p.printNewline()
+				p.printExprCommentsAfterCloseTokenAtLoc(closeParenLoc)
+				p.options.Indent--
+				p.printIndent()
+			}
+			if closeParenLoc.Start > record.Range.Loc.Start {
+				p.addSourceMapping(closeParenLoc)
+			}
+			p.print(")")
+
+			// Finish the call to "__toESM()"
+			if wrapWithToESM {
+				if p.moduleType.IsESM() {
+					p.print(",")
+					p.printSpace()
+					p.print("1")
+				}
+				p.print(")")
+			}
+			return
+		}
+
+		// External "import()"
+		kind := ast.ImportDynamic
+		if !p.options.UnsupportedFeatures.Has(compat.DynamicImport) {
+			p.printSpaceBeforeIdentifier()
+			p.print("import(")
+		} else {
+			kind = ast.ImportRequire
+			p.printSpaceBeforeIdentifier()
+			p.print("Promise.resolve()")
+			p.printDotThenPrefix()
+			defer p.printDotThenSuffix()
+
+			// Wrap this with a call to "__toESM()" if this is a CommonJS file
+			if record.Flags.Has(ast.WrapWithToESM) {
+				p.printSpaceBeforeIdentifier()
+				p.printIdentifier(p.renamer.NameForSymbol(p.options.ToESMRef))
+				p.print("(")
+				defer func() {
+					if p.moduleType.IsESM() {
+						p.print(",")
+						p.printSpace()
+						p.print("1")
+					}
+					p.print(")")
+				}()
+			}
+
+			// Potentially substitute our own "__require" stub for "require"
+			p.printSpaceBeforeIdentifier()
+			if record.Flags.Has(ast.CallRuntimeRequire) {
+				p.printIdentifier(p.renamer.NameForSymbol(p.options.RuntimeRequireRef))
+			} else {
+				p.print("require")
+			}
+
+			p.print("(")
+		}
+		isMultiLine := p.willPrintExprCommentsAtLoc(record.Range.Loc) ||
+			p.willPrintExprCommentsAtLoc(closeParenLoc) ||
+			(record.AssertOrWith != nil &&
+				!p.options.UnsupportedFeatures.Has(compat.DynamicImport) &&
+				(!p.options.UnsupportedFeatures.Has(compat.ImportAssertions) ||
+					!p.options.UnsupportedFeatures.Has(compat.ImportAttributes)) &&
+				p.willPrintExprCommentsAtLoc(record.AssertOrWith.OuterOpenBraceLoc))
+		if isMultiLine {
+			p.printNewline()
+			p.options.Indent++
+			p.printIndent()
+		}
+		p.printExprCommentsAtLoc(record.Range.Loc)
+		p.printPath(importRecordIndex, kind)
+		if !p.options.UnsupportedFeatures.Has(compat.DynamicImport) {
+			p.printImportCallAssertOrWith(record.AssertOrWith, isMultiLine)
+		}
+		if isMultiLine {
+			p.printNewline()
+			p.printExprCommentsAfterCloseTokenAtLoc(closeParenLoc)
+			p.options.Indent--
+			p.printIndent()
+		}
+		if closeParenLoc.Start > record.Range.Loc.Start {
+			p.addSourceMapping(closeParenLoc)
+		}
+		p.print(")")
+		return
+	}
+
+	meta := p.options.RequireOrImportMetaForSource(record.SourceIndex.GetIndex())
+
+	// Don't need the namespace object if the result is unused anyway
+	if (flags & exprResultIsUnused) != 0 {
+		meta.ExportsRef = ast.InvalidRef
+	}
+
+	// Internal "import()" of async ESM
+	if record.Kind == ast.ImportDynamic && meta.IsWrapperAsync {
+		p.printSpaceBeforeIdentifier()
+		p.printIdentifier(p.renamer.NameForSymbol(meta.WrapperRef))
+		p.print("()")
+		if meta.ExportsRef != ast.InvalidRef {
+			p.printDotThenPrefix()
+			p.printSpaceBeforeIdentifier()
+			p.printIdentifier(p.renamer.NameForSymbol(meta.ExportsRef))
+			p.printDotThenSuffix()
+		}
+		return
+	}
+
+	// Internal "require()" or "import()"
+	if record.Kind == ast.ImportDynamic {
+		p.printSpaceBeforeIdentifier()
+		p.print("Promise.resolve()")
+		level = p.printDotThenPrefix()
+		defer p.printDotThenSuffix()
+	}
+
+	// Make sure the comma operator is properly wrapped
+	if meta.ExportsRef != ast.InvalidRef && level >= js_ast.LComma {
+		p.print("(")
+		defer p.print(")")
+	}
+
+	// Wrap this with a call to "__toESM()" if this is a CommonJS file
+	wrapWithToESM := record.Flags.Has(ast.WrapWithToESM)
+	if wrapWithToESM {
+		p.printSpaceBeforeIdentifier()
+		p.printIdentifier(p.renamer.NameForSymbol(p.options.ToESMRef))
+		p.print("(")
+	}
+
+	// Call the wrapper
+	p.printSpaceBeforeIdentifier()
+	p.printIdentifier(p.renamer.NameForSymbol(meta.WrapperRef))
+	p.print("()")
+
+	// Return the namespace object if this is an ESM file
+	if meta.ExportsRef != ast.InvalidRef {
+		p.print(",")
+		p.printSpace()
+
+		// Wrap this with a call to "__toCommonJS()" if this is an ESM file
+		wrapWithTpCJS := record.Flags.Has(ast.WrapWithToCJS)
+		if wrapWithTpCJS {
+			p.printIdentifier(p.renamer.NameForSymbol(p.options.ToCommonJSRef))
+			p.print("(")
+		}
+		p.printIdentifier(p.renamer.NameForSymbol(meta.ExportsRef))
+		if wrapWithTpCJS {
+			p.print(")")
+		}
+	}
+
+	// Finish the call to "__toESM()"
+	if wrapWithToESM {
+		if p.moduleType.IsESM() {
+			p.print(",")
+			p.printSpace()
+			p.print("1")
+		}
+		p.print(")")
+	}
+}
+
+func (p *printer) printDotThenPrefix() js_ast.L {
+	if p.options.UnsupportedFeatures.Has(compat.Arrow) {
+		p.print(".then(function()")
+		p.printSpace()
+		p.print("{")
+		p.printNewline()
+		p.options.Indent++
+		p.printIndent()
+		p.print("return")
+		p.printSpace()
+		return js_ast.LLowest
+	} else {
+		p.print(".then(()")
+		p.printSpace()
+		p.print("=>")
+		p.printSpace()
+		return js_ast.LComma
+	}
+}
+
+func (p *printer) printDotThenSuffix() {
+	if p.options.UnsupportedFeatures.Has(compat.Arrow) {
+		if !p.options.MinifyWhitespace {
+			p.print(";")
+		}
+		p.printNewline()
+		p.options.Indent--
+		p.printIndent()
+		p.print("})")
+	} else {
+		p.print(")")
+	}
+}
+
+func (p *printer) printUndefined(loc logger.Loc, level js_ast.L) {
+	if level >= js_ast.LPrefix {
+		p.addSourceMapping(loc)
+		p.print("(void 0)")
+	} else {
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(loc)
+		p.print("void 0")
+	}
+}
+
+// Call this before printing an expression to see if it turned out to be empty.
+// We use this to do inlining of empty functions at print time. It can't happen
+// during parse time because a) parse time only has two passes and we only know
+// if a function can be inlined at the end of the second pass (due to is-mutated
+// analysis) and b) we want to enable cross-module inlining of empty functions
+// which has to happen after linking.
+//
+// This function returns "nil" to indicate that the expression should be removed
+// completely.
+//
+// This function doesn't need to search everywhere inside the entire expression
+// for calls to inline. Calls are automatically inlined when printed. However,
+// the printer replaces the call with "undefined" since the result may still
+// be needed by the caller. If the caller knows that it doesn't need the result,
+// it should call this function first instead so we don't print "undefined".
+//
+// This is a separate function instead of trying to work this logic into the
+// printer because it's too late to eliminate the expression entirely when we're
+// in the printer. We may have already printed the leading indent, for example.
+func (p *printer) simplifyUnusedExpr(expr js_ast.Expr) js_ast.Expr {
+	switch e := expr.Data.(type) {
+	case *js_ast.EBinary:
+		// Calls to be inlined may be hidden inside a comma operator chain
+		if e.Op == js_ast.BinOpComma {
+			left := p.simplifyUnusedExpr(e.Left)
+			right := p.simplifyUnusedExpr(e.Right)
+			if left.Data != e.Left.Data || right.Data != e.Right.Data {
+				return js_ast.JoinWithComma(left, right)
+			}
+		}
+
+	case *js_ast.ECall:
+		var symbolFlags ast.SymbolFlags
+		switch target := e.Target.Data.(type) {
+		case *js_ast.EIdentifier:
+			symbolFlags = p.symbols.Get(target.Ref).Flags
+		case *js_ast.EImportIdentifier:
+			ref := ast.FollowSymbols(p.symbols, target.Ref)
+			symbolFlags = p.symbols.Get(ref).Flags
+		}
+
+		// Replace non-mutated empty functions with their arguments at print time
+		if (symbolFlags & (ast.IsEmptyFunction | ast.CouldPotentiallyBeMutated)) == ast.IsEmptyFunction {
+			var replacement js_ast.Expr
+			for _, arg := range e.Args {
+				if _, ok := arg.Data.(*js_ast.ESpread); ok {
+					arg.Data = &js_ast.EArray{Items: []js_ast.Expr{arg}, IsSingleLine: true}
+				}
+				replacement = js_ast.JoinWithComma(replacement, p.astHelpers.SimplifyUnusedExpr(p.simplifyUnusedExpr(arg), p.options.UnsupportedFeatures))
+			}
+			return replacement // Don't add "undefined" here because the result isn't used
+		}
+
+		// Inline non-mutated identity functions at print time
+		if (symbolFlags&(ast.IsIdentityFunction|ast.CouldPotentiallyBeMutated)) == ast.IsIdentityFunction && len(e.Args) == 1 {
+			arg := e.Args[0]
+			if _, ok := arg.Data.(*js_ast.ESpread); !ok {
+				return p.astHelpers.SimplifyUnusedExpr(p.simplifyUnusedExpr(arg), p.options.UnsupportedFeatures)
+			}
+		}
+	}
+
+	return expr
+}
+
+// This assumes the original expression was some form of indirect value, such
+// as a value returned from a function call or the result of a comma operator.
+// In this case, there is no special behavior with the "delete" operator or
+// with function calls. If we substitute this indirect value for another value
+// due to inlining, we have to make sure we don't accidentally introduce special
+// behavior.
+func (p *printer) guardAgainstBehaviorChangeDueToSubstitution(expr js_ast.Expr, flags printExprFlags) js_ast.Expr {
+	wrap := false
+
+	if (flags & isDeleteTarget) != 0 {
+		// "delete id(x)" must not become "delete x"
+		// "delete (empty(), x)" must not become "delete x"
+		if binary, ok := expr.Data.(*js_ast.EBinary); !ok || binary.Op != js_ast.BinOpComma {
+			wrap = true
+		}
+	} else if (flags & isCallTargetOrTemplateTag) != 0 {
+		// "id(x.y)()" must not become "x.y()"
+		// "id(x.y)``" must not become "x.y``"
+		// "(empty(), x.y)()" must not become "x.y()"
+		// "(empty(), eval)()" must not become "eval()"
+		switch expr.Data.(type) {
+		case *js_ast.EDot, *js_ast.EIndex:
+			wrap = true
+		case *js_ast.EIdentifier:
+			if p.isUnboundEvalIdentifier(expr) {
+				wrap = true
+			}
+		}
+	}
+
+	if wrap {
+		expr.Data = &js_ast.EBinary{
+			Op:    js_ast.BinOpComma,
+			Left:  js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENumber{Value: 0}},
+			Right: expr,
+		}
+	}
+
+	return expr
+}
+
+// Constant folding is already implemented once in the parser. A smaller form
+// of constant folding (just for numbers) is implemented here to clean up cross-
+// module numeric constants and bitwise operations. This is not an general-
+// purpose/optimal approach and never will be. For example, we can't affect
+// tree shaking at this stage because it has already happened.
+func (p *printer) lateConstantFoldUnaryOrBinaryOrIfExpr(expr js_ast.Expr) js_ast.Expr {
+	switch e := expr.Data.(type) {
+	case *js_ast.EImportIdentifier:
+		ref := ast.FollowSymbols(p.symbols, e.Ref)
+		if value := p.options.ConstValues[ref]; value.Kind != js_ast.ConstValueNone {
+			return js_ast.ConstValueToExpr(expr.Loc, value)
+		}
+
+	case *js_ast.EDot:
+		if value, ok := p.tryToGetImportedEnumValue(e.Target, e.Name); ok {
+			var inlinedValue js_ast.Expr
+			if value.String != nil {
+				inlinedValue = js_ast.Expr{Loc: expr.Loc, Data: &js_ast.EString{Value: value.String}}
+			} else {
+				inlinedValue = js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENumber{Value: value.Number}}
+			}
+
+			if strings.Contains(e.Name, "*/") {
+				// Don't wrap with a comment
+				return inlinedValue
+			}
+
+			// Wrap with a comment
+			return js_ast.Expr{Loc: inlinedValue.Loc, Data: &js_ast.EInlinedEnum{
+				Value:   inlinedValue,
+				Comment: e.Name,
+			}}
+		}
+
+	case *js_ast.EUnary:
+		value := p.lateConstantFoldUnaryOrBinaryOrIfExpr(e.Value)
+
+		// Only fold again if something chained
+		if value.Data != e.Value.Data {
+			// Only fold certain operations (just like the parser)
+			if v, ok := js_ast.ToNumberWithoutSideEffects(value.Data); ok {
+				switch e.Op {
+				case js_ast.UnOpPos:
+					return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENumber{Value: v}}
+
+				case js_ast.UnOpNeg:
+					return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENumber{Value: -v}}
+
+				case js_ast.UnOpCpl:
+					return js_ast.Expr{Loc: expr.Loc, Data: &js_ast.ENumber{Value: float64(^js_ast.ToInt32(v))}}
+				}
+			}
+
+			// Don't mutate the original AST
+			expr.Data = &js_ast.EUnary{Op: e.Op, Value: value}
+		}
+
+	case *js_ast.EBinary:
+		left := p.lateConstantFoldUnaryOrBinaryOrIfExpr(e.Left)
+		right := p.lateConstantFoldUnaryOrBinaryOrIfExpr(e.Right)
+
+		// Only fold again if something changed
+		if left.Data != e.Left.Data || right.Data != e.Right.Data {
+			binary := &js_ast.EBinary{Op: e.Op, Left: left, Right: right}
+
+			// Only fold certain operations (just like the parser)
+			if js_ast.ShouldFoldBinaryOperatorWhenMinifying(binary) {
+				if result := js_ast.FoldBinaryOperator(expr.Loc, binary); result.Data != nil {
+					return result
+				}
+			}
+
+			// Don't mutate the original AST
+			expr.Data = binary
+		}
+
+	case *js_ast.EIf:
+		test := p.lateConstantFoldUnaryOrBinaryOrIfExpr(e.Test)
+
+		// Only fold again if something changed
+		if test.Data != e.Test.Data {
+			if boolean, sideEffects, ok := js_ast.ToBooleanWithSideEffects(test.Data); ok && sideEffects == js_ast.NoSideEffects {
+				if boolean {
+					return p.lateConstantFoldUnaryOrBinaryOrIfExpr(e.Yes)
+				} else {
+					return p.lateConstantFoldUnaryOrBinaryOrIfExpr(e.No)
+				}
+			}
+
+			// Don't mutate the original AST
+			expr.Data = &js_ast.EIf{Test: test, Yes: e.Yes, No: e.No}
+		}
+	}
+
+	return expr
+}
+
+func (p *printer) isUnboundIdentifier(expr js_ast.Expr) bool {
+	id, ok := expr.Data.(*js_ast.EIdentifier)
+	return ok && p.symbols.Get(ast.FollowSymbols(p.symbols, id.Ref)).Kind == ast.SymbolUnbound
+}
+
+func (p *printer) isIdentifierOrNumericConstantOrPropertyAccess(expr js_ast.Expr) bool {
+	switch e := expr.Data.(type) {
+	case *js_ast.EIdentifier, *js_ast.EDot, *js_ast.EIndex:
+		return true
+	case *js_ast.ENumber:
+		return math.IsInf(e.Value, 1) || math.IsNaN(e.Value)
+	}
+	return false
+}
+
+type exprStartFlags uint8
+
+const (
+	stmtStartFlag exprStartFlags = 1 << iota
+	exportDefaultStartFlag
+	arrowExprStartFlag
+	forOfInitStartFlag
+)
+
+func (p *printer) saveExprStartFlags() (flags exprStartFlags) {
+	n := len(p.js)
+	if p.stmtStart == n {
+		flags |= stmtStartFlag
+	}
+	if p.exportDefaultStart == n {
+		flags |= exportDefaultStartFlag
+	}
+	if p.arrowExprStart == n {
+		flags |= arrowExprStartFlag
+	}
+	if p.forOfInitStart == n {
+		flags |= forOfInitStartFlag
+	}
+	return
+}
+
+func (p *printer) restoreExprStartFlags(flags exprStartFlags) {
+	if flags != 0 {
+		n := len(p.js)
+		if (flags & stmtStartFlag) != 0 {
+			p.stmtStart = n
+		}
+		if (flags & exportDefaultStartFlag) != 0 {
+			p.exportDefaultStart = n
+		}
+		if (flags & arrowExprStartFlag) != 0 {
+			p.arrowExprStart = n
+		}
+		if (flags & forOfInitStartFlag) != 0 {
+			p.forOfInitStart = n
+		}
+	}
+}
+
+// Print any stored comments that are associated with this location
+func (p *printer) printExprCommentsAtLoc(loc logger.Loc) {
+	if p.options.MinifyWhitespace {
+		return
+	}
+	if comments := p.exprComments[loc]; comments != nil && !p.printedExprComments[loc] {
+		flags := p.saveExprStartFlags()
+
+		// We must never generate a newline before certain expressions. For example,
+		// generating a newline before the expression in a "return" statement will
+		// cause a semicolon to be inserted, which would change the code's behavior.
+		if p.noLeadingNewlineHere == len(p.js) {
+			for _, comment := range comments {
+				if strings.HasPrefix(comment, "//") {
+					p.print("/*")
+					p.print(comment[2:])
+					if strings.HasPrefix(comment, "// ") {
+						p.print(" ")
+					}
+					p.print("*/")
+				} else {
+					p.print(strings.Join(strings.Split(comment, "\n"), ""))
+				}
+				p.printSpace()
+			}
+		} else {
+			for _, comment := range comments {
+				p.printIndentedComment(comment)
+				p.printIndent()
+			}
+		}
+
+		// Mark these comments as printed so we don't print them again
+		p.printedExprComments[loc] = true
+
+		p.restoreExprStartFlags(flags)
+	}
+}
+
+func (p *printer) printExprCommentsAfterCloseTokenAtLoc(loc logger.Loc) {
+	if comments := p.exprComments[loc]; comments != nil && !p.printedExprComments[loc] {
+		flags := p.saveExprStartFlags()
+
+		for _, comment := range comments {
+			p.printIndent()
+			p.printIndentedComment(comment)
+		}
+
+		// Mark these comments as printed so we don't print them again
+		p.printedExprComments[loc] = true
+
+		p.restoreExprStartFlags(flags)
+	}
+}
+
+func (p *printer) printExprWithoutLeadingNewline(expr js_ast.Expr, level js_ast.L, flags printExprFlags) {
+	if !p.options.MinifyWhitespace && p.willPrintExprCommentsAtLoc(expr.Loc) {
+		p.print("(")
+		p.printNewline()
+		p.options.Indent++
+		p.printIndent()
+		p.printExpr(expr, level, flags)
+		p.printNewline()
+		p.options.Indent--
+		p.printIndent()
+		p.print(")")
+		return
+	}
+
+	p.noLeadingNewlineHere = len(p.js)
+	p.printExpr(expr, level, flags)
+}
+
+type printExprFlags uint16
+
+const (
+	forbidCall printExprFlags = 1 << iota
+	forbidIn
+	hasNonOptionalChainParent
+	exprResultIsUnused
+	didAlreadySimplifyUnusedExprs
+	isFollowedByOf
+	isInsideForAwait
+	isDeleteTarget
+	isCallTargetOrTemplateTag
+	isPropertyAccessTarget
+	parentWasUnaryOrBinaryOrIfTest
+)
+
+func (p *printer) printExpr(expr js_ast.Expr, level js_ast.L, flags printExprFlags) {
+	// If syntax compression is enabled, do a pre-pass over unary and binary
+	// operators to inline bitwise operations of cross-module inlined constants.
+	// This makes the output a little tighter if people construct bit masks in
+	// other files. This is not a general-purpose constant folding pass. In
+	// particular, it has no effect on tree shaking because that pass has already
+	// been run.
+	//
+	// This sets a flag to avoid doing this when the parent is a unary or binary
+	// operator so that we don't trigger O(n^2) behavior when traversing over a
+	// large expression tree.
+	if p.options.MinifySyntax && (flags&parentWasUnaryOrBinaryOrIfTest) == 0 {
+		switch expr.Data.(type) {
+		case *js_ast.EUnary, *js_ast.EBinary, *js_ast.EIf:
+			expr = p.lateConstantFoldUnaryOrBinaryOrIfExpr(expr)
+		}
+	}
+
+	p.printExprCommentsAtLoc(expr.Loc)
+
+	switch e := expr.Data.(type) {
+	case *js_ast.EMissing:
+		p.addSourceMapping(expr.Loc)
+
+	case *js_ast.EAnnotation:
+		p.printExpr(e.Value, level, flags)
+
+	case *js_ast.EUndefined:
+		p.printUndefined(expr.Loc, level)
+
+	case *js_ast.ESuper:
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(expr.Loc)
+		p.print("super")
+
+	case *js_ast.ENull:
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(expr.Loc)
+		p.print("null")
+
+	case *js_ast.EThis:
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(expr.Loc)
+		p.print("this")
+
+	case *js_ast.ESpread:
+		p.addSourceMapping(expr.Loc)
+		p.print("...")
+		p.printExpr(e.Value, js_ast.LComma, 0)
+
+	case *js_ast.ENewTarget:
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(expr.Loc)
+		p.print("new.target")
+
+	case *js_ast.EImportMeta:
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(expr.Loc)
+		p.print("import.meta")
+
+	case *js_ast.ENameOfSymbol:
+		name := p.mangledPropName(e.Ref)
+		p.addSourceMappingForName(expr.Loc, name, e.Ref)
+
+		if !p.options.MinifyWhitespace && e.HasPropertyKeyComment {
+			p.print("/* @__KEY__ */ ")
+		}
+
+		p.printQuotedUTF8(name, printQuotedAllowBacktick)
+
+	case *js_ast.EJSXElement:
+		// Start the opening tag
+		p.addSourceMapping(expr.Loc)
+		p.print("<")
+		p.printJSXTag(e.TagOrNil)
+		if !e.IsTagSingleLine {
+			p.options.Indent++
+		}
+
+		// Print the attributes
+		for _, property := range e.Properties {
+			if e.IsTagSingleLine {
+				p.printSpace()
+			} else {
+				p.printNewline()
+				p.printIndent()
+			}
+
+			if property.Kind == js_ast.PropertySpread {
+				if p.willPrintExprCommentsAtLoc(property.Loc) {
+					p.print("{")
+					p.printNewline()
+					p.options.Indent++
+					p.printIndent()
+					p.printExprCommentsAtLoc(property.Loc)
+					p.print("...")
+					p.printExpr(property.ValueOrNil, js_ast.LComma, 0)
+					p.printNewline()
+					p.options.Indent--
+					p.printIndent()
+					p.print("}")
+				} else {
+					p.print("{...")
+					p.printExpr(property.ValueOrNil, js_ast.LComma, 0)
+					p.print("}")
+				}
+				continue
+			}
+
+			p.printSpaceBeforeIdentifier()
+			if mangled, ok := property.Key.Data.(*js_ast.ENameOfSymbol); ok {
+				name := p.mangledPropName(mangled.Ref)
+				p.addSourceMappingForName(property.Key.Loc, name, mangled.Ref)
+				p.printIdentifier(name)
+			} else if str, ok := property.Key.Data.(*js_ast.EString); ok {
+				p.addSourceMapping(property.Key.Loc)
+				p.print(helpers.UTF16ToString(str.Value))
+			} else {
+				p.print("{...{")
+				p.printSpace()
+				p.print("[")
+				p.printExpr(property.Key, js_ast.LComma, 0)
+				p.print("]:")
+				p.printSpace()
+				p.printExpr(property.ValueOrNil, js_ast.LComma, 0)
+				p.printSpace()
+				p.print("}}")
+				continue
+			}
+
+			isMultiLine := p.willPrintExprCommentsAtLoc(property.ValueOrNil.Loc)
+
+			if property.Flags.Has(js_ast.PropertyWasShorthand) {
+				// Implicit "true" value
+				if boolean, ok := property.ValueOrNil.Data.(*js_ast.EBoolean); ok && boolean.Value {
+					continue
+				}
+
+				// JSX element as JSX attribute value
+				if _, ok := property.ValueOrNil.Data.(*js_ast.EJSXElement); ok {
+					p.print("=")
+					p.printExpr(property.ValueOrNil, js_ast.LLowest, 0)
+					continue
+				}
+			}
+
+			// Special-case raw text
+			if text, ok := property.ValueOrNil.Data.(*js_ast.EJSXText); ok {
+				p.print("=")
+				p.addSourceMapping(property.ValueOrNil.Loc)
+				p.print(text.Raw)
+				continue
+			}
+
+			// Generic JS value
+			p.print("={")
+			if isMultiLine {
+				p.printNewline()
+				p.options.Indent++
+				p.printIndent()
+			}
+			p.printExpr(property.ValueOrNil, js_ast.LComma, 0)
+			if isMultiLine {
+				p.printNewline()
+				p.options.Indent--
+				p.printIndent()
+			}
+			p.print("}")
+		}
+
+		// End the opening tag
+		if !e.IsTagSingleLine {
+			p.options.Indent--
+			if len(e.Properties) > 0 {
+				p.printNewline()
+				p.printIndent()
+			}
+		}
+		if e.TagOrNil.Data != nil && len(e.NullableChildren) == 0 {
+			if e.IsTagSingleLine || len(e.Properties) == 0 {
+				p.printSpace()
+			}
+			p.addSourceMapping(e.CloseLoc)
+			p.print("/>")
+			break
+		}
+		p.print(">")
+
+		// Print the children
+		for _, childOrNil := range e.NullableChildren {
+			if _, ok := childOrNil.Data.(*js_ast.EJSXElement); ok {
+				p.printExpr(childOrNil, js_ast.LLowest, 0)
+			} else if text, ok := childOrNil.Data.(*js_ast.EJSXText); ok {
+				p.addSourceMapping(childOrNil.Loc)
+				p.print(text.Raw)
+			} else if childOrNil.Data != nil {
+				isMultiLine := p.willPrintExprCommentsAtLoc(childOrNil.Loc)
+				p.print("{")
+				if isMultiLine {
+					p.printNewline()
+					p.options.Indent++
+					p.printIndent()
+				}
+				p.printExpr(childOrNil, js_ast.LComma, 0)
+				if isMultiLine {
+					p.printNewline()
+					p.options.Indent--
+					p.printIndent()
+				}
+				p.print("}")
+			} else {
+				p.print("{")
+				if p.willPrintExprCommentsAtLoc(childOrNil.Loc) {
+					// Note: Some people use these comments for AST transformations
+					p.printNewline()
+					p.options.Indent++
+					p.printExprCommentsAfterCloseTokenAtLoc(childOrNil.Loc)
+					p.options.Indent--
+					p.printIndent()
+				}
+				p.print("}")
+			}
+		}
+
+		// Print the closing tag
+		p.addSourceMapping(e.CloseLoc)
+		p.print("</")
+		p.printJSXTag(e.TagOrNil)
+		p.print(">")
+
+	case *js_ast.ENew:
+		wrap := level >= js_ast.LCall
+
+		hasPureComment := !p.options.MinifyWhitespace && e.CanBeUnwrappedIfUnused
+		if hasPureComment && level >= js_ast.LPostfix {
+			wrap = true
+		}
+
+		if wrap {
+			p.print("(")
+		}
+
+		if hasPureComment {
+			p.addSourceMapping(expr.Loc)
+			p.print("/* @__PURE__ */ ")
+		}
+
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(expr.Loc)
+		p.print("new")
+		p.printSpace()
+		p.printExpr(e.Target, js_ast.LNew, forbidCall)
+
+		// Omit the "()" when minifying, but only when safe to do so
+		isMultiLine := !p.options.MinifyWhitespace && ((e.IsMultiLine && len(e.Args) > 0) ||
+			p.willPrintExprCommentsForAnyOf(e.Args) ||
+			p.willPrintExprCommentsAtLoc(e.CloseParenLoc))
+		if !p.options.MinifyWhitespace || len(e.Args) > 0 || level >= js_ast.LPostfix || isMultiLine {
+			needsNewline := true
+			p.print("(")
+			if isMultiLine {
+				p.options.Indent++
+			}
+			for i, arg := range e.Args {
+				if i != 0 {
+					p.print(",")
+				}
+				if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit() {
+					if isMultiLine {
+						if needsNewline {
+							p.printNewline()
+						}
+						p.printIndent()
+					} else if i != 0 {
+						p.printSpace()
+					}
+				}
+				p.printExpr(arg, js_ast.LComma, 0)
+				needsNewline = true
+			}
+			if isMultiLine {
+				if needsNewline || p.willPrintExprCommentsAtLoc(e.CloseParenLoc) {
+					p.printNewline()
+				}
+				p.printExprCommentsAfterCloseTokenAtLoc(e.CloseParenLoc)
+				p.options.Indent--
+				p.printIndent()
+			}
+			if e.CloseParenLoc.Start > expr.Loc.Start {
+				p.addSourceMapping(e.CloseParenLoc)
+			}
+			p.print(")")
+		}
+
+		if wrap {
+			p.print(")")
+		}
+
+	case *js_ast.ECall:
+		if p.options.MinifySyntax {
+			var symbolFlags ast.SymbolFlags
+			switch target := e.Target.Data.(type) {
+			case *js_ast.EIdentifier:
+				symbolFlags = p.symbols.Get(target.Ref).Flags
+			case *js_ast.EImportIdentifier:
+				ref := ast.FollowSymbols(p.symbols, target.Ref)
+				symbolFlags = p.symbols.Get(ref).Flags
+			}
+
+			// Replace non-mutated empty functions with their arguments at print time
+			if (symbolFlags & (ast.IsEmptyFunction | ast.CouldPotentiallyBeMutated)) == ast.IsEmptyFunction {
+				var replacement js_ast.Expr
+				for _, arg := range e.Args {
+					if _, ok := arg.Data.(*js_ast.ESpread); ok {
+						arg.Data = &js_ast.EArray{Items: []js_ast.Expr{arg}, IsSingleLine: true}
+					}
+					replacement = js_ast.JoinWithComma(replacement, p.astHelpers.SimplifyUnusedExpr(arg, p.options.UnsupportedFeatures))
+				}
+				if replacement.Data == nil || (flags&exprResultIsUnused) == 0 {
+					replacement = js_ast.JoinWithComma(replacement, js_ast.Expr{Loc: expr.Loc, Data: js_ast.EUndefinedShared})
+				}
+				p.printExpr(p.guardAgainstBehaviorChangeDueToSubstitution(replacement, flags), level, flags)
+				break
+			}
+
+			// Inline non-mutated identity functions at print time
+			if (symbolFlags&(ast.IsIdentityFunction|ast.CouldPotentiallyBeMutated)) == ast.IsIdentityFunction && len(e.Args) == 1 {
+				arg := e.Args[0]
+				if _, ok := arg.Data.(*js_ast.ESpread); !ok {
+					if (flags & exprResultIsUnused) != 0 {
+						arg = p.astHelpers.SimplifyUnusedExpr(arg, p.options.UnsupportedFeatures)
+					}
+					p.printExpr(p.guardAgainstBehaviorChangeDueToSubstitution(arg, flags), level, flags)
+					break
+				}
+			}
+
+			// Inline IIFEs that return expressions at print time
+			if len(e.Args) == 0 {
+				// Note: Do not inline async arrow functions as they are not IIFEs. In
+				// particular, they are not necessarily invoked immediately, and any
+				// exceptions involved in their evaluation will be swallowed without
+				// bubbling up to the surrounding context.
+				if arrow, ok := e.Target.Data.(*js_ast.EArrow); ok && len(arrow.Args) == 0 && !arrow.IsAsync {
+					stmts := arrow.Body.Block.Stmts
+
+					// "(() => {})()" => "void 0"
+					if len(stmts) == 0 {
+						value := js_ast.Expr{Loc: expr.Loc, Data: js_ast.EUndefinedShared}
+						p.printExpr(p.guardAgainstBehaviorChangeDueToSubstitution(value, flags), level, flags)
+						break
+					}
+
+					// "(() => 123)()" => "123"
+					if len(stmts) == 1 {
+						if stmt, ok := stmts[0].Data.(*js_ast.SReturn); ok {
+							value := stmt.ValueOrNil
+							if value.Data == nil {
+								value.Data = js_ast.EUndefinedShared
+							}
+							p.printExpr(p.guardAgainstBehaviorChangeDueToSubstitution(value, flags), level, flags)
+							break
+						}
+					}
+				}
+			}
+		}
+
+		wrap := level >= js_ast.LNew || (flags&forbidCall) != 0
+		var targetFlags printExprFlags
+		if e.OptionalChain == js_ast.OptionalChainNone {
+			targetFlags = hasNonOptionalChainParent
+		} else if (flags & hasNonOptionalChainParent) != 0 {
+			wrap = true
+		}
+
+		hasPureComment := !p.options.MinifyWhitespace && e.CanBeUnwrappedIfUnused
+		if hasPureComment && level >= js_ast.LPostfix {
+			wrap = true
+		}
+
+		if wrap {
+			p.print("(")
+		}
+
+		if hasPureComment {
+			flags := p.saveExprStartFlags()
+			p.addSourceMapping(expr.Loc)
+			p.print("/* @__PURE__ */ ")
+			p.restoreExprStartFlags(flags)
+		}
+
+		// We don't ever want to accidentally generate a direct eval expression here
+		p.callTarget = e.Target.Data
+		if (e.Kind != js_ast.DirectEval && p.isUnboundEvalIdentifier(e.Target) && e.OptionalChain == js_ast.OptionalChainNone) ||
+			(e.Kind != js_ast.TargetWasOriginallyPropertyAccess && js_ast.IsPropertyAccess(e.Target)) {
+			p.print("(0,")
+			p.printSpace()
+			p.printExpr(e.Target, js_ast.LPostfix, isCallTargetOrTemplateTag)
+			p.print(")")
+		} else {
+			p.printExpr(e.Target, js_ast.LPostfix, isCallTargetOrTemplateTag|targetFlags)
+		}
+
+		if e.OptionalChain == js_ast.OptionalChainStart {
+			p.print("?.")
+		}
+
+		isMultiLine := !p.options.MinifyWhitespace && ((e.IsMultiLine && len(e.Args) > 0) ||
+			p.willPrintExprCommentsForAnyOf(e.Args) ||
+			p.willPrintExprCommentsAtLoc(e.CloseParenLoc))
+		p.print("(")
+		if isMultiLine {
+			p.options.Indent++
+		}
+		for i, arg := range e.Args {
+			if i != 0 {
+				p.print(",")
+			}
+			if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit() {
+				if isMultiLine {
+					p.printNewline()
+					p.printIndent()
+				} else if i != 0 {
+					p.printSpace()
+				}
+			}
+			p.printExpr(arg, js_ast.LComma, 0)
+		}
+		if isMultiLine {
+			p.printNewline()
+			p.printExprCommentsAfterCloseTokenAtLoc(e.CloseParenLoc)
+			p.options.Indent--
+			p.printIndent()
+		}
+		if e.CloseParenLoc.Start > expr.Loc.Start {
+			p.addSourceMapping(e.CloseParenLoc)
+		}
+		p.print(")")
+
+		if wrap {
+			p.print(")")
+		}
+
+	case *js_ast.ERequireString:
+		p.addSourceMapping(expr.Loc)
+		p.printRequireOrImportExpr(e.ImportRecordIndex, level, flags, e.CloseParenLoc)
+
+	case *js_ast.ERequireResolveString:
+		recordLoc := p.importRecords[e.ImportRecordIndex].Range.Loc
+		isMultiLine := p.willPrintExprCommentsAtLoc(recordLoc) || p.willPrintExprCommentsAtLoc(e.CloseParenLoc)
+		wrap := level >= js_ast.LNew || (flags&forbidCall) != 0
+		if wrap {
+			p.print("(")
+		}
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(expr.Loc)
+		p.print("require.resolve(")
+		if isMultiLine {
+			p.printNewline()
+			p.options.Indent++
+			p.printIndent()
+			p.printExprCommentsAtLoc(recordLoc)
+		}
+		p.printPath(e.ImportRecordIndex, ast.ImportRequireResolve)
+		if isMultiLine {
+			p.printNewline()
+			p.printExprCommentsAfterCloseTokenAtLoc(e.CloseParenLoc)
+			p.options.Indent--
+			p.printIndent()
+		}
+		if e.CloseParenLoc.Start > expr.Loc.Start {
+			p.addSourceMapping(e.CloseParenLoc)
+		}
+		p.print(")")
+		if wrap {
+			p.print(")")
+		}
+
+	case *js_ast.EImportString:
+		p.addSourceMapping(expr.Loc)
+		p.printRequireOrImportExpr(e.ImportRecordIndex, level, flags, e.CloseParenLoc)
+
+	case *js_ast.EImportCall:
+		// Only print the second argument if either import assertions or import attributes are supported
+		printImportAssertOrWith := e.OptionsOrNil.Data != nil && (!p.options.UnsupportedFeatures.Has(compat.ImportAssertions) || !p.options.UnsupportedFeatures.Has(compat.ImportAttributes))
+		isMultiLine := !p.options.MinifyWhitespace &&
+			(p.willPrintExprCommentsAtLoc(e.Expr.Loc) ||
+				(printImportAssertOrWith && p.willPrintExprCommentsAtLoc(e.OptionsOrNil.Loc)) ||
+				p.willPrintExprCommentsAtLoc(e.CloseParenLoc))
+		wrap := level >= js_ast.LNew || (flags&forbidCall) != 0
+		if wrap {
+			p.print("(")
+		}
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(expr.Loc)
+		p.print("import(")
+		if isMultiLine {
+			p.printNewline()
+			p.options.Indent++
+			p.printIndent()
+		}
+		p.printExpr(e.Expr, js_ast.LComma, 0)
+
+		if printImportAssertOrWith {
+			p.print(",")
+			if isMultiLine {
+				p.printNewline()
+				p.printIndent()
+			} else {
+				p.printSpace()
+			}
+			p.printExpr(e.OptionsOrNil, js_ast.LComma, 0)
+		}
+
+		if isMultiLine {
+			p.printNewline()
+			p.printExprCommentsAfterCloseTokenAtLoc(e.CloseParenLoc)
+			p.options.Indent--
+			p.printIndent()
+		}
+		p.print(")")
+		if wrap {
+			p.print(")")
+		}
+
+	case *js_ast.EDot:
+		wrap := false
+		if e.OptionalChain == js_ast.OptionalChainNone {
+			flags |= hasNonOptionalChainParent
+
+			// Inline cross-module TypeScript enum references here
+			if value, ok := p.tryToGetImportedEnumValue(e.Target, e.Name); ok {
+				if value.String != nil {
+					p.printQuotedUTF16(value.String, printQuotedAllowBacktick)
+				} else {
+					p.printNumber(value.Number, level)
+				}
+				if !p.options.MinifyWhitespace && !p.options.MinifyIdentifiers && !strings.Contains(e.Name, "*/") {
+					p.print(" /* ")
+					p.print(e.Name)
+					p.print(" */")
+				}
+				break
+			}
+		} else {
+			if (flags & hasNonOptionalChainParent) != 0 {
+				wrap = true
+				p.print("(")
+			}
+			flags &= ^hasNonOptionalChainParent
+		}
+		p.printExpr(e.Target, js_ast.LPostfix, (flags&(forbidCall|hasNonOptionalChainParent))|isPropertyAccessTarget)
+		if p.canPrintIdentifier(e.Name) {
+			if e.OptionalChain != js_ast.OptionalChainStart && p.needSpaceBeforeDot == len(p.js) {
+				// "1.toString" is a syntax error, so print "1 .toString" instead
+				p.print(" ")
+			}
+			if e.OptionalChain == js_ast.OptionalChainStart {
+				p.print("?.")
+			} else {
+				p.print(".")
+			}
+			if p.options.LineLimit > 0 {
+				p.printNewlinePastLineLimit()
+			}
+			p.addSourceMapping(e.NameLoc)
+			p.printIdentifier(e.Name)
+		} else {
+			if e.OptionalChain == js_ast.OptionalChainStart {
+				p.print("?.")
+			}
+			p.print("[")
+			p.addSourceMapping(e.NameLoc)
+			p.printQuotedUTF8(e.Name, printQuotedAllowBacktick)
+			p.print("]")
+		}
+		if wrap {
+			p.print(")")
+		}
+
+	case *js_ast.EIndex:
+		if e.OptionalChain == js_ast.OptionalChainNone {
+			flags |= hasNonOptionalChainParent
+
+			// Inline cross-module TypeScript enum references here
+			if index, ok := e.Index.Data.(*js_ast.EString); ok {
+				if value, name, ok := p.tryToGetImportedEnumValueUTF16(e.Target, index.Value); ok {
+					if value.String != nil {
+						p.printQuotedUTF16(value.String, printQuotedAllowBacktick)
+					} else {
+						p.printNumber(value.Number, level)
+					}
+					if !p.options.MinifyWhitespace && !p.options.MinifyIdentifiers && !strings.Contains(name, "*/") {
+						p.print(" /* ")
+						p.print(name)
+						p.print(" */")
+					}
+					break
+				}
+			}
+		} else {
+			if (flags & hasNonOptionalChainParent) != 0 {
+				p.print("(")
+				defer p.print(")")
+			}
+			flags &= ^hasNonOptionalChainParent
+		}
+		p.printExpr(e.Target, js_ast.LPostfix, (flags&(forbidCall|hasNonOptionalChainParent))|isPropertyAccessTarget)
+		if e.OptionalChain == js_ast.OptionalChainStart {
+			p.print("?.")
+		}
+
+		switch index := e.Index.Data.(type) {
+		case *js_ast.EPrivateIdentifier:
+			if e.OptionalChain != js_ast.OptionalChainStart {
+				p.print(".")
+			}
+			name := p.renamer.NameForSymbol(index.Ref)
+			p.addSourceMappingForName(e.Index.Loc, name, index.Ref)
+			p.printIdentifier(name)
+			return
+
+		case *js_ast.ENameOfSymbol:
+			if name := p.mangledPropName(index.Ref); p.canPrintIdentifier(name) {
+				if e.OptionalChain != js_ast.OptionalChainStart {
+					p.print(".")
+				}
+				p.addSourceMappingForName(e.Index.Loc, name, index.Ref)
+				p.printIdentifier(name)
+				return
+			}
+
+		case *js_ast.EInlinedEnum:
+			if p.options.MinifySyntax {
+				if str, ok := index.Value.Data.(*js_ast.EString); ok && p.canPrintIdentifierUTF16(str.Value) {
+					if e.OptionalChain != js_ast.OptionalChainStart {
+						p.print(".")
+					}
+					p.addSourceMapping(index.Value.Loc)
+					p.printIdentifierUTF16(str.Value)
+					return
+				}
+			}
+
+		case *js_ast.EDot:
+			if p.options.MinifySyntax {
+				if value, ok := p.tryToGetImportedEnumValue(index.Target, index.Name); ok && value.String != nil && p.canPrintIdentifierUTF16(value.String) {
+					if e.OptionalChain != js_ast.OptionalChainStart {
+						p.print(".")
+					}
+					p.addSourceMapping(e.Index.Loc)
+					p.printIdentifierUTF16(value.String)
+					return
+				}
+			}
+		}
+
+		isMultiLine := p.willPrintExprCommentsAtLoc(e.Index.Loc) || p.willPrintExprCommentsAtLoc(e.CloseBracketLoc)
+		p.print("[")
+		if isMultiLine {
+			p.printNewline()
+			p.options.Indent++
+			p.printIndent()
+		}
+		p.printExpr(e.Index, js_ast.LLowest, 0)
+		if isMultiLine {
+			p.printNewline()
+			p.printExprCommentsAfterCloseTokenAtLoc(e.CloseBracketLoc)
+			p.options.Indent--
+			p.printIndent()
+		}
+		if e.CloseBracketLoc.Start > expr.Loc.Start {
+			p.addSourceMapping(e.CloseBracketLoc)
+		}
+		p.print("]")
+
+	case *js_ast.EIf:
+		wrap := level >= js_ast.LConditional
+		if wrap {
+			p.print("(")
+			flags &= ^forbidIn
+		}
+		p.printExpr(e.Test, js_ast.LConditional, (flags&forbidIn)|parentWasUnaryOrBinaryOrIfTest)
+		p.printSpace()
+		p.print("?")
+		if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit() {
+			p.printSpace()
+		}
+		p.printExprWithoutLeadingNewline(e.Yes, js_ast.LYield, 0)
+		p.printSpace()
+		p.print(":")
+		if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit() {
+			p.printSpace()
+		}
+		p.printExprWithoutLeadingNewline(e.No, js_ast.LYield, flags&forbidIn)
+		if wrap {
+			p.print(")")
+		}
+
+	case *js_ast.EArrow:
+		wrap := level >= js_ast.LAssign
+
+		if wrap {
+			p.print("(")
+		}
+		if !p.options.MinifyWhitespace && e.HasNoSideEffectsComment {
+			p.print("/* @__NO_SIDE_EFFECTS__ */ ")
+		}
+		if e.IsAsync {
+			p.addSourceMapping(expr.Loc)
+			p.printSpaceBeforeIdentifier()
+			p.print("async")
+			p.printSpace()
+		}
+
+		p.printFnArgs(e.Args, fnArgsOpts{
+			openParenLoc:              expr.Loc,
+			addMappingForOpenParenLoc: !e.IsAsync,
+			hasRestArg:                e.HasRestArg,
+			isArrow:                   true,
+		})
+		p.printSpace()
+		p.print("=>")
+		p.printSpace()
+
+		wasPrinted := false
+		if len(e.Body.Block.Stmts) == 1 && e.PreferExpr {
+			if s, ok := e.Body.Block.Stmts[0].Data.(*js_ast.SReturn); ok && s.ValueOrNil.Data != nil {
+				p.arrowExprStart = len(p.js)
+				p.printExprWithoutLeadingNewline(s.ValueOrNil, js_ast.LComma, flags&forbidIn)
+				wasPrinted = true
+			}
+		}
+		if !wasPrinted {
+			p.printBlock(e.Body.Loc, e.Body.Block)
+		}
+		if wrap {
+			p.print(")")
+		}
+
+	case *js_ast.EFunction:
+		n := len(p.js)
+		wrap := p.stmtStart == n || p.exportDefaultStart == n ||
+			((flags&isPropertyAccessTarget) != 0 && p.options.UnsupportedFeatures.Has(compat.FunctionOrClassPropertyAccess))
+		if wrap {
+			p.print("(")
+		}
+		if !p.options.MinifyWhitespace && e.Fn.HasNoSideEffectsComment {
+			p.print("/* @__NO_SIDE_EFFECTS__ */ ")
+		}
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(expr.Loc)
+		if e.Fn.IsAsync {
+			p.print("async ")
+		}
+		p.print("function")
+		if e.Fn.IsGenerator {
+			p.print("*")
+			p.printSpace()
+		}
+		if e.Fn.Name != nil {
+			p.printSpaceBeforeIdentifier()
+			name := p.renamer.NameForSymbol(e.Fn.Name.Ref)
+			p.addSourceMappingForName(e.Fn.Name.Loc, name, e.Fn.Name.Ref)
+			p.printIdentifier(name)
+		}
+		p.printFn(e.Fn)
+		if wrap {
+			p.print(")")
+		}
+
+	case *js_ast.EClass:
+		n := len(p.js)
+		wrap := p.stmtStart == n || p.exportDefaultStart == n ||
+			((flags&isPropertyAccessTarget) != 0 && p.options.UnsupportedFeatures.Has(compat.FunctionOrClassPropertyAccess))
+		if wrap {
+			p.print("(")
+		}
+		p.printDecorators(e.Class.Decorators, printSpaceAfterDecorator)
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(expr.Loc)
+		p.print("class")
+		if e.Class.Name != nil {
+			p.print(" ")
+			name := p.renamer.NameForSymbol(e.Class.Name.Ref)
+			p.addSourceMappingForName(e.Class.Name.Loc, name, e.Class.Name.Ref)
+			p.printIdentifier(name)
+		}
+		p.printClass(e.Class)
+		if wrap {
+			p.print(")")
+		}
+
+	case *js_ast.EArray:
+		isMultiLine := (len(e.Items) > 0 && !e.IsSingleLine) || p.willPrintExprCommentsForAnyOf(e.Items) || p.willPrintExprCommentsAtLoc(e.CloseBracketLoc)
+		p.addSourceMapping(expr.Loc)
+		p.print("[")
+		if len(e.Items) > 0 || isMultiLine {
+			if isMultiLine {
+				p.options.Indent++
+			}
+
+			for i, item := range e.Items {
+				if i != 0 {
+					p.print(",")
+				}
+				if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit() {
+					if isMultiLine {
+						p.printNewline()
+						p.printIndent()
+					} else if i != 0 {
+						p.printSpace()
+					}
+				}
+				p.printExpr(item, js_ast.LComma, 0)
+
+				// Make sure there's a comma after trailing missing items
+				_, ok := item.Data.(*js_ast.EMissing)
+				if ok && i == len(e.Items)-1 {
+					p.print(",")
+				}
+			}
+
+			if isMultiLine {
+				p.printNewline()
+				p.printExprCommentsAfterCloseTokenAtLoc(e.CloseBracketLoc)
+				p.options.Indent--
+				p.printIndent()
+			}
+		}
+		if e.CloseBracketLoc.Start > expr.Loc.Start {
+			p.addSourceMapping(e.CloseBracketLoc)
+		}
+		p.print("]")
+
+	case *js_ast.EObject:
+		isMultiLine := (len(e.Properties) > 0 && !e.IsSingleLine) || p.willPrintExprCommentsAtLoc(e.CloseBraceLoc)
+		if !p.options.MinifyWhitespace && !isMultiLine {
+			for _, property := range e.Properties {
+				if p.willPrintExprCommentsAtLoc(property.Loc) {
+					isMultiLine = true
+					break
+				}
+			}
+		}
+		n := len(p.js)
+		wrap := p.stmtStart == n || p.arrowExprStart == n
+		if wrap {
+			p.print("(")
+		}
+		p.addSourceMapping(expr.Loc)
+		p.print("{")
+		if len(e.Properties) > 0 || isMultiLine {
+			if isMultiLine {
+				p.options.Indent++
+			}
+
+			for i, item := range e.Properties {
+				if i != 0 {
+					p.print(",")
+				}
+				if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit() {
+					if isMultiLine {
+						p.printNewline()
+						p.printIndent()
+					} else {
+						p.printSpace()
+					}
+				}
+				p.printProperty(item)
+			}
+
+			if isMultiLine {
+				p.printNewline()
+				p.printExprCommentsAfterCloseTokenAtLoc(e.CloseBraceLoc)
+				p.options.Indent--
+				p.printIndent()
+			} else if len(e.Properties) > 0 {
+				p.printSpace()
+			}
+		}
+		if e.CloseBraceLoc.Start > expr.Loc.Start {
+			p.addSourceMapping(e.CloseBraceLoc)
+		}
+		p.print("}")
+		if wrap {
+			p.print(")")
+		}
+
+	case *js_ast.EBoolean:
+		p.addSourceMapping(expr.Loc)
+		if p.options.MinifySyntax {
+			if level >= js_ast.LPrefix {
+				if e.Value {
+					p.print("(!0)")
+				} else {
+					p.print("(!1)")
+				}
+			} else {
+				if e.Value {
+					p.print("!0")
+				} else {
+					p.print("!1")
+				}
+			}
+		} else {
+			p.printSpaceBeforeIdentifier()
+			if e.Value {
+				p.print("true")
+			} else {
+				p.print("false")
+			}
+		}
+
+	case *js_ast.EString:
+		var flags printQuotedFlags
+		if e.ContainsUniqueKey {
+			flags = printQuotedNoWrap
+		}
+		p.addSourceMapping(expr.Loc)
+
+		if !p.options.MinifyWhitespace && e.HasPropertyKeyComment {
+			p.print("/* @__KEY__ */ ")
+		}
+
+		// If this was originally a template literal, print it as one as long as we're not minifying
+		if e.PreferTemplate && !p.options.MinifySyntax && !p.options.UnsupportedFeatures.Has(compat.TemplateLiteral) {
+			p.print("`")
+			p.printUnquotedUTF16(e.Value, '`', flags)
+			p.print("`")
+			return
+		}
+
+		p.printQuotedUTF16(e.Value, flags|printQuotedAllowBacktick)
+
+	case *js_ast.ETemplate:
+		if e.TagOrNil.Data == nil && (p.options.MinifySyntax || p.wasLazyExport) {
+			// Inline enums and mangled properties when minifying
+			var replaced []js_ast.TemplatePart
+			for i, part := range e.Parts {
+				var inlinedValue js_ast.E
+				switch e2 := part.Value.Data.(type) {
+				case *js_ast.ENameOfSymbol:
+					inlinedValue = &js_ast.EString{Value: helpers.StringToUTF16(p.mangledPropName(e2.Ref))}
+				case *js_ast.EDot:
+					if value, ok := p.tryToGetImportedEnumValue(e2.Target, e2.Name); ok {
+						if value.String != nil {
+							inlinedValue = &js_ast.EString{Value: value.String}
+						} else {
+							inlinedValue = &js_ast.ENumber{Value: value.Number}
+						}
+					}
+				}
+				if inlinedValue != nil {
+					if replaced == nil {
+						replaced = make([]js_ast.TemplatePart, 0, len(e.Parts))
+						replaced = append(replaced, e.Parts[:i]...)
+					}
+					part.Value.Data = inlinedValue
+					replaced = append(replaced, part)
+				} else if replaced != nil {
+					replaced = append(replaced, part)
+				}
+			}
+			if replaced != nil {
+				copy := *e
+				copy.Parts = replaced
+				switch e2 := js_ast.InlinePrimitivesIntoTemplate(logger.Loc{}, &copy).Data.(type) {
+				case *js_ast.EString:
+					p.printQuotedUTF16(e2.Value, printQuotedAllowBacktick)
+					return
+				case *js_ast.ETemplate:
+					e = e2
+				}
+			}
+
+			// Convert no-substitution template literals into strings if it's smaller
+			if len(e.Parts) == 0 {
+				p.addSourceMapping(expr.Loc)
+				p.printQuotedUTF16(e.HeadCooked, printQuotedAllowBacktick)
+				return
+			}
+		}
+
+		if e.TagOrNil.Data != nil {
+			tagIsPropertyAccess := false
+			switch e.TagOrNil.Data.(type) {
+			case *js_ast.EDot, *js_ast.EIndex:
+				tagIsPropertyAccess = true
+			}
+			if !e.TagWasOriginallyPropertyAccess && tagIsPropertyAccess {
+				// Prevent "x``" from becoming "y.z``"
+				p.print("(0,")
+				p.printSpace()
+				p.printExpr(e.TagOrNil, js_ast.LLowest, isCallTargetOrTemplateTag)
+				p.print(")")
+			} else if js_ast.IsOptionalChain(e.TagOrNil) {
+				// Optional chains are forbidden in template tags
+				p.print("(")
+				p.printExpr(e.TagOrNil, js_ast.LLowest, isCallTargetOrTemplateTag)
+				p.print(")")
+			} else {
+				p.printExpr(e.TagOrNil, js_ast.LPostfix, isCallTargetOrTemplateTag)
+			}
+		} else {
+			p.addSourceMapping(expr.Loc)
+		}
+		p.print("`")
+		if e.TagOrNil.Data != nil {
+			p.print(e.HeadRaw)
+		} else {
+			p.printUnquotedUTF16(e.HeadCooked, '`', 0)
+		}
+		for _, part := range e.Parts {
+			p.print("${")
+			p.printExpr(part.Value, js_ast.LLowest, 0)
+			p.addSourceMapping(part.TailLoc)
+			p.print("}")
+			if e.TagOrNil.Data != nil {
+				p.print(part.TailRaw)
+			} else {
+				p.printUnquotedUTF16(part.TailCooked, '`', 0)
+			}
+		}
+		p.print("`")
+
+	case *js_ast.ERegExp:
+		buffer := p.js
+		n := len(buffer)
+
+		// Avoid forming a single-line comment or "</script" sequence
+		if !p.options.UnsupportedFeatures.Has(compat.InlineScript) && n > 0 {
+			if last := buffer[n-1]; last == '/' || (last == '<' && len(e.Value) >= 7 && strings.EqualFold(e.Value[:7], "/script")) {
+				p.print(" ")
+			}
+		}
+
+		p.addSourceMapping(expr.Loc)
+		p.print(e.Value)
+
+		// Need a space before the next identifier to avoid it turning into flags
+		p.prevRegExpEnd = len(p.js)
+
+	case *js_ast.EInlinedEnum:
+		p.printExpr(e.Value, level, flags)
+
+		if !p.options.MinifyWhitespace && !p.options.MinifyIdentifiers {
+			p.print(" /* ")
+			p.print(e.Comment)
+			p.print(" */")
+		}
+
+	case *js_ast.EBigInt:
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(expr.Loc)
+		p.print(e.Value)
+		p.print("n")
+
+	case *js_ast.ENumber:
+		p.addSourceMapping(expr.Loc)
+		p.printNumber(e.Value, level)
+
+	case *js_ast.EIdentifier:
+		name := p.renamer.NameForSymbol(e.Ref)
+		wrap := len(p.js) == p.forOfInitStart && (name == "let" ||
+			((flags&isFollowedByOf) != 0 && (flags&isInsideForAwait) == 0 && name == "async"))
+
+		if wrap {
+			p.print("(")
+		}
+
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMappingForName(expr.Loc, name, e.Ref)
+		p.printIdentifier(name)
+
+		if wrap {
+			p.print(")")
+		}
+
+	case *js_ast.EImportIdentifier:
+		// Potentially use a property access instead of an identifier
+		ref := ast.FollowSymbols(p.symbols, e.Ref)
+		symbol := p.symbols.Get(ref)
+
+		if symbol.ImportItemStatus == ast.ImportItemMissing {
+			p.printUndefined(expr.Loc, level)
+		} else if symbol.NamespaceAlias != nil {
+			wrap := p.callTarget == e && e.WasOriginallyIdentifier
+			if wrap {
+				p.print("(0,")
+				p.printSpace()
+			}
+			p.printSpaceBeforeIdentifier()
+			p.addSourceMapping(expr.Loc)
+			p.printIdentifier(p.renamer.NameForSymbol(symbol.NamespaceAlias.NamespaceRef))
+			alias := symbol.NamespaceAlias.Alias
+			if !e.PreferQuotedKey && p.canPrintIdentifier(alias) {
+				p.print(".")
+				p.addSourceMappingForName(expr.Loc, alias, ref)
+				p.printIdentifier(alias)
+			} else {
+				p.print("[")
+				p.addSourceMappingForName(expr.Loc, alias, ref)
+				p.printQuotedUTF8(alias, printQuotedAllowBacktick)
+				p.print("]")
+			}
+			if wrap {
+				p.print(")")
+			}
+		} else if value := p.options.ConstValues[ref]; value.Kind != js_ast.ConstValueNone {
+			// Handle inlined constants
+			p.printExpr(js_ast.ConstValueToExpr(expr.Loc, value), level, flags)
+		} else {
+			p.printSpaceBeforeIdentifier()
+			name := p.renamer.NameForSymbol(ref)
+			p.addSourceMappingForName(expr.Loc, name, ref)
+			p.printIdentifier(name)
+		}
+
+	case *js_ast.EAwait:
+		wrap := level >= js_ast.LPrefix
+
+		if wrap {
+			p.print("(")
+		}
+
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(expr.Loc)
+		p.print("await")
+		p.printSpace()
+		p.printExpr(e.Value, js_ast.LPrefix-1, 0)
+
+		if wrap {
+			p.print(")")
+		}
+
+	case *js_ast.EYield:
+		wrap := level >= js_ast.LAssign
+
+		if wrap {
+			p.print("(")
+		}
+
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(expr.Loc)
+		p.print("yield")
+
+		if e.ValueOrNil.Data != nil {
+			if e.IsStar {
+				p.print("*")
+			}
+			p.printSpace()
+			p.printExprWithoutLeadingNewline(e.ValueOrNil, js_ast.LYield, 0)
+		}
+
+		if wrap {
+			p.print(")")
+		}
+
+	case *js_ast.EUnary:
+		entry := js_ast.OpTable[e.Op]
+		wrap := level >= entry.Level
+
+		if wrap {
+			p.print("(")
+		}
+
+		if !e.Op.IsPrefix() {
+			p.printExpr(e.Value, js_ast.LPostfix-1, parentWasUnaryOrBinaryOrIfTest)
+		}
+
+		if entry.IsKeyword {
+			p.printSpaceBeforeIdentifier()
+			if e.Op.IsPrefix() {
+				p.addSourceMapping(expr.Loc)
+			}
+			p.print(entry.Text)
+			p.printSpace()
+		} else {
+			p.printSpaceBeforeOperator(e.Op)
+			if e.Op.IsPrefix() {
+				p.addSourceMapping(expr.Loc)
+			}
+			p.print(entry.Text)
+			p.prevOp = e.Op
+			p.prevOpEnd = len(p.js)
+		}
+
+		if e.Op.IsPrefix() {
+			valueFlags := parentWasUnaryOrBinaryOrIfTest
+			if e.Op == js_ast.UnOpDelete {
+				valueFlags |= isDeleteTarget
+			}
+
+			// Never turn "typeof (0, x)" into "typeof x" or "delete (0, x)" into "delete x"
+			if (e.Op == js_ast.UnOpTypeof && !e.WasOriginallyTypeofIdentifier && p.isUnboundIdentifier(e.Value)) ||
+				(e.Op == js_ast.UnOpDelete && !e.WasOriginallyDeleteOfIdentifierOrPropertyAccess && p.isIdentifierOrNumericConstantOrPropertyAccess(e.Value)) {
+				p.print("(0,")
+				p.printSpace()
+				p.printExpr(e.Value, js_ast.LPrefix-1, valueFlags)
+				p.print(")")
+			} else {
+				p.printExpr(e.Value, js_ast.LPrefix-1, valueFlags)
+			}
+		}
+
+		if wrap {
+			p.print(")")
+		}
+
+	case *js_ast.EBinary:
+		// The handling of binary expressions is convoluted because we're using
+		// iteration on the heap instead of recursion on the call stack to avoid
+		// stack overflow for deeply-nested ASTs. See the comments for the similar
+		// code in the JavaScript parser for details.
+		v := binaryExprVisitor{
+			e:     e,
+			level: level,
+			flags: flags,
+		}
+
+		// Use a single stack to reduce allocation overhead
+		stackBottom := len(p.binaryExprStack)
+
+		for {
+			// Check whether this node is a special case, and stop if it is
+			if !v.checkAndPrepare(p) {
+				break
+			}
+
+			left := v.e.Left
+			leftBinary, ok := left.Data.(*js_ast.EBinary)
+
+			// Stop iterating if iteration doesn't apply to the left node
+			if !ok {
+				p.printExpr(left, v.leftLevel, v.leftFlags)
+				v.visitRightAndFinish(p)
+				break
+			}
+
+			// Manually run the code at the start of "printExpr"
+			p.printExprCommentsAtLoc(left.Loc)
+
+			// Only allocate heap memory on the stack for nested binary expressions
+			p.binaryExprStack = append(p.binaryExprStack, v)
+			v = binaryExprVisitor{
+				e:     leftBinary,
+				level: v.leftLevel,
+				flags: v.leftFlags,
+			}
+		}
+
+		// Process all binary operations from the deepest-visited node back toward
+		// our original top-level binary operation
+		for {
+			n := len(p.binaryExprStack) - 1
+			if n < stackBottom {
+				break
+			}
+			v := p.binaryExprStack[n]
+			p.binaryExprStack = p.binaryExprStack[:n]
+			v.visitRightAndFinish(p)
+		}
+
+	default:
+		panic(fmt.Sprintf("Unexpected expression of type %T", expr.Data))
+	}
+}
+
+// The handling of binary expressions is convoluted because we're using
+// iteration on the heap instead of recursion on the call stack to avoid
+// stack overflow for deeply-nested ASTs. See the comments for the similar
+// code in the JavaScript parser for details.
+type binaryExprVisitor struct {
+	// Inputs
+	e     *js_ast.EBinary
+	level js_ast.L
+	flags printExprFlags
+
+	// Input for visiting the left child
+	leftLevel js_ast.L
+	leftFlags printExprFlags
+
+	// "Local variables" passed from "checkAndPrepare" to "visitRightAndFinish"
+	entry      js_ast.OpTableEntry
+	wrap       bool
+	rightLevel js_ast.L
+}
+
+func (v *binaryExprVisitor) checkAndPrepare(p *printer) bool {
+	e := v.e
+
+	// If this is a comma operator then either the result is unused (and we
+	// should have already simplified unused expressions), or the result is used
+	// (and we can still simplify unused expressions inside the left operand)
+	if e.Op == js_ast.BinOpComma {
+		if (v.flags & didAlreadySimplifyUnusedExprs) == 0 {
+			left := p.simplifyUnusedExpr(e.Left)
+			right := e.Right
+			if (v.flags & exprResultIsUnused) != 0 {
+				right = p.simplifyUnusedExpr(right)
+			}
+			if left.Data != e.Left.Data || right.Data != e.Right.Data {
+				// Pass a flag so we don't needlessly re-simplify the same expression
+				p.printExpr(p.guardAgainstBehaviorChangeDueToSubstitution(js_ast.JoinWithComma(left, right), v.flags), v.level, v.flags|didAlreadySimplifyUnusedExprs)
+				return false
+			}
+		} else {
+			// Pass a flag so we don't needlessly re-simplify the same expression
+			v.flags |= didAlreadySimplifyUnusedExprs
+		}
+	}
+
+	v.entry = js_ast.OpTable[e.Op]
+	v.wrap = v.level >= v.entry.Level || (e.Op == js_ast.BinOpIn && (v.flags&forbidIn) != 0)
+
+	// Destructuring assignments must be parenthesized
+	if n := len(p.js); p.stmtStart == n || p.arrowExprStart == n {
+		if _, ok := e.Left.Data.(*js_ast.EObject); ok {
+			v.wrap = true
+		}
+	}
+
+	if v.wrap {
+		p.print("(")
+		v.flags &= ^forbidIn
+	}
+
+	v.leftLevel = v.entry.Level - 1
+	v.rightLevel = v.entry.Level - 1
+
+	if e.Op.IsRightAssociative() {
+		v.leftLevel = v.entry.Level
+	}
+	if e.Op.IsLeftAssociative() {
+		v.rightLevel = v.entry.Level
+	}
+
+	switch e.Op {
+	case js_ast.BinOpNullishCoalescing:
+		// "??" can't directly contain "||" or "&&" without being wrapped in parentheses
+		if left, ok := e.Left.Data.(*js_ast.EBinary); ok && (left.Op == js_ast.BinOpLogicalOr || left.Op == js_ast.BinOpLogicalAnd) {
+			v.leftLevel = js_ast.LPrefix
+		}
+		if right, ok := e.Right.Data.(*js_ast.EBinary); ok && (right.Op == js_ast.BinOpLogicalOr || right.Op == js_ast.BinOpLogicalAnd) {
+			v.rightLevel = js_ast.LPrefix
+		}
+
+	case js_ast.BinOpPow:
+		// "**" can't contain certain unary expressions
+		if left, ok := e.Left.Data.(*js_ast.EUnary); ok && left.Op.UnaryAssignTarget() == js_ast.AssignTargetNone {
+			v.leftLevel = js_ast.LCall
+		} else if _, ok := e.Left.Data.(*js_ast.EAwait); ok {
+			v.leftLevel = js_ast.LCall
+		} else if _, ok := e.Left.Data.(*js_ast.EUndefined); ok {
+			// Undefined is printed as "void 0"
+			v.leftLevel = js_ast.LCall
+		} else if _, ok := e.Left.Data.(*js_ast.ENumber); ok {
+			// Negative numbers are printed using a unary operator
+			v.leftLevel = js_ast.LCall
+		} else if p.options.MinifySyntax {
+			// When minifying, booleans are printed as "!0 and "!1"
+			if _, ok := e.Left.Data.(*js_ast.EBoolean); ok {
+				v.leftLevel = js_ast.LCall
+			}
+		}
+	}
+
+	// Special-case "#foo in bar"
+	if private, ok := e.Left.Data.(*js_ast.EPrivateIdentifier); ok && e.Op == js_ast.BinOpIn {
+		name := p.renamer.NameForSymbol(private.Ref)
+		p.addSourceMappingForName(e.Left.Loc, name, private.Ref)
+		p.printIdentifier(name)
+		v.visitRightAndFinish(p)
+		return false
+	}
+
+	if e.Op == js_ast.BinOpComma {
+		// The result of the left operand of the comma operator is unused
+		v.leftFlags = (v.flags & forbidIn) | exprResultIsUnused | parentWasUnaryOrBinaryOrIfTest
+	} else {
+		v.leftFlags = (v.flags & forbidIn) | parentWasUnaryOrBinaryOrIfTest
+	}
+	return true
+}
+
+func (v *binaryExprVisitor) visitRightAndFinish(p *printer) {
+	e := v.e
+
+	if e.Op != js_ast.BinOpComma {
+		p.printSpace()
+	}
+
+	if v.entry.IsKeyword {
+		p.printSpaceBeforeIdentifier()
+		p.print(v.entry.Text)
+	} else {
+		p.printSpaceBeforeOperator(e.Op)
+		p.print(v.entry.Text)
+		p.prevOp = e.Op
+		p.prevOpEnd = len(p.js)
+	}
+
+	if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit() {
+		p.printSpace()
+	}
+
+	if e.Op == js_ast.BinOpComma {
+		// The result of the right operand of the comma operator is unused if the caller doesn't use it
+		p.printExpr(e.Right, v.rightLevel, (v.flags&(forbidIn|exprResultIsUnused))|parentWasUnaryOrBinaryOrIfTest)
+	} else {
+		p.printExpr(e.Right, v.rightLevel, (v.flags&forbidIn)|parentWasUnaryOrBinaryOrIfTest)
+	}
+
+	if v.wrap {
+		p.print(")")
+	}
+}
+
+func (p *printer) isUnboundEvalIdentifier(value js_ast.Expr) bool {
+	if id, ok := value.Data.(*js_ast.EIdentifier); ok {
+		// Using the original name here is ok since unbound symbols are not renamed
+		symbol := p.symbols.Get(ast.FollowSymbols(p.symbols, id.Ref))
+		return symbol.Kind == ast.SymbolUnbound && symbol.OriginalName == "eval"
+	}
+	return false
+}
+
+// Convert an integer to a byte slice without any allocations
+func (p *printer) smallIntToBytes(n int) []byte {
+	wasNegative := n < 0
+	if wasNegative {
+		// This assumes that -math.MinInt isn't a problem. This is fine because
+		// these integers are floating-point exponents which never go up that high.
+		n = -n
+	}
+
+	bytes := p.intToBytesBuffer[:]
+	start := len(bytes)
+
+	// Write out the number from the end to the front
+	for {
+		start--
+		bytes[start] = '0' + byte(n%10)
+		n /= 10
+		if n == 0 {
+			break
+		}
+	}
+
+	// Stick a negative sign on the front if needed
+	if wasNegative {
+		start--
+		bytes[start] = '-'
+	}
+
+	return bytes[start:]
+}
+
+func parseSmallInt(bytes []byte) int {
+	wasNegative := bytes[0] == '-'
+	if wasNegative {
+		bytes = bytes[1:]
+	}
+
+	// Parse the integer without any error checking. This doesn't need to handle
+	// integer overflow because these integers are floating-point exponents which
+	// never go up that high.
+	n := 0
+	for _, c := range bytes {
+		n = n*10 + int(c-'0')
+	}
+
+	if wasNegative {
+		return -n
+	}
+	return n
+}
+
+func (p *printer) printNonNegativeFloat(absValue float64) {
+	// We can avoid the slow call to strconv.FormatFloat() for integers less than
+	// 1000 because we know that exponential notation will always be longer than
+	// the integer representation. This is not the case for 1000 which is "1e3".
+	if absValue < 1000 {
+		if asInt := int64(absValue); absValue == float64(asInt) {
+			p.printBytes(p.smallIntToBytes(int(asInt)))
+
+			// Integers always need a space before "." to avoid making a decimal point
+			p.needSpaceBeforeDot = len(p.js)
+			return
+		}
+	}
+
+	// Format this number into a byte slice so we can mutate it in place without
+	// further reallocation
+	result := []byte(strconv.FormatFloat(absValue, 'g', -1, 64))
+
+	// Simplify the exponent
+	// "e+05" => "e5"
+	// "e-05" => "e-5"
+	if e := bytes.LastIndexByte(result, 'e'); e != -1 {
+		from := e + 1
+		to := from
+
+		switch result[from] {
+		case '+':
+			// Strip off the leading "+"
+			from++
+
+		case '-':
+			// Skip past the leading "-"
+			to++
+			from++
+		}
+
+		// Strip off leading zeros
+		for from < len(result) && result[from] == '0' {
+			from++
+		}
+
+		result = append(result[:to], result[from:]...)
+	}
+
+	dot := bytes.IndexByte(result, '.')
+
+	if dot == 1 && result[0] == '0' {
+		// Simplify numbers starting with "0."
+		afterDot := 2
+
+		// Strip off the leading zero when minifying
+		// "0.5" => ".5"
+		if p.options.MinifyWhitespace {
+			result = result[1:]
+			afterDot--
+		}
+
+		// Try using an exponent
+		// "0.001" => "1e-3"
+		if result[afterDot] == '0' {
+			i := afterDot + 1
+			for result[i] == '0' {
+				i++
+			}
+			remaining := result[i:]
+			exponent := p.smallIntToBytes(afterDot - i - len(remaining))
+
+			// Only switch if it's actually shorter
+			if len(result) > len(remaining)+1+len(exponent) {
+				result = append(append(remaining, 'e'), exponent...)
+			}
+		}
+	} else if dot != -1 {
+		// Try to get rid of a "." and maybe also an "e"
+		if e := bytes.LastIndexByte(result, 'e'); e != -1 {
+			integer := result[:dot]
+			fraction := result[dot+1 : e]
+			exponent := parseSmallInt(result[e+1:]) - len(fraction)
+
+			// Handle small exponents by appending zeros instead
+			if exponent >= 0 && exponent <= 2 {
+				// "1.2e1" => "12"
+				// "1.2e2" => "120"
+				// "1.2e3" => "1200"
+				if len(result) >= len(integer)+len(fraction)+exponent {
+					result = append(integer, fraction...)
+					for i := 0; i < exponent; i++ {
+						result = append(result, '0')
+					}
+				}
+			} else {
+				// "1.2e4" => "12e3"
+				exponent := p.smallIntToBytes(exponent)
+				if len(result) >= len(integer)+len(fraction)+1+len(exponent) {
+					result = append(append(append(integer, fraction...), 'e'), exponent...)
+				}
+			}
+		}
+	} else if result[len(result)-1] == '0' {
+		// Simplify numbers ending with "0" by trying to use an exponent
+		// "1000" => "1e3"
+		i := len(result) - 1
+		for i > 0 && result[i-1] == '0' {
+			i--
+		}
+		remaining := result[:i]
+		exponent := p.smallIntToBytes(len(result) - i)
+
+		// Only switch if it's actually shorter
+		if len(result) > len(remaining)+1+len(exponent) {
+			result = append(append(remaining, 'e'), exponent...)
+		}
+	}
+
+	// Numbers in this range can potentially be printed with one fewer byte as
+	// hex. This compares against 0xFFFF_FFFF_FFFF_F800 instead of comparing
+	// against 0xFFFF_FFFF_FFFF_FFFF because 0xFFFF_FFFF_FFFF_FFFF when converted
+	// to float64 rounds up to 0x1_0000_0000_0000_0180, which can no longer fit
+	// into uint64. In Go, the result of converting float64 to uint64 outside of
+	// the uint64 range is implementation-dependent and is different on amd64 vs.
+	// arm64. The float64 value 0xFFFF_FFFF_FFFF_F800 is the biggest value that
+	// is below the float64 value 0x1_0000_0000_0000_0180, so we use that instead.
+	if p.options.MinifyWhitespace && absValue >= 1_000_000_000_000 && absValue <= 0xFFFF_FFFF_FFFF_F800 {
+		if asInt := uint64(absValue); absValue == float64(asInt) {
+			if hex := strconv.FormatUint(asInt, 16); 2+len(hex) < len(result) {
+				result = append(append(result[:0], '0', 'x'), hex...)
+			}
+		}
+	}
+
+	p.printBytes(result)
+
+	// We'll need a space before "." if it could be parsed as a decimal point
+	if !bytes.ContainsAny(result, ".ex") {
+		p.needSpaceBeforeDot = len(p.js)
+	}
+}
+
+func (p *printer) printDeclStmt(isExport bool, keyword string, decls []js_ast.Decl) {
+	p.printIndent()
+	p.printSpaceBeforeIdentifier()
+	if isExport {
+		p.print("export ")
+	}
+	p.printDecls(keyword, decls, 0)
+	p.printSemicolonAfterStatement()
+}
+
+func (p *printer) printForLoopInit(init js_ast.Stmt, flags printExprFlags) {
+	switch s := init.Data.(type) {
+	case *js_ast.SExpr:
+		p.printExpr(s.Value, js_ast.LLowest, flags|exprResultIsUnused)
+	case *js_ast.SLocal:
+		switch s.Kind {
+		case js_ast.LocalAwaitUsing:
+			p.printDecls("await using", s.Decls, flags)
+		case js_ast.LocalConst:
+			p.printDecls("const", s.Decls, flags)
+		case js_ast.LocalLet:
+			p.printDecls("let", s.Decls, flags)
+		case js_ast.LocalUsing:
+			p.printDecls("using", s.Decls, flags)
+		case js_ast.LocalVar:
+			p.printDecls("var", s.Decls, flags)
+		}
+	default:
+		panic("Internal error")
+	}
+}
+
+func (p *printer) printDecls(keyword string, decls []js_ast.Decl, flags printExprFlags) {
+	p.print(keyword)
+	p.printSpace()
+
+	for i, decl := range decls {
+		if i != 0 {
+			p.print(",")
+			if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit() {
+				p.printSpace()
+			}
+		}
+		p.printBinding(decl.Binding)
+
+		if decl.ValueOrNil.Data != nil {
+			p.printSpace()
+			p.print("=")
+			p.printSpace()
+			p.printExprWithoutLeadingNewline(decl.ValueOrNil, js_ast.LComma, flags)
+		}
+	}
+}
+
+func (p *printer) printBody(body js_ast.Stmt, isSingleLine bool) {
+	if block, ok := body.Data.(*js_ast.SBlock); ok {
+		p.printSpace()
+		p.printBlock(body.Loc, *block)
+		p.printNewline()
+	} else if isSingleLine {
+		p.printNextIndentAsSpace = true
+		p.printStmt(body, 0)
+	} else {
+		p.printNewline()
+		p.options.Indent++
+		p.printStmt(body, 0)
+		p.options.Indent--
+	}
+}
+
+func (p *printer) printBlock(loc logger.Loc, block js_ast.SBlock) {
+	p.addSourceMapping(loc)
+	p.print("{")
+	p.printNewline()
+
+	p.options.Indent++
+	for _, stmt := range block.Stmts {
+		p.printSemicolonIfNeeded()
+		p.printStmt(stmt, canOmitStatement)
+	}
+	p.options.Indent--
+	p.needsSemicolon = false
+
+	p.printIndent()
+	if block.CloseBraceLoc.Start > loc.Start {
+		p.addSourceMapping(block.CloseBraceLoc)
+	}
+	p.print("}")
+}
+
+func wrapToAvoidAmbiguousElse(s js_ast.S) bool {
+	for {
+		switch current := s.(type) {
+		case *js_ast.SIf:
+			if current.NoOrNil.Data == nil {
+				return true
+			}
+			s = current.NoOrNil.Data
+
+		case *js_ast.SFor:
+			s = current.Body.Data
+
+		case *js_ast.SForIn:
+			s = current.Body.Data
+
+		case *js_ast.SForOf:
+			s = current.Body.Data
+
+		case *js_ast.SWhile:
+			s = current.Body.Data
+
+		case *js_ast.SWith:
+			s = current.Body.Data
+
+		case *js_ast.SLabel:
+			s = current.Stmt.Data
+
+		default:
+			return false
+		}
+	}
+}
+
+func (p *printer) printIf(s *js_ast.SIf) {
+	p.printSpaceBeforeIdentifier()
+	p.print("if")
+	p.printSpace()
+	p.print("(")
+	if p.willPrintExprCommentsAtLoc(s.Test.Loc) {
+		p.printNewline()
+		p.options.Indent++
+		p.printIndent()
+		p.printExpr(s.Test, js_ast.LLowest, 0)
+		p.printNewline()
+		p.options.Indent--
+		p.printIndent()
+	} else {
+		p.printExpr(s.Test, js_ast.LLowest, 0)
+	}
+	p.print(")")
+
+	// Simplify the else branch, which may disappear entirely
+	no := s.NoOrNil
+	if expr, ok := no.Data.(*js_ast.SExpr); ok {
+		if value := p.simplifyUnusedExpr(expr.Value); value.Data == nil {
+			no.Data = nil
+		} else if value.Data != expr.Value.Data {
+			no.Data = &js_ast.SExpr{Value: value}
+		}
+	}
+
+	if yes, ok := s.Yes.Data.(*js_ast.SBlock); ok {
+		p.printSpace()
+		p.printBlock(s.Yes.Loc, *yes)
+
+		if no.Data != nil {
+			p.printSpace()
+		} else {
+			p.printNewline()
+		}
+	} else if wrapToAvoidAmbiguousElse(s.Yes.Data) {
+		p.printSpace()
+		p.print("{")
+		p.printNewline()
+
+		p.options.Indent++
+		p.printStmt(s.Yes, canOmitStatement)
+		p.options.Indent--
+		p.needsSemicolon = false
+
+		p.printIndent()
+		p.print("}")
+
+		if no.Data != nil {
+			p.printSpace()
+		} else {
+			p.printNewline()
+		}
+	} else {
+		p.printBody(s.Yes, s.IsSingleLineYes)
+
+		if no.Data != nil {
+			p.printIndent()
+		}
+	}
+
+	if no.Data != nil {
+		p.printSemicolonIfNeeded()
+		p.printSpaceBeforeIdentifier()
+		p.print("else")
+
+		if block, ok := no.Data.(*js_ast.SBlock); ok {
+			p.printSpace()
+			p.printBlock(no.Loc, *block)
+			p.printNewline()
+		} else if ifStmt, ok := no.Data.(*js_ast.SIf); ok {
+			p.printIf(ifStmt)
+		} else {
+			p.printBody(no, s.IsSingleLineNo)
+		}
+	}
+}
+
+func (p *printer) printIndentedComment(text string) {
+	// Avoid generating a comment containing the character sequence "</script"
+	if !p.options.UnsupportedFeatures.Has(compat.InlineScript) {
+		text = helpers.EscapeClosingTag(text, "/script")
+	}
+
+	if strings.HasPrefix(text, "/*") {
+		// Re-indent multi-line comments
+		for {
+			newline := strings.IndexByte(text, '\n')
+			if newline == -1 {
+				break
+			}
+			p.print(text[:newline+1])
+			p.printIndent()
+			text = text[newline+1:]
+		}
+		p.print(text)
+		p.printNewline()
+	} else {
+		// Print a mandatory newline after single-line comments
+		p.print(text)
+		p.print("\n")
+	}
+}
+
+func (p *printer) printPath(importRecordIndex uint32, importKind ast.ImportKind) {
+	record := p.importRecords[importRecordIndex]
+	p.addSourceMapping(record.Range.Loc)
+	p.printQuotedUTF8(record.Path.Text, printQuotedNoWrap)
+
+	if p.options.NeedsMetafile {
+		external := ""
+		if (record.Flags & ast.ShouldNotBeExternalInMetafile) == 0 {
+			external = ",\n          \"external\": true"
+		}
+		p.jsonMetadataImports = append(p.jsonMetadataImports, fmt.Sprintf("\n        {\n          \"path\": %s,\n          \"kind\": %s%s\n        }",
+			helpers.QuoteForJSON(record.Path.Text, p.options.ASCIIOnly),
+			helpers.QuoteForJSON(importKind.StringForMetafile(), p.options.ASCIIOnly),
+			external))
+	}
+
+	if record.AssertOrWith != nil && importKind == ast.ImportStmt {
+		feature := compat.ImportAttributes
+		if record.AssertOrWith.Keyword == ast.AssertKeyword {
+			feature = compat.ImportAssertions
+		}
+
+		// Omit import assertions/attributes on this import statement if they would cause a syntax error
+		if p.options.UnsupportedFeatures.Has(feature) {
+			return
+		}
+
+		p.printSpace()
+		p.addSourceMapping(record.AssertOrWith.KeywordLoc)
+		p.print(record.AssertOrWith.Keyword.String())
+		p.printSpace()
+		p.printImportAssertOrWithClause(*record.AssertOrWith)
+	}
+}
+
+func (p *printer) printImportCallAssertOrWith(assertOrWith *ast.ImportAssertOrWith, outerIsMultiLine bool) {
+	// Omit import assertions/attributes if we know the "import()" syntax doesn't
+	// support a second argument (i.e. both import assertions and import
+	// attributes aren't supported) and doing so would cause a syntax error
+	if assertOrWith == nil || (p.options.UnsupportedFeatures.Has(compat.ImportAssertions) && p.options.UnsupportedFeatures.Has(compat.ImportAttributes)) {
+		return
+	}
+
+	isMultiLine := p.willPrintExprCommentsAtLoc(assertOrWith.KeywordLoc) ||
+		p.willPrintExprCommentsAtLoc(assertOrWith.InnerOpenBraceLoc) ||
+		p.willPrintExprCommentsAtLoc(assertOrWith.OuterCloseBraceLoc)
+
+	p.print(",")
+	if outerIsMultiLine {
+		p.printNewline()
+		p.printIndent()
+	} else {
+		p.printSpace()
+	}
+	p.printExprCommentsAtLoc(assertOrWith.OuterOpenBraceLoc)
+	p.addSourceMapping(assertOrWith.OuterOpenBraceLoc)
+	p.print("{")
+
+	if isMultiLine {
+		p.printNewline()
+		p.options.Indent++
+		p.printIndent()
+	} else {
+		p.printSpace()
+	}
+
+	p.printExprCommentsAtLoc(assertOrWith.KeywordLoc)
+	p.addSourceMapping(assertOrWith.KeywordLoc)
+	p.print(assertOrWith.Keyword.String())
+	p.print(":")
+
+	if p.willPrintExprCommentsAtLoc(assertOrWith.InnerOpenBraceLoc) {
+		p.printNewline()
+		p.options.Indent++
+		p.printIndent()
+		p.printExprCommentsAtLoc(assertOrWith.InnerOpenBraceLoc)
+		p.printImportAssertOrWithClause(*assertOrWith)
+		p.options.Indent--
+	} else {
+		p.printSpace()
+		p.printImportAssertOrWithClause(*assertOrWith)
+	}
+
+	if isMultiLine {
+		p.printNewline()
+		p.printExprCommentsAfterCloseTokenAtLoc(assertOrWith.OuterCloseBraceLoc)
+		p.options.Indent--
+		p.printIndent()
+	} else {
+		p.printSpace()
+	}
+
+	p.addSourceMapping(assertOrWith.OuterCloseBraceLoc)
+	p.print("}")
+}
+
+func (p *printer) printImportAssertOrWithClause(assertOrWith ast.ImportAssertOrWith) {
+	isMultiLine := p.willPrintExprCommentsAtLoc(assertOrWith.InnerCloseBraceLoc)
+	if !isMultiLine {
+		for _, entry := range assertOrWith.Entries {
+			if p.willPrintExprCommentsAtLoc(entry.KeyLoc) || p.willPrintExprCommentsAtLoc(entry.ValueLoc) {
+				isMultiLine = true
+				break
+			}
+		}
+	}
+
+	p.addSourceMapping(assertOrWith.InnerOpenBraceLoc)
+	p.print("{")
+	if isMultiLine {
+		p.options.Indent++
+	}
+
+	for i, entry := range assertOrWith.Entries {
+		if i > 0 {
+			p.print(",")
+		}
+		if isMultiLine {
+			p.printNewline()
+			p.printIndent()
+		} else {
+			p.printSpace()
+		}
+
+		p.printExprCommentsAtLoc(entry.KeyLoc)
+		p.addSourceMapping(entry.KeyLoc)
+		if !entry.PreferQuotedKey && p.canPrintIdentifierUTF16(entry.Key) {
+			p.printSpaceBeforeIdentifier()
+			p.printIdentifierUTF16(entry.Key)
+		} else {
+			p.printQuotedUTF16(entry.Key, 0)
+		}
+
+		p.print(":")
+
+		if p.willPrintExprCommentsAtLoc(entry.ValueLoc) {
+			p.printNewline()
+			p.options.Indent++
+			p.printIndent()
+			p.printExprCommentsAtLoc(entry.ValueLoc)
+			p.addSourceMapping(entry.ValueLoc)
+			p.printQuotedUTF16(entry.Value, 0)
+			p.options.Indent--
+		} else {
+			p.printSpace()
+			p.addSourceMapping(entry.ValueLoc)
+			p.printQuotedUTF16(entry.Value, 0)
+		}
+	}
+
+	if isMultiLine {
+		p.printNewline()
+		p.printExprCommentsAfterCloseTokenAtLoc(assertOrWith.InnerCloseBraceLoc)
+		p.options.Indent--
+		p.printIndent()
+	} else if len(assertOrWith.Entries) > 0 {
+		p.printSpace()
+	}
+
+	p.addSourceMapping(assertOrWith.InnerCloseBraceLoc)
+	p.print("}")
+}
+
+type printStmtFlags uint8
+
+const (
+	canOmitStatement printStmtFlags = 1 << iota
+)
+
+func (p *printer) printStmt(stmt js_ast.Stmt, flags printStmtFlags) {
+	if p.options.LineLimit > 0 {
+		p.printNewlinePastLineLimit()
+	}
+
+	switch s := stmt.Data.(type) {
+	case *js_ast.SComment:
+		text := s.Text
+
+		if s.IsLegalComment {
+			switch p.options.LegalComments {
+			case config.LegalCommentsNone:
+				return
+
+			case config.LegalCommentsEndOfFile,
+				config.LegalCommentsLinkedWithComment,
+				config.LegalCommentsExternalWithoutComment:
+
+				// Don't record the same legal comment more than once per file
+				if p.hasLegalComment == nil {
+					p.hasLegalComment = make(map[string]struct{})
+				} else if _, ok := p.hasLegalComment[text]; ok {
+					return
+				}
+				p.hasLegalComment[text] = struct{}{}
+				p.extractedLegalComments = append(p.extractedLegalComments, text)
+				return
+			}
+		}
+
+		p.printIndent()
+		p.addSourceMapping(stmt.Loc)
+		p.printIndentedComment(text)
+
+	case *js_ast.SFunction:
+		if !p.options.MinifyWhitespace && s.Fn.HasNoSideEffectsComment {
+			p.printIndent()
+			p.print("// @__NO_SIDE_EFFECTS__\n")
+		}
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		if s.IsExport {
+			p.print("export ")
+		}
+		if s.Fn.IsAsync {
+			p.print("async ")
+		}
+		p.print("function")
+		if s.Fn.IsGenerator {
+			p.print("*")
+			p.printSpace()
+		}
+		p.printSpaceBeforeIdentifier()
+		name := p.renamer.NameForSymbol(s.Fn.Name.Ref)
+		p.addSourceMappingForName(s.Fn.Name.Loc, name, s.Fn.Name.Ref)
+		p.printIdentifier(name)
+		p.printFn(s.Fn)
+		p.printNewline()
+
+	case *js_ast.SClass:
+		omitIndent := p.printDecorators(s.Class.Decorators, printNewlineAfterDecorator)
+		if !omitIndent {
+			p.printIndent()
+		}
+		p.printSpaceBeforeIdentifier()
+		p.addSourceMapping(stmt.Loc)
+		if s.IsExport {
+			p.print("export ")
+		}
+		p.print("class ")
+		name := p.renamer.NameForSymbol(s.Class.Name.Ref)
+		p.addSourceMappingForName(s.Class.Name.Loc, name, s.Class.Name.Ref)
+		p.printIdentifier(name)
+		p.printClass(s.Class)
+		p.printNewline()
+
+	case *js_ast.SEmpty:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.print(";")
+		p.printNewline()
+
+	case *js_ast.SExportDefault:
+		if !p.options.MinifyWhitespace {
+			if s2, ok := s.Value.Data.(*js_ast.SFunction); ok && s2.Fn.HasNoSideEffectsComment {
+				p.printIndent()
+				p.print("// @__NO_SIDE_EFFECTS__\n")
+			}
+		}
+		omitIndent := false
+		if s2, ok := s.Value.Data.(*js_ast.SClass); ok {
+			omitIndent = p.printDecorators(s2.Class.Decorators, printNewlineAfterDecorator)
+		}
+		p.addSourceMapping(stmt.Loc)
+		if !omitIndent {
+			p.printIndent()
+		}
+		p.printSpaceBeforeIdentifier()
+		p.print("export default")
+		p.printSpace()
+
+		switch s2 := s.Value.Data.(type) {
+		case *js_ast.SExpr:
+			// Functions and classes must be wrapped to avoid confusion with their statement forms
+			p.exportDefaultStart = len(p.js)
+
+			p.printExprWithoutLeadingNewline(s2.Value, js_ast.LComma, 0)
+			p.printSemicolonAfterStatement()
+			return
+
+		case *js_ast.SFunction:
+			p.printSpaceBeforeIdentifier()
+			if s2.Fn.IsAsync {
+				p.print("async ")
+			}
+			p.print("function")
+			if s2.Fn.IsGenerator {
+				p.print("*")
+				p.printSpace()
+			}
+			if s2.Fn.Name != nil {
+				p.printSpaceBeforeIdentifier()
+				name := p.renamer.NameForSymbol(s2.Fn.Name.Ref)
+				p.addSourceMappingForName(s2.Fn.Name.Loc, name, s2.Fn.Name.Ref)
+				p.printIdentifier(name)
+			}
+			p.printFn(s2.Fn)
+			p.printNewline()
+
+		case *js_ast.SClass:
+			p.printSpaceBeforeIdentifier()
+			p.print("class")
+			if s2.Class.Name != nil {
+				p.print(" ")
+				name := p.renamer.NameForSymbol(s2.Class.Name.Ref)
+				p.addSourceMappingForName(s2.Class.Name.Loc, name, s2.Class.Name.Ref)
+				p.printIdentifier(name)
+			}
+			p.printClass(s2.Class)
+			p.printNewline()
+
+		default:
+			panic("Internal error")
+		}
+
+	case *js_ast.SExportStar:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("export")
+		p.printSpace()
+		p.print("*")
+		p.printSpace()
+		if s.Alias != nil {
+			p.print("as")
+			p.printSpace()
+			p.printClauseAlias(s.Alias.Loc, s.Alias.OriginalName)
+			p.printSpace()
+			p.printSpaceBeforeIdentifier()
+		}
+		p.print("from")
+		p.printSpace()
+		p.printPath(s.ImportRecordIndex, ast.ImportStmt)
+		p.printSemicolonAfterStatement()
+
+	case *js_ast.SExportClause:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("export")
+		p.printSpace()
+		p.print("{")
+
+		if !s.IsSingleLine {
+			p.options.Indent++
+		}
+
+		for i, item := range s.Items {
+			if i != 0 {
+				p.print(",")
+			}
+
+			if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit() {
+				if s.IsSingleLine {
+					p.printSpace()
+				} else {
+					p.printNewline()
+					p.printIndent()
+				}
+			}
+
+			name := p.renamer.NameForSymbol(item.Name.Ref)
+			p.addSourceMappingForName(item.Name.Loc, name, item.Name.Ref)
+			p.printIdentifier(name)
+			if name != item.Alias {
+				p.print(" as")
+				p.printSpace()
+				p.printClauseAlias(item.AliasLoc, item.Alias)
+			}
+		}
+
+		if !s.IsSingleLine {
+			p.options.Indent--
+			p.printNewline()
+			p.printIndent()
+		} else if len(s.Items) > 0 {
+			p.printSpace()
+		}
+
+		p.print("}")
+		p.printSemicolonAfterStatement()
+
+	case *js_ast.SExportFrom:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("export")
+		p.printSpace()
+		p.print("{")
+
+		if !s.IsSingleLine {
+			p.options.Indent++
+		}
+
+		for i, item := range s.Items {
+			if i != 0 {
+				p.print(",")
+			}
+
+			if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit() {
+				if s.IsSingleLine {
+					p.printSpace()
+				} else {
+					p.printNewline()
+					p.printIndent()
+				}
+			}
+
+			p.printClauseAlias(item.Name.Loc, item.OriginalName)
+			if item.OriginalName != item.Alias {
+				p.printSpace()
+				p.printSpaceBeforeIdentifier()
+				p.print("as")
+				p.printSpace()
+				p.printClauseAlias(item.AliasLoc, item.Alias)
+			}
+		}
+
+		if !s.IsSingleLine {
+			p.options.Indent--
+			p.printNewline()
+			p.printIndent()
+		} else if len(s.Items) > 0 {
+			p.printSpace()
+		}
+
+		p.print("}")
+		p.printSpace()
+		p.print("from")
+		p.printSpace()
+		p.printPath(s.ImportRecordIndex, ast.ImportStmt)
+		p.printSemicolonAfterStatement()
+
+	case *js_ast.SLocal:
+		p.addSourceMapping(stmt.Loc)
+		switch s.Kind {
+		case js_ast.LocalAwaitUsing:
+			p.printDeclStmt(s.IsExport, "await using", s.Decls)
+		case js_ast.LocalConst:
+			p.printDeclStmt(s.IsExport, "const", s.Decls)
+		case js_ast.LocalLet:
+			p.printDeclStmt(s.IsExport, "let", s.Decls)
+		case js_ast.LocalUsing:
+			p.printDeclStmt(s.IsExport, "using", s.Decls)
+		case js_ast.LocalVar:
+			p.printDeclStmt(s.IsExport, "var", s.Decls)
+		}
+
+	case *js_ast.SIf:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printIf(s)
+
+	case *js_ast.SDoWhile:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("do")
+		if block, ok := s.Body.Data.(*js_ast.SBlock); ok {
+			p.printSpace()
+			p.printBlock(s.Body.Loc, *block)
+			p.printSpace()
+		} else {
+			p.printNewline()
+			p.options.Indent++
+			p.printStmt(s.Body, 0)
+			p.printSemicolonIfNeeded()
+			p.options.Indent--
+			p.printIndent()
+		}
+		p.print("while")
+		p.printSpace()
+		p.print("(")
+		if p.willPrintExprCommentsAtLoc(s.Test.Loc) {
+			p.printNewline()
+			p.options.Indent++
+			p.printIndent()
+			p.printExpr(s.Test, js_ast.LLowest, 0)
+			p.printNewline()
+			p.options.Indent--
+			p.printIndent()
+		} else {
+			p.printExpr(s.Test, js_ast.LLowest, 0)
+		}
+		p.print(")")
+		p.printSemicolonAfterStatement()
+
+	case *js_ast.SForIn:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("for")
+		p.printSpace()
+		p.print("(")
+		hasInitComment := p.willPrintExprCommentsAtLoc(s.Init.Loc)
+		hasValueComment := p.willPrintExprCommentsAtLoc(s.Value.Loc)
+		if hasInitComment || hasValueComment {
+			p.printNewline()
+			p.options.Indent++
+			p.printIndent()
+		}
+		p.printForLoopInit(s.Init, forbidIn)
+		p.printSpace()
+		p.printSpaceBeforeIdentifier()
+		p.print("in")
+		if hasValueComment {
+			p.printNewline()
+			p.printIndent()
+		} else {
+			p.printSpace()
+		}
+		p.printExpr(s.Value, js_ast.LLowest, 0)
+		if hasInitComment || hasValueComment {
+			p.printNewline()
+			p.options.Indent--
+			p.printIndent()
+		}
+		p.print(")")
+		p.printBody(s.Body, s.IsSingleLineBody)
+
+	case *js_ast.SForOf:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("for")
+		if s.Await.Len > 0 {
+			p.print(" await")
+		}
+		p.printSpace()
+		p.print("(")
+		hasInitComment := p.willPrintExprCommentsAtLoc(s.Init.Loc)
+		hasValueComment := p.willPrintExprCommentsAtLoc(s.Value.Loc)
+		flags := forbidIn | isFollowedByOf
+		if s.Await.Len > 0 {
+			flags |= isInsideForAwait
+		}
+		if hasInitComment || hasValueComment {
+			p.printNewline()
+			p.options.Indent++
+			p.printIndent()
+		}
+		p.forOfInitStart = len(p.js)
+		p.printForLoopInit(s.Init, flags)
+		p.printSpace()
+		p.printSpaceBeforeIdentifier()
+		p.print("of")
+		if hasValueComment {
+			p.printNewline()
+			p.printIndent()
+		} else {
+			p.printSpace()
+		}
+		p.printExpr(s.Value, js_ast.LComma, 0)
+		if hasInitComment || hasValueComment {
+			p.printNewline()
+			p.options.Indent--
+			p.printIndent()
+		}
+		p.print(")")
+		p.printBody(s.Body, s.IsSingleLineBody)
+
+	case *js_ast.SWhile:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("while")
+		p.printSpace()
+		p.print("(")
+		if p.willPrintExprCommentsAtLoc(s.Test.Loc) {
+			p.printNewline()
+			p.options.Indent++
+			p.printIndent()
+			p.printExpr(s.Test, js_ast.LLowest, 0)
+			p.printNewline()
+			p.options.Indent--
+			p.printIndent()
+		} else {
+			p.printExpr(s.Test, js_ast.LLowest, 0)
+		}
+		p.print(")")
+		p.printBody(s.Body, s.IsSingleLineBody)
+
+	case *js_ast.SWith:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("with")
+		p.printSpace()
+		p.print("(")
+		if p.willPrintExprCommentsAtLoc(s.Value.Loc) {
+			p.printNewline()
+			p.options.Indent++
+			p.printIndent()
+			p.printExpr(s.Value, js_ast.LLowest, 0)
+			p.printNewline()
+			p.options.Indent--
+			p.printIndent()
+		} else {
+			p.printExpr(s.Value, js_ast.LLowest, 0)
+		}
+		p.print(")")
+		p.withNesting++
+		p.printBody(s.Body, s.IsSingleLineBody)
+		p.withNesting--
+
+	case *js_ast.SLabel:
+		// Avoid printing a source mapping that masks the one from the label
+		if !p.options.MinifyWhitespace && (p.options.Indent > 0 || p.printNextIndentAsSpace) {
+			p.addSourceMapping(stmt.Loc)
+			p.printIndent()
+		}
+
+		p.printSpaceBeforeIdentifier()
+		name := p.renamer.NameForSymbol(s.Name.Ref)
+		p.addSourceMappingForName(s.Name.Loc, name, s.Name.Ref)
+		p.printIdentifier(name)
+		p.print(":")
+		p.printBody(s.Stmt, s.IsSingleLineStmt)
+
+	case *js_ast.STry:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("try")
+		p.printSpace()
+		p.printBlock(s.BlockLoc, s.Block)
+
+		if s.Catch != nil {
+			p.printSpace()
+			p.print("catch")
+			if s.Catch.BindingOrNil.Data != nil {
+				p.printSpace()
+				p.print("(")
+				p.printBinding(s.Catch.BindingOrNil)
+				p.print(")")
+			}
+			p.printSpace()
+			p.printBlock(s.Catch.BlockLoc, s.Catch.Block)
+		}
+
+		if s.Finally != nil {
+			p.printSpace()
+			p.print("finally")
+			p.printSpace()
+			p.printBlock(s.Finally.Loc, s.Finally.Block)
+		}
+
+		p.printNewline()
+
+	case *js_ast.SFor:
+		init := s.InitOrNil
+		update := s.UpdateOrNil
+
+		// Omit calls to empty functions from the output completely
+		if p.options.MinifySyntax {
+			if expr, ok := init.Data.(*js_ast.SExpr); ok {
+				if value := p.simplifyUnusedExpr(expr.Value); value.Data == nil {
+					init.Data = nil
+				} else if value.Data != expr.Value.Data {
+					init.Data = &js_ast.SExpr{Value: value}
+				}
+			}
+			if update.Data != nil {
+				update = p.simplifyUnusedExpr(update)
+			}
+		}
+
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("for")
+		p.printSpace()
+		p.print("(")
+		isMultiLine :=
+			(init.Data != nil && p.willPrintExprCommentsAtLoc(init.Loc)) ||
+				(s.TestOrNil.Data != nil && p.willPrintExprCommentsAtLoc(s.TestOrNil.Loc)) ||
+				(update.Data != nil && p.willPrintExprCommentsAtLoc(update.Loc))
+		if isMultiLine {
+			p.printNewline()
+			p.options.Indent++
+			p.printIndent()
+		}
+		if init.Data != nil {
+			p.printForLoopInit(init, forbidIn)
+		}
+		p.print(";")
+		if isMultiLine {
+			p.printNewline()
+			p.printIndent()
+		} else {
+			p.printSpace()
+		}
+		if s.TestOrNil.Data != nil {
+			p.printExpr(s.TestOrNil, js_ast.LLowest, 0)
+		}
+		p.print(";")
+		if !isMultiLine {
+			p.printSpace()
+		} else if update.Data != nil {
+			p.printNewline()
+			p.printIndent()
+		}
+		if update.Data != nil {
+			p.printExpr(update, js_ast.LLowest, exprResultIsUnused)
+		}
+		if isMultiLine {
+			p.printNewline()
+			p.options.Indent--
+			p.printIndent()
+		}
+		p.print(")")
+		p.printBody(s.Body, s.IsSingleLineBody)
+
+	case *js_ast.SSwitch:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("switch")
+		p.printSpace()
+		p.print("(")
+		if p.willPrintExprCommentsAtLoc(s.Test.Loc) {
+			p.printNewline()
+			p.options.Indent++
+			p.printIndent()
+			p.printExpr(s.Test, js_ast.LLowest, 0)
+			p.printNewline()
+			p.options.Indent--
+			p.printIndent()
+		} else {
+			p.printExpr(s.Test, js_ast.LLowest, 0)
+		}
+		p.print(")")
+		p.printSpace()
+		p.addSourceMapping(s.BodyLoc)
+		p.print("{")
+		p.printNewline()
+		p.options.Indent++
+
+		for _, c := range s.Cases {
+			p.printSemicolonIfNeeded()
+			p.printIndent()
+			p.printExprCommentsAtLoc(c.Loc)
+			p.addSourceMapping(c.Loc)
+
+			if c.ValueOrNil.Data != nil {
+				p.print("case")
+				p.printSpace()
+				p.printExpr(c.ValueOrNil, js_ast.LLogicalAnd, 0)
+			} else {
+				p.print("default")
+			}
+			p.print(":")
+
+			if len(c.Body) == 1 {
+				if block, ok := c.Body[0].Data.(*js_ast.SBlock); ok {
+					p.printSpace()
+					p.printBlock(c.Body[0].Loc, *block)
+					p.printNewline()
+					continue
+				}
+			}
+
+			p.printNewline()
+			p.options.Indent++
+			for _, stmt := range c.Body {
+				p.printSemicolonIfNeeded()
+				p.printStmt(stmt, canOmitStatement)
+			}
+			p.options.Indent--
+		}
+
+		p.options.Indent--
+		p.printIndent()
+		p.addSourceMapping(s.CloseBraceLoc)
+		p.print("}")
+		p.printNewline()
+		p.needsSemicolon = false
+
+	case *js_ast.SImport:
+		itemCount := 0
+
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("import")
+		p.printSpace()
+
+		if s.DefaultName != nil {
+			p.printSpaceBeforeIdentifier()
+			name := p.renamer.NameForSymbol(s.DefaultName.Ref)
+			p.addSourceMappingForName(s.DefaultName.Loc, name, s.DefaultName.Ref)
+			p.printIdentifier(name)
+			itemCount++
+		}
+
+		if s.Items != nil {
+			if itemCount > 0 {
+				p.print(",")
+				p.printSpace()
+			}
+
+			p.print("{")
+			if !s.IsSingleLine {
+				p.options.Indent++
+			}
+
+			for i, item := range *s.Items {
+				if i != 0 {
+					p.print(",")
+				}
+
+				if p.options.LineLimit <= 0 || !p.printNewlinePastLineLimit() {
+					if s.IsSingleLine {
+						p.printSpace()
+					} else {
+						p.printNewline()
+						p.printIndent()
+					}
+				}
+
+				p.printClauseAlias(item.AliasLoc, item.Alias)
+
+				name := p.renamer.NameForSymbol(item.Name.Ref)
+				if name != item.Alias {
+					p.printSpace()
+					p.printSpaceBeforeIdentifier()
+					p.print("as ")
+					p.addSourceMappingForName(item.Name.Loc, name, item.Name.Ref)
+					p.printIdentifier(name)
+				}
+			}
+
+			if !s.IsSingleLine {
+				p.options.Indent--
+				p.printNewline()
+				p.printIndent()
+			} else if len(*s.Items) > 0 {
+				p.printSpace()
+			}
+
+			p.print("}")
+			itemCount++
+		}
+
+		if s.StarNameLoc != nil {
+			if itemCount > 0 {
+				p.print(",")
+				p.printSpace()
+			}
+
+			p.print("*")
+			p.printSpace()
+			p.print("as ")
+			name := p.renamer.NameForSymbol(s.NamespaceRef)
+			p.addSourceMappingForName(*s.StarNameLoc, name, s.NamespaceRef)
+			p.printIdentifier(name)
+			itemCount++
+		}
+
+		if itemCount > 0 {
+			p.printSpace()
+			p.printSpaceBeforeIdentifier()
+			p.print("from")
+			p.printSpace()
+		}
+
+		p.printPath(s.ImportRecordIndex, ast.ImportStmt)
+		p.printSemicolonAfterStatement()
+
+	case *js_ast.SBlock:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printBlock(stmt.Loc, *s)
+		p.printNewline()
+
+	case *js_ast.SDebugger:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("debugger")
+		p.printSemicolonAfterStatement()
+
+	case *js_ast.SDirective:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.printQuotedUTF16(s.Value, 0)
+		p.printSemicolonAfterStatement()
+
+	case *js_ast.SBreak:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("break")
+		if s.Label != nil {
+			p.print(" ")
+			name := p.renamer.NameForSymbol(s.Label.Ref)
+			p.addSourceMappingForName(s.Label.Loc, name, s.Label.Ref)
+			p.printIdentifier(name)
+		}
+		p.printSemicolonAfterStatement()
+
+	case *js_ast.SContinue:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("continue")
+		if s.Label != nil {
+			p.print(" ")
+			name := p.renamer.NameForSymbol(s.Label.Ref)
+			p.addSourceMappingForName(s.Label.Loc, name, s.Label.Ref)
+			p.printIdentifier(name)
+		}
+		p.printSemicolonAfterStatement()
+
+	case *js_ast.SReturn:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("return")
+		if s.ValueOrNil.Data != nil {
+			p.printSpace()
+			p.printExprWithoutLeadingNewline(s.ValueOrNil, js_ast.LLowest, 0)
+		}
+		p.printSemicolonAfterStatement()
+
+	case *js_ast.SThrow:
+		p.addSourceMapping(stmt.Loc)
+		p.printIndent()
+		p.printSpaceBeforeIdentifier()
+		p.print("throw")
+		p.printSpace()
+		p.printExprWithoutLeadingNewline(s.Value, js_ast.LLowest, 0)
+		p.printSemicolonAfterStatement()
+
+	case *js_ast.SExpr:
+		value := s.Value
+
+		// Omit calls to empty functions from the output completely
+		if p.options.MinifySyntax {
+			value = p.simplifyUnusedExpr(value)
+			if value.Data == nil {
+				// If this statement is not in a block, then we still need to emit something
+				if (flags & canOmitStatement) == 0 {
+					// "if (x) empty();" => "if (x) ;"
+					p.addSourceMapping(stmt.Loc)
+					p.printIndent()
+					p.print(";")
+					p.printNewline()
+				} else {
+					// "if (x) { empty(); }" => "if (x) {}"
+				}
+				break
+			}
+		}
+
+		// Avoid printing a source mapping when the expression would print one in
+		// the same spot. We don't want to accidentally mask the mapping it emits.
+		if !p.options.MinifyWhitespace && (p.options.Indent > 0 || p.printNextIndentAsSpace) {
+			p.addSourceMapping(stmt.Loc)
+			p.printIndent()
+		}
+
+		p.stmtStart = len(p.js)
+		p.printExpr(value, js_ast.LLowest, exprResultIsUnused)
+		p.printSemicolonAfterStatement()
+
+	default:
+		panic(fmt.Sprintf("Unexpected statement of type %T", stmt.Data))
+	}
+}
+
+type Options struct {
+	RequireOrImportMetaForSource func(uint32) RequireOrImportMeta
+
+	// Cross-module inlining of TypeScript enums is actually done during printing
+	TSEnums map[ast.Ref]map[string]js_ast.TSEnumValue
+
+	// Cross-module inlining of detected inlinable constants is also done during printing
+	ConstValues map[ast.Ref]js_ast.ConstValue
+
+	// Property mangling results go here
+	MangledProps map[ast.Ref]string
+
+	// This will be present if the input file had a source map. In that case we
+	// want to map all the way back to the original input file(s).
+	InputSourceMap *sourcemap.SourceMap
+
+	// If we're writing out a source map, this table of line start indices lets
+	// us do binary search on to figure out what line a given AST node came from
+	LineOffsetTables []sourcemap.LineOffsetTable
+
+	ToCommonJSRef       ast.Ref
+	ToESMRef            ast.Ref
+	RuntimeRequireRef   ast.Ref
+	UnsupportedFeatures compat.JSFeature
+	Indent              int
+	LineLimit           int
+	OutputFormat        config.Format
+	MinifyWhitespace    bool
+	MinifyIdentifiers   bool
+	MinifySyntax        bool
+	ASCIIOnly           bool
+	LegalComments       config.LegalComments
+	SourceMap           config.SourceMap
+	AddSourceMappings   bool
+	NeedsMetafile       bool
+}
+
+type RequireOrImportMeta struct {
+	// CommonJS files will return the "require_*" wrapper function and an invalid
+	// exports object reference. Lazily-initialized ESM files will return the
+	// "init_*" wrapper function and the exports object for that file.
+	WrapperRef     ast.Ref
+	ExportsRef     ast.Ref
+	IsWrapperAsync bool
+}
+
+type PrintResult struct {
+	JS                     []byte
+	ExtractedLegalComments []string
+	JSONMetadataImports    []string
+
+	// This source map chunk just contains the VLQ-encoded offsets for the "JS"
+	// field above. It's not a full source map. The bundler will be joining many
+	// source map chunks together to form the final source map.
+	SourceMapChunk sourcemap.Chunk
+}
+
+func Print(tree js_ast.AST, symbols ast.SymbolMap, r renamer.Renamer, options Options) PrintResult {
+	p := &printer{
+		symbols:       symbols,
+		renamer:       r,
+		importRecords: tree.ImportRecords,
+		options:       options,
+		moduleType:    tree.ModuleTypeData.Type,
+		exprComments:  tree.ExprComments,
+		wasLazyExport: tree.HasLazyExport,
+
+		stmtStart:          -1,
+		exportDefaultStart: -1,
+		arrowExprStart:     -1,
+		forOfInitStart:     -1,
+
+		prevOpEnd:            -1,
+		needSpaceBeforeDot:   -1,
+		prevRegExpEnd:        -1,
+		noLeadingNewlineHere: -1,
+		builder:              sourcemap.MakeChunkBuilder(options.InputSourceMap, options.LineOffsetTables, options.ASCIIOnly),
+	}
+
+	if p.exprComments != nil {
+		p.printedExprComments = make(map[logger.Loc]bool)
+	}
+
+	p.astHelpers = js_ast.MakeHelperContext(func(ref ast.Ref) bool {
+		ref = ast.FollowSymbols(symbols, ref)
+		return symbols.Get(ref).Kind == ast.SymbolUnbound
+	})
+
+	// Add the top-level directive if present
+	for _, directive := range tree.Directives {
+		p.printIndent()
+		p.printQuotedUTF8(directive, 0)
+		p.print(";")
+		p.printNewline()
+	}
+
+	for _, part := range tree.Parts {
+		for _, stmt := range part.Stmts {
+			p.printStmt(stmt, canOmitStatement)
+			p.printSemicolonIfNeeded()
+		}
+	}
+
+	result := PrintResult{
+		JS:                     p.js,
+		JSONMetadataImports:    p.jsonMetadataImports,
+		ExtractedLegalComments: p.extractedLegalComments,
+	}
+	if options.SourceMap != config.SourceMapNone {
+		// This is expensive. Only do this if it's necessary.
+		result.SourceMapChunk = p.builder.GenerateChunk(p.js)
+	}
+	return result
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/linker/debug.go b/source/vendor/github.com/evanw/esbuild/internal/linker/debug.go
new file mode 100644
index 0000000..04d0a39
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/linker/debug.go
@@ -0,0 +1,148 @@
+package linker
+
+import (
+	"fmt"
+	"strings"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/graph"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+)
+
+// Set this to true and then load the resulting metafile in "graph-debugger.html"
+// to debug graph information.
+//
+// This is deliberately not exposed in the final binary. It is *very* internal
+// and only exists to help debug esbuild itself. Make sure this is always set
+// back to false before committing.
+const debugVerboseMetafile = false
+
+func (c *linkerContext) generateExtraDataForFileJS(sourceIndex uint32) string {
+	if !debugVerboseMetafile {
+		return ""
+	}
+
+	file := &c.graph.Files[sourceIndex]
+	repr := file.InputFile.Repr.(*graph.JSRepr)
+	sb := strings.Builder{}
+	isFirstPartWithStmts := true
+
+	quoteSym := func(ref ast.Ref) string {
+		name := fmt.Sprintf("%d:%d [%s]", ref.SourceIndex, ref.InnerIndex, c.graph.Symbols.Get(ref).OriginalName)
+		return string(helpers.QuoteForJSON(name, c.options.ASCIIOnly))
+	}
+
+	sb.WriteString(`,"parts":[`)
+	for partIndex, part := range repr.AST.Parts {
+		if partIndex > 0 {
+			sb.WriteByte(',')
+		}
+		var isFirst bool
+		code := ""
+
+		sb.WriteString(fmt.Sprintf(`{"isLive":%v`, part.IsLive))
+		sb.WriteString(fmt.Sprintf(`,"canBeRemovedIfUnused":%v`, part.CanBeRemovedIfUnused))
+
+		if partIndex == int(js_ast.NSExportPartIndex) {
+			sb.WriteString(`,"nsExportPartIndex":true`)
+		} else if ast.MakeIndex32(uint32(partIndex)) == repr.Meta.WrapperPartIndex {
+			sb.WriteString(`,"wrapperPartIndex":true`)
+		} else if len(part.Stmts) > 0 {
+			contents := file.InputFile.Source.Contents
+			start := int(part.Stmts[0].Loc.Start)
+			if isFirstPartWithStmts {
+				start = 0
+				isFirstPartWithStmts = false
+			}
+			end := len(contents)
+			if partIndex+1 < len(repr.AST.Parts) {
+				if nextStmts := repr.AST.Parts[partIndex+1].Stmts; len(nextStmts) > 0 {
+					if nextStart := int(nextStmts[0].Loc.Start); nextStart >= start {
+						end = int(nextStart)
+					}
+				}
+			}
+			start = moveBeforeExport(contents, start)
+			end = moveBeforeExport(contents, end)
+			code = contents[start:end]
+		}
+
+		// importRecords
+		sb.WriteString(`,"importRecords":[`)
+		isFirst = true
+		for _, importRecordIndex := range part.ImportRecordIndices {
+			record := repr.AST.ImportRecords[importRecordIndex]
+			if !record.SourceIndex.IsValid() {
+				continue
+			}
+			if isFirst {
+				isFirst = false
+			} else {
+				sb.WriteByte(',')
+			}
+			path := c.graph.Files[record.SourceIndex.GetIndex()].InputFile.Source.PrettyPath
+			sb.WriteString(fmt.Sprintf(`{"source":%s}`, helpers.QuoteForJSON(path, c.options.ASCIIOnly)))
+		}
+		sb.WriteByte(']')
+
+		// declaredSymbols
+		sb.WriteString(`,"declaredSymbols":[`)
+		isFirst = true
+		for _, declSym := range part.DeclaredSymbols {
+			if !declSym.IsTopLevel {
+				continue
+			}
+			if isFirst {
+				isFirst = false
+			} else {
+				sb.WriteByte(',')
+			}
+			sb.WriteString(fmt.Sprintf(`{"name":%s}`, quoteSym(declSym.Ref)))
+		}
+		sb.WriteByte(']')
+
+		// symbolUses
+		sb.WriteString(`,"symbolUses":[`)
+		isFirst = true
+		for ref, uses := range part.SymbolUses {
+			if isFirst {
+				isFirst = false
+			} else {
+				sb.WriteByte(',')
+			}
+			sb.WriteString(fmt.Sprintf(`{"name":%s,"countEstimate":%d}`, quoteSym(ref), uses.CountEstimate))
+		}
+		sb.WriteByte(']')
+
+		// dependencies
+		sb.WriteString(`,"dependencies":[`)
+		for i, dep := range part.Dependencies {
+			if i > 0 {
+				sb.WriteByte(',')
+			}
+			sb.WriteString(fmt.Sprintf(`{"source":%s,"partIndex":%d}`,
+				helpers.QuoteForJSON(c.graph.Files[dep.SourceIndex].InputFile.Source.PrettyPath, c.options.ASCIIOnly),
+				dep.PartIndex,
+			))
+		}
+		sb.WriteByte(']')
+
+		// code
+		sb.WriteString(`,"code":`)
+		sb.Write(helpers.QuoteForJSON(code, c.options.ASCIIOnly))
+
+		sb.WriteByte('}')
+	}
+	sb.WriteString(`]`)
+
+	return sb.String()
+}
+
+func moveBeforeExport(contents string, i int) int {
+	contents = strings.TrimRight(contents[:i], " \t\r\n")
+	if strings.HasSuffix(contents, "export") {
+		return len(contents) - 6
+	}
+	return i
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/linker/linker.go b/source/vendor/github.com/evanw/esbuild/internal/linker/linker.go
new file mode 100644
index 0000000..219fe8e
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/linker/linker.go
@@ -0,0 +1,7154 @@
+package linker
+
+// This package implements the second phase of the bundling operation that
+// generates the output files when given a module graph. It has been split off
+// into separate package to allow two linkers to cleanly exist in the same code
+// base. This will be useful when rewriting the linker because the new one can
+// be off by default to minimize disruption, but can still be enabled by anyone
+// to assist in giving feedback on the rewrite.
+
+import (
+	"bytes"
+	"encoding/base64"
+	"encoding/binary"
+	"fmt"
+	"hash"
+	"path"
+	"sort"
+	"strconv"
+	"strings"
+	"sync"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/bundler"
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/config"
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/css_lexer"
+	"github.com/evanw/esbuild/internal/css_parser"
+	"github.com/evanw/esbuild/internal/css_printer"
+	"github.com/evanw/esbuild/internal/fs"
+	"github.com/evanw/esbuild/internal/graph"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/js_lexer"
+	"github.com/evanw/esbuild/internal/js_printer"
+	"github.com/evanw/esbuild/internal/logger"
+	"github.com/evanw/esbuild/internal/renamer"
+	"github.com/evanw/esbuild/internal/resolver"
+	"github.com/evanw/esbuild/internal/runtime"
+	"github.com/evanw/esbuild/internal/sourcemap"
+	"github.com/evanw/esbuild/internal/xxhash"
+)
+
+type linkerContext struct {
+	options *config.Options
+	timer   *helpers.Timer
+	log     logger.Log
+	fs      fs.FS
+	res     *resolver.Resolver
+	graph   graph.LinkerGraph
+	chunks  []chunkInfo
+
+	// This helps avoid an infinite loop when matching imports to exports
+	cycleDetector []importTracker
+
+	// This represents the parallel computation of source map related data.
+	// Calling this will block until the computation is done. The resulting value
+	// is shared between threads and must be treated as immutable.
+	dataForSourceMaps func() []bundler.DataForSourceMap
+
+	// This is passed to us from the bundling phase
+	uniqueKeyPrefix      string
+	uniqueKeyPrefixBytes []byte // This is just "uniqueKeyPrefix" in byte form
+
+	// Property mangling results go here
+	mangledProps map[ast.Ref]string
+
+	// We may need to refer to the CommonJS "module" symbol for exports
+	unboundModuleRef ast.Ref
+
+	// We may need to refer to the "__esm" and/or "__commonJS" runtime symbols
+	cjsRuntimeRef ast.Ref
+	esmRuntimeRef ast.Ref
+}
+
+type partRange struct {
+	sourceIndex    uint32
+	partIndexBegin uint32
+	partIndexEnd   uint32
+}
+
+type chunkInfo struct {
+	// This is a random string and is used to represent the output path of this
+	// chunk before the final output path has been computed.
+	uniqueKey string
+
+	filesWithPartsInChunk map[uint32]bool
+	entryBits             helpers.BitSet
+
+	// For code splitting
+	crossChunkImports []chunkImport
+
+	// This is the representation-specific information
+	chunkRepr chunkRepr
+
+	// This is the final path of this chunk relative to the output directory, but
+	// without the substitution of the final hash (since it hasn't been computed).
+	finalTemplate []config.PathTemplate
+
+	// This is the final path of this chunk relative to the output directory. It
+	// is the substitution of the final hash into "finalTemplate".
+	finalRelPath string
+
+	// If non-empty, this chunk needs to generate an external legal comments file.
+	externalLegalComments []byte
+
+	// This contains the hash for just this chunk without including information
+	// from the hashes of other chunks. Later on in the linking process, the
+	// final hash for this chunk will be constructed by merging the isolated
+	// hashes of all transitive dependencies of this chunk. This is separated
+	// into two phases like this to handle cycles in the chunk import graph.
+	waitForIsolatedHash func() []byte
+
+	// Other fields relating to the output file for this chunk
+	jsonMetadataChunkCallback func(finalOutputSize int) helpers.Joiner
+	outputSourceMap           sourcemap.SourceMapPieces
+
+	// When this chunk is initially generated in isolation, the output pieces
+	// will contain slices of the output with the unique keys of other chunks
+	// omitted.
+	intermediateOutput intermediateOutput
+
+	// This information is only useful if "isEntryPoint" is true
+	entryPointBit uint   // An index into "c.graph.EntryPoints"
+	sourceIndex   uint32 // An index into "c.sources"
+	isEntryPoint  bool
+
+	isExecutable bool
+}
+
+type chunkImport struct {
+	chunkIndex uint32
+	importKind ast.ImportKind
+}
+
+type outputPieceIndexKind uint8
+
+const (
+	outputPieceNone outputPieceIndexKind = iota
+	outputPieceAssetIndex
+	outputPieceChunkIndex
+)
+
+// This is a chunk of source code followed by a reference to another chunk. For
+// example, the file "@import 'CHUNK0001'; body { color: black; }" would be
+// represented by two pieces, one with the data "@import '" and another with the
+// data "'; body { color: black; }". The first would have the chunk index 1 and
+// the second would have an invalid chunk index.
+type outputPiece struct {
+	data []byte
+
+	// Note: The "kind" may be "outputPieceNone" in which case there is one piece
+	// with data and no chunk index. For example, the chunk may not contain any
+	// imports.
+	index uint32
+	kind  outputPieceIndexKind
+}
+
+type intermediateOutput struct {
+	// If the chunk has references to other chunks, then "pieces" contains the
+	// contents of the chunk and "joiner" should not be used. Another joiner
+	// will have to be constructed later when merging the pieces together.
+	pieces []outputPiece
+
+	// If the chunk doesn't have any references to other chunks, then "pieces" is
+	// nil and "joiner" contains the contents of the chunk. This is more efficient
+	// because it avoids doing a join operation twice.
+	joiner helpers.Joiner
+}
+
+type chunkRepr interface{ isChunk() }
+
+func (*chunkReprJS) isChunk()  {}
+func (*chunkReprCSS) isChunk() {}
+
+type chunkReprJS struct {
+	filesInChunkInOrder []uint32
+	partsInChunkInOrder []partRange
+
+	// For code splitting
+	exportsToOtherChunks   map[ast.Ref]string
+	importsFromOtherChunks map[uint32]crossChunkImportItemArray
+	crossChunkPrefixStmts  []js_ast.Stmt
+	crossChunkSuffixStmts  []js_ast.Stmt
+
+	cssChunkIndex uint32
+	hasCSSChunk   bool
+}
+
+type chunkReprCSS struct {
+	importsInChunkInOrder []cssImportOrder
+}
+
+type externalImportCSS struct {
+	path                   logger.Path
+	conditions             []css_ast.ImportConditions
+	conditionImportRecords []ast.ImportRecord
+}
+
+// Returns a log where "log.HasErrors()" only returns true if any errors have
+// been logged since this call. This is useful when there have already been
+// errors logged by other linkers that share the same log.
+func wrappedLog(log logger.Log) logger.Log {
+	var mutex sync.Mutex
+	var hasErrors bool
+	addMsg := log.AddMsg
+
+	log.AddMsg = func(msg logger.Msg) {
+		if msg.Kind == logger.Error {
+			mutex.Lock()
+			defer mutex.Unlock()
+			hasErrors = true
+		}
+		addMsg(msg)
+	}
+
+	log.HasErrors = func() bool {
+		mutex.Lock()
+		defer mutex.Unlock()
+		return hasErrors
+	}
+
+	return log
+}
+
+func Link(
+	options *config.Options,
+	timer *helpers.Timer,
+	log logger.Log,
+	fs fs.FS,
+	res *resolver.Resolver,
+	inputFiles []graph.InputFile,
+	entryPoints []graph.EntryPoint,
+	uniqueKeyPrefix string,
+	reachableFiles []uint32,
+	dataForSourceMaps func() []bundler.DataForSourceMap,
+) []graph.OutputFile {
+	timer.Begin("Link")
+	defer timer.End("Link")
+
+	log = wrappedLog(log)
+
+	timer.Begin("Clone linker graph")
+	c := linkerContext{
+		options:              options,
+		timer:                timer,
+		log:                  log,
+		fs:                   fs,
+		res:                  res,
+		dataForSourceMaps:    dataForSourceMaps,
+		uniqueKeyPrefix:      uniqueKeyPrefix,
+		uniqueKeyPrefixBytes: []byte(uniqueKeyPrefix),
+		graph: graph.CloneLinkerGraph(
+			inputFiles,
+			reachableFiles,
+			entryPoints,
+			options.CodeSplitting,
+		),
+	}
+	timer.End("Clone linker graph")
+
+	// Use a smaller version of these functions if we don't need profiler names
+	runtimeRepr := c.graph.Files[runtime.SourceIndex].InputFile.Repr.(*graph.JSRepr)
+	if c.options.ProfilerNames {
+		c.cjsRuntimeRef = runtimeRepr.AST.NamedExports["__commonJS"].Ref
+		c.esmRuntimeRef = runtimeRepr.AST.NamedExports["__esm"].Ref
+	} else {
+		c.cjsRuntimeRef = runtimeRepr.AST.NamedExports["__commonJSMin"].Ref
+		c.esmRuntimeRef = runtimeRepr.AST.NamedExports["__esmMin"].Ref
+	}
+
+	var additionalFiles []graph.OutputFile
+	for _, entryPoint := range entryPoints {
+		file := &c.graph.Files[entryPoint.SourceIndex].InputFile
+		switch repr := file.Repr.(type) {
+		case *graph.JSRepr:
+			// Loaders default to CommonJS when they are the entry point and the output
+			// format is not ESM-compatible since that avoids generating the ESM-to-CJS
+			// machinery.
+			if repr.AST.HasLazyExport && (c.options.Mode == config.ModePassThrough ||
+				(c.options.Mode == config.ModeConvertFormat && !c.options.OutputFormat.KeepESMImportExportSyntax())) {
+				repr.AST.ExportsKind = js_ast.ExportsCommonJS
+			}
+
+			// Entry points with ES6 exports must generate an exports object when
+			// targeting non-ES6 formats. Note that the IIFE format only needs this
+			// when the global name is present, since that's the only way the exports
+			// can actually be observed externally.
+			if repr.AST.ExportKeyword.Len > 0 && (options.OutputFormat == config.FormatCommonJS ||
+				(options.OutputFormat == config.FormatIIFE && len(options.GlobalName) > 0)) {
+				repr.AST.UsesExportsRef = true
+				repr.Meta.ForceIncludeExportsForEntryPoint = true
+			}
+
+		case *graph.CopyRepr:
+			// If an entry point uses the copy loader, then copy the file manually
+			// here. Other uses of the copy loader will automatically be included
+			// along with the corresponding bundled chunk but that doesn't happen
+			// for entry points.
+			additionalFiles = append(additionalFiles, file.AdditionalFiles...)
+		}
+	}
+
+	// Allocate a new unbound symbol called "module" in case we need it later
+	if c.options.OutputFormat == config.FormatCommonJS {
+		c.unboundModuleRef = c.graph.GenerateNewSymbol(runtime.SourceIndex, ast.SymbolUnbound, "module")
+	} else {
+		c.unboundModuleRef = ast.InvalidRef
+	}
+
+	c.scanImportsAndExports()
+
+	// Stop now if there were errors
+	if c.log.HasErrors() {
+		c.options.ExclusiveMangleCacheUpdate(func(map[string]interface{}, map[string]bool) {
+			// Always do this so that we don't cause other entry points when there are errors
+		})
+		return []graph.OutputFile{}
+	}
+
+	c.treeShakingAndCodeSplitting()
+
+	if c.options.Mode == config.ModePassThrough {
+		for _, entryPoint := range c.graph.EntryPoints() {
+			c.preventExportsFromBeingRenamed(entryPoint.SourceIndex)
+		}
+	}
+
+	c.computeChunks()
+	c.computeCrossChunkDependencies()
+
+	// Merge mangled properties before chunks are generated since the names must
+	// be consistent across all chunks, or the generated code will break
+	c.timer.Begin("Waiting for mangle cache")
+	c.options.ExclusiveMangleCacheUpdate(func(
+		mangleCache map[string]interface{},
+		cssUsedLocalNames map[string]bool,
+	) {
+		c.timer.End("Waiting for mangle cache")
+		c.mangleProps(mangleCache)
+		c.mangleLocalCSS(cssUsedLocalNames)
+	})
+
+	// Make sure calls to "ast.FollowSymbols()" in parallel goroutines after this
+	// won't hit concurrent map mutation hazards
+	ast.FollowAllSymbols(c.graph.Symbols)
+
+	return c.generateChunksInParallel(additionalFiles)
+}
+
+func (c *linkerContext) mangleProps(mangleCache map[string]interface{}) {
+	c.timer.Begin("Mangle props")
+	defer c.timer.End("Mangle props")
+
+	mangledProps := make(map[ast.Ref]string)
+	c.mangledProps = mangledProps
+
+	// Reserve all JS keywords
+	reservedProps := make(map[string]bool)
+	for keyword := range js_lexer.Keywords {
+		reservedProps[keyword] = true
+	}
+
+	// Reserve all target properties in the cache
+	for original, remapped := range mangleCache {
+		if remapped == false {
+			reservedProps[original] = true
+		} else {
+			reservedProps[remapped.(string)] = true
+		}
+	}
+
+	// Merge all mangled property symbols together
+	freq := ast.CharFreq{}
+	mergedProps := make(map[string]ast.Ref)
+	for _, sourceIndex := range c.graph.ReachableFiles {
+		// Don't mangle anything in the runtime code
+		if sourceIndex == runtime.SourceIndex {
+			continue
+		}
+
+		// For each file
+		if repr, ok := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr); ok {
+			// Reserve all non-mangled properties
+			for prop := range repr.AST.ReservedProps {
+				reservedProps[prop] = true
+			}
+
+			// Merge each mangled property with other ones of the same name
+			for name, ref := range repr.AST.MangledProps {
+				if existing, ok := mergedProps[name]; ok {
+					ast.MergeSymbols(c.graph.Symbols, ref, existing)
+				} else {
+					mergedProps[name] = ref
+				}
+			}
+
+			// Include this file's frequency histogram, which affects the mangled names
+			if repr.AST.CharFreq != nil {
+				freq.Include(repr.AST.CharFreq)
+			}
+		}
+	}
+
+	// Sort by use count (note: does not currently account for live vs. dead code)
+	sorted := make(renamer.StableSymbolCountArray, 0, len(mergedProps))
+	stableSourceIndices := c.graph.StableSourceIndices
+	for _, ref := range mergedProps {
+		sorted = append(sorted, renamer.StableSymbolCount{
+			StableSourceIndex: stableSourceIndices[ref.SourceIndex],
+			Ref:               ref,
+			Count:             c.graph.Symbols.Get(ref).UseCountEstimate,
+		})
+	}
+	sort.Sort(sorted)
+
+	// Assign names in order of use count
+	minifier := ast.DefaultNameMinifierJS.ShuffleByCharFreq(freq)
+	nextName := 0
+	for _, symbolCount := range sorted {
+		symbol := c.graph.Symbols.Get(symbolCount.Ref)
+
+		// Don't change existing mappings
+		if existing, ok := mangleCache[symbol.OriginalName]; ok {
+			if existing != false {
+				mangledProps[symbolCount.Ref] = existing.(string)
+			}
+			continue
+		}
+
+		// Generate a new name
+		name := minifier.NumberToMinifiedName(nextName)
+		nextName++
+
+		// Avoid reserved properties
+		for reservedProps[name] {
+			name = minifier.NumberToMinifiedName(nextName)
+			nextName++
+		}
+
+		// Track the new mapping
+		if mangleCache != nil {
+			mangleCache[symbol.OriginalName] = name
+		}
+		mangledProps[symbolCount.Ref] = name
+	}
+}
+
+func (c *linkerContext) mangleLocalCSS(usedLocalNames map[string]bool) {
+	c.timer.Begin("Mangle local CSS")
+	defer c.timer.End("Mangle local CSS")
+
+	mangledProps := c.mangledProps
+	globalNames := make(map[string]bool)
+	localNames := make(map[ast.Ref]struct{})
+
+	// Collect all local and global CSS names
+	freq := ast.CharFreq{}
+	for _, sourceIndex := range c.graph.ReachableFiles {
+		if repr, ok := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.CSSRepr); ok {
+			for innerIndex, symbol := range c.graph.Symbols.SymbolsForSource[sourceIndex] {
+				if symbol.Kind == ast.SymbolGlobalCSS {
+					globalNames[symbol.OriginalName] = true
+				} else {
+					ref := ast.Ref{SourceIndex: sourceIndex, InnerIndex: uint32(innerIndex)}
+					ref = ast.FollowSymbols(c.graph.Symbols, ref)
+					localNames[ref] = struct{}{}
+				}
+			}
+
+			// Include this file's frequency histogram, which affects the mangled names
+			if repr.AST.CharFreq != nil {
+				freq.Include(repr.AST.CharFreq)
+			}
+		}
+	}
+
+	// Sort by use count (note: does not currently account for live vs. dead code)
+	sorted := make(renamer.StableSymbolCountArray, 0, len(localNames))
+	stableSourceIndices := c.graph.StableSourceIndices
+	for ref := range localNames {
+		sorted = append(sorted, renamer.StableSymbolCount{
+			StableSourceIndex: stableSourceIndices[ref.SourceIndex],
+			Ref:               ref,
+			Count:             c.graph.Symbols.Get(ref).UseCountEstimate,
+		})
+	}
+	sort.Sort(sorted)
+
+	// Rename all local names to avoid collisions
+	if c.options.MinifyIdentifiers {
+		minifier := ast.DefaultNameMinifierCSS.ShuffleByCharFreq(freq)
+		nextName := 0
+
+		for _, symbolCount := range sorted {
+			name := minifier.NumberToMinifiedName(nextName)
+			for globalNames[name] || usedLocalNames[name] {
+				nextName++
+				name = minifier.NumberToMinifiedName(nextName)
+			}
+
+			// Turn this local name into a global one
+			mangledProps[symbolCount.Ref] = name
+			usedLocalNames[name] = true
+		}
+	} else {
+		nameCounts := make(map[string]uint32)
+
+		for _, symbolCount := range sorted {
+			symbol := c.graph.Symbols.Get(symbolCount.Ref)
+			name := fmt.Sprintf("%s_%s", c.graph.Files[symbolCount.Ref.SourceIndex].InputFile.Source.IdentifierName, symbol.OriginalName)
+
+			// If the name is already in use, generate a new name by appending a number
+			if globalNames[name] || usedLocalNames[name] {
+				// To avoid O(n^2) behavior, the number must start off being the number
+				// that we used last time there was a collision with this name. Otherwise
+				// if there are many collisions with the same name, each name collision
+				// would have to increment the counter past all previous name collisions
+				// which is a O(n^2) time algorithm.
+				tries, ok := nameCounts[name]
+				if !ok {
+					tries = 1
+				}
+				prefix := name
+
+				// Keep incrementing the number until the name is unused
+				for {
+					tries++
+					name = prefix + strconv.Itoa(int(tries))
+
+					// Make sure this new name is unused
+					if !globalNames[name] && !usedLocalNames[name] {
+						// Store the count so we can start here next time instead of starting
+						// from 1. This means we avoid O(n^2) behavior.
+						nameCounts[prefix] = tries
+						break
+					}
+				}
+			}
+
+			// Turn this local name into a global one
+			mangledProps[symbolCount.Ref] = name
+			usedLocalNames[name] = true
+		}
+	}
+}
+
+// Currently the automatic chunk generation algorithm should by construction
+// never generate chunks that import each other since files are allocated to
+// chunks based on which entry points they are reachable from.
+//
+// This will change in the future when we allow manual chunk labels. But before
+// we allow manual chunk labels, we'll need to rework module initialization to
+// allow code splitting chunks to be lazily-initialized.
+//
+// Since that work hasn't been finished yet, cycles in the chunk import graph
+// can cause initialization bugs. So let's forbid these cycles for now to guard
+// against code splitting bugs that could cause us to generate buggy chunks.
+func (c *linkerContext) enforceNoCyclicChunkImports() {
+	var validate func(int, map[int]int) bool
+
+	// DFS memoization with 3-colors, more space efficient
+	// 0: white (unvisited), 1: gray (visiting), 2: black (visited)
+	colors := make(map[int]int)
+	validate = func(chunkIndex int, colors map[int]int) bool {
+		if colors[chunkIndex] == 1 {
+			c.log.AddError(nil, logger.Range{}, "Internal error: generated chunks contain a circular import")
+			return true
+		}
+
+		if colors[chunkIndex] == 2 {
+			return false
+		}
+
+		colors[chunkIndex] = 1
+
+		for _, chunkImport := range c.chunks[chunkIndex].crossChunkImports {
+			// Ignore cycles caused by dynamic "import()" expressions. These are fine
+			// because they don't necessarily cause initialization order issues and
+			// they don't indicate a bug in our chunk generation algorithm. They arise
+			// normally in real code (e.g. two files that import each other).
+			if chunkImport.importKind != ast.ImportDynamic {
+
+				// Recursively validate otherChunkIndex
+				if validate(int(chunkImport.chunkIndex), colors) {
+					return true
+				}
+			}
+		}
+
+		colors[chunkIndex] = 2
+		return false
+	}
+
+	for i := range c.chunks {
+		if validate(i, colors) {
+			break
+		}
+	}
+}
+
+func (c *linkerContext) generateChunksInParallel(additionalFiles []graph.OutputFile) []graph.OutputFile {
+	c.timer.Begin("Generate chunks")
+	defer c.timer.End("Generate chunks")
+
+	// Generate each chunk on a separate goroutine. When a chunk needs to
+	// reference the path of another chunk, it will use a temporary path called
+	// the "uniqueKey" since the final path hasn't been computed yet (and is
+	// in general uncomputable at this point because paths have hashes that
+	// include information about chunk dependencies, and chunk dependencies
+	// can be cyclic due to dynamic imports).
+	generateWaitGroup := sync.WaitGroup{}
+	generateWaitGroup.Add(len(c.chunks))
+	for chunkIndex := range c.chunks {
+		switch c.chunks[chunkIndex].chunkRepr.(type) {
+		case *chunkReprJS:
+			go c.generateChunkJS(chunkIndex, &generateWaitGroup)
+		case *chunkReprCSS:
+			go c.generateChunkCSS(chunkIndex, &generateWaitGroup)
+		}
+	}
+	c.enforceNoCyclicChunkImports()
+	generateWaitGroup.Wait()
+
+	// Compute the final hashes of each chunk, then use those to create the final
+	// paths of each chunk. This can technically be done in parallel but it
+	// probably doesn't matter so much because we're not hashing that much data.
+	visited := make([]uint32, len(c.chunks))
+	var finalBytes []byte
+	for chunkIndex := range c.chunks {
+		chunk := &c.chunks[chunkIndex]
+		var hashSubstitution *string
+
+		// Only wait for the hash if necessary
+		if config.HasPlaceholder(chunk.finalTemplate, config.HashPlaceholder) {
+			// Compute the final hash using the isolated hashes of the dependencies
+			hash := xxhash.New()
+			c.appendIsolatedHashesForImportedChunks(hash, uint32(chunkIndex), visited, ^uint32(chunkIndex))
+			finalBytes = hash.Sum(finalBytes[:0])
+			finalString := bundler.HashForFileName(finalBytes)
+			hashSubstitution = &finalString
+		}
+
+		// Render the last remaining placeholder in the template
+		chunk.finalRelPath = config.TemplateToString(config.SubstituteTemplate(chunk.finalTemplate, config.PathPlaceholders{
+			Hash: hashSubstitution,
+		}))
+	}
+
+	// Generate the final output files by joining file pieces together and
+	// substituting the temporary paths for the final paths. This substitution
+	// can be done in parallel for each chunk.
+	c.timer.Begin("Generate final output files")
+	var resultsWaitGroup sync.WaitGroup
+	results := make([][]graph.OutputFile, len(c.chunks))
+	resultsWaitGroup.Add(len(c.chunks))
+	for chunkIndex, chunk := range c.chunks {
+		go func(chunkIndex int, chunk chunkInfo) {
+			var outputFiles []graph.OutputFile
+
+			// Each file may optionally contain additional files to be copied to the
+			// output directory. This is used by the "file" and "copy" loaders.
+			var commentPrefix string
+			var commentSuffix string
+			switch chunkRepr := chunk.chunkRepr.(type) {
+			case *chunkReprJS:
+				for _, sourceIndex := range chunkRepr.filesInChunkInOrder {
+					outputFiles = append(outputFiles, c.graph.Files[sourceIndex].InputFile.AdditionalFiles...)
+				}
+				commentPrefix = "//"
+
+			case *chunkReprCSS:
+				for _, entry := range chunkRepr.importsInChunkInOrder {
+					if entry.kind == cssImportSourceIndex {
+						outputFiles = append(outputFiles, c.graph.Files[entry.sourceIndex].InputFile.AdditionalFiles...)
+					}
+				}
+				commentPrefix = "/*"
+				commentSuffix = " */"
+			}
+
+			// Path substitution for the chunk itself
+			finalRelDir := c.fs.Dir(chunk.finalRelPath)
+			outputContentsJoiner, outputSourceMapShifts := c.substituteFinalPaths(chunk.intermediateOutput,
+				func(finalRelPathForImport string) string {
+					return c.pathBetweenChunks(finalRelDir, finalRelPathForImport)
+				})
+
+			// Generate the optional legal comments file for this chunk
+			if chunk.externalLegalComments != nil {
+				finalRelPathForLegalComments := chunk.finalRelPath + ".LEGAL.txt"
+
+				// Link the file to the legal comments
+				if c.options.LegalComments == config.LegalCommentsLinkedWithComment {
+					importPath := c.pathBetweenChunks(finalRelDir, finalRelPathForLegalComments)
+					importPath = strings.TrimPrefix(importPath, "./")
+					outputContentsJoiner.EnsureNewlineAtEnd()
+					outputContentsJoiner.AddString("/*! For license information please see ")
+					outputContentsJoiner.AddString(importPath)
+					outputContentsJoiner.AddString(" */\n")
+				}
+
+				// Write the external legal comments file
+				outputFiles = append(outputFiles, graph.OutputFile{
+					AbsPath:  c.fs.Join(c.options.AbsOutputDir, finalRelPathForLegalComments),
+					Contents: chunk.externalLegalComments,
+					JSONMetadataChunk: fmt.Sprintf(
+						"{\n      \"imports\": [],\n      \"exports\": [],\n      \"inputs\": {},\n      \"bytes\": %d\n    }", len(chunk.externalLegalComments)),
+				})
+			}
+
+			// Generate the optional source map for this chunk
+			if c.options.SourceMap != config.SourceMapNone && chunk.outputSourceMap.HasContent() {
+				outputSourceMap := chunk.outputSourceMap.Finalize(outputSourceMapShifts)
+				finalRelPathForSourceMap := chunk.finalRelPath + ".map"
+
+				// Potentially write a trailing source map comment
+				switch c.options.SourceMap {
+				case config.SourceMapLinkedWithComment:
+					importPath := c.pathBetweenChunks(finalRelDir, finalRelPathForSourceMap)
+					importPath = strings.TrimPrefix(importPath, "./")
+					outputContentsJoiner.EnsureNewlineAtEnd()
+					outputContentsJoiner.AddString(commentPrefix)
+					outputContentsJoiner.AddString("# sourceMappingURL=")
+					outputContentsJoiner.AddString(importPath)
+					outputContentsJoiner.AddString(commentSuffix)
+					outputContentsJoiner.AddString("\n")
+
+				case config.SourceMapInline, config.SourceMapInlineAndExternal:
+					outputContentsJoiner.EnsureNewlineAtEnd()
+					outputContentsJoiner.AddString(commentPrefix)
+					outputContentsJoiner.AddString("# sourceMappingURL=data:application/json;base64,")
+					outputContentsJoiner.AddString(base64.StdEncoding.EncodeToString(outputSourceMap))
+					outputContentsJoiner.AddString(commentSuffix)
+					outputContentsJoiner.AddString("\n")
+				}
+
+				// Potentially write the external source map file
+				switch c.options.SourceMap {
+				case config.SourceMapLinkedWithComment, config.SourceMapInlineAndExternal, config.SourceMapExternalWithoutComment:
+					outputFiles = append(outputFiles, graph.OutputFile{
+						AbsPath:  c.fs.Join(c.options.AbsOutputDir, finalRelPathForSourceMap),
+						Contents: outputSourceMap,
+						JSONMetadataChunk: fmt.Sprintf(
+							"{\n      \"imports\": [],\n      \"exports\": [],\n      \"inputs\": {},\n      \"bytes\": %d\n    }", len(outputSourceMap)),
+					})
+				}
+			}
+
+			// Finalize the output contents
+			outputContents := outputContentsJoiner.Done()
+
+			// Path substitution for the JSON metadata
+			var jsonMetadataChunk string
+			if c.options.NeedsMetafile {
+				jsonMetadataChunkPieces := c.breakJoinerIntoPieces(chunk.jsonMetadataChunkCallback(len(outputContents)))
+				jsonMetadataChunkBytes, _ := c.substituteFinalPaths(jsonMetadataChunkPieces, func(finalRelPathForImport string) string {
+					return resolver.PrettyPath(c.fs, logger.Path{Text: c.fs.Join(c.options.AbsOutputDir, finalRelPathForImport), Namespace: "file"})
+				})
+				jsonMetadataChunk = string(jsonMetadataChunkBytes.Done())
+			}
+
+			// Generate the output file for this chunk
+			outputFiles = append(outputFiles, graph.OutputFile{
+				AbsPath:           c.fs.Join(c.options.AbsOutputDir, chunk.finalRelPath),
+				Contents:          outputContents,
+				JSONMetadataChunk: jsonMetadataChunk,
+				IsExecutable:      chunk.isExecutable,
+			})
+
+			results[chunkIndex] = outputFiles
+			resultsWaitGroup.Done()
+		}(chunkIndex, chunk)
+	}
+	resultsWaitGroup.Wait()
+	c.timer.End("Generate final output files")
+
+	// Merge the output files from the different goroutines together in order
+	outputFilesLen := len(additionalFiles)
+	for _, result := range results {
+		outputFilesLen += len(result)
+	}
+	outputFiles := make([]graph.OutputFile, 0, outputFilesLen)
+	outputFiles = append(outputFiles, additionalFiles...)
+	for _, result := range results {
+		outputFiles = append(outputFiles, result...)
+	}
+	return outputFiles
+}
+
+// Given a set of output pieces (i.e. a buffer already divided into the spans
+// between import paths), substitute the final import paths in and then join
+// everything into a single byte buffer.
+func (c *linkerContext) substituteFinalPaths(
+	intermediateOutput intermediateOutput,
+	modifyPath func(string) string,
+) (j helpers.Joiner, shifts []sourcemap.SourceMapShift) {
+	// Optimization: If there can be no substitutions, just reuse the initial
+	// joiner that was used when generating the intermediate chunk output
+	// instead of creating another one and copying the whole file into it.
+	if intermediateOutput.pieces == nil {
+		return intermediateOutput.joiner, []sourcemap.SourceMapShift{{}}
+	}
+
+	var shift sourcemap.SourceMapShift
+	shifts = make([]sourcemap.SourceMapShift, 0, len(intermediateOutput.pieces))
+	shifts = append(shifts, shift)
+
+	for _, piece := range intermediateOutput.pieces {
+		var dataOffset sourcemap.LineColumnOffset
+		j.AddBytes(piece.data)
+		dataOffset.AdvanceBytes(piece.data)
+		shift.Before.Add(dataOffset)
+		shift.After.Add(dataOffset)
+
+		switch piece.kind {
+		case outputPieceAssetIndex:
+			file := c.graph.Files[piece.index]
+			if len(file.InputFile.AdditionalFiles) != 1 {
+				panic("Internal error")
+			}
+			relPath, _ := c.fs.Rel(c.options.AbsOutputDir, file.InputFile.AdditionalFiles[0].AbsPath)
+
+			// Make sure to always use forward slashes, even on Windows
+			relPath = strings.ReplaceAll(relPath, "\\", "/")
+
+			importPath := modifyPath(relPath)
+			j.AddString(importPath)
+			shift.Before.AdvanceString(file.InputFile.UniqueKeyForAdditionalFile)
+			shift.After.AdvanceString(importPath)
+			shifts = append(shifts, shift)
+
+		case outputPieceChunkIndex:
+			chunk := c.chunks[piece.index]
+			importPath := modifyPath(chunk.finalRelPath)
+			j.AddString(importPath)
+			shift.Before.AdvanceString(chunk.uniqueKey)
+			shift.After.AdvanceString(importPath)
+			shifts = append(shifts, shift)
+		}
+	}
+
+	return
+}
+
+func (c *linkerContext) accurateFinalByteCount(output intermediateOutput, chunkFinalRelDir string) int {
+	count := 0
+
+	// Note: The paths generated here must match "substituteFinalPaths" above
+	for _, piece := range output.pieces {
+		count += len(piece.data)
+
+		switch piece.kind {
+		case outputPieceAssetIndex:
+			file := c.graph.Files[piece.index]
+			if len(file.InputFile.AdditionalFiles) != 1 {
+				panic("Internal error")
+			}
+			relPath, _ := c.fs.Rel(c.options.AbsOutputDir, file.InputFile.AdditionalFiles[0].AbsPath)
+
+			// Make sure to always use forward slashes, even on Windows
+			relPath = strings.ReplaceAll(relPath, "\\", "/")
+
+			importPath := c.pathBetweenChunks(chunkFinalRelDir, relPath)
+			count += len(importPath)
+
+		case outputPieceChunkIndex:
+			chunk := c.chunks[piece.index]
+			importPath := c.pathBetweenChunks(chunkFinalRelDir, chunk.finalRelPath)
+			count += len(importPath)
+		}
+	}
+
+	return count
+}
+
+func (c *linkerContext) pathBetweenChunks(fromRelDir string, toRelPath string) string {
+	// Join with the public path if it has been configured
+	if c.options.PublicPath != "" {
+		return joinWithPublicPath(c.options.PublicPath, toRelPath)
+	}
+
+	// Otherwise, return a relative path
+	relPath, ok := c.fs.Rel(fromRelDir, toRelPath)
+	if !ok {
+		c.log.AddError(nil, logger.Range{},
+			fmt.Sprintf("Cannot traverse from directory %q to chunk %q", fromRelDir, toRelPath))
+		return ""
+	}
+
+	// Make sure to always use forward slashes, even on Windows
+	relPath = strings.ReplaceAll(relPath, "\\", "/")
+
+	// Make sure the relative path doesn't start with a name, since that could
+	// be interpreted as a package path instead of a relative path
+	if !strings.HasPrefix(relPath, "./") && !strings.HasPrefix(relPath, "../") {
+		relPath = "./" + relPath
+	}
+
+	return relPath
+}
+
+func (c *linkerContext) computeCrossChunkDependencies() {
+	c.timer.Begin("Compute cross-chunk dependencies")
+	defer c.timer.End("Compute cross-chunk dependencies")
+
+	if !c.options.CodeSplitting {
+		// No need to compute cross-chunk dependencies if there can't be any
+		return
+	}
+
+	type chunkMeta struct {
+		imports        map[ast.Ref]bool
+		exports        map[ast.Ref]bool
+		dynamicImports map[int]bool
+	}
+
+	chunkMetas := make([]chunkMeta, len(c.chunks))
+
+	// For each chunk, see what symbols it uses from other chunks. Do this in
+	// parallel because it's the most expensive part of this function.
+	waitGroup := sync.WaitGroup{}
+	waitGroup.Add(len(c.chunks))
+	for chunkIndex, chunk := range c.chunks {
+		go func(chunkIndex int, chunk chunkInfo) {
+			chunkMeta := &chunkMetas[chunkIndex]
+			imports := make(map[ast.Ref]bool)
+			chunkMeta.imports = imports
+			chunkMeta.exports = make(map[ast.Ref]bool)
+
+			// Go over each file in this chunk
+			for sourceIndex := range chunk.filesWithPartsInChunk {
+				// Go over each part in this file that's marked for inclusion in this chunk
+				switch repr := c.graph.Files[sourceIndex].InputFile.Repr.(type) {
+				case *graph.JSRepr:
+					for partIndex, partMeta := range repr.AST.Parts {
+						if !partMeta.IsLive {
+							continue
+						}
+						part := &repr.AST.Parts[partIndex]
+
+						// Rewrite external dynamic imports to point to the chunk for that entry point
+						for _, importRecordIndex := range part.ImportRecordIndices {
+							record := &repr.AST.ImportRecords[importRecordIndex]
+							if record.SourceIndex.IsValid() && c.isExternalDynamicImport(record, sourceIndex) {
+								otherChunkIndex := c.graph.Files[record.SourceIndex.GetIndex()].EntryPointChunkIndex
+								record.Path.Text = c.chunks[otherChunkIndex].uniqueKey
+								record.SourceIndex = ast.Index32{}
+								record.Flags |= ast.ShouldNotBeExternalInMetafile | ast.ContainsUniqueKey
+
+								// Track this cross-chunk dynamic import so we make sure to
+								// include its hash when we're calculating the hashes of all
+								// dependencies of this chunk.
+								if int(otherChunkIndex) != chunkIndex {
+									if chunkMeta.dynamicImports == nil {
+										chunkMeta.dynamicImports = make(map[int]bool)
+									}
+									chunkMeta.dynamicImports[int(otherChunkIndex)] = true
+								}
+							}
+						}
+
+						// Remember what chunk each top-level symbol is declared in. Symbols
+						// with multiple declarations such as repeated "var" statements with
+						// the same name should already be marked as all being in a single
+						// chunk. In that case this will overwrite the same value below which
+						// is fine.
+						for _, declared := range part.DeclaredSymbols {
+							if declared.IsTopLevel {
+								c.graph.Symbols.Get(declared.Ref).ChunkIndex = ast.MakeIndex32(uint32(chunkIndex))
+							}
+						}
+
+						// Record each symbol used in this part. This will later be matched up
+						// with our map of which chunk a given symbol is declared in to
+						// determine if the symbol needs to be imported from another chunk.
+						for ref := range part.SymbolUses {
+							symbol := c.graph.Symbols.Get(ref)
+
+							// Ignore unbound symbols, which don't have declarations
+							if symbol.Kind == ast.SymbolUnbound {
+								continue
+							}
+
+							// Ignore symbols that are going to be replaced by undefined
+							if symbol.ImportItemStatus == ast.ImportItemMissing {
+								continue
+							}
+
+							// If this is imported from another file, follow the import
+							// reference and reference the symbol in that file instead
+							if importData, ok := repr.Meta.ImportsToBind[ref]; ok {
+								ref = importData.Ref
+								symbol = c.graph.Symbols.Get(ref)
+							} else if repr.Meta.Wrap == graph.WrapCJS && ref != repr.AST.WrapperRef {
+								// The only internal symbol that wrapped CommonJS files export
+								// is the wrapper itself.
+								continue
+							}
+
+							// If this is an ES6 import from a CommonJS file, it will become a
+							// property access off the namespace symbol instead of a bare
+							// identifier. In that case we want to pull in the namespace symbol
+							// instead. The namespace symbol stores the result of "require()".
+							if symbol.NamespaceAlias != nil {
+								ref = symbol.NamespaceAlias.NamespaceRef
+							}
+
+							// We must record this relationship even for symbols that are not
+							// imports. Due to code splitting, the definition of a symbol may
+							// be moved to a separate chunk than the use of a symbol even if
+							// the definition and use of that symbol are originally from the
+							// same source file.
+							imports[ref] = true
+						}
+					}
+				}
+			}
+
+			// Include the exports if this is an entry point chunk
+			if chunk.isEntryPoint {
+				if repr, ok := c.graph.Files[chunk.sourceIndex].InputFile.Repr.(*graph.JSRepr); ok {
+					if repr.Meta.Wrap != graph.WrapCJS {
+						for _, alias := range repr.Meta.SortedAndFilteredExportAliases {
+							export := repr.Meta.ResolvedExports[alias]
+							targetRef := export.Ref
+
+							// If this is an import, then target what the import points to
+							if importData, ok := c.graph.Files[export.SourceIndex].InputFile.Repr.(*graph.JSRepr).Meta.ImportsToBind[targetRef]; ok {
+								targetRef = importData.Ref
+							}
+
+							// If this is an ES6 import from a CommonJS file, it will become a
+							// property access off the namespace symbol instead of a bare
+							// identifier. In that case we want to pull in the namespace symbol
+							// instead. The namespace symbol stores the result of "require()".
+							if symbol := c.graph.Symbols.Get(targetRef); symbol.NamespaceAlias != nil {
+								targetRef = symbol.NamespaceAlias.NamespaceRef
+							}
+
+							imports[targetRef] = true
+						}
+					}
+
+					// Ensure "exports" is included if the current output format needs it
+					if repr.Meta.ForceIncludeExportsForEntryPoint {
+						imports[repr.AST.ExportsRef] = true
+					}
+
+					// Include the wrapper if present
+					if repr.Meta.Wrap != graph.WrapNone {
+						imports[repr.AST.WrapperRef] = true
+					}
+				}
+			}
+
+			waitGroup.Done()
+		}(chunkIndex, chunk)
+	}
+	waitGroup.Wait()
+
+	// Mark imported symbols as exported in the chunk from which they are declared
+	for chunkIndex := range c.chunks {
+		chunk := &c.chunks[chunkIndex]
+		chunkRepr, ok := chunk.chunkRepr.(*chunkReprJS)
+		if !ok {
+			continue
+		}
+		chunkMeta := chunkMetas[chunkIndex]
+
+		// Find all uses in this chunk of symbols from other chunks
+		chunkRepr.importsFromOtherChunks = make(map[uint32]crossChunkImportItemArray)
+		for importRef := range chunkMeta.imports {
+			// Ignore uses that aren't top-level symbols
+			if otherChunkIndex := c.graph.Symbols.Get(importRef).ChunkIndex; otherChunkIndex.IsValid() {
+				if otherChunkIndex := otherChunkIndex.GetIndex(); otherChunkIndex != uint32(chunkIndex) {
+					chunkRepr.importsFromOtherChunks[otherChunkIndex] =
+						append(chunkRepr.importsFromOtherChunks[otherChunkIndex], crossChunkImportItem{ref: importRef})
+					chunkMetas[otherChunkIndex].exports[importRef] = true
+				}
+			}
+		}
+
+		// If this is an entry point, make sure we import all chunks belonging to
+		// this entry point, even if there are no imports. We need to make sure
+		// these chunks are evaluated for their side effects too.
+		if chunk.isEntryPoint {
+			for otherChunkIndex, otherChunk := range c.chunks {
+				if _, ok := otherChunk.chunkRepr.(*chunkReprJS); ok && chunkIndex != otherChunkIndex && otherChunk.entryBits.HasBit(chunk.entryPointBit) {
+					imports := chunkRepr.importsFromOtherChunks[uint32(otherChunkIndex)]
+					chunkRepr.importsFromOtherChunks[uint32(otherChunkIndex)] = imports
+				}
+			}
+		}
+
+		// Make sure we also track dynamic cross-chunk imports. These need to be
+		// tracked so we count them as dependencies of this chunk for the purpose
+		// of hash calculation.
+		if chunkMeta.dynamicImports != nil {
+			sortedDynamicImports := make([]int, 0, len(chunkMeta.dynamicImports))
+			for chunkIndex := range chunkMeta.dynamicImports {
+				sortedDynamicImports = append(sortedDynamicImports, chunkIndex)
+			}
+			sort.Ints(sortedDynamicImports)
+			for _, chunkIndex := range sortedDynamicImports {
+				chunk.crossChunkImports = append(chunk.crossChunkImports, chunkImport{
+					importKind: ast.ImportDynamic,
+					chunkIndex: uint32(chunkIndex),
+				})
+			}
+		}
+	}
+
+	// Generate cross-chunk exports. These must be computed before cross-chunk
+	// imports because of export alias renaming, which must consider all export
+	// aliases simultaneously to avoid collisions.
+	for chunkIndex := range c.chunks {
+		chunk := &c.chunks[chunkIndex]
+		chunkRepr, ok := chunk.chunkRepr.(*chunkReprJS)
+		if !ok {
+			continue
+		}
+
+		chunkRepr.exportsToOtherChunks = make(map[ast.Ref]string)
+		switch c.options.OutputFormat {
+		case config.FormatESModule:
+			r := renamer.ExportRenamer{}
+			var items []js_ast.ClauseItem
+			for _, export := range c.sortedCrossChunkExportItems(chunkMetas[chunkIndex].exports) {
+				var alias string
+				if c.options.MinifyIdentifiers {
+					alias = r.NextMinifiedName()
+				} else {
+					alias = r.NextRenamedName(c.graph.Symbols.Get(export.Ref).OriginalName)
+				}
+				items = append(items, js_ast.ClauseItem{Name: ast.LocRef{Ref: export.Ref}, Alias: alias})
+				chunkRepr.exportsToOtherChunks[export.Ref] = alias
+			}
+			if len(items) > 0 {
+				chunkRepr.crossChunkSuffixStmts = []js_ast.Stmt{{Data: &js_ast.SExportClause{
+					Items: items,
+				}}}
+			}
+
+		default:
+			panic("Internal error")
+		}
+	}
+
+	// Generate cross-chunk imports. These must be computed after cross-chunk
+	// exports because the export aliases must already be finalized so they can
+	// be embedded in the generated import statements.
+	for chunkIndex := range c.chunks {
+		chunk := &c.chunks[chunkIndex]
+		chunkRepr, ok := chunk.chunkRepr.(*chunkReprJS)
+		if !ok {
+			continue
+		}
+
+		var crossChunkPrefixStmts []js_ast.Stmt
+
+		for _, crossChunkImport := range c.sortedCrossChunkImports(chunkRepr.importsFromOtherChunks) {
+			switch c.options.OutputFormat {
+			case config.FormatESModule:
+				var items []js_ast.ClauseItem
+				for _, item := range crossChunkImport.sortedImportItems {
+					items = append(items, js_ast.ClauseItem{Name: ast.LocRef{Ref: item.ref}, Alias: item.exportAlias})
+				}
+				importRecordIndex := uint32(len(chunk.crossChunkImports))
+				chunk.crossChunkImports = append(chunk.crossChunkImports, chunkImport{
+					importKind: ast.ImportStmt,
+					chunkIndex: crossChunkImport.chunkIndex,
+				})
+				if len(items) > 0 {
+					// "import {a, b} from './chunk.js'"
+					crossChunkPrefixStmts = append(crossChunkPrefixStmts, js_ast.Stmt{Data: &js_ast.SImport{
+						Items:             &items,
+						ImportRecordIndex: importRecordIndex,
+					}})
+				} else {
+					// "import './chunk.js'"
+					crossChunkPrefixStmts = append(crossChunkPrefixStmts, js_ast.Stmt{Data: &js_ast.SImport{
+						ImportRecordIndex: importRecordIndex,
+					}})
+				}
+
+			default:
+				panic("Internal error")
+			}
+		}
+
+		chunkRepr.crossChunkPrefixStmts = crossChunkPrefixStmts
+	}
+}
+
+type crossChunkImport struct {
+	sortedImportItems crossChunkImportItemArray
+	chunkIndex        uint32
+}
+
+// This type is just so we can use Go's native sort function
+type crossChunkImportArray []crossChunkImport
+
+func (a crossChunkImportArray) Len() int          { return len(a) }
+func (a crossChunkImportArray) Swap(i int, j int) { a[i], a[j] = a[j], a[i] }
+
+func (a crossChunkImportArray) Less(i int, j int) bool {
+	return a[i].chunkIndex < a[j].chunkIndex
+}
+
+// Sort cross-chunk imports by chunk name for determinism
+func (c *linkerContext) sortedCrossChunkImports(importsFromOtherChunks map[uint32]crossChunkImportItemArray) crossChunkImportArray {
+	result := make(crossChunkImportArray, 0, len(importsFromOtherChunks))
+
+	for otherChunkIndex, importItems := range importsFromOtherChunks {
+		// Sort imports from a single chunk by alias for determinism
+		otherChunk := &c.chunks[otherChunkIndex]
+		exportsToOtherChunks := otherChunk.chunkRepr.(*chunkReprJS).exportsToOtherChunks
+		for i, item := range importItems {
+			importItems[i].exportAlias = exportsToOtherChunks[item.ref]
+		}
+		sort.Sort(importItems)
+		result = append(result, crossChunkImport{
+			chunkIndex:        otherChunkIndex,
+			sortedImportItems: importItems,
+		})
+	}
+
+	sort.Sort(result)
+	return result
+}
+
+type crossChunkImportItem struct {
+	exportAlias string
+	ref         ast.Ref
+}
+
+// This type is just so we can use Go's native sort function
+type crossChunkImportItemArray []crossChunkImportItem
+
+func (a crossChunkImportItemArray) Len() int          { return len(a) }
+func (a crossChunkImportItemArray) Swap(i int, j int) { a[i], a[j] = a[j], a[i] }
+
+func (a crossChunkImportItemArray) Less(i int, j int) bool {
+	return a[i].exportAlias < a[j].exportAlias
+}
+
+// The sort order here is arbitrary but needs to be consistent between builds.
+// The InnerIndex should be stable because the parser for a single file is
+// single-threaded and deterministically assigns out InnerIndex values
+// sequentially. But the SourceIndex should be unstable because the main thread
+// assigns out source index values sequentially to newly-discovered dependencies
+// in a multi-threaded producer/consumer relationship. So instead we use the
+// index of the source in the DFS order over all entry points for stability.
+type stableRef struct {
+	StableSourceIndex uint32
+	Ref               ast.Ref
+}
+
+// This type is just so we can use Go's native sort function
+type stableRefArray []stableRef
+
+func (a stableRefArray) Len() int          { return len(a) }
+func (a stableRefArray) Swap(i int, j int) { a[i], a[j] = a[j], a[i] }
+func (a stableRefArray) Less(i int, j int) bool {
+	ai, aj := a[i], a[j]
+	return ai.StableSourceIndex < aj.StableSourceIndex ||
+		(ai.StableSourceIndex == aj.StableSourceIndex && ai.Ref.InnerIndex < aj.Ref.InnerIndex)
+}
+
+// Sort cross-chunk exports by chunk name for determinism
+func (c *linkerContext) sortedCrossChunkExportItems(exportRefs map[ast.Ref]bool) stableRefArray {
+	result := make(stableRefArray, 0, len(exportRefs))
+	for ref := range exportRefs {
+		result = append(result, stableRef{
+			StableSourceIndex: c.graph.StableSourceIndices[ref.SourceIndex],
+			Ref:               ref,
+		})
+	}
+	sort.Sort(result)
+	return result
+}
+
+func (c *linkerContext) scanImportsAndExports() {
+	c.timer.Begin("Scan imports and exports")
+	defer c.timer.End("Scan imports and exports")
+
+	// Step 1: Figure out what modules must be CommonJS
+	c.timer.Begin("Step 1")
+	for _, sourceIndex := range c.graph.ReachableFiles {
+		file := &c.graph.Files[sourceIndex]
+		additionalFiles := file.InputFile.AdditionalFiles
+
+		switch repr := file.InputFile.Repr.(type) {
+		case *graph.CSSRepr:
+			// Inline URLs for non-CSS files into the CSS file
+			for importRecordIndex := range repr.AST.ImportRecords {
+				if record := &repr.AST.ImportRecords[importRecordIndex]; record.SourceIndex.IsValid() {
+					otherFile := &c.graph.Files[record.SourceIndex.GetIndex()]
+					if otherRepr, ok := otherFile.InputFile.Repr.(*graph.JSRepr); ok {
+						record.Path.Text = otherRepr.AST.URLForCSS
+						record.Path.Namespace = ""
+						record.SourceIndex = ast.Index32{}
+						if otherFile.InputFile.Loader == config.LoaderEmpty {
+							record.Flags |= ast.WasLoadedWithEmptyLoader
+						} else {
+							record.Flags |= ast.ShouldNotBeExternalInMetafile
+						}
+						if strings.Contains(otherRepr.AST.URLForCSS, c.uniqueKeyPrefix) {
+							record.Flags |= ast.ContainsUniqueKey
+						}
+
+						// Copy the additional files to the output directory
+						additionalFiles = append(additionalFiles, otherFile.InputFile.AdditionalFiles...)
+					}
+				} else if record.CopySourceIndex.IsValid() {
+					otherFile := &c.graph.Files[record.CopySourceIndex.GetIndex()]
+					if otherRepr, ok := otherFile.InputFile.Repr.(*graph.CopyRepr); ok {
+						record.Path.Text = otherRepr.URLForCode
+						record.Path.Namespace = ""
+						record.CopySourceIndex = ast.Index32{}
+						record.Flags |= ast.ShouldNotBeExternalInMetafile | ast.ContainsUniqueKey
+
+						// Copy the additional files to the output directory
+						additionalFiles = append(additionalFiles, otherFile.InputFile.AdditionalFiles...)
+					}
+				}
+			}
+
+			// Validate cross-file "composes: ... from" named imports
+			for _, composes := range repr.AST.Composes {
+				for _, name := range composes.ImportedNames {
+					if record := repr.AST.ImportRecords[name.ImportRecordIndex]; record.SourceIndex.IsValid() {
+						otherFile := &c.graph.Files[record.SourceIndex.GetIndex()]
+						if otherRepr, ok := otherFile.InputFile.Repr.(*graph.CSSRepr); ok {
+							if _, ok := otherRepr.AST.LocalScope[name.Alias]; !ok {
+								if global, ok := otherRepr.AST.GlobalScope[name.Alias]; ok {
+									var hint string
+									if otherFile.InputFile.Loader == config.LoaderCSS {
+										hint = fmt.Sprintf("Use the \"local-css\" loader for %q to enable local names.", otherFile.InputFile.Source.PrettyPath)
+									} else {
+										hint = fmt.Sprintf("Use the \":local\" selector to change %q into a local name.", name.Alias)
+									}
+									c.log.AddErrorWithNotes(file.LineColumnTracker(),
+										css_lexer.RangeOfIdentifier(file.InputFile.Source, name.AliasLoc),
+										fmt.Sprintf("Cannot use global name %q with \"composes\"", name.Alias),
+										[]logger.MsgData{
+											otherFile.LineColumnTracker().MsgData(
+												css_lexer.RangeOfIdentifier(otherFile.InputFile.Source, global.Loc),
+												fmt.Sprintf("The global name %q is defined here:", name.Alias),
+											),
+											{Text: hint},
+										})
+								} else {
+									c.log.AddError(file.LineColumnTracker(),
+										css_lexer.RangeOfIdentifier(file.InputFile.Source, name.AliasLoc),
+										fmt.Sprintf("The name %q never appears in %q",
+											name.Alias, otherFile.InputFile.Source.PrettyPath))
+								}
+							}
+						}
+					}
+				}
+			}
+
+			c.validateComposesFromProperties(file, repr)
+
+		case *graph.JSRepr:
+			for importRecordIndex := range repr.AST.ImportRecords {
+				record := &repr.AST.ImportRecords[importRecordIndex]
+				if !record.SourceIndex.IsValid() {
+					if record.CopySourceIndex.IsValid() {
+						otherFile := &c.graph.Files[record.CopySourceIndex.GetIndex()]
+						if otherRepr, ok := otherFile.InputFile.Repr.(*graph.CopyRepr); ok {
+							record.Path.Text = otherRepr.URLForCode
+							record.Path.Namespace = ""
+							record.CopySourceIndex = ast.Index32{}
+							record.Flags |= ast.ShouldNotBeExternalInMetafile | ast.ContainsUniqueKey
+
+							// Copy the additional files to the output directory
+							additionalFiles = append(additionalFiles, otherFile.InputFile.AdditionalFiles...)
+						}
+					}
+					continue
+				}
+
+				otherFile := &c.graph.Files[record.SourceIndex.GetIndex()]
+				otherRepr := otherFile.InputFile.Repr.(*graph.JSRepr)
+
+				switch record.Kind {
+				case ast.ImportStmt:
+					// Importing using ES6 syntax from a file without any ES6 syntax
+					// causes that module to be considered CommonJS-style, even if it
+					// doesn't have any CommonJS exports.
+					//
+					// That means the ES6 imports will become undefined instead of
+					// causing errors. This is for compatibility with older CommonJS-
+					// style bundlers.
+					//
+					// We emit a warning in this case but try to avoid turning the module
+					// into a CommonJS module if possible. This is possible with named
+					// imports (the module stays an ECMAScript module but the imports are
+					// rewritten with undefined) but is not possible with star or default
+					// imports:
+					//
+					//   import * as ns from './empty-file'
+					//   import defVal from './empty-file'
+					//   console.log(ns, defVal)
+					//
+					// In that case the module *is* considered a CommonJS module because
+					// the namespace object must be created.
+					if (record.Flags.Has(ast.ContainsImportStar) || record.Flags.Has(ast.ContainsDefaultAlias)) &&
+						otherRepr.AST.ExportsKind == js_ast.ExportsNone && !otherRepr.AST.HasLazyExport {
+						otherRepr.Meta.Wrap = graph.WrapCJS
+						otherRepr.AST.ExportsKind = js_ast.ExportsCommonJS
+					}
+
+				case ast.ImportRequire:
+					// Files that are imported with require() must be wrapped so that
+					// they can be lazily-evaluated
+					if otherRepr.AST.ExportsKind == js_ast.ExportsESM {
+						otherRepr.Meta.Wrap = graph.WrapESM
+					} else {
+						otherRepr.Meta.Wrap = graph.WrapCJS
+						otherRepr.AST.ExportsKind = js_ast.ExportsCommonJS
+					}
+
+				case ast.ImportDynamic:
+					if !c.options.CodeSplitting {
+						// If we're not splitting, then import() is just a require() that
+						// returns a promise, so the imported file must also be wrapped
+						if otherRepr.AST.ExportsKind == js_ast.ExportsESM {
+							otherRepr.Meta.Wrap = graph.WrapESM
+						} else {
+							otherRepr.Meta.Wrap = graph.WrapCJS
+							otherRepr.AST.ExportsKind = js_ast.ExportsCommonJS
+						}
+					}
+				}
+			}
+
+			// If the output format doesn't have an implicit CommonJS wrapper, any file
+			// that uses CommonJS features will need to be wrapped, even though the
+			// resulting wrapper won't be invoked by other files. An exception is made
+			// for entry point files in CommonJS format (or when in pass-through mode).
+			if repr.AST.ExportsKind == js_ast.ExportsCommonJS && (!file.IsEntryPoint() ||
+				c.options.OutputFormat == config.FormatIIFE || c.options.OutputFormat == config.FormatESModule) {
+				repr.Meta.Wrap = graph.WrapCJS
+			}
+		}
+
+		file.InputFile.AdditionalFiles = additionalFiles
+	}
+	c.timer.End("Step 1")
+
+	// Step 2: Propagate dynamic export status for export star statements that
+	// are re-exports from a module whose exports are not statically analyzable.
+	// In this case the export star must be evaluated at run time instead of at
+	// bundle time.
+	c.timer.Begin("Step 2")
+	for _, sourceIndex := range c.graph.ReachableFiles {
+		repr, ok := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr)
+		if !ok {
+			continue
+		}
+
+		if repr.Meta.Wrap != graph.WrapNone {
+			c.recursivelyWrapDependencies(sourceIndex)
+		}
+
+		if len(repr.AST.ExportStarImportRecords) > 0 {
+			visited := make(map[uint32]bool)
+			c.hasDynamicExportsDueToExportStar(sourceIndex, visited)
+		}
+
+		// Even if the output file is CommonJS-like, we may still need to wrap
+		// CommonJS-style files. Any file that imports a CommonJS-style file will
+		// cause that file to need to be wrapped. This is because the import
+		// method, whatever it is, will need to invoke the wrapper. Note that
+		// this can include entry points (e.g. an entry point that imports a file
+		// that imports that entry point).
+		for _, record := range repr.AST.ImportRecords {
+			if record.SourceIndex.IsValid() {
+				otherRepr := c.graph.Files[record.SourceIndex.GetIndex()].InputFile.Repr.(*graph.JSRepr)
+				if otherRepr.AST.ExportsKind == js_ast.ExportsCommonJS {
+					c.recursivelyWrapDependencies(record.SourceIndex.GetIndex())
+				}
+			}
+		}
+	}
+	c.timer.End("Step 2")
+
+	// Step 3: Resolve "export * from" statements. This must be done after we
+	// discover all modules that can have dynamic exports because export stars
+	// are ignored for those modules.
+	c.timer.Begin("Step 3")
+	exportStarStack := make([]uint32, 0, 32)
+	for _, sourceIndex := range c.graph.ReachableFiles {
+		repr, ok := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr)
+		if !ok {
+			continue
+		}
+
+		// Expression-style loaders defer code generation until linking. Code
+		// generation is done here because at this point we know that the
+		// "ExportsKind" field has its final value and will not be changed.
+		if repr.AST.HasLazyExport {
+			c.generateCodeForLazyExport(sourceIndex)
+		}
+
+		// Propagate exports for export star statements
+		if len(repr.AST.ExportStarImportRecords) > 0 {
+			c.addExportsForExportStar(repr.Meta.ResolvedExports, sourceIndex, exportStarStack)
+		}
+
+		// Also add a special export so import stars can bind to it. This must be
+		// done in this step because it must come after CommonJS module discovery
+		// but before matching imports with exports.
+		repr.Meta.ResolvedExportStar = &graph.ExportData{
+			Ref:         repr.AST.ExportsRef,
+			SourceIndex: sourceIndex,
+		}
+	}
+	c.timer.End("Step 3")
+
+	// Step 4: Match imports with exports. This must be done after we process all
+	// export stars because imports can bind to export star re-exports.
+	c.timer.Begin("Step 4")
+	for _, sourceIndex := range c.graph.ReachableFiles {
+		file := &c.graph.Files[sourceIndex]
+		repr, ok := file.InputFile.Repr.(*graph.JSRepr)
+		if !ok {
+			continue
+		}
+
+		if len(repr.AST.NamedImports) > 0 {
+			c.matchImportsWithExportsForFile(uint32(sourceIndex))
+		}
+
+		// If we're exporting as CommonJS and this file was originally CommonJS,
+		// then we'll be using the actual CommonJS "exports" and/or "module"
+		// symbols. In that case make sure to mark them as such so they don't
+		// get minified.
+		if file.IsEntryPoint() && repr.AST.ExportsKind == js_ast.ExportsCommonJS && repr.Meta.Wrap == graph.WrapNone &&
+			(c.options.OutputFormat == config.FormatPreserve || c.options.OutputFormat == config.FormatCommonJS) {
+			exportsRef := ast.FollowSymbols(c.graph.Symbols, repr.AST.ExportsRef)
+			moduleRef := ast.FollowSymbols(c.graph.Symbols, repr.AST.ModuleRef)
+			c.graph.Symbols.Get(exportsRef).Kind = ast.SymbolUnbound
+			c.graph.Symbols.Get(moduleRef).Kind = ast.SymbolUnbound
+		} else if repr.Meta.ForceIncludeExportsForEntryPoint || repr.AST.ExportsKind != js_ast.ExportsCommonJS {
+			repr.Meta.NeedsExportsVariable = true
+		}
+
+		// Create the wrapper part for wrapped files. This is needed by a later step.
+		c.createWrapperForFile(uint32(sourceIndex))
+	}
+	c.timer.End("Step 4")
+
+	// Step 5: Create namespace exports for every file. This is always necessary
+	// for CommonJS files, and is also necessary for other files if they are
+	// imported using an import star statement.
+	c.timer.Begin("Step 5")
+	waitGroup := sync.WaitGroup{}
+	for _, sourceIndex := range c.graph.ReachableFiles {
+		repr, ok := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr)
+		if !ok {
+			continue
+		}
+
+		// This is the slowest step and is also parallelizable, so do this in parallel.
+		waitGroup.Add(1)
+		go func(sourceIndex uint32, repr *graph.JSRepr) {
+			// Now that all exports have been resolved, sort and filter them to create
+			// something we can iterate over later.
+			aliases := make([]string, 0, len(repr.Meta.ResolvedExports))
+		nextAlias:
+			for alias, export := range repr.Meta.ResolvedExports {
+				otherFile := &c.graph.Files[export.SourceIndex].InputFile
+				otherRepr := otherFile.Repr.(*graph.JSRepr)
+
+				// Re-exporting multiple symbols with the same name causes an ambiguous
+				// export. These names cannot be used and should not end up in generated code.
+				if len(export.PotentiallyAmbiguousExportStarRefs) > 0 {
+					mainRef := export.Ref
+					mainLoc := export.NameLoc
+					if imported, ok := otherRepr.Meta.ImportsToBind[export.Ref]; ok {
+						mainRef = imported.Ref
+						mainLoc = imported.NameLoc
+					}
+
+					for _, ambiguousExport := range export.PotentiallyAmbiguousExportStarRefs {
+						ambiguousFile := &c.graph.Files[ambiguousExport.SourceIndex].InputFile
+						ambiguousRepr := ambiguousFile.Repr.(*graph.JSRepr)
+						ambiguousRef := ambiguousExport.Ref
+						ambiguousLoc := ambiguousExport.NameLoc
+						if imported, ok := ambiguousRepr.Meta.ImportsToBind[ambiguousExport.Ref]; ok {
+							ambiguousRef = imported.Ref
+							ambiguousLoc = imported.NameLoc
+						}
+
+						if mainRef != ambiguousRef {
+							file := &c.graph.Files[sourceIndex].InputFile
+							otherTracker := logger.MakeLineColumnTracker(&otherFile.Source)
+							ambiguousTracker := logger.MakeLineColumnTracker(&ambiguousFile.Source)
+							c.log.AddIDWithNotes(logger.MsgID_Bundler_AmbiguousReexport, logger.Debug, nil, logger.Range{},
+								fmt.Sprintf("Re-export of %q in %q is ambiguous and has been removed", alias, file.Source.PrettyPath),
+								[]logger.MsgData{
+									otherTracker.MsgData(js_lexer.RangeOfIdentifier(otherFile.Source, mainLoc),
+										fmt.Sprintf("One definition of %q comes from %q here:", alias, otherFile.Source.PrettyPath)),
+									ambiguousTracker.MsgData(js_lexer.RangeOfIdentifier(ambiguousFile.Source, ambiguousLoc),
+										fmt.Sprintf("Another definition of %q comes from %q here:", alias, ambiguousFile.Source.PrettyPath)),
+								},
+							)
+							continue nextAlias
+						}
+					}
+				}
+
+				// Ignore re-exported imports in TypeScript files that failed to be
+				// resolved. These are probably just type-only imports so the best thing to
+				// do is to silently omit them from the export list.
+				if otherRepr.Meta.IsProbablyTypeScriptType[export.Ref] {
+					continue
+				}
+
+				if c.options.OutputFormat == config.FormatESModule && c.options.UnsupportedJSFeatures.Has(compat.ArbitraryModuleNamespaceNames) && c.graph.Files[sourceIndex].IsEntryPoint() {
+					c.maybeForbidArbitraryModuleNamespaceIdentifier("export", export.SourceIndex, export.NameLoc, alias)
+				}
+
+				aliases = append(aliases, alias)
+			}
+			sort.Strings(aliases)
+			repr.Meta.SortedAndFilteredExportAliases = aliases
+
+			// Export creation uses "sortedAndFilteredExportAliases" so this must
+			// come second after we fill in that array
+			c.createExportsForFile(uint32(sourceIndex))
+
+			// Each part tracks the other parts it depends on within this file
+			localDependencies := make(map[uint32]uint32)
+			parts := repr.AST.Parts
+			namedImports := repr.AST.NamedImports
+			graph := c.graph
+			for partIndex := range parts {
+				part := &parts[partIndex]
+
+				// Now that all files have been parsed, determine which property
+				// accesses off of imported symbols are inlined enum values and
+				// which ones aren't
+				for ref, properties := range part.ImportSymbolPropertyUses {
+					use := part.SymbolUses[ref]
+
+					// Rare path: this import is a TypeScript enum
+					if importData, ok := repr.Meta.ImportsToBind[ref]; ok {
+						if symbol := graph.Symbols.Get(importData.Ref); symbol.Kind == ast.SymbolTSEnum {
+							if enum, ok := graph.TSEnums[importData.Ref]; ok {
+								foundNonInlinedEnum := false
+								for name, propertyUse := range properties {
+									if _, ok := enum[name]; !ok {
+										foundNonInlinedEnum = true
+										use.CountEstimate += propertyUse.CountEstimate
+									}
+								}
+								if foundNonInlinedEnum {
+									part.SymbolUses[ref] = use
+								}
+							}
+							continue
+						}
+					}
+
+					// Common path: this import isn't a TypeScript enum
+					for _, propertyUse := range properties {
+						use.CountEstimate += propertyUse.CountEstimate
+					}
+					part.SymbolUses[ref] = use
+				}
+
+				// Also determine which function calls will be inlined (and so should
+				// not count as uses), and which ones will not be (and so should count
+				// as uses)
+				for ref, callUse := range part.SymbolCallUses {
+					use := part.SymbolUses[ref]
+
+					// Find the symbol that was called
+					symbol := graph.Symbols.Get(ref)
+					if symbol.Kind == ast.SymbolImport {
+						if importData, ok := repr.Meta.ImportsToBind[ref]; ok {
+							symbol = graph.Symbols.Get(importData.Ref)
+						}
+					}
+					flags := symbol.Flags
+
+					// Rare path: this is a function that will be inlined
+					if (flags & (ast.IsEmptyFunction | ast.CouldPotentiallyBeMutated)) == ast.IsEmptyFunction {
+						// Every call will be inlined
+						continue
+					} else if (flags & (ast.IsIdentityFunction | ast.CouldPotentiallyBeMutated)) == ast.IsIdentityFunction {
+						// Every single-argument call will be inlined as long as it's not a spread
+						callUse.CallCountEstimate -= callUse.SingleArgNonSpreadCallCountEstimate
+						if callUse.CallCountEstimate == 0 {
+							continue
+						}
+					}
+
+					// Common path: this isn't a function that will be inlined
+					use.CountEstimate += callUse.CallCountEstimate
+					part.SymbolUses[ref] = use
+				}
+
+				// Now that we know this, we can determine cross-part dependencies
+				for ref := range part.SymbolUses {
+
+					// Rare path: this import is an inlined const value
+					if graph.ConstValues != nil {
+						if importData, ok := repr.Meta.ImportsToBind[ref]; ok {
+							if _, isConstValue := graph.ConstValues[importData.Ref]; isConstValue {
+								delete(part.SymbolUses, importData.Ref)
+								continue
+							}
+						}
+					}
+
+					for _, otherPartIndex := range repr.TopLevelSymbolToParts(ref) {
+						if oldPartIndex, ok := localDependencies[otherPartIndex]; !ok || oldPartIndex != uint32(partIndex) {
+							localDependencies[otherPartIndex] = uint32(partIndex)
+							part.Dependencies = append(part.Dependencies, js_ast.Dependency{
+								SourceIndex: sourceIndex,
+								PartIndex:   otherPartIndex,
+							})
+						}
+					}
+
+					// Also map from imports to parts that use them
+					if namedImport, ok := namedImports[ref]; ok {
+						namedImport.LocalPartsWithUses = append(namedImport.LocalPartsWithUses, uint32(partIndex))
+						namedImports[ref] = namedImport
+					}
+				}
+			}
+
+			waitGroup.Done()
+		}(sourceIndex, repr)
+	}
+	waitGroup.Wait()
+	c.timer.End("Step 5")
+
+	// Step 6: Bind imports to exports. This adds non-local dependencies on the
+	// parts that declare the export to all parts that use the import. Also
+	// generate wrapper parts for wrapped files.
+	c.timer.Begin("Step 6")
+	for _, sourceIndex := range c.graph.ReachableFiles {
+		file := &c.graph.Files[sourceIndex]
+		repr, ok := file.InputFile.Repr.(*graph.JSRepr)
+		if !ok {
+			continue
+		}
+
+		// Pre-generate symbols for re-exports CommonJS symbols in case they
+		// are necessary later. This is done now because the symbols map cannot be
+		// mutated later due to parallelism.
+		if file.IsEntryPoint() && c.options.OutputFormat == config.FormatESModule {
+			copies := make([]ast.Ref, len(repr.Meta.SortedAndFilteredExportAliases))
+			for i, alias := range repr.Meta.SortedAndFilteredExportAliases {
+				copies[i] = c.graph.GenerateNewSymbol(sourceIndex, ast.SymbolOther, "export_"+alias)
+			}
+			repr.Meta.CJSExportCopies = copies
+		}
+
+		// Use "init_*" for ESM wrappers instead of "require_*"
+		if repr.Meta.Wrap == graph.WrapESM {
+			c.graph.Symbols.Get(repr.AST.WrapperRef).OriginalName = "init_" + file.InputFile.Source.IdentifierName
+		}
+
+		// If this isn't CommonJS, then rename the unused "exports" and "module"
+		// variables to avoid them causing the identically-named variables in
+		// actual CommonJS files from being renamed. This is purely about
+		// aesthetics and is not about correctness. This is done here because by
+		// this point, we know the CommonJS status will not change further.
+		if repr.Meta.Wrap != graph.WrapCJS && repr.AST.ExportsKind != js_ast.ExportsCommonJS {
+			name := file.InputFile.Source.IdentifierName
+			c.graph.Symbols.Get(repr.AST.ExportsRef).OriginalName = name + "_exports"
+			c.graph.Symbols.Get(repr.AST.ModuleRef).OriginalName = name + "_module"
+		}
+
+		// Include the "__export" symbol from the runtime if it was used in the
+		// previous step. The previous step can't do this because it's running in
+		// parallel and can't safely mutate the "importsToBind" map of another file.
+		if repr.Meta.NeedsExportSymbolFromRuntime {
+			runtimeRepr := c.graph.Files[runtime.SourceIndex].InputFile.Repr.(*graph.JSRepr)
+			exportRef := runtimeRepr.AST.ModuleScope.Members["__export"].Ref
+			c.graph.GenerateSymbolImportAndUse(sourceIndex, js_ast.NSExportPartIndex, exportRef, 1, runtime.SourceIndex)
+		}
+
+		for importRef, importData := range repr.Meta.ImportsToBind {
+			resolvedRepr := c.graph.Files[importData.SourceIndex].InputFile.Repr.(*graph.JSRepr)
+			partsDeclaringSymbol := resolvedRepr.TopLevelSymbolToParts(importData.Ref)
+
+			for _, partIndex := range repr.AST.NamedImports[importRef].LocalPartsWithUses {
+				part := &repr.AST.Parts[partIndex]
+
+				// Depend on the file containing the imported symbol
+				for _, resolvedPartIndex := range partsDeclaringSymbol {
+					part.Dependencies = append(part.Dependencies, js_ast.Dependency{
+						SourceIndex: importData.SourceIndex,
+						PartIndex:   resolvedPartIndex,
+					})
+				}
+
+				// Also depend on any files that re-exported this symbol in between the
+				// file containing the import and the file containing the imported symbol
+				part.Dependencies = append(part.Dependencies, importData.ReExports...)
+			}
+
+			// Merge these symbols so they will share the same name
+			ast.MergeSymbols(c.graph.Symbols, importRef, importData.Ref)
+		}
+
+		// If this is an entry point, depend on all exports so they are included
+		if file.IsEntryPoint() {
+			var dependencies []js_ast.Dependency
+
+			for _, alias := range repr.Meta.SortedAndFilteredExportAliases {
+				export := repr.Meta.ResolvedExports[alias]
+				targetSourceIndex := export.SourceIndex
+				targetRef := export.Ref
+
+				// If this is an import, then target what the import points to
+				targetRepr := c.graph.Files[targetSourceIndex].InputFile.Repr.(*graph.JSRepr)
+				if importData, ok := targetRepr.Meta.ImportsToBind[targetRef]; ok {
+					targetSourceIndex = importData.SourceIndex
+					targetRef = importData.Ref
+					targetRepr = c.graph.Files[targetSourceIndex].InputFile.Repr.(*graph.JSRepr)
+					dependencies = append(dependencies, importData.ReExports...)
+				}
+
+				// Pull in all declarations of this symbol
+				for _, partIndex := range targetRepr.TopLevelSymbolToParts(targetRef) {
+					dependencies = append(dependencies, js_ast.Dependency{
+						SourceIndex: targetSourceIndex,
+						PartIndex:   partIndex,
+					})
+				}
+			}
+
+			// Ensure "exports" is included if the current output format needs it
+			if repr.Meta.ForceIncludeExportsForEntryPoint {
+				dependencies = append(dependencies, js_ast.Dependency{
+					SourceIndex: sourceIndex,
+					PartIndex:   js_ast.NSExportPartIndex,
+				})
+			}
+
+			// Include the wrapper if present
+			if repr.Meta.Wrap != graph.WrapNone {
+				dependencies = append(dependencies, js_ast.Dependency{
+					SourceIndex: sourceIndex,
+					PartIndex:   repr.Meta.WrapperPartIndex.GetIndex(),
+				})
+			}
+
+			// Represent these constraints with a dummy part
+			entryPointPartIndex := c.graph.AddPartToFile(sourceIndex, js_ast.Part{
+				Dependencies:         dependencies,
+				CanBeRemovedIfUnused: false,
+			})
+			repr.Meta.EntryPointPartIndex = ast.MakeIndex32(entryPointPartIndex)
+
+			// Pull in the "__toCommonJS" symbol if we need it due to being an entry point
+			if repr.Meta.ForceIncludeExportsForEntryPoint {
+				c.graph.GenerateRuntimeSymbolImportAndUse(sourceIndex, entryPointPartIndex, "__toCommonJS", 1)
+			}
+		}
+
+		// Encode import-specific constraints in the dependency graph
+		for partIndex, part := range repr.AST.Parts {
+			toESMUses := uint32(0)
+			toCommonJSUses := uint32(0)
+			runtimeRequireUses := uint32(0)
+
+			// Imports of wrapped files must depend on the wrapper
+			for _, importRecordIndex := range part.ImportRecordIndices {
+				record := &repr.AST.ImportRecords[importRecordIndex]
+
+				// Don't follow external imports (this includes import() expressions)
+				if !record.SourceIndex.IsValid() || c.isExternalDynamicImport(record, sourceIndex) {
+					// This is an external import. Check if it will be a "require()" call.
+					if record.Kind == ast.ImportRequire || !c.options.OutputFormat.KeepESMImportExportSyntax() ||
+						(record.Kind == ast.ImportDynamic && c.options.UnsupportedJSFeatures.Has(compat.DynamicImport)) {
+						// We should use "__require" instead of "require" if we're not
+						// generating a CommonJS output file, since it won't exist otherwise
+						if config.ShouldCallRuntimeRequire(c.options.Mode, c.options.OutputFormat) {
+							record.Flags |= ast.CallRuntimeRequire
+							runtimeRequireUses++
+						}
+
+						// If this wasn't originally a "require()" call, then we may need
+						// to wrap this in a call to the "__toESM" wrapper to convert from
+						// CommonJS semantics to ESM semantics.
+						//
+						// Unfortunately this adds some additional code since the conversion
+						// is somewhat complex. As an optimization, we can avoid this if the
+						// following things are true:
+						//
+						// - The import is an ES module statement (e.g. not an "import()" expression)
+						// - The ES module namespace object must not be captured
+						// - The "default" and "__esModule" exports must not be accessed
+						//
+						if record.Kind != ast.ImportRequire &&
+							(record.Kind != ast.ImportStmt ||
+								record.Flags.Has(ast.ContainsImportStar) ||
+								record.Flags.Has(ast.ContainsDefaultAlias) ||
+								record.Flags.Has(ast.ContainsESModuleAlias)) {
+							record.Flags |= ast.WrapWithToESM
+							toESMUses++
+						}
+					}
+					continue
+				}
+
+				otherSourceIndex := record.SourceIndex.GetIndex()
+				otherRepr := c.graph.Files[otherSourceIndex].InputFile.Repr.(*graph.JSRepr)
+
+				if otherRepr.Meta.Wrap != graph.WrapNone {
+					// Depend on the automatically-generated require wrapper symbol
+					wrapperRef := otherRepr.AST.WrapperRef
+					c.graph.GenerateSymbolImportAndUse(sourceIndex, uint32(partIndex), wrapperRef, 1, otherSourceIndex)
+
+					// This is an ES6 import of a CommonJS module, so it needs the
+					// "__toESM" wrapper as long as it's not a bare "require()"
+					if record.Kind != ast.ImportRequire && otherRepr.AST.ExportsKind == js_ast.ExportsCommonJS {
+						record.Flags |= ast.WrapWithToESM
+						toESMUses++
+					}
+
+					// If this is an ESM wrapper, also depend on the exports object
+					// since the final code will contain an inline reference to it.
+					// This must be done for "require()" and "import()" expressions
+					// but does not need to be done for "import" statements since
+					// those just cause us to reference the exports directly.
+					if otherRepr.Meta.Wrap == graph.WrapESM && record.Kind != ast.ImportStmt {
+						c.graph.GenerateSymbolImportAndUse(sourceIndex, uint32(partIndex), otherRepr.AST.ExportsRef, 1, otherSourceIndex)
+
+						// If this is a "require()" call, then we should add the
+						// "__esModule" marker to behave as if the module was converted
+						// from ESM to CommonJS. This is done via a wrapper instead of
+						// by modifying the exports object itself because the same ES
+						// module may be simultaneously imported and required, and the
+						// importing code should not see "__esModule" while the requiring
+						// code should see "__esModule". This is an extremely complex
+						// and subtle set of bundler interop issues. See for example
+						// https://github.com/evanw/esbuild/issues/1591.
+						if record.Kind == ast.ImportRequire {
+							record.Flags |= ast.WrapWithToCJS
+							toCommonJSUses++
+						}
+					}
+				} else if record.Kind == ast.ImportStmt && otherRepr.AST.ExportsKind == js_ast.ExportsESMWithDynamicFallback {
+					// This is an import of a module that has a dynamic export fallback
+					// object. In that case we need to depend on that object in case
+					// something ends up needing to use it later. This could potentially
+					// be omitted in some cases with more advanced analysis if this
+					// dynamic export fallback object doesn't end up being needed.
+					c.graph.GenerateSymbolImportAndUse(sourceIndex, uint32(partIndex), otherRepr.AST.ExportsRef, 1, otherSourceIndex)
+				}
+			}
+
+			// If there's an ES6 import of a non-ES6 module, then we're going to need the
+			// "__toESM" symbol from the runtime to wrap the result of "require()"
+			c.graph.GenerateRuntimeSymbolImportAndUse(sourceIndex, uint32(partIndex), "__toESM", toESMUses)
+
+			// If there's a CommonJS require of an ES6 module, then we're going to need the
+			// "__toCommonJS" symbol from the runtime to wrap the exports object
+			c.graph.GenerateRuntimeSymbolImportAndUse(sourceIndex, uint32(partIndex), "__toCommonJS", toCommonJSUses)
+
+			// If there are unbundled calls to "require()" and we're not generating
+			// code for node, then substitute a "__require" wrapper for "require".
+			c.graph.GenerateRuntimeSymbolImportAndUse(sourceIndex, uint32(partIndex), "__require", runtimeRequireUses)
+
+			// If there's an ES6 export star statement of a non-ES6 module, then we're
+			// going to need the "__reExport" symbol from the runtime
+			reExportUses := uint32(0)
+			for _, importRecordIndex := range repr.AST.ExportStarImportRecords {
+				record := &repr.AST.ImportRecords[importRecordIndex]
+
+				// Is this export star evaluated at run time?
+				happensAtRunTime := !record.SourceIndex.IsValid() && (!file.IsEntryPoint() || !c.options.OutputFormat.KeepESMImportExportSyntax())
+				if record.SourceIndex.IsValid() {
+					otherSourceIndex := record.SourceIndex.GetIndex()
+					otherRepr := c.graph.Files[otherSourceIndex].InputFile.Repr.(*graph.JSRepr)
+					if otherSourceIndex != sourceIndex && otherRepr.AST.ExportsKind.IsDynamic() {
+						happensAtRunTime = true
+					}
+					if otherRepr.AST.ExportsKind == js_ast.ExportsESMWithDynamicFallback {
+						// This looks like "__reExport(exports_a, exports_b)". Make sure to
+						// pull in the "exports_b" symbol into this export star. This matters
+						// in code splitting situations where the "export_b" symbol might live
+						// in a different chunk than this export star.
+						c.graph.GenerateSymbolImportAndUse(sourceIndex, uint32(partIndex), otherRepr.AST.ExportsRef, 1, otherSourceIndex)
+					}
+				}
+				if happensAtRunTime {
+					// Depend on this file's "exports" object for the first argument to "__reExport"
+					c.graph.GenerateSymbolImportAndUse(sourceIndex, uint32(partIndex), repr.AST.ExportsRef, 1, sourceIndex)
+					record.Flags |= ast.CallsRunTimeReExportFn
+					repr.AST.UsesExportsRef = true
+					reExportUses++
+				}
+			}
+			c.graph.GenerateRuntimeSymbolImportAndUse(sourceIndex, uint32(partIndex), "__reExport", reExportUses)
+		}
+	}
+	c.timer.End("Step 6")
+}
+
+func (c *linkerContext) validateComposesFromProperties(rootFile *graph.LinkerFile, rootRepr *graph.CSSRepr) {
+	for _, local := range rootRepr.AST.LocalSymbols {
+		type propertyInFile struct {
+			file *graph.LinkerFile
+			loc  logger.Loc
+		}
+
+		visited := make(map[ast.Ref]bool)
+		properties := make(map[string]propertyInFile)
+		var visit func(*graph.LinkerFile, *graph.CSSRepr, ast.Ref)
+
+		visit = func(file *graph.LinkerFile, repr *graph.CSSRepr, ref ast.Ref) {
+			if visited[ref] {
+				return
+			}
+			visited[ref] = true
+
+			composes, ok := repr.AST.Composes[ref]
+			if !ok {
+				return
+			}
+
+			for _, name := range composes.ImportedNames {
+				if record := repr.AST.ImportRecords[name.ImportRecordIndex]; record.SourceIndex.IsValid() {
+					otherFile := &c.graph.Files[record.SourceIndex.GetIndex()]
+					if otherRepr, ok := otherFile.InputFile.Repr.(*graph.CSSRepr); ok {
+						if otherName, ok := otherRepr.AST.LocalScope[name.Alias]; ok {
+							visit(otherFile, otherRepr, otherName.Ref)
+						}
+					}
+				}
+			}
+
+			for _, name := range composes.Names {
+				visit(file, repr, name.Ref)
+			}
+
+			// Warn about cross-file composition with the same CSS properties
+			for keyText, keyLoc := range composes.Properties {
+				property, ok := properties[keyText]
+				if !ok {
+					properties[keyText] = propertyInFile{file, keyLoc}
+					continue
+				}
+				if property.file == file || property.file == nil {
+					continue
+				}
+
+				localOriginalName := c.graph.Symbols.Get(local.Ref).OriginalName
+				c.log.AddMsgID(logger.MsgID_CSS_UndefinedComposesFrom, logger.Msg{
+					Kind: logger.Warning,
+					Data: rootFile.LineColumnTracker().MsgData(
+						css_lexer.RangeOfIdentifier(rootFile.InputFile.Source, local.Loc),
+						fmt.Sprintf("The value of %q in the %q class is undefined", keyText, localOriginalName),
+					),
+					Notes: []logger.MsgData{
+						property.file.LineColumnTracker().MsgData(
+							css_lexer.RangeOfIdentifier(property.file.InputFile.Source, property.loc),
+							fmt.Sprintf("The first definition of %q is here:", keyText),
+						),
+						file.LineColumnTracker().MsgData(
+							css_lexer.RangeOfIdentifier(file.InputFile.Source, keyLoc),
+							fmt.Sprintf("The second definition of %q is here:", keyText),
+						),
+						{Text: fmt.Sprintf("The specification of \"composes\" does not define an order when class declarations from separate files are composed together. "+
+							"The value of the %q property for %q may change unpredictably as the code is edited. "+
+							"Make sure that all definitions of %q for %q are in a single file.", keyText, localOriginalName, keyText, localOriginalName)},
+					},
+				})
+
+				// Don't warn more than once
+				property.file = nil
+				properties[keyText] = property
+			}
+		}
+
+		visit(rootFile, rootRepr, local.Ref)
+	}
+}
+
+func (c *linkerContext) generateCodeForLazyExport(sourceIndex uint32) {
+	file := &c.graph.Files[sourceIndex]
+	repr := file.InputFile.Repr.(*graph.JSRepr)
+
+	// Grab the lazy expression
+	if len(repr.AST.Parts) < 1 {
+		panic("Internal error")
+	}
+	part := &repr.AST.Parts[len(repr.AST.Parts)-1]
+	if len(part.Stmts) != 1 {
+		panic("Internal error")
+	}
+	lazyValue := part.Stmts[0].Data.(*js_ast.SLazyExport).Value
+
+	// If this JavaScript file is a stub from a CSS file, populate the exports of
+	// this JavaScript stub with the local names from that CSS file. This is done
+	// now instead of earlier because we need the whole bundle to be present.
+	if repr.CSSSourceIndex.IsValid() {
+		cssSourceIndex := repr.CSSSourceIndex.GetIndex()
+		if css, ok := c.graph.Files[cssSourceIndex].InputFile.Repr.(*graph.CSSRepr); ok {
+			exports := js_ast.EObject{}
+
+			for _, local := range css.AST.LocalSymbols {
+				value := js_ast.Expr{Loc: local.Loc, Data: &js_ast.ENameOfSymbol{Ref: local.Ref}}
+				visited := map[ast.Ref]bool{local.Ref: true}
+				var parts []js_ast.TemplatePart
+				var visitName func(*graph.CSSRepr, ast.Ref)
+				var visitComposes func(*graph.CSSRepr, ast.Ref)
+
+				visitName = func(repr *graph.CSSRepr, ref ast.Ref) {
+					if !visited[ref] {
+						visited[ref] = true
+						visitComposes(repr, ref)
+						parts = append(parts, js_ast.TemplatePart{
+							Value:      js_ast.Expr{Data: &js_ast.ENameOfSymbol{Ref: ref}},
+							TailCooked: []uint16{' '},
+						})
+					}
+				}
+
+				visitComposes = func(repr *graph.CSSRepr, ref ast.Ref) {
+					if composes, ok := repr.AST.Composes[ref]; ok {
+						for _, name := range composes.ImportedNames {
+							if record := repr.AST.ImportRecords[name.ImportRecordIndex]; record.SourceIndex.IsValid() {
+								otherFile := &c.graph.Files[record.SourceIndex.GetIndex()]
+								if otherRepr, ok := otherFile.InputFile.Repr.(*graph.CSSRepr); ok {
+									if otherName, ok := otherRepr.AST.LocalScope[name.Alias]; ok {
+										visitName(otherRepr, otherName.Ref)
+									}
+								}
+							}
+						}
+
+						for _, name := range composes.Names {
+							visitName(repr, name.Ref)
+						}
+					}
+				}
+
+				visitComposes(css, local.Ref)
+
+				if len(parts) > 0 {
+					value.Data = &js_ast.ETemplate{Parts: append(parts, js_ast.TemplatePart{Value: value})}
+				}
+
+				exports.Properties = append(exports.Properties, js_ast.Property{
+					Key:        js_ast.Expr{Loc: local.Loc, Data: &js_ast.EString{Value: helpers.StringToUTF16(c.graph.Symbols.Get(local.Ref).OriginalName)}},
+					ValueOrNil: value,
+				})
+			}
+
+			lazyValue.Data = &exports
+		}
+	}
+
+	// Use "module.exports = value" for CommonJS-style modules
+	if repr.AST.ExportsKind == js_ast.ExportsCommonJS {
+		part.Stmts = []js_ast.Stmt{js_ast.AssignStmt(
+			js_ast.Expr{Loc: lazyValue.Loc, Data: &js_ast.EDot{
+				Target:  js_ast.Expr{Loc: lazyValue.Loc, Data: &js_ast.EIdentifier{Ref: repr.AST.ModuleRef}},
+				Name:    "exports",
+				NameLoc: lazyValue.Loc,
+			}},
+			lazyValue,
+		)}
+		c.graph.GenerateSymbolImportAndUse(sourceIndex, 0, repr.AST.ModuleRef, 1, sourceIndex)
+		return
+	}
+
+	// Otherwise, generate ES6 export statements. These are added as additional
+	// parts so they can be tree shaken individually.
+	part.Stmts = nil
+
+	// Generate a new symbol and link the export into the graph for tree shaking
+	generateExport := func(loc logger.Loc, name string, alias string) (ast.Ref, uint32) {
+		ref := c.graph.GenerateNewSymbol(sourceIndex, ast.SymbolOther, name)
+		partIndex := c.graph.AddPartToFile(sourceIndex, js_ast.Part{
+			DeclaredSymbols:      []js_ast.DeclaredSymbol{{Ref: ref, IsTopLevel: true}},
+			CanBeRemovedIfUnused: true,
+		})
+		c.graph.GenerateSymbolImportAndUse(sourceIndex, partIndex, repr.AST.ModuleRef, 1, sourceIndex)
+		repr.Meta.TopLevelSymbolToPartsOverlay[ref] = []uint32{partIndex}
+		repr.Meta.ResolvedExports[alias] = graph.ExportData{
+			Ref:         ref,
+			NameLoc:     loc,
+			SourceIndex: sourceIndex,
+		}
+		return ref, partIndex
+	}
+
+	// Unwrap JSON objects into separate top-level variables. This improves tree-
+	// shaking by letting you only import part of a JSON file.
+	//
+	// But don't do this for files loaded via "with { type: 'json' }" as that
+	// behavior is specified to not export anything except for the "default"
+	// export: https://github.com/tc39/proposal-json-modules
+	if object, ok := lazyValue.Data.(*js_ast.EObject); ok && file.InputFile.Loader != config.LoaderWithTypeJSON {
+		for _, property := range object.Properties {
+			if str, ok := property.Key.Data.(*js_ast.EString); ok &&
+				(!file.IsEntryPoint() || js_ast.IsIdentifierUTF16(str.Value) ||
+					!c.options.UnsupportedJSFeatures.Has(compat.ArbitraryModuleNamespaceNames)) {
+				if name := helpers.UTF16ToString(str.Value); name != "default" {
+					ref, partIndex := generateExport(property.Key.Loc, name, name)
+
+					// This initializes the generated variable with a copy of the property
+					// value, which is INCORRECT for values that are objects/arrays because
+					// they will have separate object identity. This is fixed up later in
+					// "generateCodeForFileInChunkJS" by changing the object literal to
+					// reference this generated variable instead.
+					//
+					// Changing the object literal is deferred until that point instead of
+					// doing it now because we only want to do this for top-level variables
+					// that actually end up being used, and we don't know which ones will
+					// end up actually being used at this point (since import binding hasn't
+					// happened yet). So we need to wait until after tree shaking happens.
+					repr.AST.Parts[partIndex].Stmts = []js_ast.Stmt{{Loc: property.Key.Loc, Data: &js_ast.SLocal{
+						IsExport: true,
+						Decls: []js_ast.Decl{{
+							Binding:    js_ast.Binding{Loc: property.Key.Loc, Data: &js_ast.BIdentifier{Ref: ref}},
+							ValueOrNil: property.ValueOrNil,
+						}},
+					}}}
+				}
+			}
+		}
+	}
+
+	// Generate the default export
+	ref, partIndex := generateExport(lazyValue.Loc, file.InputFile.Source.IdentifierName+"_default", "default")
+	repr.AST.Parts[partIndex].Stmts = []js_ast.Stmt{{Loc: lazyValue.Loc, Data: &js_ast.SExportDefault{
+		DefaultName: ast.LocRef{Loc: lazyValue.Loc, Ref: ref},
+		Value:       js_ast.Stmt{Loc: lazyValue.Loc, Data: &js_ast.SExpr{Value: lazyValue}},
+	}}}
+}
+
+func (c *linkerContext) createExportsForFile(sourceIndex uint32) {
+	////////////////////////////////////////////////////////////////////////////////
+	// WARNING: This method is run in parallel over all files. Do not mutate data
+	// for other files within this method or you will create a data race.
+	////////////////////////////////////////////////////////////////////////////////
+
+	file := &c.graph.Files[sourceIndex]
+	repr := file.InputFile.Repr.(*graph.JSRepr)
+
+	// Generate a getter per export
+	properties := []js_ast.Property{}
+	nsExportDependencies := []js_ast.Dependency{}
+	nsExportSymbolUses := make(map[ast.Ref]js_ast.SymbolUse)
+	for _, alias := range repr.Meta.SortedAndFilteredExportAliases {
+		export := repr.Meta.ResolvedExports[alias]
+
+		// If this is an export of an import, reference the symbol that the import
+		// was eventually resolved to. We need to do this because imports have
+		// already been resolved by this point, so we can't generate a new import
+		// and have that be resolved later.
+		if importData, ok := c.graph.Files[export.SourceIndex].InputFile.Repr.(*graph.JSRepr).Meta.ImportsToBind[export.Ref]; ok {
+			export.Ref = importData.Ref
+			export.SourceIndex = importData.SourceIndex
+			nsExportDependencies = append(nsExportDependencies, importData.ReExports...)
+		}
+
+		// Exports of imports need EImportIdentifier in case they need to be re-
+		// written to a property access later on
+		var value js_ast.Expr
+		if c.graph.Symbols.Get(export.Ref).NamespaceAlias != nil {
+			value = js_ast.Expr{Data: &js_ast.EImportIdentifier{Ref: export.Ref}}
+		} else {
+			value = js_ast.Expr{Data: &js_ast.EIdentifier{Ref: export.Ref}}
+		}
+
+		// Add a getter property
+		var getter js_ast.Expr
+		body := js_ast.FnBody{Block: js_ast.SBlock{Stmts: []js_ast.Stmt{{Loc: value.Loc, Data: &js_ast.SReturn{ValueOrNil: value}}}}}
+		if c.options.UnsupportedJSFeatures.Has(compat.Arrow) {
+			getter = js_ast.Expr{Data: &js_ast.EFunction{Fn: js_ast.Fn{Body: body}}}
+		} else {
+			getter = js_ast.Expr{Data: &js_ast.EArrow{PreferExpr: true, Body: body}}
+		}
+		properties = append(properties, js_ast.Property{
+			Key:        js_ast.Expr{Data: &js_ast.EString{Value: helpers.StringToUTF16(alias)}},
+			ValueOrNil: getter,
+		})
+		nsExportSymbolUses[export.Ref] = js_ast.SymbolUse{CountEstimate: 1}
+
+		// Make sure the part that declares the export is included
+		for _, partIndex := range c.graph.Files[export.SourceIndex].InputFile.Repr.(*graph.JSRepr).TopLevelSymbolToParts(export.Ref) {
+			// Use a non-local dependency since this is likely from a different
+			// file if it came in through an export star
+			nsExportDependencies = append(nsExportDependencies, js_ast.Dependency{
+				SourceIndex: export.SourceIndex,
+				PartIndex:   partIndex,
+			})
+		}
+	}
+
+	declaredSymbols := []js_ast.DeclaredSymbol{}
+	var nsExportStmts []js_ast.Stmt
+
+	// Prefix this part with "var exports = {}" if this isn't a CommonJS entry point
+	if repr.Meta.NeedsExportsVariable {
+		nsExportStmts = append(nsExportStmts, js_ast.Stmt{Data: &js_ast.SLocal{Decls: []js_ast.Decl{{
+			Binding:    js_ast.Binding{Data: &js_ast.BIdentifier{Ref: repr.AST.ExportsRef}},
+			ValueOrNil: js_ast.Expr{Data: &js_ast.EObject{}},
+		}}}})
+		declaredSymbols = append(declaredSymbols, js_ast.DeclaredSymbol{
+			Ref:        repr.AST.ExportsRef,
+			IsTopLevel: true,
+		})
+	}
+
+	// "__export(exports, { foo: () => foo })"
+	exportRef := ast.InvalidRef
+	if len(properties) > 0 {
+		runtimeRepr := c.graph.Files[runtime.SourceIndex].InputFile.Repr.(*graph.JSRepr)
+		exportRef = runtimeRepr.AST.ModuleScope.Members["__export"].Ref
+		nsExportStmts = append(nsExportStmts, js_ast.Stmt{Data: &js_ast.SExpr{Value: js_ast.Expr{Data: &js_ast.ECall{
+			Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: exportRef}},
+			Args: []js_ast.Expr{
+				{Data: &js_ast.EIdentifier{Ref: repr.AST.ExportsRef}},
+				{Data: &js_ast.EObject{
+					Properties: properties,
+				}},
+			},
+		}}}})
+
+		// Make sure this file depends on the "__export" symbol
+		for _, partIndex := range runtimeRepr.TopLevelSymbolToParts(exportRef) {
+			nsExportDependencies = append(nsExportDependencies, js_ast.Dependency{
+				SourceIndex: runtime.SourceIndex,
+				PartIndex:   partIndex,
+			})
+		}
+
+		// Make sure the CommonJS closure, if there is one, includes "exports"
+		repr.AST.UsesExportsRef = true
+	}
+
+	// Decorate "module.exports" with the "__esModule" flag to indicate that
+	// we used to be an ES module. This is done by wrapping the exports object
+	// instead of by mutating the exports object because other modules in the
+	// bundle (including the entry point module) may do "import * as" to get
+	// access to the exports object and should NOT see the "__esModule" flag.
+	if repr.Meta.ForceIncludeExportsForEntryPoint &&
+		c.options.OutputFormat == config.FormatCommonJS {
+
+		runtimeRepr := c.graph.Files[runtime.SourceIndex].InputFile.Repr.(*graph.JSRepr)
+		toCommonJSRef := runtimeRepr.AST.NamedExports["__toCommonJS"].Ref
+
+		// "module.exports = __toCommonJS(exports);"
+		nsExportStmts = append(nsExportStmts, js_ast.AssignStmt(
+			js_ast.Expr{Data: &js_ast.EDot{
+				Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: c.unboundModuleRef}},
+				Name:   "exports",
+			}},
+
+			js_ast.Expr{Data: &js_ast.ECall{
+				Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: toCommonJSRef}},
+				Args:   []js_ast.Expr{{Data: &js_ast.EIdentifier{Ref: repr.AST.ExportsRef}}},
+			}},
+		))
+	}
+
+	// No need to generate a part if it'll be empty
+	if len(nsExportStmts) > 0 {
+		// Initialize the part that was allocated for us earlier. The information
+		// here will be used after this during tree shaking.
+		repr.AST.Parts[js_ast.NSExportPartIndex] = js_ast.Part{
+			Stmts:           nsExportStmts,
+			SymbolUses:      nsExportSymbolUses,
+			Dependencies:    nsExportDependencies,
+			DeclaredSymbols: declaredSymbols,
+
+			// This can be removed if nothing uses it
+			CanBeRemovedIfUnused: true,
+
+			// Make sure this is trimmed if unused even if tree shaking is disabled
+			ForceTreeShaking: true,
+		}
+
+		// Pull in the "__export" symbol if it was used
+		if exportRef != ast.InvalidRef {
+			repr.Meta.NeedsExportSymbolFromRuntime = true
+		}
+	}
+}
+
+func (c *linkerContext) createWrapperForFile(sourceIndex uint32) {
+	repr := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr)
+
+	switch repr.Meta.Wrap {
+	// If this is a CommonJS file, we're going to need to generate a wrapper
+	// for the CommonJS closure. That will end up looking something like this:
+	//
+	//   var require_foo = __commonJS((exports, module) => {
+	//     ...
+	//   });
+	//
+	// However, that generation is special-cased for various reasons and is
+	// done later on. Still, we're going to need to ensure that this file
+	// both depends on the "__commonJS" symbol and declares the "require_foo"
+	// symbol. Instead of special-casing this during the reachablity analysis
+	// below, we just append a dummy part to the end of the file with these
+	// dependencies and let the general-purpose reachablity analysis take care
+	// of it.
+	case graph.WrapCJS:
+		runtimeRepr := c.graph.Files[runtime.SourceIndex].InputFile.Repr.(*graph.JSRepr)
+		commonJSParts := runtimeRepr.TopLevelSymbolToParts(c.cjsRuntimeRef)
+
+		// Generate the dummy part
+		dependencies := make([]js_ast.Dependency, len(commonJSParts))
+		for i, partIndex := range commonJSParts {
+			dependencies[i] = js_ast.Dependency{
+				SourceIndex: runtime.SourceIndex,
+				PartIndex:   partIndex,
+			}
+		}
+		partIndex := c.graph.AddPartToFile(sourceIndex, js_ast.Part{
+			SymbolUses: map[ast.Ref]js_ast.SymbolUse{
+				repr.AST.WrapperRef: {CountEstimate: 1},
+			},
+			DeclaredSymbols: []js_ast.DeclaredSymbol{
+				{Ref: repr.AST.ExportsRef, IsTopLevel: true},
+				{Ref: repr.AST.ModuleRef, IsTopLevel: true},
+				{Ref: repr.AST.WrapperRef, IsTopLevel: true},
+			},
+			Dependencies: dependencies,
+		})
+		repr.Meta.WrapperPartIndex = ast.MakeIndex32(partIndex)
+		c.graph.GenerateSymbolImportAndUse(sourceIndex, partIndex, c.cjsRuntimeRef, 1, runtime.SourceIndex)
+
+	// If this is a lazily-initialized ESM file, we're going to need to
+	// generate a wrapper for the ESM closure. That will end up looking
+	// something like this:
+	//
+	//   var init_foo = __esm(() => {
+	//     ...
+	//   });
+	//
+	// This depends on the "__esm" symbol and declares the "init_foo" symbol
+	// for similar reasons to the CommonJS closure above.
+	case graph.WrapESM:
+		runtimeRepr := c.graph.Files[runtime.SourceIndex].InputFile.Repr.(*graph.JSRepr)
+		esmParts := runtimeRepr.TopLevelSymbolToParts(c.esmRuntimeRef)
+
+		// Generate the dummy part
+		dependencies := make([]js_ast.Dependency, len(esmParts))
+		for i, partIndex := range esmParts {
+			dependencies[i] = js_ast.Dependency{
+				SourceIndex: runtime.SourceIndex,
+				PartIndex:   partIndex,
+			}
+		}
+		partIndex := c.graph.AddPartToFile(sourceIndex, js_ast.Part{
+			SymbolUses: map[ast.Ref]js_ast.SymbolUse{
+				repr.AST.WrapperRef: {CountEstimate: 1},
+			},
+			DeclaredSymbols: []js_ast.DeclaredSymbol{
+				{Ref: repr.AST.WrapperRef, IsTopLevel: true},
+			},
+			Dependencies: dependencies,
+		})
+		repr.Meta.WrapperPartIndex = ast.MakeIndex32(partIndex)
+		c.graph.GenerateSymbolImportAndUse(sourceIndex, partIndex, c.esmRuntimeRef, 1, runtime.SourceIndex)
+	}
+}
+
+func (c *linkerContext) matchImportsWithExportsForFile(sourceIndex uint32) {
+	file := &c.graph.Files[sourceIndex]
+	repr := file.InputFile.Repr.(*graph.JSRepr)
+
+	// Sort imports for determinism. Otherwise our unit tests will randomly
+	// fail sometimes when error messages are reordered.
+	sortedImportRefs := make([]int, 0, len(repr.AST.NamedImports))
+	for ref := range repr.AST.NamedImports {
+		sortedImportRefs = append(sortedImportRefs, int(ref.InnerIndex))
+	}
+	sort.Ints(sortedImportRefs)
+
+	// Pair imports with their matching exports
+	for _, innerIndex := range sortedImportRefs {
+		// Re-use memory for the cycle detector
+		c.cycleDetector = c.cycleDetector[:0]
+
+		importRef := ast.Ref{SourceIndex: sourceIndex, InnerIndex: uint32(innerIndex)}
+		result, reExports := c.matchImportWithExport(importTracker{sourceIndex: sourceIndex, importRef: importRef}, nil)
+		switch result.kind {
+		case matchImportIgnore:
+
+		case matchImportNormal:
+			repr.Meta.ImportsToBind[importRef] = graph.ImportData{
+				ReExports:   reExports,
+				SourceIndex: result.sourceIndex,
+				Ref:         result.ref,
+			}
+
+		case matchImportNamespace:
+			c.graph.Symbols.Get(importRef).NamespaceAlias = &ast.NamespaceAlias{
+				NamespaceRef: result.namespaceRef,
+				Alias:        result.alias,
+			}
+
+		case matchImportNormalAndNamespace:
+			repr.Meta.ImportsToBind[importRef] = graph.ImportData{
+				ReExports:   reExports,
+				SourceIndex: result.sourceIndex,
+				Ref:         result.ref,
+			}
+
+			c.graph.Symbols.Get(importRef).NamespaceAlias = &ast.NamespaceAlias{
+				NamespaceRef: result.namespaceRef,
+				Alias:        result.alias,
+			}
+
+		case matchImportCycle:
+			namedImport := repr.AST.NamedImports[importRef]
+			c.log.AddError(file.LineColumnTracker(), js_lexer.RangeOfIdentifier(file.InputFile.Source, namedImport.AliasLoc),
+				fmt.Sprintf("Detected cycle while resolving import %q", namedImport.Alias))
+
+		case matchImportProbablyTypeScriptType:
+			repr.Meta.IsProbablyTypeScriptType[importRef] = true
+
+		case matchImportAmbiguous:
+			namedImport := repr.AST.NamedImports[importRef]
+			r := js_lexer.RangeOfIdentifier(file.InputFile.Source, namedImport.AliasLoc)
+			var notes []logger.MsgData
+
+			// Provide the locations of both ambiguous exports if possible
+			if result.nameLoc.Start != 0 && result.otherNameLoc.Start != 0 {
+				a := c.graph.Files[result.sourceIndex]
+				b := c.graph.Files[result.otherSourceIndex]
+				ra := js_lexer.RangeOfIdentifier(a.InputFile.Source, result.nameLoc)
+				rb := js_lexer.RangeOfIdentifier(b.InputFile.Source, result.otherNameLoc)
+				notes = []logger.MsgData{
+					a.LineColumnTracker().MsgData(ra, "One matching export is here:"),
+					b.LineColumnTracker().MsgData(rb, "Another matching export is here:"),
+				}
+			}
+
+			symbol := c.graph.Symbols.Get(importRef)
+			if symbol.ImportItemStatus == ast.ImportItemGenerated {
+				// This is a warning instead of an error because although it appears
+				// to be a named import, it's actually an automatically-generated
+				// named import that was originally a property access on an import
+				// star namespace object. Normally this property access would just
+				// resolve to undefined at run-time instead of failing at binding-
+				// time, so we emit a warning and rewrite the value to the literal
+				// "undefined" instead of emitting an error.
+				symbol.ImportItemStatus = ast.ImportItemMissing
+				msg := fmt.Sprintf("Import %q will always be undefined because there are multiple matching exports", namedImport.Alias)
+				c.log.AddIDWithNotes(logger.MsgID_Bundler_ImportIsUndefined, logger.Warning, file.LineColumnTracker(), r, msg, notes)
+			} else {
+				msg := fmt.Sprintf("Ambiguous import %q has multiple matching exports", namedImport.Alias)
+				c.log.AddErrorWithNotes(file.LineColumnTracker(), r, msg, notes)
+			}
+		}
+	}
+}
+
+type matchImportKind uint8
+
+const (
+	// The import is either external or undefined
+	matchImportIgnore matchImportKind = iota
+
+	// "sourceIndex" and "ref" are in use
+	matchImportNormal
+
+	// "namespaceRef" and "alias" are in use
+	matchImportNamespace
+
+	// Both "matchImportNormal" and "matchImportNamespace"
+	matchImportNormalAndNamespace
+
+	// The import could not be evaluated due to a cycle
+	matchImportCycle
+
+	// The import is missing but came from a TypeScript file
+	matchImportProbablyTypeScriptType
+
+	// The import resolved to multiple symbols via "export * from"
+	matchImportAmbiguous
+)
+
+type matchImportResult struct {
+	alias            string
+	kind             matchImportKind
+	namespaceRef     ast.Ref
+	sourceIndex      uint32
+	nameLoc          logger.Loc // Optional, goes with sourceIndex, ignore if zero
+	otherSourceIndex uint32
+	otherNameLoc     logger.Loc // Optional, goes with otherSourceIndex, ignore if zero
+	ref              ast.Ref
+}
+
+func (c *linkerContext) matchImportWithExport(
+	tracker importTracker, reExportsIn []js_ast.Dependency,
+) (result matchImportResult, reExports []js_ast.Dependency) {
+	var ambiguousResults []matchImportResult
+	reExports = reExportsIn
+
+loop:
+	for {
+		// Make sure we avoid infinite loops trying to resolve cycles:
+		//
+		//   // foo.js
+		//   export {a as b} from './foo.js'
+		//   export {b as c} from './foo.js'
+		//   export {c as a} from './foo.js'
+		//
+		// This uses a O(n^2) array scan instead of a O(n) map because the vast
+		// majority of cases have one or two elements and Go arrays are cheap to
+		// reuse without allocating.
+		for _, previousTracker := range c.cycleDetector {
+			if tracker == previousTracker {
+				result = matchImportResult{kind: matchImportCycle}
+				break loop
+			}
+		}
+		c.cycleDetector = append(c.cycleDetector, tracker)
+
+		// Resolve the import by one step
+		nextTracker, status, potentiallyAmbiguousExportStarRefs := c.advanceImportTracker(tracker)
+		switch status {
+		case importCommonJS, importCommonJSWithoutExports, importExternal, importDisabled:
+			if status == importExternal && c.options.OutputFormat.KeepESMImportExportSyntax() {
+				// Imports from external modules should not be converted to CommonJS
+				// if the output format preserves the original ES6 import statements
+				break
+			}
+
+			// If it's a CommonJS or external file, rewrite the import to a
+			// property access. Don't do this if the namespace reference is invalid
+			// though. This is the case for star imports, where the import is the
+			// namespace.
+			trackerFile := &c.graph.Files[tracker.sourceIndex]
+			namedImport := trackerFile.InputFile.Repr.(*graph.JSRepr).AST.NamedImports[tracker.importRef]
+			if namedImport.NamespaceRef != ast.InvalidRef {
+				if result.kind == matchImportNormal {
+					result.kind = matchImportNormalAndNamespace
+					result.namespaceRef = namedImport.NamespaceRef
+					result.alias = namedImport.Alias
+				} else {
+					result = matchImportResult{
+						kind:         matchImportNamespace,
+						namespaceRef: namedImport.NamespaceRef,
+						alias:        namedImport.Alias,
+					}
+				}
+			}
+
+			// Warn about importing from a file that is known to not have any exports
+			if status == importCommonJSWithoutExports {
+				symbol := c.graph.Symbols.Get(tracker.importRef)
+				symbol.ImportItemStatus = ast.ImportItemMissing
+				kind := logger.Warning
+				if helpers.IsInsideNodeModules(trackerFile.InputFile.Source.KeyPath.Text) {
+					kind = logger.Debug
+				}
+				c.log.AddID(logger.MsgID_Bundler_ImportIsUndefined, kind,
+					trackerFile.LineColumnTracker(),
+					js_lexer.RangeOfIdentifier(trackerFile.InputFile.Source, namedImport.AliasLoc),
+					fmt.Sprintf("Import %q will always be undefined because the file %q has no exports",
+						namedImport.Alias, c.graph.Files[nextTracker.sourceIndex].InputFile.Source.PrettyPath))
+			}
+
+		case importDynamicFallback:
+			// If it's a file with dynamic export fallback, rewrite the import to a property access
+			trackerFile := &c.graph.Files[tracker.sourceIndex]
+			namedImport := trackerFile.InputFile.Repr.(*graph.JSRepr).AST.NamedImports[tracker.importRef]
+			if result.kind == matchImportNormal {
+				result.kind = matchImportNormalAndNamespace
+				result.namespaceRef = nextTracker.importRef
+				result.alias = namedImport.Alias
+			} else {
+				result = matchImportResult{
+					kind:         matchImportNamespace,
+					namespaceRef: nextTracker.importRef,
+					alias:        namedImport.Alias,
+				}
+			}
+
+		case importNoMatch:
+			symbol := c.graph.Symbols.Get(tracker.importRef)
+			trackerFile := &c.graph.Files[tracker.sourceIndex]
+			namedImport := trackerFile.InputFile.Repr.(*graph.JSRepr).AST.NamedImports[tracker.importRef]
+			r := js_lexer.RangeOfIdentifier(trackerFile.InputFile.Source, namedImport.AliasLoc)
+
+			// Report mismatched imports and exports
+			if symbol.ImportItemStatus == ast.ImportItemGenerated {
+				// This is not an error because although it appears to be a named
+				// import, it's actually an automatically-generated named import
+				// that was originally a property access on an import star
+				// namespace object:
+				//
+				//   import * as ns from 'foo'
+				//   const undefinedValue = ns.notAnExport
+				//
+				// If this code wasn't bundled, this property access would just resolve
+				// to undefined at run-time instead of failing at binding-time, so we
+				// emit rewrite the value to the literal "undefined" instead of
+				// emitting an error.
+				symbol.ImportItemStatus = ast.ImportItemMissing
+
+				// Don't emit a log message if this symbol isn't used, since then the
+				// log message isn't helpful. This can happen with "import" assignment
+				// statements in TypeScript code since they are ambiguously either a
+				// type or a value. We consider them to be a type if they aren't used.
+				//
+				//   import * as ns from 'foo'
+				//
+				//   // There's no warning here because this is dead code
+				//   if (false) ns.notAnExport
+				//
+				//   // There's no warning here because this is never used
+				//   import unused = ns.notAnExport
+				//
+				if symbol.UseCountEstimate > 0 {
+					nextFile := &c.graph.Files[nextTracker.sourceIndex].InputFile
+					msg := logger.Msg{
+						Kind: logger.Warning,
+						Data: trackerFile.LineColumnTracker().MsgData(r, fmt.Sprintf(
+							"Import %q will always be undefined because there is no matching export in %q",
+							namedImport.Alias, nextFile.Source.PrettyPath)),
+					}
+					if helpers.IsInsideNodeModules(trackerFile.InputFile.Source.KeyPath.Text) {
+						msg.Kind = logger.Debug
+					}
+					c.maybeCorrectObviousTypo(nextFile.Repr.(*graph.JSRepr), namedImport.Alias, &msg)
+					c.log.AddMsgID(logger.MsgID_Bundler_ImportIsUndefined, msg)
+				}
+			} else {
+				nextFile := &c.graph.Files[nextTracker.sourceIndex].InputFile
+				msg := logger.Msg{
+					Kind: logger.Error,
+					Data: trackerFile.LineColumnTracker().MsgData(r, fmt.Sprintf(
+						"No matching export in %q for import %q",
+						nextFile.Source.PrettyPath, namedImport.Alias)),
+				}
+				c.maybeCorrectObviousTypo(nextFile.Repr.(*graph.JSRepr), namedImport.Alias, &msg)
+				c.log.AddMsg(msg)
+			}
+
+		case importProbablyTypeScriptType:
+			// Omit this import from any namespace export code we generate for
+			// import star statements (i.e. "import * as ns from 'path'")
+			result = matchImportResult{kind: matchImportProbablyTypeScriptType}
+
+		case importFound:
+			// If there are multiple ambiguous results due to use of "export * from"
+			// statements, trace them all to see if they point to different things.
+			for _, ambiguousTracker := range potentiallyAmbiguousExportStarRefs {
+				// If this is a re-export of another import, follow the import
+				if _, ok := c.graph.Files[ambiguousTracker.SourceIndex].InputFile.Repr.(*graph.JSRepr).AST.NamedImports[ambiguousTracker.Ref]; ok {
+					// Save and restore the cycle detector to avoid mixing information
+					oldCycleDetector := c.cycleDetector
+					ambiguousResult, newReExportFiles := c.matchImportWithExport(importTracker{
+						sourceIndex: ambiguousTracker.SourceIndex,
+						importRef:   ambiguousTracker.Ref,
+					}, reExports)
+					c.cycleDetector = oldCycleDetector
+					ambiguousResults = append(ambiguousResults, ambiguousResult)
+					reExports = newReExportFiles
+				} else {
+					ambiguousResults = append(ambiguousResults, matchImportResult{
+						kind:        matchImportNormal,
+						sourceIndex: ambiguousTracker.SourceIndex,
+						ref:         ambiguousTracker.Ref,
+						nameLoc:     ambiguousTracker.NameLoc,
+					})
+				}
+			}
+
+			// Defer the actual binding of this import until after we generate
+			// namespace export code for all files. This has to be done for all
+			// import-to-export matches, not just the initial import to the final
+			// export, since all imports and re-exports must be merged together
+			// for correctness.
+			result = matchImportResult{
+				kind:        matchImportNormal,
+				sourceIndex: nextTracker.sourceIndex,
+				ref:         nextTracker.importRef,
+				nameLoc:     nextTracker.nameLoc,
+			}
+
+			// Depend on the statement(s) that declared this import symbol in the
+			// original file
+			for _, resolvedPartIndex := range c.graph.Files[tracker.sourceIndex].InputFile.Repr.(*graph.JSRepr).TopLevelSymbolToParts(tracker.importRef) {
+				reExports = append(reExports, js_ast.Dependency{
+					SourceIndex: tracker.sourceIndex,
+					PartIndex:   resolvedPartIndex,
+				})
+			}
+
+			// If this is a re-export of another import, continue for another
+			// iteration of the loop to resolve that import as well
+			if _, ok := c.graph.Files[nextTracker.sourceIndex].InputFile.Repr.(*graph.JSRepr).AST.NamedImports[nextTracker.importRef]; ok {
+				tracker = nextTracker
+				continue
+			}
+
+		default:
+			panic("Internal error")
+		}
+
+		// Stop now if we didn't explicitly "continue" above
+		break
+	}
+
+	// If there is a potential ambiguity, all results must be the same
+	for _, ambiguousResult := range ambiguousResults {
+		if ambiguousResult != result {
+			if result.kind == matchImportNormal && ambiguousResult.kind == matchImportNormal &&
+				result.nameLoc.Start != 0 && ambiguousResult.nameLoc.Start != 0 {
+				return matchImportResult{
+					kind:             matchImportAmbiguous,
+					sourceIndex:      result.sourceIndex,
+					nameLoc:          result.nameLoc,
+					otherSourceIndex: ambiguousResult.sourceIndex,
+					otherNameLoc:     ambiguousResult.nameLoc,
+				}, nil
+			}
+			return matchImportResult{kind: matchImportAmbiguous}, nil
+		}
+	}
+
+	return
+}
+
+func (c *linkerContext) maybeForbidArbitraryModuleNamespaceIdentifier(kind string, sourceIndex uint32, loc logger.Loc, alias string) {
+	if !js_ast.IsIdentifier(alias) {
+		file := &c.graph.Files[sourceIndex]
+		where := config.PrettyPrintTargetEnvironment(c.options.OriginalTargetEnv, c.options.UnsupportedJSFeatureOverridesMask)
+		c.log.AddError(file.LineColumnTracker(), file.InputFile.Source.RangeOfString(loc), fmt.Sprintf(
+			"Using the string %q as an %s name is not supported in %s", alias, kind, where))
+	}
+}
+
+// Attempt to correct an import name with a typo
+func (c *linkerContext) maybeCorrectObviousTypo(repr *graph.JSRepr, name string, msg *logger.Msg) {
+	if repr.Meta.ResolvedExportTypos == nil {
+		valid := make([]string, 0, len(repr.Meta.ResolvedExports))
+		for alias := range repr.Meta.ResolvedExports {
+			valid = append(valid, alias)
+		}
+		sort.Strings(valid)
+		typos := helpers.MakeTypoDetector(valid)
+		repr.Meta.ResolvedExportTypos = &typos
+	}
+
+	if corrected, ok := repr.Meta.ResolvedExportTypos.MaybeCorrectTypo(name); ok {
+		msg.Data.Location.Suggestion = corrected
+		export := repr.Meta.ResolvedExports[corrected]
+		importedFile := &c.graph.Files[export.SourceIndex]
+		text := fmt.Sprintf("Did you mean to import %q instead?", corrected)
+		var note logger.MsgData
+		if export.NameLoc.Start == 0 {
+			// Don't report a source location for definitions without one. This can
+			// happen with automatically-generated exports from non-JavaScript files.
+			note.Text = text
+		} else {
+			var r logger.Range
+			if importedFile.InputFile.Loader.IsCSS() {
+				r = css_lexer.RangeOfIdentifier(importedFile.InputFile.Source, export.NameLoc)
+			} else {
+				r = js_lexer.RangeOfIdentifier(importedFile.InputFile.Source, export.NameLoc)
+			}
+			note = importedFile.LineColumnTracker().MsgData(r, text)
+		}
+		msg.Notes = append(msg.Notes, note)
+	}
+}
+
+func (c *linkerContext) recursivelyWrapDependencies(sourceIndex uint32) {
+	repr := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr)
+	if repr.Meta.DidWrapDependencies {
+		return
+	}
+	repr.Meta.DidWrapDependencies = true
+
+	// Never wrap the runtime file since it always comes first
+	if sourceIndex == runtime.SourceIndex {
+		return
+	}
+
+	// This module must be wrapped
+	if repr.Meta.Wrap == graph.WrapNone {
+		if repr.AST.ExportsKind == js_ast.ExportsCommonJS {
+			repr.Meta.Wrap = graph.WrapCJS
+		} else {
+			repr.Meta.Wrap = graph.WrapESM
+		}
+	}
+
+	// All dependencies must also be wrapped
+	for _, record := range repr.AST.ImportRecords {
+		if record.SourceIndex.IsValid() {
+			c.recursivelyWrapDependencies(record.SourceIndex.GetIndex())
+		}
+	}
+}
+
+func (c *linkerContext) hasDynamicExportsDueToExportStar(sourceIndex uint32, visited map[uint32]bool) bool {
+	// Terminate the traversal now if this file already has dynamic exports
+	repr := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr)
+	if repr.AST.ExportsKind == js_ast.ExportsCommonJS || repr.AST.ExportsKind == js_ast.ExportsESMWithDynamicFallback {
+		return true
+	}
+
+	// Avoid infinite loops due to cycles in the export star graph
+	if visited[sourceIndex] {
+		return false
+	}
+	visited[sourceIndex] = true
+
+	// Scan over the export star graph
+	for _, importRecordIndex := range repr.AST.ExportStarImportRecords {
+		record := &repr.AST.ImportRecords[importRecordIndex]
+
+		// This file has dynamic exports if the exported imports are from a file
+		// that either has dynamic exports directly or transitively by itself
+		// having an export star from a file with dynamic exports.
+		if (!record.SourceIndex.IsValid() && (!c.graph.Files[sourceIndex].IsEntryPoint() || !c.options.OutputFormat.KeepESMImportExportSyntax())) ||
+			(record.SourceIndex.IsValid() && record.SourceIndex.GetIndex() != sourceIndex && c.hasDynamicExportsDueToExportStar(record.SourceIndex.GetIndex(), visited)) {
+			repr.AST.ExportsKind = js_ast.ExportsESMWithDynamicFallback
+			return true
+		}
+	}
+
+	return false
+}
+
+func (c *linkerContext) addExportsForExportStar(
+	resolvedExports map[string]graph.ExportData,
+	sourceIndex uint32,
+	sourceIndexStack []uint32,
+) {
+	// Avoid infinite loops due to cycles in the export star graph
+	for _, prevSourceIndex := range sourceIndexStack {
+		if prevSourceIndex == sourceIndex {
+			return
+		}
+	}
+	sourceIndexStack = append(sourceIndexStack, sourceIndex)
+	repr := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr)
+
+	for _, importRecordIndex := range repr.AST.ExportStarImportRecords {
+		record := &repr.AST.ImportRecords[importRecordIndex]
+		if !record.SourceIndex.IsValid() {
+			// This will be resolved at run time instead
+			continue
+		}
+		otherSourceIndex := record.SourceIndex.GetIndex()
+
+		// Export stars from a CommonJS module don't work because they can't be
+		// statically discovered. Just silently ignore them in this case.
+		//
+		// We could attempt to check whether the imported file still has ES6
+		// exports even though it still uses CommonJS features. However, when
+		// doing this we'd also have to rewrite any imports of these export star
+		// re-exports as property accesses off of a generated require() call.
+		otherRepr := c.graph.Files[otherSourceIndex].InputFile.Repr.(*graph.JSRepr)
+		if otherRepr.AST.ExportsKind == js_ast.ExportsCommonJS {
+			// All exports will be resolved at run time instead
+			continue
+		}
+
+		// Accumulate this file's exports
+	nextExport:
+		for alias, name := range otherRepr.AST.NamedExports {
+			// ES6 export star statements ignore exports named "default"
+			if alias == "default" {
+				continue
+			}
+
+			// This export star is shadowed if any file in the stack has a matching real named export
+			for _, prevSourceIndex := range sourceIndexStack {
+				prevRepr := c.graph.Files[prevSourceIndex].InputFile.Repr.(*graph.JSRepr)
+				if _, ok := prevRepr.AST.NamedExports[alias]; ok {
+					continue nextExport
+				}
+			}
+
+			if existing, ok := resolvedExports[alias]; !ok {
+				// Initialize the re-export
+				resolvedExports[alias] = graph.ExportData{
+					Ref:         name.Ref,
+					SourceIndex: otherSourceIndex,
+					NameLoc:     name.AliasLoc,
+				}
+
+				// Make sure the symbol is marked as imported so that code splitting
+				// imports it correctly if it ends up being shared with another chunk
+				repr.Meta.ImportsToBind[name.Ref] = graph.ImportData{
+					Ref:         name.Ref,
+					SourceIndex: otherSourceIndex,
+				}
+			} else if existing.SourceIndex != otherSourceIndex {
+				// Two different re-exports colliding makes it potentially ambiguous
+				existing.PotentiallyAmbiguousExportStarRefs =
+					append(existing.PotentiallyAmbiguousExportStarRefs, graph.ImportData{
+						SourceIndex: otherSourceIndex,
+						Ref:         name.Ref,
+						NameLoc:     name.AliasLoc,
+					})
+				resolvedExports[alias] = existing
+			}
+		}
+
+		// Search further through this file's export stars
+		c.addExportsForExportStar(resolvedExports, otherSourceIndex, sourceIndexStack)
+	}
+}
+
+type importTracker struct {
+	sourceIndex uint32
+	nameLoc     logger.Loc // Optional, goes with sourceIndex, ignore if zero
+	importRef   ast.Ref
+}
+
+type importStatus uint8
+
+const (
+	// The imported file has no matching export
+	importNoMatch importStatus = iota
+
+	// The imported file has a matching export
+	importFound
+
+	// The imported file is CommonJS and has unknown exports
+	importCommonJS
+
+	// The import is missing but there is a dynamic fallback object
+	importDynamicFallback
+
+	// The import was treated as a CommonJS import but the file is known to have no exports
+	importCommonJSWithoutExports
+
+	// The imported file was disabled by mapping it to false in the "browser"
+	// field of package.json
+	importDisabled
+
+	// The imported file is external and has unknown exports
+	importExternal
+
+	// This is a missing re-export in a TypeScript file, so it's probably a type
+	importProbablyTypeScriptType
+)
+
+func (c *linkerContext) advanceImportTracker(tracker importTracker) (importTracker, importStatus, []graph.ImportData) {
+	file := &c.graph.Files[tracker.sourceIndex]
+	repr := file.InputFile.Repr.(*graph.JSRepr)
+	namedImport := repr.AST.NamedImports[tracker.importRef]
+
+	// Is this an external file?
+	record := &repr.AST.ImportRecords[namedImport.ImportRecordIndex]
+	if !record.SourceIndex.IsValid() {
+		return importTracker{}, importExternal, nil
+	}
+
+	// Is this a named import of a file without any exports?
+	otherSourceIndex := record.SourceIndex.GetIndex()
+	otherRepr := c.graph.Files[otherSourceIndex].InputFile.Repr.(*graph.JSRepr)
+	if !namedImport.AliasIsStar && !otherRepr.AST.HasLazyExport &&
+		// CommonJS exports
+		otherRepr.AST.ExportKeyword.Len == 0 && namedImport.Alias != "default" &&
+		// ESM exports
+		!otherRepr.AST.UsesExportsRef && !otherRepr.AST.UsesModuleRef {
+		// Just warn about it and replace the import with "undefined"
+		return importTracker{sourceIndex: otherSourceIndex, importRef: ast.InvalidRef}, importCommonJSWithoutExports, nil
+	}
+
+	// Is this a CommonJS file?
+	if otherRepr.AST.ExportsKind == js_ast.ExportsCommonJS {
+		return importTracker{sourceIndex: otherSourceIndex, importRef: ast.InvalidRef}, importCommonJS, nil
+	}
+
+	// Match this import star with an export star from the imported file
+	if matchingExport := otherRepr.Meta.ResolvedExportStar; namedImport.AliasIsStar && matchingExport != nil {
+		// Check to see if this is a re-export of another import
+		return importTracker{
+			sourceIndex: matchingExport.SourceIndex,
+			importRef:   matchingExport.Ref,
+			nameLoc:     matchingExport.NameLoc,
+		}, importFound, matchingExport.PotentiallyAmbiguousExportStarRefs
+	}
+
+	// Match this import up with an export from the imported file
+	if matchingExport, ok := otherRepr.Meta.ResolvedExports[namedImport.Alias]; ok {
+		// Check to see if this is a re-export of another import
+		return importTracker{
+			sourceIndex: matchingExport.SourceIndex,
+			importRef:   matchingExport.Ref,
+			nameLoc:     matchingExport.NameLoc,
+		}, importFound, matchingExport.PotentiallyAmbiguousExportStarRefs
+	}
+
+	// Is this a file with dynamic exports?
+	if otherRepr.AST.ExportsKind == js_ast.ExportsESMWithDynamicFallback {
+		return importTracker{sourceIndex: otherSourceIndex, importRef: otherRepr.AST.ExportsRef}, importDynamicFallback, nil
+	}
+
+	// Missing re-exports in TypeScript files are indistinguishable from types
+	if file.InputFile.Loader.IsTypeScript() && namedImport.IsExported {
+		return importTracker{}, importProbablyTypeScriptType, nil
+	}
+
+	return importTracker{sourceIndex: otherSourceIndex}, importNoMatch, nil
+}
+
+func (c *linkerContext) treeShakingAndCodeSplitting() {
+	// Tree shaking: Each entry point marks all files reachable from itself
+	c.timer.Begin("Tree shaking")
+	for _, entryPoint := range c.graph.EntryPoints() {
+		c.markFileLiveForTreeShaking(entryPoint.SourceIndex)
+	}
+	c.timer.End("Tree shaking")
+
+	// Code splitting: Determine which entry points can reach which files. This
+	// has to happen after tree shaking because there is an implicit dependency
+	// between live parts within the same file. All liveness has to be computed
+	// first before determining which entry points can reach which files.
+	c.timer.Begin("Code splitting")
+	for i, entryPoint := range c.graph.EntryPoints() {
+		c.markFileReachableForCodeSplitting(entryPoint.SourceIndex, uint(i), 0)
+	}
+	c.timer.End("Code splitting")
+}
+
+func (c *linkerContext) markFileReachableForCodeSplitting(sourceIndex uint32, entryPointBit uint, distanceFromEntryPoint uint32) {
+	file := &c.graph.Files[sourceIndex]
+	if !file.IsLive {
+		return
+	}
+	traverseAgain := false
+
+	// Track the minimum distance to an entry point
+	if distanceFromEntryPoint < file.DistanceFromEntryPoint {
+		file.DistanceFromEntryPoint = distanceFromEntryPoint
+		traverseAgain = true
+	}
+	distanceFromEntryPoint++
+
+	// Don't mark this file more than once
+	if file.EntryBits.HasBit(entryPointBit) && !traverseAgain {
+		return
+	}
+	file.EntryBits.SetBit(entryPointBit)
+
+	switch repr := file.InputFile.Repr.(type) {
+	case *graph.JSRepr:
+		// If the JavaScript stub for a CSS file is included, also include the CSS file
+		if repr.CSSSourceIndex.IsValid() {
+			c.markFileReachableForCodeSplitting(repr.CSSSourceIndex.GetIndex(), entryPointBit, distanceFromEntryPoint)
+		}
+
+		// Traverse into all imported files
+		for _, record := range repr.AST.ImportRecords {
+			if record.SourceIndex.IsValid() && !c.isExternalDynamicImport(&record, sourceIndex) {
+				c.markFileReachableForCodeSplitting(record.SourceIndex.GetIndex(), entryPointBit, distanceFromEntryPoint)
+			}
+		}
+
+		// Traverse into all dependencies of all parts in this file
+		for _, part := range repr.AST.Parts {
+			for _, dependency := range part.Dependencies {
+				if dependency.SourceIndex != sourceIndex {
+					c.markFileReachableForCodeSplitting(dependency.SourceIndex, entryPointBit, distanceFromEntryPoint)
+				}
+			}
+		}
+
+	case *graph.CSSRepr:
+		// Traverse into all dependencies
+		for _, record := range repr.AST.ImportRecords {
+			if record.SourceIndex.IsValid() {
+				c.markFileReachableForCodeSplitting(record.SourceIndex.GetIndex(), entryPointBit, distanceFromEntryPoint)
+			}
+		}
+	}
+}
+
+func (c *linkerContext) markFileLiveForTreeShaking(sourceIndex uint32) {
+	file := &c.graph.Files[sourceIndex]
+
+	// Don't mark this file more than once
+	if file.IsLive {
+		return
+	}
+	file.IsLive = true
+
+	switch repr := file.InputFile.Repr.(type) {
+	case *graph.JSRepr:
+		// If the JavaScript stub for a CSS file is included, also include the CSS file
+		if repr.CSSSourceIndex.IsValid() {
+			c.markFileLiveForTreeShaking(repr.CSSSourceIndex.GetIndex())
+		}
+
+		for partIndex, part := range repr.AST.Parts {
+			canBeRemovedIfUnused := part.CanBeRemovedIfUnused
+
+			// Also include any statement-level imports
+			for _, importRecordIndex := range part.ImportRecordIndices {
+				record := &repr.AST.ImportRecords[importRecordIndex]
+				if record.Kind != ast.ImportStmt {
+					continue
+				}
+
+				if record.SourceIndex.IsValid() {
+					otherSourceIndex := record.SourceIndex.GetIndex()
+
+					// Don't include this module for its side effects if it can be
+					// considered to have no side effects
+					if otherFile := &c.graph.Files[otherSourceIndex]; otherFile.InputFile.SideEffects.Kind != graph.HasSideEffects && !c.options.IgnoreDCEAnnotations {
+						continue
+					}
+
+					// Otherwise, include this module for its side effects
+					c.markFileLiveForTreeShaking(otherSourceIndex)
+				} else if record.Flags.Has(ast.IsExternalWithoutSideEffects) {
+					// This can be removed if it's unused
+					continue
+				}
+
+				// If we get here then the import was included for its side effects, so
+				// we must also keep this part
+				canBeRemovedIfUnused = false
+			}
+
+			// Include all parts in this file with side effects, or just include
+			// everything if tree-shaking is disabled. Note that we still want to
+			// perform tree-shaking on the runtime even if tree-shaking is disabled.
+			if !canBeRemovedIfUnused || (!part.ForceTreeShaking && !c.options.TreeShaking && file.IsEntryPoint()) {
+				c.markPartLiveForTreeShaking(sourceIndex, uint32(partIndex))
+			}
+		}
+
+	case *graph.CSSRepr:
+		// Include all "@import" rules
+		for _, record := range repr.AST.ImportRecords {
+			if record.SourceIndex.IsValid() {
+				c.markFileLiveForTreeShaking(record.SourceIndex.GetIndex())
+			}
+		}
+	}
+}
+
+func (c *linkerContext) isExternalDynamicImport(record *ast.ImportRecord, sourceIndex uint32) bool {
+	return c.options.CodeSplitting &&
+		record.Kind == ast.ImportDynamic &&
+		c.graph.Files[record.SourceIndex.GetIndex()].IsEntryPoint() &&
+		record.SourceIndex.GetIndex() != sourceIndex
+}
+
+func (c *linkerContext) markPartLiveForTreeShaking(sourceIndex uint32, partIndex uint32) {
+	file := &c.graph.Files[sourceIndex]
+	repr := file.InputFile.Repr.(*graph.JSRepr)
+	part := &repr.AST.Parts[partIndex]
+
+	// Don't mark this part more than once
+	if part.IsLive {
+		return
+	}
+	part.IsLive = true
+
+	// Include the file containing this part
+	c.markFileLiveForTreeShaking(sourceIndex)
+
+	// Also include any dependencies
+	for _, dep := range part.Dependencies {
+		c.markPartLiveForTreeShaking(dep.SourceIndex, dep.PartIndex)
+	}
+}
+
+// JavaScript modules are traversed in depth-first postorder. This is the
+// order that JavaScript modules were evaluated in before the top-level await
+// feature was introduced.
+//
+//	  A
+//	 / \
+//	B   C
+//	 \ /
+//	  D
+//
+// If A imports B and then C, B imports D, and C imports D, then the JavaScript
+// traversal order is D B C A.
+//
+// This function may deviate from ESM import order for dynamic imports (both
+// "require()" and "import()"). This is because the import order is impossible
+// to determine since the imports happen at run-time instead of compile-time.
+// In this case we just pick an arbitrary but consistent order.
+func (c *linkerContext) findImportedCSSFilesInJSOrder(entryPoint uint32) (order []uint32) {
+	visited := make(map[uint32]bool)
+	var visit func(uint32)
+
+	// Include this file and all files it imports
+	visit = func(sourceIndex uint32) {
+		if visited[sourceIndex] {
+			return
+		}
+		visited[sourceIndex] = true
+		file := &c.graph.Files[sourceIndex]
+		repr := file.InputFile.Repr.(*graph.JSRepr)
+
+		// Iterate over each part in the file in order
+		for _, part := range repr.AST.Parts {
+			// Traverse any files imported by this part. Note that CommonJS calls
+			// to "require()" count as imports too, sort of as if the part has an
+			// ESM "import" statement in it. This may seem weird because ESM imports
+			// are a compile-time concept while CommonJS imports are a run-time
+			// concept. But we don't want to manipulate <style> tags at run-time so
+			// this is the only way to do it.
+			for _, importRecordIndex := range part.ImportRecordIndices {
+				if record := &repr.AST.ImportRecords[importRecordIndex]; record.SourceIndex.IsValid() {
+					visit(record.SourceIndex.GetIndex())
+				}
+			}
+		}
+
+		// Iterate over the associated CSS imports in postorder
+		if repr.CSSSourceIndex.IsValid() {
+			order = append(order, repr.CSSSourceIndex.GetIndex())
+		}
+	}
+
+	// Include all files reachable from the entry point
+	visit(entryPoint)
+
+	return
+}
+
+type cssImportKind uint8
+
+const (
+	cssImportNone cssImportKind = iota
+	cssImportSourceIndex
+	cssImportExternalPath
+	cssImportLayers
+)
+
+type cssImportOrder struct {
+	conditions             []css_ast.ImportConditions
+	conditionImportRecords []ast.ImportRecord
+
+	layers       [][]string  // kind == cssImportAtLayer
+	externalPath logger.Path // kind == cssImportExternal
+	sourceIndex  uint32      // kind == cssImportSourceIndex
+
+	kind cssImportKind
+}
+
+// CSS files are traversed in depth-first postorder just like JavaScript. But
+// unlike JavaScript import statements, CSS "@import" rules are evaluated every
+// time instead of just the first time.
+//
+//	  A
+//	 / \
+//	B   C
+//	 \ /
+//	  D
+//
+// If A imports B and then C, B imports D, and C imports D, then the CSS
+// traversal order is D B D C A.
+//
+// However, evaluating a CSS file multiple times is sort of equivalent to
+// evaluating it once at the last location. So we basically drop all but the
+// last evaluation in the order.
+//
+// The only exception to this is "@layer". Evaluating a CSS file multiple
+// times is sort of equivalent to evaluating it once at the first location
+// as far as "@layer" is concerned. So we may in some cases keep both the
+// first and last locations and only write out the "@layer" information
+// for the first location.
+func (c *linkerContext) findImportedFilesInCSSOrder(entryPoints []uint32) (order []cssImportOrder) {
+	var visit func(uint32, []uint32, []css_ast.ImportConditions, []ast.ImportRecord)
+	hasExternalImport := false
+
+	// Include this file and all files it imports
+	visit = func(
+		sourceIndex uint32,
+		visited []uint32,
+		wrappingConditions []css_ast.ImportConditions,
+		wrappingImportRecords []ast.ImportRecord,
+	) {
+		// The CSS specification strangely does not describe what to do when there
+		// is a cycle. So we are left with reverse-engineering the behavior from a
+		// real browser. Here's what the WebKit code base has to say about this:
+		//
+		//   "Check for a cycle in our import chain. If we encounter a stylesheet
+		//   in our parent chain with the same URL, then just bail."
+		//
+		// So that's what we do here. See "StyleRuleImport::requestStyleSheet()" in
+		// WebKit for more information.
+		for _, visitedSourceIndex := range visited {
+			if visitedSourceIndex == sourceIndex {
+				return
+			}
+		}
+		visited = append(visited, sourceIndex)
+
+		repr := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.CSSRepr)
+		topLevelRules := repr.AST.Rules
+
+		// Any pre-import layers come first
+		if len(repr.AST.LayersPreImport) > 0 {
+			order = append(order, cssImportOrder{
+				kind:                   cssImportLayers,
+				layers:                 repr.AST.LayersPreImport,
+				conditions:             wrappingConditions,
+				conditionImportRecords: wrappingImportRecords,
+			})
+		}
+
+		// Iterate over the top-level "@import" rules
+		for _, rule := range topLevelRules {
+			if atImport, ok := rule.Data.(*css_ast.RAtImport); ok {
+				record := &repr.AST.ImportRecords[atImport.ImportRecordIndex]
+
+				// Follow internal dependencies
+				if record.SourceIndex.IsValid() {
+					nestedConditions := wrappingConditions
+					nestedImportRecords := wrappingImportRecords
+
+					// If this import has conditions, fork our state so that the entire
+					// imported stylesheet subtree is wrapped in all of the conditions
+					if atImport.ImportConditions != nil {
+						// Fork our state
+						nestedConditions = append([]css_ast.ImportConditions{}, nestedConditions...)
+						nestedImportRecords = append([]ast.ImportRecord{}, nestedImportRecords...)
+
+						// Clone these import conditions and append them to the state
+						var conditions css_ast.ImportConditions
+						conditions, nestedImportRecords = atImport.ImportConditions.CloneWithImportRecords(repr.AST.ImportRecords, nestedImportRecords)
+						nestedConditions = append(nestedConditions, conditions)
+					}
+
+					visit(record.SourceIndex.GetIndex(), visited, nestedConditions, nestedImportRecords)
+					continue
+				}
+
+				// Record external dependencies
+				if (record.Flags & ast.WasLoadedWithEmptyLoader) == 0 {
+					allConditions := wrappingConditions
+					allImportRecords := wrappingImportRecords
+
+					// If this import has conditions, append it to the list of overall
+					// conditions for this external import. Note that an external import
+					// may actually have multiple sets of conditions that can't be
+					// merged. When this happens we need to generate a nested imported
+					// CSS file using a data URL.
+					if atImport.ImportConditions != nil {
+						var conditions css_ast.ImportConditions
+						allConditions = append([]css_ast.ImportConditions{}, allConditions...)
+						allImportRecords = append([]ast.ImportRecord{}, allImportRecords...)
+						conditions, allImportRecords = atImport.ImportConditions.CloneWithImportRecords(repr.AST.ImportRecords, allImportRecords)
+						allConditions = append(allConditions, conditions)
+					}
+
+					order = append(order, cssImportOrder{
+						kind:                   cssImportExternalPath,
+						externalPath:           record.Path,
+						conditions:             allConditions,
+						conditionImportRecords: allImportRecords,
+					})
+					hasExternalImport = true
+				}
+			}
+		}
+
+		// Iterate over the "composes" directives. Note that the order doesn't
+		// matter for these because the output order is explicitly undefined
+		// in the specification.
+		for _, record := range repr.AST.ImportRecords {
+			if record.Kind == ast.ImportComposesFrom && record.SourceIndex.IsValid() {
+				visit(record.SourceIndex.GetIndex(), visited, wrappingConditions, wrappingImportRecords)
+			}
+		}
+
+		// Accumulate imports in depth-first postorder
+		order = append(order, cssImportOrder{
+			kind:                   cssImportSourceIndex,
+			sourceIndex:            sourceIndex,
+			conditions:             wrappingConditions,
+			conditionImportRecords: wrappingImportRecords,
+		})
+	}
+
+	// Include all files reachable from any entry point
+	var visited [16]uint32 // Preallocate some space for the visited set
+	for _, sourceIndex := range entryPoints {
+		visit(sourceIndex, visited[:], nil, nil)
+	}
+
+	// Create a temporary array that we can use for filtering
+	wipOrder := make([]cssImportOrder, 0, len(order))
+
+	// CSS syntax unfortunately only allows "@import" rules at the top of the
+	// file. This means we must hoist all external "@import" rules to the top of
+	// the file when bundling, even though doing so will change the order of CSS
+	// evaluation.
+	if hasExternalImport {
+		// Pass 1: Pull out leading "@layer" and external "@import" rules
+		isAtLayerPrefix := true
+		for _, entry := range order {
+			if (entry.kind == cssImportLayers && isAtLayerPrefix) || entry.kind == cssImportExternalPath {
+				wipOrder = append(wipOrder, entry)
+			}
+			if entry.kind != cssImportLayers {
+				isAtLayerPrefix = false
+			}
+		}
+
+		// Pass 2: Append everything that we didn't pull out in pass 1
+		isAtLayerPrefix = true
+		for _, entry := range order {
+			if (entry.kind != cssImportLayers || !isAtLayerPrefix) && entry.kind != cssImportExternalPath {
+				wipOrder = append(wipOrder, entry)
+			}
+			if entry.kind != cssImportLayers {
+				isAtLayerPrefix = false
+			}
+		}
+
+		order, wipOrder = wipOrder, order[:0]
+	}
+
+	// Next, optimize import order. If there are duplicate copies of an imported
+	// file, replace all but the last copy with just the layers that are in that
+	// file. This works because in CSS, the last instance of a declaration
+	// overrides all previous instances of that declaration.
+	{
+		sourceIndexDuplicates := make(map[uint32][]int)
+		externalPathDuplicates := make(map[logger.Path][]int)
+
+	nextBackward:
+		for i := len(order) - 1; i >= 0; i-- {
+			entry := order[i]
+			switch entry.kind {
+			case cssImportSourceIndex:
+				duplicates := sourceIndexDuplicates[entry.sourceIndex]
+				for _, j := range duplicates {
+					if isConditionalImportRedundant(entry.conditions, order[j].conditions) {
+						order[i].kind = cssImportLayers
+						order[i].layers = c.graph.Files[entry.sourceIndex].InputFile.Repr.(*graph.CSSRepr).AST.LayersPostImport
+						continue nextBackward
+					}
+				}
+				sourceIndexDuplicates[entry.sourceIndex] = append(duplicates, i)
+
+			case cssImportExternalPath:
+				duplicates := externalPathDuplicates[entry.externalPath]
+				for _, j := range duplicates {
+					if isConditionalImportRedundant(entry.conditions, order[j].conditions) {
+						// Don't remove duplicates entirely. The import conditions may
+						// still introduce layers to the layer order. Represent this as a
+						// file with an empty layer list.
+						order[i].kind = cssImportLayers
+						continue nextBackward
+					}
+				}
+				externalPathDuplicates[entry.externalPath] = append(duplicates, i)
+			}
+		}
+	}
+
+	// Then optimize "@layer" rules by removing redundant ones. This loop goes
+	// forward instead of backward because "@layer" takes effect at the first
+	// copy instead of the last copy like other things in CSS.
+	{
+		type duplicateEntry struct {
+			layers  [][]string
+			indices []int
+		}
+		var layerDuplicates []duplicateEntry
+
+	nextForward:
+		for i := range order {
+			entry := order[i]
+
+			// Simplify the conditions since we know they only wrap "@layer"
+			if entry.kind == cssImportLayers {
+				// Truncate the conditions at the first anonymous layer
+				for i, conditions := range entry.conditions {
+					// The layer is anonymous if it's a "layer" token without any
+					// children instead of a "layer(...)" token with children:
+					//
+					//   /* entry.css */
+					//   @import "foo.css" layer;
+					//
+					//   /* foo.css */
+					//   @layer foo;
+					//
+					// We don't need to generate this (as far as I can tell):
+					//
+					//   @layer {
+					//     @layer foo;
+					//   }
+					//
+					if conditions.Layers != nil && len(conditions.Layers) == 1 && conditions.Layers[0].Children == nil {
+						entry.conditions = entry.conditions[:i]
+						entry.layers = nil
+						break
+					}
+				}
+
+				// If there are no layer names for this file, trim all conditions
+				// without layers because we know they have no effect.
+				//
+				//   /* entry.css */
+				//   @import "foo.css" layer(foo) supports(display: flex);
+				//
+				//   /* foo.css */
+				//   @import "empty.css" supports(display: grid);
+				//
+				// That would result in this:
+				//
+				//   @supports (display: flex) {
+				//     @layer foo {
+				//       @supports (display: grid) {}
+				//     }
+				//   }
+				//
+				// Here we can trim "supports(display: grid)" to generate this:
+				//
+				//   @supports (display: flex) {
+				//     @layer foo;
+				//   }
+				//
+				if len(entry.layers) == 0 {
+					for i := len(entry.conditions) - 1; i >= 0; i-- {
+						if len(entry.conditions[i].Layers) > 0 {
+							break
+						}
+						entry.conditions = entry.conditions[:i]
+					}
+				}
+
+				// Remove unnecessary entries entirely
+				if len(entry.conditions) == 0 && len(entry.layers) == 0 {
+					continue
+				}
+			}
+
+			// Omit redundant "@layer" rules with the same set of layer names. Note
+			// that this tests all import order entries (not just layer ones) because
+			// sometimes non-layer ones can make following layer ones redundant.
+			layersKey := entry.layers
+			if entry.kind == cssImportSourceIndex {
+				layersKey = c.graph.Files[entry.sourceIndex].InputFile.Repr.(*graph.CSSRepr).AST.LayersPostImport
+			}
+			index := 0
+			for index < len(layerDuplicates) {
+				if helpers.StringArrayArraysEqual(layersKey, layerDuplicates[index].layers) {
+					break
+				}
+				index++
+			}
+			if index == len(layerDuplicates) {
+				// This is the first time we've seen this combination of layer names.
+				// Allocate a new set of duplicate indices to track this combination.
+				layerDuplicates = append(layerDuplicates, duplicateEntry{layers: layersKey})
+			}
+			duplicates := layerDuplicates[index].indices
+			for j := len(duplicates) - 1; j >= 0; j-- {
+				if index := duplicates[j]; isConditionalImportRedundant(entry.conditions, wipOrder[index].conditions) {
+					if entry.kind != cssImportLayers {
+						// If an empty layer is followed immediately by a full layer and
+						// everything else is identical, then we don't need to emit the
+						// empty layer. For example:
+						//
+						//   @media screen {
+						//     @supports (display: grid) {
+						//       @layer foo;
+						//     }
+						//   }
+						//   @media screen {
+						//     @supports (display: grid) {
+						//       @layer foo {
+						//         div {
+						//           color: red;
+						//         }
+						//       }
+						//     }
+						//   }
+						//
+						// This can be improved by dropping the empty layer. But we can
+						// only do this if there's nothing in between these two rules.
+						if j == len(duplicates)-1 && index == len(wipOrder)-1 {
+							if other := wipOrder[index]; other.kind == cssImportLayers && importConditionsAreEqual(entry.conditions, other.conditions) {
+								// Remove the previous entry and then overwrite it below
+								duplicates = duplicates[:j]
+								wipOrder = wipOrder[:index]
+								break
+							}
+						}
+
+						// Non-layer entries still need to be present because they have
+						// other side effects beside inserting things in the layer order
+						wipOrder = append(wipOrder, entry)
+					}
+
+					// Don't add this to the duplicate list below because it's redundant
+					continue nextForward
+				}
+			}
+			layerDuplicates[index].indices = append(duplicates, len(wipOrder))
+			wipOrder = append(wipOrder, entry)
+		}
+
+		order, wipOrder = wipOrder, order[:0]
+	}
+
+	// Finally, merge adjacent "@layer" rules with identical conditions together.
+	{
+		didClone := -1
+		for _, entry := range order {
+			if entry.kind == cssImportLayers && len(wipOrder) > 0 {
+				prevIndex := len(wipOrder) - 1
+				prev := wipOrder[prevIndex]
+				if prev.kind == cssImportLayers && importConditionsAreEqual(prev.conditions, entry.conditions) {
+					if didClone != prevIndex {
+						didClone = prevIndex
+						prev.layers = append([][]string{}, prev.layers...)
+					}
+					wipOrder[prevIndex].layers = append(prev.layers, entry.layers...)
+					continue
+				}
+			}
+			wipOrder = append(wipOrder, entry)
+		}
+		order = wipOrder
+	}
+
+	return
+}
+
+func importConditionsAreEqual(a []css_ast.ImportConditions, b []css_ast.ImportConditions) bool {
+	if len(a) != len(b) {
+		return false
+	}
+	for i := 0; i < len(a); i++ {
+		ai := a[i]
+		bi := b[i]
+		if !css_ast.TokensEqualIgnoringWhitespace(ai.Layers, bi.Layers) ||
+			!css_ast.TokensEqualIgnoringWhitespace(ai.Supports, bi.Supports) ||
+			!css_ast.TokensEqualIgnoringWhitespace(ai.Media, bi.Media) {
+			return false
+		}
+	}
+	return true
+}
+
+// Given two "@import" rules for the same source index (an earlier one and a
+// later one), the earlier one is masked by the later one if the later one's
+// condition list is a prefix of the earlier one's condition list.
+//
+// For example:
+//
+//	// entry.css
+//	@import "foo.css" supports(display: flex);
+//	@import "bar.css" supports(display: flex);
+//
+//	// foo.css
+//	@import "lib.css" screen;
+//
+//	// bar.css
+//	@import "lib.css";
+//
+// When we bundle this code we'll get an import order as follows:
+//
+//  1. lib.css [supports(display: flex), screen]
+//  2. foo.css [supports(display: flex)]
+//  3. lib.css [supports(display: flex)]
+//  4. bar.css [supports(display: flex)]
+//  5. entry.css []
+//
+// For "lib.css", the entry with the conditions [supports(display: flex)] should
+// make the entry with the conditions [supports(display: flex), screen] redundant.
+//
+// Note that all of this deliberately ignores the existence of "@layer" because
+// that is handled separately. All of this is only for handling unlayered styles.
+func isConditionalImportRedundant(earlier []css_ast.ImportConditions, later []css_ast.ImportConditions) bool {
+	if len(later) > len(earlier) {
+		return false
+	}
+
+	for i := 0; i < len(later); i++ {
+		a := earlier[i]
+		b := later[i]
+
+		// Only compare "@supports" and "@media" if "@layers" is equal
+		if css_ast.TokensEqualIgnoringWhitespace(a.Layers, b.Layers) {
+			sameSupports := css_ast.TokensEqualIgnoringWhitespace(a.Supports, b.Supports)
+			sameMedia := css_ast.TokensEqualIgnoringWhitespace(a.Media, b.Media)
+
+			// If the import conditions are exactly equal, then only keep
+			// the later one. The earlier one is redundant. Example:
+			//
+			//   @import "foo.css" layer(abc) supports(display: flex) screen;
+			//   @import "foo.css" layer(abc) supports(display: flex) screen;
+			//
+			// The later one makes the earlier one redundant.
+			if sameSupports && sameMedia {
+				continue
+			}
+
+			// If the media conditions are exactly equal and the later one
+			// doesn't have any supports conditions, then the later one will
+			// apply in all cases where the earlier one applies. Example:
+			//
+			//   @import "foo.css" layer(abc) supports(display: flex) screen;
+			//   @import "foo.css" layer(abc) screen;
+			//
+			// The later one makes the earlier one redundant.
+			if sameMedia && len(b.Supports) == 0 {
+				continue
+			}
+
+			// If the supports conditions are exactly equal and the later one
+			// doesn't have any media conditions, then the later one will
+			// apply in all cases where the earlier one applies. Example:
+			//
+			//   @import "foo.css" layer(abc) supports(display: flex) screen;
+			//   @import "foo.css" layer(abc) supports(display: flex);
+			//
+			// The later one makes the earlier one redundant.
+			if sameSupports && len(b.Media) == 0 {
+				continue
+			}
+		}
+
+		return false
+	}
+
+	return true
+}
+
+func (c *linkerContext) computeChunks() {
+	c.timer.Begin("Compute chunks")
+	defer c.timer.End("Compute chunks")
+
+	jsChunks := make(map[string]chunkInfo)
+	cssChunks := make(map[string]chunkInfo)
+
+	// Create chunks for entry points
+	for i, entryPoint := range c.graph.EntryPoints() {
+		file := &c.graph.Files[entryPoint.SourceIndex]
+
+		// Create a chunk for the entry point here to ensure that the chunk is
+		// always generated even if the resulting file is empty
+		entryBits := helpers.NewBitSet(uint(len(c.graph.EntryPoints())))
+		entryBits.SetBit(uint(i))
+		key := entryBits.String()
+		chunk := chunkInfo{
+			entryBits:             entryBits,
+			isEntryPoint:          true,
+			sourceIndex:           entryPoint.SourceIndex,
+			entryPointBit:         uint(i),
+			filesWithPartsInChunk: make(map[uint32]bool),
+		}
+
+		switch file.InputFile.Repr.(type) {
+		case *graph.JSRepr:
+			chunkRepr := &chunkReprJS{}
+			chunk.chunkRepr = chunkRepr
+			jsChunks[key] = chunk
+
+			// If this JS entry point has an associated CSS entry point, generate it
+			// now. This is essentially done by generating a virtual CSS file that
+			// only contains "@import" statements in the order that the files were
+			// discovered in JS source order, where JS source order is arbitrary but
+			// consistent for dynamic imports. Then we run the CSS import order
+			// algorithm to determine the final CSS file order for the chunk.
+
+			if cssSourceIndices := c.findImportedCSSFilesInJSOrder(entryPoint.SourceIndex); len(cssSourceIndices) > 0 {
+				order := c.findImportedFilesInCSSOrder(cssSourceIndices)
+				cssFilesWithPartsInChunk := make(map[uint32]bool)
+				for _, entry := range order {
+					if entry.kind == cssImportSourceIndex {
+						cssFilesWithPartsInChunk[uint32(entry.sourceIndex)] = true
+					}
+				}
+				cssChunks[key] = chunkInfo{
+					entryBits:             entryBits,
+					isEntryPoint:          true,
+					sourceIndex:           entryPoint.SourceIndex,
+					entryPointBit:         uint(i),
+					filesWithPartsInChunk: cssFilesWithPartsInChunk,
+					chunkRepr: &chunkReprCSS{
+						importsInChunkInOrder: order,
+					},
+				}
+				chunkRepr.hasCSSChunk = true
+			}
+
+		case *graph.CSSRepr:
+			order := c.findImportedFilesInCSSOrder([]uint32{entryPoint.SourceIndex})
+			for _, entry := range order {
+				if entry.kind == cssImportSourceIndex {
+					chunk.filesWithPartsInChunk[uint32(entry.sourceIndex)] = true
+				}
+			}
+			chunk.chunkRepr = &chunkReprCSS{
+				importsInChunkInOrder: order,
+			}
+			cssChunks[key] = chunk
+		}
+	}
+
+	// Figure out which JS files are in which chunk
+	for _, sourceIndex := range c.graph.ReachableFiles {
+		if file := &c.graph.Files[sourceIndex]; file.IsLive {
+			if _, ok := file.InputFile.Repr.(*graph.JSRepr); ok {
+				key := file.EntryBits.String()
+				chunk, ok := jsChunks[key]
+				if !ok {
+					chunk.entryBits = file.EntryBits
+					chunk.filesWithPartsInChunk = make(map[uint32]bool)
+					chunk.chunkRepr = &chunkReprJS{}
+					jsChunks[key] = chunk
+				}
+				chunk.filesWithPartsInChunk[uint32(sourceIndex)] = true
+			}
+		}
+	}
+
+	// Sort the chunks for determinism. This matters because we use chunk indices
+	// as sorting keys in a few places.
+	sortedChunks := make([]chunkInfo, 0, len(jsChunks)+len(cssChunks))
+	sortedKeys := make([]string, 0, len(jsChunks)+len(cssChunks))
+	for key := range jsChunks {
+		sortedKeys = append(sortedKeys, key)
+	}
+	sort.Strings(sortedKeys)
+	jsChunkIndicesForCSS := make(map[string]uint32)
+	for _, key := range sortedKeys {
+		chunk := jsChunks[key]
+		if chunk.chunkRepr.(*chunkReprJS).hasCSSChunk {
+			jsChunkIndicesForCSS[key] = uint32(len(sortedChunks))
+		}
+		sortedChunks = append(sortedChunks, chunk)
+	}
+	sortedKeys = sortedKeys[:0]
+	for key := range cssChunks {
+		sortedKeys = append(sortedKeys, key)
+	}
+	sort.Strings(sortedKeys)
+	for _, key := range sortedKeys {
+		chunk := cssChunks[key]
+		if jsChunkIndex, ok := jsChunkIndicesForCSS[key]; ok {
+			sortedChunks[jsChunkIndex].chunkRepr.(*chunkReprJS).cssChunkIndex = uint32(len(sortedChunks))
+		}
+		sortedChunks = append(sortedChunks, chunk)
+	}
+
+	// Map from the entry point file to its chunk. We will need this later if
+	// a file contains a dynamic import to this entry point, since we'll need
+	// to look up the path for this chunk to use with the import.
+	for chunkIndex, chunk := range sortedChunks {
+		if chunk.isEntryPoint {
+			file := &c.graph.Files[chunk.sourceIndex]
+
+			// JS entry points that import CSS files generate two chunks, a JS chunk
+			// and a CSS chunk. Don't link the CSS chunk to the JS file since the CSS
+			// chunk is secondary (the JS chunk is primary).
+			if _, ok := chunk.chunkRepr.(*chunkReprCSS); ok {
+				if _, ok := file.InputFile.Repr.(*graph.JSRepr); ok {
+					continue
+				}
+			}
+
+			file.EntryPointChunkIndex = uint32(chunkIndex)
+		}
+	}
+
+	// Determine the order of JS files (and parts) within the chunk ahead of time
+	for _, chunk := range sortedChunks {
+		if chunkRepr, ok := chunk.chunkRepr.(*chunkReprJS); ok {
+			chunkRepr.filesInChunkInOrder, chunkRepr.partsInChunkInOrder = c.findImportedPartsInJSOrder(&chunk)
+		}
+	}
+
+	// Assign general information to each chunk
+	for chunkIndex := range sortedChunks {
+		chunk := &sortedChunks[chunkIndex]
+
+		// Assign a unique key to each chunk. This key encodes the index directly so
+		// we can easily recover it later without needing to look it up in a map. The
+		// last 8 numbers of the key are the chunk index.
+		chunk.uniqueKey = fmt.Sprintf("%sC%08d", c.uniqueKeyPrefix, chunkIndex)
+
+		// Determine the standard file extension
+		var stdExt string
+		switch chunk.chunkRepr.(type) {
+		case *chunkReprJS:
+			stdExt = c.options.OutputExtensionJS
+		case *chunkReprCSS:
+			stdExt = c.options.OutputExtensionCSS
+		}
+
+		// Compute the template substitutions
+		var dir, base, ext string
+		var template []config.PathTemplate
+		if chunk.isEntryPoint {
+			// Only use the entry path template for user-specified entry points
+			file := &c.graph.Files[chunk.sourceIndex]
+			if file.IsUserSpecifiedEntryPoint() {
+				template = c.options.EntryPathTemplate
+			} else {
+				template = c.options.ChunkPathTemplate
+			}
+
+			if c.options.AbsOutputFile != "" {
+				// If the output path was configured explicitly, use it verbatim
+				dir = "/"
+				base = c.fs.Base(c.options.AbsOutputFile)
+				originalExt := c.fs.Ext(base)
+				base = base[:len(base)-len(originalExt)]
+
+				// Use the extension from the explicit output file path. However, don't do
+				// that if this is a CSS chunk but the entry point file is not CSS. In that
+				// case use the standard extension. This happens when importing CSS into JS.
+				if _, ok := file.InputFile.Repr.(*graph.CSSRepr); ok || stdExt != c.options.OutputExtensionCSS {
+					ext = originalExt
+				} else {
+					ext = stdExt
+				}
+			} else {
+				// Otherwise, derive the output path from the input path
+				dir, base = bundler.PathRelativeToOutbase(
+					&c.graph.Files[chunk.sourceIndex].InputFile,
+					c.options,
+					c.fs,
+					!file.IsUserSpecifiedEntryPoint(),
+					c.graph.EntryPoints()[chunk.entryPointBit].OutputPath,
+				)
+				ext = stdExt
+			}
+		} else {
+			dir = "/"
+			base = "chunk"
+			ext = stdExt
+			template = c.options.ChunkPathTemplate
+		}
+
+		// Determine the output path template
+		templateExt := strings.TrimPrefix(ext, ".")
+		template = append(append(make([]config.PathTemplate, 0, len(template)+1), template...), config.PathTemplate{Data: ext})
+		chunk.finalTemplate = config.SubstituteTemplate(template, config.PathPlaceholders{
+			Dir:  &dir,
+			Name: &base,
+			Ext:  &templateExt,
+		})
+	}
+
+	c.chunks = sortedChunks
+}
+
+type chunkOrder struct {
+	sourceIndex uint32
+	distance    uint32
+	tieBreaker  uint32
+}
+
+// This type is just so we can use Go's native sort function
+type chunkOrderArray []chunkOrder
+
+func (a chunkOrderArray) Len() int          { return len(a) }
+func (a chunkOrderArray) Swap(i int, j int) { a[i], a[j] = a[j], a[i] }
+
+func (a chunkOrderArray) Less(i int, j int) bool {
+	ai := a[i]
+	aj := a[j]
+	return ai.distance < aj.distance || (ai.distance == aj.distance && ai.tieBreaker < aj.tieBreaker)
+}
+
+func appendOrExtendPartRange(ranges []partRange, sourceIndex uint32, partIndex uint32) []partRange {
+	if i := len(ranges) - 1; i >= 0 {
+		if r := &ranges[i]; r.sourceIndex == sourceIndex && r.partIndexEnd == partIndex {
+			r.partIndexEnd = partIndex + 1
+			return ranges
+		}
+	}
+
+	return append(ranges, partRange{
+		sourceIndex:    sourceIndex,
+		partIndexBegin: partIndex,
+		partIndexEnd:   partIndex + 1,
+	})
+}
+
+func (c *linkerContext) shouldIncludePart(repr *graph.JSRepr, part js_ast.Part) bool {
+	// As an optimization, ignore parts containing a single import statement to
+	// an internal non-wrapped file. These will be ignored anyway and it's a
+	// performance hit to spin up a goroutine only to discover this later.
+	if len(part.Stmts) == 1 {
+		if s, ok := part.Stmts[0].Data.(*js_ast.SImport); ok {
+			record := &repr.AST.ImportRecords[s.ImportRecordIndex]
+			if record.SourceIndex.IsValid() && c.graph.Files[record.SourceIndex.GetIndex()].InputFile.Repr.(*graph.JSRepr).Meta.Wrap == graph.WrapNone {
+				return false
+			}
+		}
+	}
+	return true
+}
+
+func (c *linkerContext) findImportedPartsInJSOrder(chunk *chunkInfo) (js []uint32, jsParts []partRange) {
+	sorted := make(chunkOrderArray, 0, len(chunk.filesWithPartsInChunk))
+
+	// Attach information to the files for use with sorting
+	for sourceIndex := range chunk.filesWithPartsInChunk {
+		file := &c.graph.Files[sourceIndex]
+		sorted = append(sorted, chunkOrder{
+			sourceIndex: sourceIndex,
+			distance:    file.DistanceFromEntryPoint,
+			tieBreaker:  c.graph.StableSourceIndices[sourceIndex],
+		})
+	}
+
+	// Sort so files closest to an entry point come first. If two files are
+	// equidistant to an entry point, then break the tie by sorting on the
+	// stable source index derived from the DFS over all entry points.
+	sort.Sort(sorted)
+
+	visited := make(map[uint32]bool)
+	jsPartsPrefix := []partRange{}
+
+	// Traverse the graph using this stable order and linearize the files with
+	// dependencies before dependents
+	var visit func(uint32)
+	visit = func(sourceIndex uint32) {
+		if visited[sourceIndex] {
+			return
+		}
+
+		visited[sourceIndex] = true
+		file := &c.graph.Files[sourceIndex]
+
+		if repr, ok := file.InputFile.Repr.(*graph.JSRepr); ok {
+			isFileInThisChunk := chunk.entryBits.Equals(file.EntryBits)
+
+			// Wrapped files can't be split because they are all inside the wrapper
+			canFileBeSplit := repr.Meta.Wrap == graph.WrapNone
+
+			// Make sure the generated call to "__export(exports, ...)" comes first
+			// before anything else in this file
+			if canFileBeSplit && isFileInThisChunk && repr.AST.Parts[js_ast.NSExportPartIndex].IsLive {
+				jsParts = appendOrExtendPartRange(jsParts, sourceIndex, js_ast.NSExportPartIndex)
+			}
+
+			for partIndex, part := range repr.AST.Parts {
+				isPartInThisChunk := isFileInThisChunk && repr.AST.Parts[partIndex].IsLive
+
+				// Also traverse any files imported by this part
+				for _, importRecordIndex := range part.ImportRecordIndices {
+					record := &repr.AST.ImportRecords[importRecordIndex]
+					if record.SourceIndex.IsValid() && (record.Kind == ast.ImportStmt || isPartInThisChunk) {
+						if c.isExternalDynamicImport(record, sourceIndex) {
+							// Don't follow import() dependencies
+							continue
+						}
+						visit(record.SourceIndex.GetIndex())
+					}
+				}
+
+				// Then include this part after the files it imports
+				if isPartInThisChunk {
+					isFileInThisChunk = true
+					if canFileBeSplit && uint32(partIndex) != js_ast.NSExportPartIndex && c.shouldIncludePart(repr, part) {
+						if sourceIndex == runtime.SourceIndex {
+							jsPartsPrefix = appendOrExtendPartRange(jsPartsPrefix, sourceIndex, uint32(partIndex))
+						} else {
+							jsParts = appendOrExtendPartRange(jsParts, sourceIndex, uint32(partIndex))
+						}
+					}
+				}
+			}
+
+			if isFileInThisChunk {
+				js = append(js, sourceIndex)
+
+				// CommonJS files are all-or-nothing so all parts must be contiguous
+				if !canFileBeSplit {
+					jsPartsPrefix = append(jsPartsPrefix, partRange{
+						sourceIndex:    sourceIndex,
+						partIndexBegin: 0,
+						partIndexEnd:   uint32(len(repr.AST.Parts)),
+					})
+				}
+			}
+		}
+	}
+
+	// Always put the runtime code first before anything else
+	visit(runtime.SourceIndex)
+	for _, data := range sorted {
+		visit(data.sourceIndex)
+	}
+	jsParts = append(jsPartsPrefix, jsParts...)
+	return
+}
+
+func (c *linkerContext) shouldRemoveImportExportStmt(
+	sourceIndex uint32,
+	stmtList *stmtList,
+	loc logger.Loc,
+	namespaceRef ast.Ref,
+	importRecordIndex uint32,
+) bool {
+	repr := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr)
+	record := &repr.AST.ImportRecords[importRecordIndex]
+
+	// Is this an external import?
+	if !record.SourceIndex.IsValid() {
+		// Keep the "import" statement if "import" statements are supported
+		if c.options.OutputFormat.KeepESMImportExportSyntax() {
+			return false
+		}
+
+		// Otherwise, replace this statement with a call to "require()"
+		stmtList.insideWrapperPrefix = append(stmtList.insideWrapperPrefix, js_ast.Stmt{
+			Loc: loc,
+			Data: &js_ast.SLocal{Decls: []js_ast.Decl{{
+				Binding: js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: namespaceRef}},
+				ValueOrNil: js_ast.Expr{Loc: record.Range.Loc, Data: &js_ast.ERequireString{
+					ImportRecordIndex: importRecordIndex,
+				}},
+			}}},
+		})
+		return true
+	}
+
+	// We don't need a call to "require()" if this is a self-import inside a
+	// CommonJS-style module, since we can just reference the exports directly.
+	if repr.AST.ExportsKind == js_ast.ExportsCommonJS && ast.FollowSymbols(c.graph.Symbols, namespaceRef) == repr.AST.ExportsRef {
+		return true
+	}
+
+	otherFile := &c.graph.Files[record.SourceIndex.GetIndex()]
+	otherRepr := otherFile.InputFile.Repr.(*graph.JSRepr)
+	switch otherRepr.Meta.Wrap {
+	case graph.WrapNone:
+		// Remove the statement entirely if this module is not wrapped
+
+	case graph.WrapCJS:
+		// Replace the statement with a call to "require()"
+		stmtList.insideWrapperPrefix = append(stmtList.insideWrapperPrefix, js_ast.Stmt{
+			Loc: loc,
+			Data: &js_ast.SLocal{Decls: []js_ast.Decl{{
+				Binding: js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: namespaceRef}},
+				ValueOrNil: js_ast.Expr{Loc: record.Range.Loc, Data: &js_ast.ERequireString{
+					ImportRecordIndex: importRecordIndex,
+				}},
+			}}},
+		})
+
+	case graph.WrapESM:
+		// Ignore this file if it's not included in the bundle. This can happen for
+		// wrapped ESM files but not for wrapped CommonJS files because we allow
+		// tree shaking inside wrapped ESM files.
+		if !otherFile.IsLive {
+			break
+		}
+
+		// Replace the statement with a call to "init()"
+		value := js_ast.Expr{Loc: loc, Data: &js_ast.ECall{Target: js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: otherRepr.AST.WrapperRef}}}}
+		if otherRepr.Meta.IsAsyncOrHasAsyncDependency {
+			// This currently evaluates sibling dependencies in serial instead of in
+			// parallel, which is incorrect. This should be changed to store a promise
+			// and await all stored promises after all imports but before any code.
+			value.Data = &js_ast.EAwait{Value: value}
+		}
+		stmtList.insideWrapperPrefix = append(stmtList.insideWrapperPrefix, js_ast.Stmt{Loc: loc, Data: &js_ast.SExpr{Value: value}})
+	}
+
+	return true
+}
+
+func (c *linkerContext) convertStmtsForChunk(sourceIndex uint32, stmtList *stmtList, partStmts []js_ast.Stmt) {
+	file := &c.graph.Files[sourceIndex]
+	shouldStripExports := c.options.Mode != config.ModePassThrough || !file.IsEntryPoint()
+	repr := file.InputFile.Repr.(*graph.JSRepr)
+	shouldExtractESMStmtsForWrap := repr.Meta.Wrap != graph.WrapNone
+
+	// If this file is a CommonJS entry point, double-write re-exports to the
+	// external CommonJS "module.exports" object in addition to our internal ESM
+	// export namespace object. The difference between these two objects is that
+	// our internal one must not have the "__esModule" marker while the external
+	// one must have the "__esModule" marker. This is done because an ES module
+	// importing itself should not see the "__esModule" marker but a CommonJS module
+	// importing us should see the "__esModule" marker.
+	var moduleExportsForReExportOrNil js_ast.Expr
+	if c.options.OutputFormat == config.FormatCommonJS && file.IsEntryPoint() {
+		moduleExportsForReExportOrNil = js_ast.Expr{Data: &js_ast.EDot{
+			Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: c.unboundModuleRef}},
+			Name:   "exports",
+		}}
+	}
+
+	for _, stmt := range partStmts {
+		switch s := stmt.Data.(type) {
+		case *js_ast.SImport:
+			// "import * as ns from 'path'"
+			// "import {foo} from 'path'"
+			if c.shouldRemoveImportExportStmt(sourceIndex, stmtList, stmt.Loc, s.NamespaceRef, s.ImportRecordIndex) {
+				continue
+			}
+
+			if c.options.UnsupportedJSFeatures.Has(compat.ArbitraryModuleNamespaceNames) && s.Items != nil {
+				for _, item := range *s.Items {
+					c.maybeForbidArbitraryModuleNamespaceIdentifier("import", sourceIndex, item.AliasLoc, item.Alias)
+				}
+			}
+
+			// Make sure these don't end up in the wrapper closure
+			if shouldExtractESMStmtsForWrap {
+				stmtList.outsideWrapperPrefix = append(stmtList.outsideWrapperPrefix, stmt)
+				continue
+			}
+
+		case *js_ast.SExportStar:
+			// "export * as ns from 'path'"
+			if s.Alias != nil {
+				if c.shouldRemoveImportExportStmt(sourceIndex, stmtList, stmt.Loc, s.NamespaceRef, s.ImportRecordIndex) {
+					continue
+				}
+
+				if c.options.UnsupportedJSFeatures.Has(compat.ArbitraryModuleNamespaceNames) {
+					c.maybeForbidArbitraryModuleNamespaceIdentifier("export", sourceIndex, s.Alias.Loc, s.Alias.OriginalName)
+				}
+
+				if shouldStripExports {
+					// Turn this statement into "import * as ns from 'path'"
+					stmt.Data = &js_ast.SImport{
+						NamespaceRef:      s.NamespaceRef,
+						StarNameLoc:       &s.Alias.Loc,
+						ImportRecordIndex: s.ImportRecordIndex,
+					}
+				}
+
+				// Make sure these don't end up in the wrapper closure
+				if shouldExtractESMStmtsForWrap {
+					stmtList.outsideWrapperPrefix = append(stmtList.outsideWrapperPrefix, stmt)
+					continue
+				}
+				break
+			}
+
+			// "export * from 'path'"
+			if !shouldStripExports {
+				break
+			}
+			record := &repr.AST.ImportRecords[s.ImportRecordIndex]
+
+			// Is this export star evaluated at run time?
+			if !record.SourceIndex.IsValid() && c.options.OutputFormat.KeepESMImportExportSyntax() {
+				if record.Flags.Has(ast.CallsRunTimeReExportFn) {
+					// Turn this statement into "import * as ns from 'path'"
+					stmt.Data = &js_ast.SImport{
+						NamespaceRef:      s.NamespaceRef,
+						StarNameLoc:       &logger.Loc{Start: stmt.Loc.Start},
+						ImportRecordIndex: s.ImportRecordIndex,
+					}
+
+					// Prefix this module with "__reExport(exports, ns, module.exports)"
+					exportStarRef := c.graph.Files[runtime.SourceIndex].InputFile.Repr.(*graph.JSRepr).AST.ModuleScope.Members["__reExport"].Ref
+					args := []js_ast.Expr{
+						{Loc: stmt.Loc, Data: &js_ast.EIdentifier{Ref: repr.AST.ExportsRef}},
+						{Loc: stmt.Loc, Data: &js_ast.EIdentifier{Ref: s.NamespaceRef}},
+					}
+					if moduleExportsForReExportOrNil.Data != nil {
+						args = append(args, moduleExportsForReExportOrNil)
+					}
+					stmtList.insideWrapperPrefix = append(stmtList.insideWrapperPrefix, js_ast.Stmt{
+						Loc: stmt.Loc,
+						Data: &js_ast.SExpr{Value: js_ast.Expr{Loc: stmt.Loc, Data: &js_ast.ECall{
+							Target: js_ast.Expr{Loc: stmt.Loc, Data: &js_ast.EIdentifier{Ref: exportStarRef}},
+							Args:   args,
+						}}},
+					})
+
+					// Make sure these don't end up in the wrapper closure
+					if shouldExtractESMStmtsForWrap {
+						stmtList.outsideWrapperPrefix = append(stmtList.outsideWrapperPrefix, stmt)
+						continue
+					}
+				}
+			} else {
+				if record.SourceIndex.IsValid() {
+					if otherRepr := c.graph.Files[record.SourceIndex.GetIndex()].InputFile.Repr.(*graph.JSRepr); otherRepr.Meta.Wrap == graph.WrapESM {
+						stmtList.insideWrapperPrefix = append(stmtList.insideWrapperPrefix, js_ast.Stmt{Loc: stmt.Loc,
+							Data: &js_ast.SExpr{Value: js_ast.Expr{Loc: stmt.Loc, Data: &js_ast.ECall{
+								Target: js_ast.Expr{Loc: stmt.Loc, Data: &js_ast.EIdentifier{Ref: otherRepr.AST.WrapperRef}}}}}})
+					}
+				}
+
+				if record.Flags.Has(ast.CallsRunTimeReExportFn) {
+					var target js_ast.E
+					if record.SourceIndex.IsValid() {
+						if otherRepr := c.graph.Files[record.SourceIndex.GetIndex()].InputFile.Repr.(*graph.JSRepr); otherRepr.AST.ExportsKind == js_ast.ExportsESMWithDynamicFallback {
+							// Prefix this module with "__reExport(exports, otherExports, module.exports)"
+							target = &js_ast.EIdentifier{Ref: otherRepr.AST.ExportsRef}
+						}
+					}
+					if target == nil {
+						// Prefix this module with "__reExport(exports, require(path), module.exports)"
+						target = &js_ast.ERequireString{
+							ImportRecordIndex: s.ImportRecordIndex,
+						}
+					}
+					exportStarRef := c.graph.Files[runtime.SourceIndex].InputFile.Repr.(*graph.JSRepr).AST.ModuleScope.Members["__reExport"].Ref
+					args := []js_ast.Expr{
+						{Loc: stmt.Loc, Data: &js_ast.EIdentifier{Ref: repr.AST.ExportsRef}},
+						{Loc: record.Range.Loc, Data: target},
+					}
+					if moduleExportsForReExportOrNil.Data != nil {
+						args = append(args, moduleExportsForReExportOrNil)
+					}
+					stmtList.insideWrapperPrefix = append(stmtList.insideWrapperPrefix, js_ast.Stmt{
+						Loc: stmt.Loc,
+						Data: &js_ast.SExpr{Value: js_ast.Expr{Loc: stmt.Loc, Data: &js_ast.ECall{
+							Target: js_ast.Expr{Loc: stmt.Loc, Data: &js_ast.EIdentifier{Ref: exportStarRef}},
+							Args:   args,
+						}}},
+					})
+				}
+
+				// Remove the export star statement
+				continue
+			}
+
+		case *js_ast.SExportFrom:
+			// "export {foo} from 'path'"
+			if c.shouldRemoveImportExportStmt(sourceIndex, stmtList, stmt.Loc, s.NamespaceRef, s.ImportRecordIndex) {
+				continue
+			}
+
+			if c.options.UnsupportedJSFeatures.Has(compat.ArbitraryModuleNamespaceNames) {
+				for _, item := range s.Items {
+					c.maybeForbidArbitraryModuleNamespaceIdentifier("export", sourceIndex, item.AliasLoc, item.Alias)
+					if item.AliasLoc != item.Name.Loc {
+						c.maybeForbidArbitraryModuleNamespaceIdentifier("import", sourceIndex, item.Name.Loc, item.OriginalName)
+					}
+				}
+			}
+
+			if shouldStripExports {
+				// Turn this statement into "import {foo} from 'path'"
+				for i, item := range s.Items {
+					s.Items[i].Alias = item.OriginalName
+				}
+				stmt.Data = &js_ast.SImport{
+					NamespaceRef:      s.NamespaceRef,
+					Items:             &s.Items,
+					ImportRecordIndex: s.ImportRecordIndex,
+					IsSingleLine:      s.IsSingleLine,
+				}
+			}
+
+			// Make sure these don't end up in the wrapper closure
+			if shouldExtractESMStmtsForWrap {
+				stmtList.outsideWrapperPrefix = append(stmtList.outsideWrapperPrefix, stmt)
+				continue
+			}
+
+		case *js_ast.SExportClause:
+			if shouldStripExports {
+				// Remove export statements entirely
+				continue
+			}
+
+			if c.options.UnsupportedJSFeatures.Has(compat.ArbitraryModuleNamespaceNames) {
+				for _, item := range s.Items {
+					c.maybeForbidArbitraryModuleNamespaceIdentifier("export", sourceIndex, item.AliasLoc, item.Alias)
+				}
+			}
+
+			// Make sure these don't end up in the wrapper closure
+			if shouldExtractESMStmtsForWrap {
+				stmtList.outsideWrapperPrefix = append(stmtList.outsideWrapperPrefix, stmt)
+				continue
+			}
+
+		case *js_ast.SFunction:
+			// Strip the "export" keyword while bundling
+			if shouldStripExports && s.IsExport {
+				// Be careful to not modify the original statement
+				clone := *s
+				clone.IsExport = false
+				stmt.Data = &clone
+			}
+
+		case *js_ast.SClass:
+			if shouldStripExports && s.IsExport {
+				// Be careful to not modify the original statement
+				clone := *s
+				clone.IsExport = false
+				stmt.Data = &clone
+			}
+
+		case *js_ast.SLocal:
+			if shouldStripExports && s.IsExport {
+				// Be careful to not modify the original statement
+				clone := *s
+				clone.IsExport = false
+				stmt.Data = &clone
+			}
+
+		case *js_ast.SExportDefault:
+			// If we're bundling, convert "export default" into a normal declaration
+			if shouldStripExports {
+				switch s2 := s.Value.Data.(type) {
+				case *js_ast.SExpr:
+					// "export default foo;" => "var default = foo;"
+					stmt = js_ast.Stmt{Loc: stmt.Loc, Data: &js_ast.SLocal{Decls: []js_ast.Decl{
+						{Binding: js_ast.Binding{Loc: s.DefaultName.Loc, Data: &js_ast.BIdentifier{Ref: s.DefaultName.Ref}}, ValueOrNil: s2.Value},
+					}}}
+
+				case *js_ast.SFunction:
+					// "export default function() {}" => "function default() {}"
+					// "export default function foo() {}" => "function foo() {}"
+
+					// Be careful to not modify the original statement
+					s2 = &js_ast.SFunction{Fn: s2.Fn}
+					s2.Fn.Name = &s.DefaultName
+
+					stmt = js_ast.Stmt{Loc: s.Value.Loc, Data: s2}
+
+				case *js_ast.SClass:
+					// "export default class {}" => "class default {}"
+					// "export default class Foo {}" => "class Foo {}"
+
+					// Be careful to not modify the original statement
+					s2 = &js_ast.SClass{Class: s2.Class}
+					s2.Class.Name = &s.DefaultName
+
+					stmt = js_ast.Stmt{Loc: s.Value.Loc, Data: s2}
+
+				default:
+					panic("Internal error")
+				}
+			}
+		}
+
+		stmtList.insideWrapperSuffix = append(stmtList.insideWrapperSuffix, stmt)
+	}
+}
+
+// "var a = 1; var b = 2;" => "var a = 1, b = 2;"
+func mergeAdjacentLocalStmts(stmts []js_ast.Stmt) []js_ast.Stmt {
+	if len(stmts) == 0 {
+		return stmts
+	}
+
+	didMergeWithPreviousLocal := false
+	end := 1
+
+	for _, stmt := range stmts[1:] {
+		// Try to merge with the previous variable statement
+		if after, ok := stmt.Data.(*js_ast.SLocal); ok {
+			if before, ok := stmts[end-1].Data.(*js_ast.SLocal); ok {
+				// It must be the same kind of variable statement (i.e. let/var/const)
+				if before.Kind == after.Kind && before.IsExport == after.IsExport {
+					if didMergeWithPreviousLocal {
+						// Avoid O(n^2) behavior for repeated variable declarations
+						before.Decls = append(before.Decls, after.Decls...)
+					} else {
+						// Be careful to not modify the original statement
+						didMergeWithPreviousLocal = true
+						clone := *before
+						clone.Decls = make([]js_ast.Decl, 0, len(before.Decls)+len(after.Decls))
+						clone.Decls = append(clone.Decls, before.Decls...)
+						clone.Decls = append(clone.Decls, after.Decls...)
+						stmts[end-1].Data = &clone
+					}
+					continue
+				}
+			}
+		}
+
+		// Otherwise, append a normal statement
+		didMergeWithPreviousLocal = false
+		stmts[end] = stmt
+		end++
+	}
+
+	return stmts[:end]
+}
+
+type stmtList struct {
+	// These statements come first, and can be inside the wrapper
+	insideWrapperPrefix []js_ast.Stmt
+
+	// These statements come last, and can be inside the wrapper
+	insideWrapperSuffix []js_ast.Stmt
+
+	outsideWrapperPrefix []js_ast.Stmt
+}
+
+type compileResultJS struct {
+	js_printer.PrintResult
+
+	sourceIndex uint32
+
+	// This is the line and column offset since the previous JavaScript string
+	// or the start of the file if this is the first JavaScript string.
+	generatedOffset sourcemap.LineColumnOffset
+}
+
+func (c *linkerContext) requireOrImportMetaForSource(sourceIndex uint32) (meta js_printer.RequireOrImportMeta) {
+	repr := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr)
+	meta.WrapperRef = repr.AST.WrapperRef
+	meta.IsWrapperAsync = repr.Meta.IsAsyncOrHasAsyncDependency
+	if repr.Meta.Wrap == graph.WrapESM {
+		meta.ExportsRef = repr.AST.ExportsRef
+	} else {
+		meta.ExportsRef = ast.InvalidRef
+	}
+	return
+}
+
+func (c *linkerContext) generateCodeForFileInChunkJS(
+	r renamer.Renamer,
+	waitGroup *sync.WaitGroup,
+	partRange partRange,
+	toCommonJSRef ast.Ref,
+	toESMRef ast.Ref,
+	runtimeRequireRef ast.Ref,
+	result *compileResultJS,
+	dataForSourceMaps []bundler.DataForSourceMap,
+) {
+	defer c.recoverInternalError(waitGroup, partRange.sourceIndex)
+
+	file := &c.graph.Files[partRange.sourceIndex]
+	repr := file.InputFile.Repr.(*graph.JSRepr)
+	nsExportPartIndex := js_ast.NSExportPartIndex
+	needsWrapper := false
+	stmtList := stmtList{}
+
+	// The top-level directive must come first (the non-wrapped case is handled
+	// by the chunk generation code, although only for the entry point)
+	if repr.Meta.Wrap != graph.WrapNone && !file.IsEntryPoint() {
+		for _, directive := range repr.AST.Directives {
+			stmtList.insideWrapperPrefix = append(stmtList.insideWrapperPrefix, js_ast.Stmt{
+				Data: &js_ast.SDirective{Value: helpers.StringToUTF16(directive)},
+			})
+		}
+	}
+
+	// Make sure the generated call to "__export(exports, ...)" comes first
+	// before anything else.
+	if nsExportPartIndex >= partRange.partIndexBegin && nsExportPartIndex < partRange.partIndexEnd &&
+		repr.AST.Parts[nsExportPartIndex].IsLive {
+		c.convertStmtsForChunk(partRange.sourceIndex, &stmtList, repr.AST.Parts[nsExportPartIndex].Stmts)
+
+		// Move everything to the prefix list
+		if repr.Meta.Wrap == graph.WrapESM {
+			stmtList.outsideWrapperPrefix = append(stmtList.outsideWrapperPrefix, stmtList.insideWrapperSuffix...)
+		} else {
+			stmtList.insideWrapperPrefix = append(stmtList.insideWrapperPrefix, stmtList.insideWrapperSuffix...)
+		}
+		stmtList.insideWrapperSuffix = nil
+	}
+
+	var partIndexForLazyDefaultExport ast.Index32
+	if repr.AST.HasLazyExport {
+		if defaultExport, ok := repr.Meta.ResolvedExports["default"]; ok {
+			partIndexForLazyDefaultExport = ast.MakeIndex32(repr.TopLevelSymbolToParts(defaultExport.Ref)[0])
+		}
+	}
+
+	// Add all other parts in this chunk
+	for partIndex := partRange.partIndexBegin; partIndex < partRange.partIndexEnd; partIndex++ {
+		part := repr.AST.Parts[partIndex]
+		if !repr.AST.Parts[partIndex].IsLive {
+			// Skip the part if it's not in this chunk
+			continue
+		}
+
+		if uint32(partIndex) == nsExportPartIndex {
+			// Skip the generated call to "__export()" that was extracted above
+			continue
+		}
+
+		// Mark if we hit the dummy part representing the wrapper
+		if uint32(partIndex) == repr.Meta.WrapperPartIndex.GetIndex() {
+			needsWrapper = true
+			continue
+		}
+
+		stmts := part.Stmts
+
+		// If this could be a JSON file that exports a top-level object literal, go
+		// over the non-default top-level properties that ended up being imported
+		// and substitute references to them into the main top-level object literal.
+		// So this JSON file:
+		//
+		//   {
+		//     "foo": [1, 2, 3],
+		//     "bar": [4, 5, 6],
+		//   }
+		//
+		// is initially compiled into this:
+		//
+		//   export var foo = [1, 2, 3];
+		//   export var bar = [4, 5, 6];
+		//   export default {
+		//     foo: [1, 2, 3],
+		//     bar: [4, 5, 6],
+		//   };
+		//
+		// But we turn it into this if both "foo" and "default" are imported:
+		//
+		//   export var foo = [1, 2, 3];
+		//   export default {
+		//     foo,
+		//     bar: [4, 5, 6],
+		//   };
+		//
+		if partIndexForLazyDefaultExport.IsValid() && partIndex == partIndexForLazyDefaultExport.GetIndex() {
+			stmt := stmts[0]
+			defaultExport := stmt.Data.(*js_ast.SExportDefault)
+			defaultExpr := defaultExport.Value.Data.(*js_ast.SExpr)
+
+			// Be careful: the top-level value in a JSON file is not necessarily an object
+			if object, ok := defaultExpr.Value.Data.(*js_ast.EObject); ok {
+				objectClone := *object
+				objectClone.Properties = append([]js_ast.Property{}, objectClone.Properties...)
+
+				// If any top-level properties ended up being imported directly, change
+				// the property to just reference the corresponding variable instead
+				for i, property := range object.Properties {
+					if str, ok := property.Key.Data.(*js_ast.EString); ok {
+						if name := helpers.UTF16ToString(str.Value); name != "default" {
+							if export, ok := repr.Meta.ResolvedExports[name]; ok {
+								if part := repr.AST.Parts[repr.TopLevelSymbolToParts(export.Ref)[0]]; part.IsLive {
+									ref := part.Stmts[0].Data.(*js_ast.SLocal).Decls[0].Binding.Data.(*js_ast.BIdentifier).Ref
+									objectClone.Properties[i].ValueOrNil = js_ast.Expr{Loc: property.Key.Loc, Data: &js_ast.EIdentifier{Ref: ref}}
+								}
+							}
+						}
+					}
+				}
+
+				// Avoid mutating the original AST
+				defaultExprClone := *defaultExpr
+				defaultExprClone.Value.Data = &objectClone
+				defaultExportClone := *defaultExport
+				defaultExportClone.Value.Data = &defaultExprClone
+				stmt.Data = &defaultExportClone
+				stmts = []js_ast.Stmt{stmt}
+			}
+		}
+
+		c.convertStmtsForChunk(partRange.sourceIndex, &stmtList, stmts)
+	}
+
+	// Hoist all import statements before any normal statements. ES6 imports
+	// are different than CommonJS imports. All modules imported via ES6 import
+	// statements are evaluated before the module doing the importing is
+	// evaluated (well, except for cyclic import scenarios). We need to preserve
+	// these semantics even when modules imported via ES6 import statements end
+	// up being CommonJS modules.
+	stmts := stmtList.insideWrapperSuffix
+	if len(stmtList.insideWrapperPrefix) > 0 {
+		stmts = append(stmtList.insideWrapperPrefix, stmts...)
+	}
+	if c.options.MinifySyntax {
+		stmts = mergeAdjacentLocalStmts(stmts)
+	}
+
+	// Optionally wrap all statements in a closure
+	if needsWrapper {
+		switch repr.Meta.Wrap {
+		case graph.WrapCJS:
+			// Only include the arguments that are actually used
+			args := []js_ast.Arg{}
+			if repr.AST.UsesExportsRef || repr.AST.UsesModuleRef {
+				args = append(args, js_ast.Arg{Binding: js_ast.Binding{Data: &js_ast.BIdentifier{Ref: repr.AST.ExportsRef}}})
+				if repr.AST.UsesModuleRef {
+					args = append(args, js_ast.Arg{Binding: js_ast.Binding{Data: &js_ast.BIdentifier{Ref: repr.AST.ModuleRef}}})
+				}
+			}
+
+			var cjsArgs []js_ast.Expr
+			if c.options.ProfilerNames {
+				// "__commonJS({ 'file.js'(exports, module) { ... } })"
+				kind := js_ast.PropertyField
+				if !c.options.UnsupportedJSFeatures.Has(compat.ObjectExtensions) {
+					kind = js_ast.PropertyMethod
+				}
+				cjsArgs = []js_ast.Expr{{Data: &js_ast.EObject{Properties: []js_ast.Property{{
+					Kind:       kind,
+					Key:        js_ast.Expr{Data: &js_ast.EString{Value: helpers.StringToUTF16(file.InputFile.Source.PrettyPath)}},
+					ValueOrNil: js_ast.Expr{Data: &js_ast.EFunction{Fn: js_ast.Fn{Args: args, Body: js_ast.FnBody{Block: js_ast.SBlock{Stmts: stmts}}}}},
+				}}}}}
+			} else if c.options.UnsupportedJSFeatures.Has(compat.Arrow) {
+				// "__commonJS(function (exports, module) { ... })"
+				cjsArgs = []js_ast.Expr{{Data: &js_ast.EFunction{Fn: js_ast.Fn{Args: args, Body: js_ast.FnBody{Block: js_ast.SBlock{Stmts: stmts}}}}}}
+			} else {
+				// "__commonJS((exports, module) => { ... })"
+				cjsArgs = []js_ast.Expr{{Data: &js_ast.EArrow{Args: args, Body: js_ast.FnBody{Block: js_ast.SBlock{Stmts: stmts}}}}}
+			}
+			value := js_ast.Expr{Data: &js_ast.ECall{
+				Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: c.cjsRuntimeRef}},
+				Args:   cjsArgs,
+			}}
+
+			// "var require_foo = __commonJS(...);"
+			stmts = append(stmtList.outsideWrapperPrefix, js_ast.Stmt{Data: &js_ast.SLocal{
+				Decls: []js_ast.Decl{{
+					Binding:    js_ast.Binding{Data: &js_ast.BIdentifier{Ref: repr.AST.WrapperRef}},
+					ValueOrNil: value,
+				}},
+			}})
+
+		case graph.WrapESM:
+			// The wrapper only needs to be "async" if there is a transitive async
+			// dependency. For correctness, we must not use "async" if the module
+			// isn't async because then calling "require()" on that module would
+			// swallow any exceptions thrown during module initialization.
+			isAsync := repr.Meta.IsAsyncOrHasAsyncDependency
+
+			// Hoist all top-level "var" and "function" declarations out of the closure
+			var decls []js_ast.Decl
+			end := 0
+			for _, stmt := range stmts {
+				switch s := stmt.Data.(type) {
+				case *js_ast.SLocal:
+					// Convert the declarations to assignments
+					wrapIdentifier := func(loc logger.Loc, ref ast.Ref) js_ast.Expr {
+						decls = append(decls, js_ast.Decl{Binding: js_ast.Binding{Loc: loc, Data: &js_ast.BIdentifier{Ref: ref}}})
+						return js_ast.Expr{Loc: loc, Data: &js_ast.EIdentifier{Ref: ref}}
+					}
+					var value js_ast.Expr
+					for _, decl := range s.Decls {
+						binding := js_ast.ConvertBindingToExpr(decl.Binding, wrapIdentifier)
+						if decl.ValueOrNil.Data != nil {
+							value = js_ast.JoinWithComma(value, js_ast.Assign(binding, decl.ValueOrNil))
+						}
+					}
+					if value.Data == nil {
+						continue
+					}
+					stmt = js_ast.Stmt{Loc: stmt.Loc, Data: &js_ast.SExpr{Value: value}}
+
+				case *js_ast.SFunction:
+					stmtList.outsideWrapperPrefix = append(stmtList.outsideWrapperPrefix, stmt)
+					continue
+				}
+
+				stmts[end] = stmt
+				end++
+			}
+			stmts = stmts[:end]
+
+			var esmArgs []js_ast.Expr
+			if c.options.ProfilerNames {
+				// "__esm({ 'file.js'() { ... } })"
+				kind := js_ast.PropertyField
+				if !c.options.UnsupportedJSFeatures.Has(compat.ObjectExtensions) {
+					kind = js_ast.PropertyMethod
+				}
+				esmArgs = []js_ast.Expr{{Data: &js_ast.EObject{Properties: []js_ast.Property{{
+					Kind:       kind,
+					Key:        js_ast.Expr{Data: &js_ast.EString{Value: helpers.StringToUTF16(file.InputFile.Source.PrettyPath)}},
+					ValueOrNil: js_ast.Expr{Data: &js_ast.EFunction{Fn: js_ast.Fn{Body: js_ast.FnBody{Block: js_ast.SBlock{Stmts: stmts}}, IsAsync: isAsync}}},
+				}}}}}
+			} else if c.options.UnsupportedJSFeatures.Has(compat.Arrow) {
+				// "__esm(function () { ... })"
+				esmArgs = []js_ast.Expr{{Data: &js_ast.EFunction{Fn: js_ast.Fn{Body: js_ast.FnBody{Block: js_ast.SBlock{Stmts: stmts}}, IsAsync: isAsync}}}}
+			} else {
+				// "__esm(() => { ... })"
+				esmArgs = []js_ast.Expr{{Data: &js_ast.EArrow{Body: js_ast.FnBody{Block: js_ast.SBlock{Stmts: stmts}}, IsAsync: isAsync}}}
+			}
+			value := js_ast.Expr{Data: &js_ast.ECall{
+				Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: c.esmRuntimeRef}},
+				Args:   esmArgs,
+			}}
+
+			// "var foo, bar;"
+			if !c.options.MinifySyntax && len(decls) > 0 {
+				stmtList.outsideWrapperPrefix = append(stmtList.outsideWrapperPrefix, js_ast.Stmt{Data: &js_ast.SLocal{
+					Decls: decls,
+				}})
+				decls = nil
+			}
+
+			// "var init_foo = __esm(...);"
+			stmts = append(stmtList.outsideWrapperPrefix, js_ast.Stmt{Data: &js_ast.SLocal{
+				Decls: append(decls, js_ast.Decl{
+					Binding:    js_ast.Binding{Data: &js_ast.BIdentifier{Ref: repr.AST.WrapperRef}},
+					ValueOrNil: value,
+				}),
+			}})
+		}
+	}
+
+	// Only generate a source map if needed
+	var addSourceMappings bool
+	var inputSourceMap *sourcemap.SourceMap
+	var lineOffsetTables []sourcemap.LineOffsetTable
+	if file.InputFile.Loader.CanHaveSourceMap() && c.options.SourceMap != config.SourceMapNone {
+		addSourceMappings = true
+		inputSourceMap = file.InputFile.InputSourceMap
+		lineOffsetTables = dataForSourceMaps[partRange.sourceIndex].LineOffsetTables
+	}
+
+	// Indent the file if everything is wrapped in an IIFE
+	indent := 0
+	if c.options.OutputFormat == config.FormatIIFE {
+		indent++
+	}
+
+	// Convert the AST to JavaScript code
+	printOptions := js_printer.Options{
+		Indent:                       indent,
+		OutputFormat:                 c.options.OutputFormat,
+		MinifyIdentifiers:            c.options.MinifyIdentifiers,
+		MinifyWhitespace:             c.options.MinifyWhitespace,
+		MinifySyntax:                 c.options.MinifySyntax,
+		LineLimit:                    c.options.LineLimit,
+		ASCIIOnly:                    c.options.ASCIIOnly,
+		ToCommonJSRef:                toCommonJSRef,
+		ToESMRef:                     toESMRef,
+		RuntimeRequireRef:            runtimeRequireRef,
+		TSEnums:                      c.graph.TSEnums,
+		ConstValues:                  c.graph.ConstValues,
+		LegalComments:                c.options.LegalComments,
+		UnsupportedFeatures:          c.options.UnsupportedJSFeatures,
+		SourceMap:                    c.options.SourceMap,
+		AddSourceMappings:            addSourceMappings,
+		InputSourceMap:               inputSourceMap,
+		LineOffsetTables:             lineOffsetTables,
+		RequireOrImportMetaForSource: c.requireOrImportMetaForSource,
+		MangledProps:                 c.mangledProps,
+		NeedsMetafile:                c.options.NeedsMetafile,
+	}
+	tree := repr.AST
+	tree.Directives = nil // This is handled elsewhere
+	tree.Parts = []js_ast.Part{{Stmts: stmts}}
+	*result = compileResultJS{
+		PrintResult: js_printer.Print(tree, c.graph.Symbols, r, printOptions),
+		sourceIndex: partRange.sourceIndex,
+	}
+
+	if file.InputFile.Loader == config.LoaderFile {
+		result.JSONMetadataImports = append(result.JSONMetadataImports, fmt.Sprintf("\n        {\n          \"path\": %s,\n          \"kind\": \"file-loader\"\n        }",
+			helpers.QuoteForJSON(file.InputFile.UniqueKeyForAdditionalFile, c.options.ASCIIOnly)))
+	}
+
+	waitGroup.Done()
+}
+
+func (c *linkerContext) generateEntryPointTailJS(
+	r renamer.Renamer,
+	toCommonJSRef ast.Ref,
+	toESMRef ast.Ref,
+	sourceIndex uint32,
+) (result compileResultJS) {
+	file := &c.graph.Files[sourceIndex]
+	repr := file.InputFile.Repr.(*graph.JSRepr)
+	var stmts []js_ast.Stmt
+
+	switch c.options.OutputFormat {
+	case config.FormatPreserve:
+		if repr.Meta.Wrap != graph.WrapNone {
+			// "require_foo();"
+			// "init_foo();"
+			stmts = append(stmts, js_ast.Stmt{Data: &js_ast.SExpr{Value: js_ast.Expr{Data: &js_ast.ECall{
+				Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: repr.AST.WrapperRef}},
+			}}}})
+		}
+
+	case config.FormatIIFE:
+		if repr.Meta.Wrap == graph.WrapCJS {
+			if len(c.options.GlobalName) > 0 {
+				// "return require_foo();"
+				stmts = append(stmts, js_ast.Stmt{Data: &js_ast.SReturn{ValueOrNil: js_ast.Expr{Data: &js_ast.ECall{
+					Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: repr.AST.WrapperRef}},
+				}}}})
+			} else {
+				// "require_foo();"
+				stmts = append(stmts, js_ast.Stmt{Data: &js_ast.SExpr{Value: js_ast.Expr{Data: &js_ast.ECall{
+					Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: repr.AST.WrapperRef}},
+				}}}})
+			}
+		} else {
+			if repr.Meta.Wrap == graph.WrapESM {
+				// "init_foo();"
+				stmts = append(stmts, js_ast.Stmt{Data: &js_ast.SExpr{Value: js_ast.Expr{Data: &js_ast.ECall{
+					Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: repr.AST.WrapperRef}},
+				}}}})
+			}
+
+			if repr.Meta.ForceIncludeExportsForEntryPoint {
+				// "return __toCommonJS(exports);"
+				stmts = append(stmts, js_ast.Stmt{Data: &js_ast.SReturn{
+					ValueOrNil: js_ast.Expr{Data: &js_ast.ECall{
+						Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: toCommonJSRef}},
+						Args:   []js_ast.Expr{{Data: &js_ast.EIdentifier{Ref: repr.AST.ExportsRef}}},
+					}},
+				}})
+			}
+		}
+
+	case config.FormatCommonJS:
+		if repr.Meta.Wrap == graph.WrapCJS {
+			// "module.exports = require_foo();"
+			stmts = append(stmts, js_ast.AssignStmt(
+				js_ast.Expr{Data: &js_ast.EDot{
+					Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: c.unboundModuleRef}},
+					Name:   "exports",
+				}},
+				js_ast.Expr{Data: &js_ast.ECall{
+					Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: repr.AST.WrapperRef}},
+				}},
+			))
+		} else {
+			if repr.Meta.Wrap == graph.WrapESM {
+				// "init_foo();"
+				stmts = append(stmts, js_ast.Stmt{Data: &js_ast.SExpr{Value: js_ast.Expr{Data: &js_ast.ECall{
+					Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: repr.AST.WrapperRef}},
+				}}}})
+			}
+		}
+
+		// If we are generating CommonJS for node, encode the known export names in
+		// a form that node can understand them. This relies on the specific behavior
+		// of this parser, which the node project uses to detect named exports in
+		// CommonJS files: https://github.com/guybedford/cjs-module-lexer. Think of
+		// this code as an annotation for that parser.
+		if c.options.Platform == config.PlatformNode {
+			// Add a comment since otherwise people will surely wonder what this is.
+			// This annotation means you can do this and have it work:
+			//
+			//   import { name } from './file-from-esbuild.cjs'
+			//
+			// when "file-from-esbuild.cjs" looks like this:
+			//
+			//   __export(exports, { name: () => name });
+			//   0 && (module.exports = {name});
+			//
+			// The maintainer of "cjs-module-lexer" is receptive to adding esbuild-
+			// friendly patterns to this library. However, this library has already
+			// shipped in node and using existing patterns instead of defining new
+			// patterns is maximally compatible.
+			//
+			// An alternative to doing this could be to use "Object.defineProperties"
+			// instead of "__export" but support for that would need to be added to
+			// "cjs-module-lexer" and then we would need to be ok with not supporting
+			// older versions of node that don't have that newly-added support.
+
+			// "{a, b, if: null}"
+			var moduleExports []js_ast.Property
+			for _, export := range repr.Meta.SortedAndFilteredExportAliases {
+				if export == "default" {
+					// In node the default export is always "module.exports" regardless of
+					// what the annotation says. So don't bother generating "default".
+					continue
+				}
+
+				// "{if: null}"
+				var valueOrNil js_ast.Expr
+				if _, ok := js_lexer.Keywords[export]; ok {
+					// Make sure keywords don't cause a syntax error. This has to map to
+					// "null" instead of something shorter like "0" because the library
+					// "cjs-module-lexer" only supports identifiers in this position, and
+					// it thinks "null" is an identifier.
+					valueOrNil = js_ast.Expr{Data: js_ast.ENullShared}
+				}
+
+				moduleExports = append(moduleExports, js_ast.Property{
+					Key:        js_ast.Expr{Data: &js_ast.EString{Value: helpers.StringToUTF16(export)}},
+					ValueOrNil: valueOrNil,
+				})
+			}
+
+			// Add annotations for re-exports: "{...require('./foo')}"
+			for _, importRecordIndex := range repr.AST.ExportStarImportRecords {
+				if record := &repr.AST.ImportRecords[importRecordIndex]; !record.SourceIndex.IsValid() {
+					moduleExports = append(moduleExports, js_ast.Property{
+						Kind:       js_ast.PropertySpread,
+						ValueOrNil: js_ast.Expr{Data: &js_ast.ERequireString{ImportRecordIndex: importRecordIndex}},
+					})
+				}
+			}
+
+			if len(moduleExports) > 0 {
+				// "0 && (module.exports = {a, b, if: null});"
+				expr := js_ast.Expr{Data: &js_ast.EBinary{
+					Op:   js_ast.BinOpLogicalAnd,
+					Left: js_ast.Expr{Data: &js_ast.ENumber{Value: 0}},
+					Right: js_ast.Assign(
+						js_ast.Expr{Data: &js_ast.EDot{
+							Target: js_ast.Expr{Data: &js_ast.EIdentifier{Ref: c.unboundModuleRef}},
+							Name:   "exports",
+						}},
+						js_ast.Expr{Data: &js_ast.EObject{Properties: moduleExports}},
+					),
+				}}
+
+				if !c.options.MinifyWhitespace {
+					stmts = append(stmts,
+						js_ast.Stmt{Data: &js_ast.SComment{Text: `// Annotate the CommonJS export names for ESM import in node:`}},
+					)
+				}
+
+				stmts = append(stmts, js_ast.Stmt{Data: &js_ast.SExpr{Value: expr}})
+			}
+		}
+
+	case config.FormatESModule:
+		if repr.Meta.Wrap == graph.WrapCJS {
+			// "export default require_foo();"
+			stmts = append(stmts, js_ast.Stmt{
+				Data: &js_ast.SExportDefault{Value: js_ast.Stmt{
+					Data: &js_ast.SExpr{Value: js_ast.Expr{
+						Data: &js_ast.ECall{Target: js_ast.Expr{
+							Data: &js_ast.EIdentifier{Ref: repr.AST.WrapperRef}}}}}}}})
+		} else {
+			if repr.Meta.Wrap == graph.WrapESM {
+				if repr.Meta.IsAsyncOrHasAsyncDependency {
+					// "await init_foo();"
+					stmts = append(stmts, js_ast.Stmt{
+						Data: &js_ast.SExpr{Value: js_ast.Expr{
+							Data: &js_ast.EAwait{Value: js_ast.Expr{
+								Data: &js_ast.ECall{Target: js_ast.Expr{
+									Data: &js_ast.EIdentifier{Ref: repr.AST.WrapperRef}}}}}}}})
+				} else {
+					// "init_foo();"
+					stmts = append(stmts, js_ast.Stmt{
+						Data: &js_ast.SExpr{
+							Value: js_ast.Expr{Data: &js_ast.ECall{Target: js_ast.Expr{
+								Data: &js_ast.EIdentifier{Ref: repr.AST.WrapperRef}}}}}})
+				}
+			}
+
+			if len(repr.Meta.SortedAndFilteredExportAliases) > 0 {
+				// If the output format is ES6 modules and we're an entry point, generate an
+				// ES6 export statement containing all exports. Except don't do that if this
+				// entry point is a CommonJS-style module, since that would generate an ES6
+				// export statement that's not top-level. Instead, we will export the CommonJS
+				// exports as a default export later on.
+				var items []js_ast.ClauseItem
+
+				for i, alias := range repr.Meta.SortedAndFilteredExportAliases {
+					export := repr.Meta.ResolvedExports[alias]
+
+					// If this is an export of an import, reference the symbol that the import
+					// was eventually resolved to. We need to do this because imports have
+					// already been resolved by this point, so we can't generate a new import
+					// and have that be resolved later.
+					if importData, ok := c.graph.Files[export.SourceIndex].InputFile.Repr.(*graph.JSRepr).Meta.ImportsToBind[export.Ref]; ok {
+						export.Ref = importData.Ref
+						export.SourceIndex = importData.SourceIndex
+					}
+
+					// Exports of imports need EImportIdentifier in case they need to be re-
+					// written to a property access later on
+					if c.graph.Symbols.Get(export.Ref).NamespaceAlias != nil {
+						// Create both a local variable and an export clause for that variable.
+						// The local variable is initialized with the initial value of the
+						// export. This isn't fully correct because it's a "dead" binding and
+						// doesn't update with the "live" value as it changes. But ES6 modules
+						// don't have any syntax for bare named getter functions so this is the
+						// best we can do.
+						//
+						// These input files:
+						//
+						//   // entry_point.js
+						//   export {foo} from './cjs-format.js'
+						//
+						//   // cjs-format.js
+						//   Object.defineProperty(exports, 'foo', {
+						//     enumerable: true,
+						//     get: () => Math.random(),
+						//   })
+						//
+						// Become this output file:
+						//
+						//   // cjs-format.js
+						//   var require_cjs_format = __commonJS((exports) => {
+						//     Object.defineProperty(exports, "foo", {
+						//       enumerable: true,
+						//       get: () => Math.random()
+						//     });
+						//   });
+						//
+						//   // entry_point.js
+						//   var cjs_format = __toESM(require_cjs_format());
+						//   var export_foo = cjs_format.foo;
+						//   export {
+						//     export_foo as foo
+						//   };
+						//
+						tempRef := repr.Meta.CJSExportCopies[i]
+						stmts = append(stmts, js_ast.Stmt{Data: &js_ast.SLocal{
+							Decls: []js_ast.Decl{{
+								Binding:    js_ast.Binding{Data: &js_ast.BIdentifier{Ref: tempRef}},
+								ValueOrNil: js_ast.Expr{Data: &js_ast.EImportIdentifier{Ref: export.Ref}},
+							}},
+						}})
+						items = append(items, js_ast.ClauseItem{
+							Name:  ast.LocRef{Ref: tempRef},
+							Alias: alias,
+						})
+					} else {
+						// Local identifiers can be exported using an export clause. This is done
+						// this way instead of leaving the "export" keyword on the local declaration
+						// itself both because it lets the local identifier be minified and because
+						// it works transparently for re-exports across files.
+						//
+						// These input files:
+						//
+						//   // entry_point.js
+						//   export * from './esm-format.js'
+						//
+						//   // esm-format.js
+						//   export let foo = 123
+						//
+						// Become this output file:
+						//
+						//   // esm-format.js
+						//   let foo = 123;
+						//
+						//   // entry_point.js
+						//   export {
+						//     foo
+						//   };
+						//
+						items = append(items, js_ast.ClauseItem{
+							Name:  ast.LocRef{Ref: export.Ref},
+							Alias: alias,
+						})
+					}
+				}
+
+				stmts = append(stmts, js_ast.Stmt{Data: &js_ast.SExportClause{Items: items}})
+			}
+		}
+	}
+
+	if len(stmts) == 0 {
+		return
+	}
+
+	tree := repr.AST
+	tree.Directives = nil
+	tree.Parts = []js_ast.Part{{Stmts: stmts}}
+
+	// Indent the file if everything is wrapped in an IIFE
+	indent := 0
+	if c.options.OutputFormat == config.FormatIIFE {
+		indent++
+	}
+
+	// Convert the AST to JavaScript code
+	printOptions := js_printer.Options{
+		Indent:                       indent,
+		OutputFormat:                 c.options.OutputFormat,
+		MinifyIdentifiers:            c.options.MinifyIdentifiers,
+		MinifyWhitespace:             c.options.MinifyWhitespace,
+		MinifySyntax:                 c.options.MinifySyntax,
+		LineLimit:                    c.options.LineLimit,
+		ASCIIOnly:                    c.options.ASCIIOnly,
+		ToCommonJSRef:                toCommonJSRef,
+		ToESMRef:                     toESMRef,
+		LegalComments:                c.options.LegalComments,
+		UnsupportedFeatures:          c.options.UnsupportedJSFeatures,
+		RequireOrImportMetaForSource: c.requireOrImportMetaForSource,
+		MangledProps:                 c.mangledProps,
+	}
+	result.PrintResult = js_printer.Print(tree, c.graph.Symbols, r, printOptions)
+	return
+}
+
+func (c *linkerContext) renameSymbolsInChunk(chunk *chunkInfo, filesInOrder []uint32, timer *helpers.Timer) renamer.Renamer {
+	if c.options.MinifyIdentifiers {
+		timer.Begin("Minify symbols")
+		defer timer.End("Minify symbols")
+	} else {
+		timer.Begin("Rename symbols")
+		defer timer.End("Rename symbols")
+	}
+
+	// Determine the reserved names (e.g. can't generate the name "if")
+	timer.Begin("Compute reserved names")
+	moduleScopes := make([]*js_ast.Scope, len(filesInOrder))
+	for i, sourceIndex := range filesInOrder {
+		moduleScopes[i] = c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr).AST.ModuleScope
+	}
+	reservedNames := renamer.ComputeReservedNames(moduleScopes, c.graph.Symbols)
+
+	// Node contains code that scans CommonJS modules in an attempt to statically
+	// detect the  set of export names that a module will use. However, it doesn't
+	// do any scope analysis so it can be fooled by local variables with the same
+	// name as the CommonJS module-scope variables "exports" and "module". Avoid
+	// using these names in this case even if there is not a risk of a name
+	// collision because there is still a risk of node incorrectly detecting
+	// something in a nested scope as an top-level export. Here's a case where
+	// this happened: https://github.com/evanw/esbuild/issues/3544
+	if c.options.OutputFormat == config.FormatCommonJS && c.options.Platform == config.PlatformNode {
+		reservedNames["exports"] = 1
+		reservedNames["module"] = 1
+	}
+
+	// These are used to implement bundling, and need to be free for use
+	if c.options.Mode != config.ModePassThrough {
+		reservedNames["require"] = 1
+		reservedNames["Promise"] = 1
+	}
+	timer.End("Compute reserved names")
+
+	// Make sure imports get a chance to be renamed too
+	var sortedImportsFromOtherChunks stableRefArray
+	for _, imports := range chunk.chunkRepr.(*chunkReprJS).importsFromOtherChunks {
+		for _, item := range imports {
+			sortedImportsFromOtherChunks = append(sortedImportsFromOtherChunks, stableRef{
+				StableSourceIndex: c.graph.StableSourceIndices[item.ref.SourceIndex],
+				Ref:               item.ref,
+			})
+		}
+	}
+	sort.Sort(sortedImportsFromOtherChunks)
+
+	// Minification uses frequency analysis to give shorter names to more frequent symbols
+	if c.options.MinifyIdentifiers {
+		// Determine the first top-level slot (i.e. not in a nested scope)
+		var firstTopLevelSlots ast.SlotCounts
+		for _, sourceIndex := range filesInOrder {
+			firstTopLevelSlots.UnionMax(c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr).AST.NestedScopeSlotCounts)
+		}
+		r := renamer.NewMinifyRenamer(c.graph.Symbols, firstTopLevelSlots, reservedNames)
+
+		// Accumulate nested symbol usage counts
+		timer.Begin("Accumulate symbol counts")
+		timer.Begin("Parallel phase")
+		allTopLevelSymbols := make([]renamer.StableSymbolCountArray, len(filesInOrder))
+		stableSourceIndices := c.graph.StableSourceIndices
+		freq := ast.CharFreq{}
+		waitGroup := sync.WaitGroup{}
+		waitGroup.Add(len(filesInOrder))
+		for i, sourceIndex := range filesInOrder {
+			repr := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr)
+
+			// Do this outside of the goroutine because it's not atomic
+			if repr.AST.CharFreq != nil {
+				freq.Include(repr.AST.CharFreq)
+			}
+
+			go func(topLevelSymbols *renamer.StableSymbolCountArray, repr *graph.JSRepr) {
+				if repr.AST.UsesExportsRef {
+					r.AccumulateSymbolCount(topLevelSymbols, repr.AST.ExportsRef, 1, stableSourceIndices)
+				}
+				if repr.AST.UsesModuleRef {
+					r.AccumulateSymbolCount(topLevelSymbols, repr.AST.ModuleRef, 1, stableSourceIndices)
+				}
+
+				for partIndex, part := range repr.AST.Parts {
+					if !repr.AST.Parts[partIndex].IsLive {
+						// Skip the part if it's not in this chunk
+						continue
+					}
+
+					// Accumulate symbol use counts
+					r.AccumulateSymbolUseCounts(topLevelSymbols, part.SymbolUses, stableSourceIndices)
+
+					// Make sure to also count the declaration in addition to the uses
+					for _, declared := range part.DeclaredSymbols {
+						r.AccumulateSymbolCount(topLevelSymbols, declared.Ref, 1, stableSourceIndices)
+					}
+				}
+
+				sort.Sort(topLevelSymbols)
+				waitGroup.Done()
+			}(&allTopLevelSymbols[i], repr)
+		}
+		waitGroup.Wait()
+		timer.End("Parallel phase")
+
+		// Accumulate top-level symbol usage counts
+		timer.Begin("Serial phase")
+		capacity := len(sortedImportsFromOtherChunks)
+		for _, array := range allTopLevelSymbols {
+			capacity += len(array)
+		}
+		topLevelSymbols := make(renamer.StableSymbolCountArray, 0, capacity)
+		for _, stable := range sortedImportsFromOtherChunks {
+			r.AccumulateSymbolCount(&topLevelSymbols, stable.Ref, 1, stableSourceIndices)
+		}
+		for _, array := range allTopLevelSymbols {
+			topLevelSymbols = append(topLevelSymbols, array...)
+		}
+		r.AllocateTopLevelSymbolSlots(topLevelSymbols)
+		timer.End("Serial phase")
+		timer.End("Accumulate symbol counts")
+
+		// Add all of the character frequency histograms for all files in this
+		// chunk together, then use it to compute the character sequence used to
+		// generate minified names. This results in slightly better gzip compression
+		// over assigning minified names in order (i.e. "a b c ..."). Even though
+		// it's a very small win, we still do it because it's simple to do and very
+		// cheap to compute.
+		minifier := ast.DefaultNameMinifierJS.ShuffleByCharFreq(freq)
+		timer.Begin("Assign names by frequency")
+		r.AssignNamesByFrequency(&minifier)
+		timer.End("Assign names by frequency")
+		return r
+	}
+
+	// When we're not minifying, just append numbers to symbol names to avoid collisions
+	r := renamer.NewNumberRenamer(c.graph.Symbols, reservedNames)
+	nestedScopes := make(map[uint32][]*js_ast.Scope)
+
+	timer.Begin("Add top-level symbols")
+	for _, stable := range sortedImportsFromOtherChunks {
+		r.AddTopLevelSymbol(stable.Ref)
+	}
+	for _, sourceIndex := range filesInOrder {
+		repr := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr)
+		var scopes []*js_ast.Scope
+
+		// Modules wrapped in a CommonJS closure look like this:
+		//
+		//   // foo.js
+		//   var require_foo = __commonJS((exports, module) => {
+		//     exports.foo = 123;
+		//   });
+		//
+		// The symbol "require_foo" is stored in "file.ast.WrapperRef". We want
+		// to be able to minify everything inside the closure without worrying
+		// about collisions with other CommonJS modules. Set up the scopes such
+		// that it appears as if the file was structured this way all along. It's
+		// not completely accurate (e.g. we don't set the parent of the module
+		// scope to this new top-level scope) but it's good enough for the
+		// renaming code.
+		if repr.Meta.Wrap == graph.WrapCJS {
+			r.AddTopLevelSymbol(repr.AST.WrapperRef)
+
+			// External import statements will be hoisted outside of the CommonJS
+			// wrapper if the output format supports import statements. We need to
+			// add those symbols to the top-level scope to avoid causing name
+			// collisions. This code special-cases only those symbols.
+			if c.options.OutputFormat.KeepESMImportExportSyntax() {
+				for _, part := range repr.AST.Parts {
+					for _, stmt := range part.Stmts {
+						switch s := stmt.Data.(type) {
+						case *js_ast.SImport:
+							if !repr.AST.ImportRecords[s.ImportRecordIndex].SourceIndex.IsValid() {
+								r.AddTopLevelSymbol(s.NamespaceRef)
+								if s.DefaultName != nil {
+									r.AddTopLevelSymbol(s.DefaultName.Ref)
+								}
+								if s.Items != nil {
+									for _, item := range *s.Items {
+										r.AddTopLevelSymbol(item.Name.Ref)
+									}
+								}
+							}
+
+						case *js_ast.SExportStar:
+							if !repr.AST.ImportRecords[s.ImportRecordIndex].SourceIndex.IsValid() {
+								r.AddTopLevelSymbol(s.NamespaceRef)
+							}
+
+						case *js_ast.SExportFrom:
+							if !repr.AST.ImportRecords[s.ImportRecordIndex].SourceIndex.IsValid() {
+								r.AddTopLevelSymbol(s.NamespaceRef)
+								for _, item := range s.Items {
+									r.AddTopLevelSymbol(item.Name.Ref)
+								}
+							}
+						}
+					}
+				}
+			}
+
+			nestedScopes[sourceIndex] = []*js_ast.Scope{repr.AST.ModuleScope}
+			continue
+		}
+
+		// Modules wrapped in an ESM closure look like this:
+		//
+		//   // foo.js
+		//   var foo, foo_exports = {};
+		//   __export(foo_exports, {
+		//     foo: () => foo
+		//   });
+		//   let init_foo = __esm(() => {
+		//     foo = 123;
+		//   });
+		//
+		// The symbol "init_foo" is stored in "file.ast.WrapperRef". We need to
+		// minify everything inside the closure without introducing a new scope
+		// since all top-level variables will be hoisted outside of the closure.
+		if repr.Meta.Wrap == graph.WrapESM {
+			r.AddTopLevelSymbol(repr.AST.WrapperRef)
+		}
+
+		// Rename each top-level symbol declaration in this chunk
+		for partIndex, part := range repr.AST.Parts {
+			if repr.AST.Parts[partIndex].IsLive {
+				for _, declared := range part.DeclaredSymbols {
+					if declared.IsTopLevel {
+						r.AddTopLevelSymbol(declared.Ref)
+					}
+				}
+				scopes = append(scopes, part.Scopes...)
+			}
+		}
+
+		nestedScopes[sourceIndex] = scopes
+	}
+	timer.End("Add top-level symbols")
+
+	// Recursively rename symbols in child scopes now that all top-level
+	// symbols have been renamed. This is done in parallel because the symbols
+	// inside nested scopes are independent and can't conflict.
+	timer.Begin("Assign names by scope")
+	r.AssignNamesByScope(nestedScopes)
+	timer.End("Assign names by scope")
+	return r
+}
+
+func (c *linkerContext) generateChunkJS(chunkIndex int, chunkWaitGroup *sync.WaitGroup) {
+	defer c.recoverInternalError(chunkWaitGroup, runtime.SourceIndex)
+
+	chunk := &c.chunks[chunkIndex]
+
+	timer := c.timer.Fork()
+	if timer != nil {
+		timeName := fmt.Sprintf("Generate chunk %q", path.Clean(config.TemplateToString(chunk.finalTemplate)))
+		timer.Begin(timeName)
+		defer c.timer.Join(timer)
+		defer timer.End(timeName)
+	}
+
+	chunkRepr := chunk.chunkRepr.(*chunkReprJS)
+	compileResults := make([]compileResultJS, 0, len(chunkRepr.partsInChunkInOrder))
+	runtimeMembers := c.graph.Files[runtime.SourceIndex].InputFile.Repr.(*graph.JSRepr).AST.ModuleScope.Members
+	toCommonJSRef := ast.FollowSymbols(c.graph.Symbols, runtimeMembers["__toCommonJS"].Ref)
+	toESMRef := ast.FollowSymbols(c.graph.Symbols, runtimeMembers["__toESM"].Ref)
+	runtimeRequireRef := ast.FollowSymbols(c.graph.Symbols, runtimeMembers["__require"].Ref)
+	r := c.renameSymbolsInChunk(chunk, chunkRepr.filesInChunkInOrder, timer)
+	dataForSourceMaps := c.dataForSourceMaps()
+
+	// Note: This contains placeholders instead of what the placeholders are
+	// substituted with. That should be fine though because this should only
+	// ever be used for figuring out how many "../" to add to a relative path
+	// from a chunk whose final path hasn't been calculated yet to a chunk
+	// whose final path has already been calculated. That and placeholders are
+	// never substituted with something containing a "/" so substitution should
+	// never change the "../" count.
+	chunkAbsDir := c.fs.Dir(c.fs.Join(c.options.AbsOutputDir, config.TemplateToString(chunk.finalTemplate)))
+
+	// Generate JavaScript for each file in parallel
+	timer.Begin("Print JavaScript files")
+	waitGroup := sync.WaitGroup{}
+	for _, partRange := range chunkRepr.partsInChunkInOrder {
+		// Skip the runtime in test output
+		if partRange.sourceIndex == runtime.SourceIndex && c.options.OmitRuntimeForTests {
+			continue
+		}
+
+		// Create a goroutine for this file
+		compileResults = append(compileResults, compileResultJS{})
+		compileResult := &compileResults[len(compileResults)-1]
+		waitGroup.Add(1)
+		go c.generateCodeForFileInChunkJS(
+			r,
+			&waitGroup,
+			partRange,
+			toCommonJSRef,
+			toESMRef,
+			runtimeRequireRef,
+			compileResult,
+			dataForSourceMaps,
+		)
+	}
+
+	// Also generate the cross-chunk binding code
+	var crossChunkPrefix []byte
+	var crossChunkSuffix []byte
+	var jsonMetadataImports []string
+	{
+		// Indent the file if everything is wrapped in an IIFE
+		indent := 0
+		if c.options.OutputFormat == config.FormatIIFE {
+			indent++
+		}
+		printOptions := js_printer.Options{
+			Indent:            indent,
+			OutputFormat:      c.options.OutputFormat,
+			MinifyIdentifiers: c.options.MinifyIdentifiers,
+			MinifyWhitespace:  c.options.MinifyWhitespace,
+			MinifySyntax:      c.options.MinifySyntax,
+			LineLimit:         c.options.LineLimit,
+			NeedsMetafile:     c.options.NeedsMetafile,
+		}
+		crossChunkImportRecords := make([]ast.ImportRecord, len(chunk.crossChunkImports))
+		for i, chunkImport := range chunk.crossChunkImports {
+			crossChunkImportRecords[i] = ast.ImportRecord{
+				Kind:  chunkImport.importKind,
+				Path:  logger.Path{Text: c.chunks[chunkImport.chunkIndex].uniqueKey},
+				Flags: ast.ShouldNotBeExternalInMetafile | ast.ContainsUniqueKey,
+			}
+		}
+		crossChunkResult := js_printer.Print(js_ast.AST{
+			ImportRecords: crossChunkImportRecords,
+			Parts:         []js_ast.Part{{Stmts: chunkRepr.crossChunkPrefixStmts}},
+		}, c.graph.Symbols, r, printOptions)
+		crossChunkPrefix = crossChunkResult.JS
+		jsonMetadataImports = crossChunkResult.JSONMetadataImports
+		crossChunkSuffix = js_printer.Print(js_ast.AST{
+			Parts: []js_ast.Part{{Stmts: chunkRepr.crossChunkSuffixStmts}},
+		}, c.graph.Symbols, r, printOptions).JS
+	}
+
+	// Generate the exports for the entry point, if there are any
+	var entryPointTail compileResultJS
+	if chunk.isEntryPoint {
+		entryPointTail = c.generateEntryPointTailJS(
+			r,
+			toCommonJSRef,
+			toESMRef,
+			chunk.sourceIndex,
+		)
+	}
+
+	waitGroup.Wait()
+	timer.End("Print JavaScript files")
+	timer.Begin("Join JavaScript files")
+
+	j := helpers.Joiner{}
+	prevOffset := sourcemap.LineColumnOffset{}
+
+	// Optionally strip whitespace
+	indent := ""
+	space := " "
+	newline := "\n"
+	if c.options.MinifyWhitespace {
+		space = ""
+		newline = ""
+	}
+	newlineBeforeComment := false
+	isExecutable := false
+
+	// Start with the hashbang if there is one. This must be done before the
+	// banner because it only works if it's literally the first character.
+	if chunk.isEntryPoint {
+		if repr := c.graph.Files[chunk.sourceIndex].InputFile.Repr.(*graph.JSRepr); repr.AST.Hashbang != "" {
+			hashbang := repr.AST.Hashbang + "\n"
+			prevOffset.AdvanceString(hashbang)
+			j.AddString(hashbang)
+			newlineBeforeComment = true
+			isExecutable = true
+		}
+	}
+
+	// Then emit the banner after the hashbang. This must come before the
+	// "use strict" directive below because some people use the banner to
+	// emit a hashbang, which must be the first thing in the file.
+	if len(c.options.JSBanner) > 0 {
+		prevOffset.AdvanceString(c.options.JSBanner)
+		prevOffset.AdvanceString("\n")
+		j.AddString(c.options.JSBanner)
+		j.AddString("\n")
+		newlineBeforeComment = true
+	}
+
+	// Add the top-level directive if present (but omit "use strict" in ES
+	// modules because all ES modules are automatically in strict mode)
+	if chunk.isEntryPoint {
+		repr := c.graph.Files[chunk.sourceIndex].InputFile.Repr.(*graph.JSRepr)
+		for _, directive := range repr.AST.Directives {
+			if directive != "use strict" || c.options.OutputFormat != config.FormatESModule {
+				quoted := string(helpers.QuoteForJSON(directive, c.options.ASCIIOnly)) + ";" + newline
+				prevOffset.AdvanceString(quoted)
+				j.AddString(quoted)
+				newlineBeforeComment = true
+			}
+		}
+	}
+
+	// Optionally wrap with an IIFE
+	if c.options.OutputFormat == config.FormatIIFE {
+		var text string
+		indent = "  "
+		if len(c.options.GlobalName) > 0 {
+			text = c.generateGlobalNamePrefix()
+		}
+		if c.options.UnsupportedJSFeatures.Has(compat.Arrow) {
+			text += "(function()" + space + "{" + newline
+		} else {
+			text += "(()" + space + "=>" + space + "{" + newline
+		}
+		prevOffset.AdvanceString(text)
+		j.AddString(text)
+		newlineBeforeComment = false
+	}
+
+	// Put the cross-chunk prefix inside the IIFE
+	if len(crossChunkPrefix) > 0 {
+		newlineBeforeComment = true
+		prevOffset.AdvanceBytes(crossChunkPrefix)
+		j.AddBytes(crossChunkPrefix)
+	}
+
+	// Start the metadata
+	jMeta := helpers.Joiner{}
+	if c.options.NeedsMetafile {
+		// Print imports
+		isFirstMeta := true
+		jMeta.AddString("{\n      \"imports\": [")
+		for _, json := range jsonMetadataImports {
+			if isFirstMeta {
+				isFirstMeta = false
+			} else {
+				jMeta.AddString(",")
+			}
+			jMeta.AddString(json)
+		}
+		for _, compileResult := range compileResults {
+			for _, json := range compileResult.JSONMetadataImports {
+				if isFirstMeta {
+					isFirstMeta = false
+				} else {
+					jMeta.AddString(",")
+				}
+				jMeta.AddString(json)
+			}
+		}
+		if !isFirstMeta {
+			jMeta.AddString("\n      ")
+		}
+
+		// Print exports
+		jMeta.AddString("],\n      \"exports\": [")
+		var aliases []string
+		if c.options.OutputFormat.KeepESMImportExportSyntax() {
+			if chunk.isEntryPoint {
+				if fileRepr := c.graph.Files[chunk.sourceIndex].InputFile.Repr.(*graph.JSRepr); fileRepr.Meta.Wrap == graph.WrapCJS {
+					aliases = []string{"default"}
+				} else {
+					resolvedExports := fileRepr.Meta.ResolvedExports
+					aliases = make([]string, 0, len(resolvedExports))
+					for alias := range resolvedExports {
+						aliases = append(aliases, alias)
+					}
+				}
+			} else {
+				aliases = make([]string, 0, len(chunkRepr.exportsToOtherChunks))
+				for _, alias := range chunkRepr.exportsToOtherChunks {
+					aliases = append(aliases, alias)
+				}
+			}
+		}
+		isFirstMeta = true
+		sort.Strings(aliases) // Sort for determinism
+		for _, alias := range aliases {
+			if isFirstMeta {
+				isFirstMeta = false
+			} else {
+				jMeta.AddString(",")
+			}
+			jMeta.AddString(fmt.Sprintf("\n        %s",
+				helpers.QuoteForJSON(alias, c.options.ASCIIOnly)))
+		}
+		if !isFirstMeta {
+			jMeta.AddString("\n      ")
+		}
+		jMeta.AddString("],\n")
+		if chunk.isEntryPoint {
+			entryPoint := c.graph.Files[chunk.sourceIndex].InputFile.Source.PrettyPath
+			jMeta.AddString(fmt.Sprintf("      \"entryPoint\": %s,\n", helpers.QuoteForJSON(entryPoint, c.options.ASCIIOnly)))
+		}
+		if chunkRepr.hasCSSChunk {
+			jMeta.AddString(fmt.Sprintf("      \"cssBundle\": %s,\n", helpers.QuoteForJSON(c.chunks[chunkRepr.cssChunkIndex].uniqueKey, c.options.ASCIIOnly)))
+		}
+		jMeta.AddString("      \"inputs\": {")
+	}
+
+	// Concatenate the generated JavaScript chunks together
+	var compileResultsForSourceMap []compileResultForSourceMap
+	var legalCommentList []legalCommentEntry
+	var metaOrder []uint32
+	var metaBytes map[uint32][][]byte
+	prevFileNameComment := uint32(0)
+	if c.options.NeedsMetafile {
+		metaOrder = make([]uint32, 0, len(compileResults))
+		metaBytes = make(map[uint32][][]byte, len(compileResults))
+	}
+	for _, compileResult := range compileResults {
+		if len(compileResult.ExtractedLegalComments) > 0 {
+			legalCommentList = append(legalCommentList, legalCommentEntry{
+				sourceIndex: compileResult.sourceIndex,
+				comments:    compileResult.ExtractedLegalComments,
+			})
+		}
+
+		// Add a comment with the file path before the file contents
+		if c.options.Mode == config.ModeBundle && !c.options.MinifyWhitespace &&
+			prevFileNameComment != compileResult.sourceIndex && len(compileResult.JS) > 0 {
+			if newlineBeforeComment {
+				prevOffset.AdvanceString("\n")
+				j.AddString("\n")
+			}
+
+			path := c.graph.Files[compileResult.sourceIndex].InputFile.Source.PrettyPath
+
+			// Make sure newlines in the path can't cause a syntax error. This does
+			// not minimize allocations because it's expected that this case never
+			// comes up in practice.
+			path = strings.ReplaceAll(path, "\r", "\\r")
+			path = strings.ReplaceAll(path, "\n", "\\n")
+			path = strings.ReplaceAll(path, "\u2028", "\\u2028")
+			path = strings.ReplaceAll(path, "\u2029", "\\u2029")
+
+			text := fmt.Sprintf("%s// %s\n", indent, path)
+			prevOffset.AdvanceString(text)
+			j.AddString(text)
+			prevFileNameComment = compileResult.sourceIndex
+		}
+
+		// Don't include the runtime in source maps
+		if c.graph.Files[compileResult.sourceIndex].InputFile.OmitFromSourceMapsAndMetafile {
+			prevOffset.AdvanceString(string(compileResult.JS))
+			j.AddBytes(compileResult.JS)
+		} else {
+			// Save the offset to the start of the stored JavaScript
+			compileResult.generatedOffset = prevOffset
+			j.AddBytes(compileResult.JS)
+
+			// Ignore empty source map chunks
+			if compileResult.SourceMapChunk.ShouldIgnore {
+				prevOffset.AdvanceBytes(compileResult.JS)
+			} else {
+				prevOffset = sourcemap.LineColumnOffset{}
+
+				// Include this file in the source map
+				if c.options.SourceMap != config.SourceMapNone {
+					compileResultsForSourceMap = append(compileResultsForSourceMap, compileResultForSourceMap{
+						sourceMapChunk:  compileResult.SourceMapChunk,
+						generatedOffset: compileResult.generatedOffset,
+						sourceIndex:     compileResult.sourceIndex,
+					})
+				}
+			}
+
+			// Include this file in the metadata
+			if c.options.NeedsMetafile {
+				// Accumulate file sizes since a given file may be split into multiple parts
+				bytes, ok := metaBytes[compileResult.sourceIndex]
+				if !ok {
+					metaOrder = append(metaOrder, compileResult.sourceIndex)
+				}
+				metaBytes[compileResult.sourceIndex] = append(bytes, compileResult.JS)
+			}
+		}
+
+		// Put a newline before the next file path comment
+		if len(compileResult.JS) > 0 {
+			newlineBeforeComment = true
+		}
+	}
+
+	// Stick the entry point tail at the end of the file. Deliberately don't
+	// include any source mapping information for this because it's automatically
+	// generated and doesn't correspond to a location in the input file.
+	j.AddBytes(entryPointTail.JS)
+
+	// Put the cross-chunk suffix inside the IIFE
+	if len(crossChunkSuffix) > 0 {
+		if newlineBeforeComment {
+			j.AddString(newline)
+		}
+		j.AddBytes(crossChunkSuffix)
+	}
+
+	// Optionally wrap with an IIFE
+	if c.options.OutputFormat == config.FormatIIFE {
+		j.AddString("})();" + newline)
+	}
+
+	// Make sure the file ends with a newline
+	j.EnsureNewlineAtEnd()
+	slashTag := "/script"
+	if c.options.UnsupportedJSFeatures.Has(compat.InlineScript) {
+		slashTag = ""
+	}
+	c.maybeAppendLegalComments(c.options.LegalComments, legalCommentList, chunk, &j, slashTag)
+
+	if len(c.options.JSFooter) > 0 {
+		j.AddString(c.options.JSFooter)
+		j.AddString("\n")
+	}
+
+	// The JavaScript contents are done now that the source map comment is in
+	chunk.intermediateOutput = c.breakJoinerIntoPieces(j)
+	timer.End("Join JavaScript files")
+
+	if c.options.SourceMap != config.SourceMapNone {
+		timer.Begin("Generate source map")
+		canHaveShifts := chunk.intermediateOutput.pieces != nil
+		chunk.outputSourceMap = c.generateSourceMapForChunk(compileResultsForSourceMap, chunkAbsDir, dataForSourceMaps, canHaveShifts)
+		timer.End("Generate source map")
+	}
+
+	// End the metadata lazily. The final output size is not known until the
+	// final import paths are substituted into the output pieces generated below.
+	if c.options.NeedsMetafile {
+		pieces := make([][]intermediateOutput, len(metaOrder))
+		for i, sourceIndex := range metaOrder {
+			slices := metaBytes[sourceIndex]
+			outputs := make([]intermediateOutput, len(slices))
+			for j, slice := range slices {
+				outputs[j] = c.breakOutputIntoPieces(slice)
+			}
+			pieces[i] = outputs
+		}
+		chunk.jsonMetadataChunkCallback = func(finalOutputSize int) helpers.Joiner {
+			finalRelDir := c.fs.Dir(chunk.finalRelPath)
+			for i, sourceIndex := range metaOrder {
+				if i > 0 {
+					jMeta.AddString(",")
+				}
+				count := 0
+				for _, output := range pieces[i] {
+					count += c.accurateFinalByteCount(output, finalRelDir)
+				}
+				jMeta.AddString(fmt.Sprintf("\n        %s: {\n          \"bytesInOutput\": %d\n        %s}",
+					helpers.QuoteForJSON(c.graph.Files[sourceIndex].InputFile.Source.PrettyPath, c.options.ASCIIOnly),
+					count, c.generateExtraDataForFileJS(sourceIndex)))
+			}
+			if len(metaOrder) > 0 {
+				jMeta.AddString("\n      ")
+			}
+			jMeta.AddString(fmt.Sprintf("},\n      \"bytes\": %d\n    }", finalOutputSize))
+			return jMeta
+		}
+	}
+
+	c.generateIsolatedHashInParallel(chunk)
+	chunk.isExecutable = isExecutable
+	chunkWaitGroup.Done()
+}
+
+func (c *linkerContext) generateGlobalNamePrefix() string {
+	var text string
+	globalName := c.options.GlobalName
+	prefix := globalName[0]
+	space := " "
+	join := ";\n"
+
+	if c.options.MinifyWhitespace {
+		space = ""
+		join = ";"
+	}
+
+	// Use "||=" to make the code more compact when it's supported
+	if len(globalName) > 1 && !c.options.UnsupportedJSFeatures.Has(compat.LogicalAssignment) {
+		if js_printer.CanEscapeIdentifier(prefix, c.options.UnsupportedJSFeatures, c.options.ASCIIOnly) {
+			if c.options.ASCIIOnly {
+				prefix = string(js_printer.QuoteIdentifier(nil, prefix, c.options.UnsupportedJSFeatures))
+			}
+			text = fmt.Sprintf("var %s%s", prefix, join)
+		} else {
+			prefix = fmt.Sprintf("this[%s]", helpers.QuoteForJSON(prefix, c.options.ASCIIOnly))
+		}
+		for _, name := range globalName[1:] {
+			var dotOrIndex string
+			if js_printer.CanEscapeIdentifier(name, c.options.UnsupportedJSFeatures, c.options.ASCIIOnly) {
+				if c.options.ASCIIOnly {
+					name = string(js_printer.QuoteIdentifier(nil, name, c.options.UnsupportedJSFeatures))
+				}
+				dotOrIndex = fmt.Sprintf(".%s", name)
+			} else {
+				dotOrIndex = fmt.Sprintf("[%s]", helpers.QuoteForJSON(name, c.options.ASCIIOnly))
+			}
+			prefix = fmt.Sprintf("(%s%s||=%s{})%s", prefix, space, space, dotOrIndex)
+		}
+		return fmt.Sprintf("%s%s%s=%s", text, prefix, space, space)
+	}
+
+	if js_printer.CanEscapeIdentifier(prefix, c.options.UnsupportedJSFeatures, c.options.ASCIIOnly) {
+		if c.options.ASCIIOnly {
+			prefix = string(js_printer.QuoteIdentifier(nil, prefix, c.options.UnsupportedJSFeatures))
+		}
+		text = fmt.Sprintf("var %s%s=%s", prefix, space, space)
+	} else {
+		prefix = fmt.Sprintf("this[%s]", helpers.QuoteForJSON(prefix, c.options.ASCIIOnly))
+		text = fmt.Sprintf("%s%s=%s", prefix, space, space)
+	}
+
+	for _, name := range globalName[1:] {
+		oldPrefix := prefix
+		if js_printer.CanEscapeIdentifier(name, c.options.UnsupportedJSFeatures, c.options.ASCIIOnly) {
+			if c.options.ASCIIOnly {
+				name = string(js_printer.QuoteIdentifier(nil, name, c.options.UnsupportedJSFeatures))
+			}
+			prefix = fmt.Sprintf("%s.%s", prefix, name)
+		} else {
+			prefix = fmt.Sprintf("%s[%s]", prefix, helpers.QuoteForJSON(name, c.options.ASCIIOnly))
+		}
+		text += fmt.Sprintf("%s%s||%s{}%s%s%s=%s", oldPrefix, space, space, join, prefix, space, space)
+	}
+
+	return text
+}
+
+type compileResultCSS struct {
+	css_printer.PrintResult
+
+	// This is the line and column offset since the previous CSS string
+	// or the start of the file if this is the first CSS string.
+	generatedOffset sourcemap.LineColumnOffset
+
+	// The source index can be invalid for short snippets that aren't necessarily
+	// tied to any one file and/or that don't really need source mappings. The
+	// source index is really only valid for the compile result that contains the
+	// main contents of a file, which we try to only ever write out once.
+	sourceIndex ast.Index32
+	hasCharset  bool
+}
+
+func (c *linkerContext) generateChunkCSS(chunkIndex int, chunkWaitGroup *sync.WaitGroup) {
+	defer c.recoverInternalError(chunkWaitGroup, runtime.SourceIndex)
+
+	chunk := &c.chunks[chunkIndex]
+
+	timer := c.timer.Fork()
+	if timer != nil {
+		timeName := fmt.Sprintf("Generate chunk %q", path.Clean(config.TemplateToString(chunk.finalTemplate)))
+		timer.Begin(timeName)
+		defer c.timer.Join(timer)
+		defer timer.End(timeName)
+	}
+
+	chunkRepr := chunk.chunkRepr.(*chunkReprCSS)
+	compileResults := make([]compileResultCSS, len(chunkRepr.importsInChunkInOrder))
+	dataForSourceMaps := c.dataForSourceMaps()
+
+	// Note: This contains placeholders instead of what the placeholders are
+	// substituted with. That should be fine though because this should only
+	// ever be used for figuring out how many "../" to add to a relative path
+	// from a chunk whose final path hasn't been calculated yet to a chunk
+	// whose final path has already been calculated. That and placeholders are
+	// never substituted with something containing a "/" so substitution should
+	// never change the "../" count.
+	chunkAbsDir := c.fs.Dir(c.fs.Join(c.options.AbsOutputDir, config.TemplateToString(chunk.finalTemplate)))
+
+	// Remove duplicate rules across files. This must be done in serial, not
+	// in parallel, and must be done from the last rule to the first rule.
+	timer.Begin("Prepare CSS ASTs")
+	asts := make([]css_ast.AST, len(chunkRepr.importsInChunkInOrder))
+	var remover css_parser.DuplicateRuleRemover
+	if c.options.MinifySyntax {
+		remover = css_parser.MakeDuplicateRuleMangler(c.graph.Symbols)
+	}
+	for i := len(chunkRepr.importsInChunkInOrder) - 1; i >= 0; i-- {
+		entry := chunkRepr.importsInChunkInOrder[i]
+		switch entry.kind {
+		case cssImportLayers:
+			var rules []css_ast.Rule
+			if len(entry.layers) > 0 {
+				rules = append(rules, css_ast.Rule{Data: &css_ast.RAtLayer{Names: entry.layers}})
+			}
+			rules, importRecords := wrapRulesWithConditions(rules, nil, entry.conditions, entry.conditionImportRecords)
+			asts[i] = css_ast.AST{Rules: rules, ImportRecords: importRecords}
+
+		case cssImportExternalPath:
+			var conditions *css_ast.ImportConditions
+			if len(entry.conditions) > 0 {
+				conditions = &entry.conditions[0]
+
+				// Handling a chain of nested conditions is complicated. We can't
+				// necessarily join them together because a) there may be multiple
+				// layer names and b) layer names are only supposed to be inserted
+				// into the layer order if the parent conditions are applied.
+				//
+				// Instead we handle them by preserving the "@import" nesting using
+				// imports of data URL stylesheets. This may seem strange but I think
+				// this is the only way to do this in CSS.
+				for i := len(entry.conditions) - 1; i > 0; i-- {
+					astImport := css_ast.AST{
+						Rules: []css_ast.Rule{{Data: &css_ast.RAtImport{
+							ImportRecordIndex: uint32(len(entry.conditionImportRecords)),
+							ImportConditions:  &entry.conditions[i],
+						}}},
+						ImportRecords: append(entry.conditionImportRecords, ast.ImportRecord{
+							Kind: ast.ImportAt,
+							Path: entry.externalPath,
+						}),
+					}
+					astResult := css_printer.Print(astImport, c.graph.Symbols, css_printer.Options{
+						MinifyWhitespace: c.options.MinifyWhitespace,
+						ASCIIOnly:        c.options.ASCIIOnly,
+					})
+					entry.externalPath = logger.Path{Text: helpers.EncodeStringAsShortestDataURL("text/css", string(bytes.TrimSpace(astResult.CSS)))}
+				}
+			}
+			asts[i] = css_ast.AST{
+				ImportRecords: append(append([]ast.ImportRecord{}, entry.conditionImportRecords...), ast.ImportRecord{
+					Kind: ast.ImportAt,
+					Path: entry.externalPath,
+				}),
+				Rules: []css_ast.Rule{{Data: &css_ast.RAtImport{
+					ImportRecordIndex: uint32(len(entry.conditionImportRecords)),
+					ImportConditions:  conditions,
+				}}},
+			}
+
+		case cssImportSourceIndex:
+			file := &c.graph.Files[entry.sourceIndex]
+			ast := file.InputFile.Repr.(*graph.CSSRepr).AST
+
+			// Filter out "@charset", "@import", and leading "@layer" rules
+			rules := make([]css_ast.Rule, 0, len(ast.Rules))
+			didFindAtImport := false
+			didFindAtLayer := false
+			for _, rule := range ast.Rules {
+				switch rule.Data.(type) {
+				case *css_ast.RAtCharset:
+					compileResults[i].hasCharset = true
+					continue
+				case *css_ast.RAtLayer:
+					didFindAtLayer = true
+				case *css_ast.RAtImport:
+					if !didFindAtImport {
+						didFindAtImport = true
+						if didFindAtLayer {
+							// Filter out the pre-import layers once we see the first
+							// "@import". These layers are special-cased by the linker
+							// and already appear as a separate entry in import order.
+							end := 0
+							for _, rule := range rules {
+								if _, ok := rule.Data.(*css_ast.RAtLayer); !ok {
+									rules[end] = rule
+									end++
+								}
+							}
+							rules = rules[:end]
+						}
+					}
+					continue
+				}
+				rules = append(rules, rule)
+			}
+
+			rules, ast.ImportRecords = wrapRulesWithConditions(rules, ast.ImportRecords, entry.conditions, entry.conditionImportRecords)
+
+			// Remove top-level duplicate rules across files
+			if c.options.MinifySyntax {
+				rules = remover.RemoveDuplicateRulesInPlace(entry.sourceIndex, rules, ast.ImportRecords)
+			}
+
+			ast.Rules = rules
+			asts[i] = ast
+		}
+	}
+	timer.End("Prepare CSS ASTs")
+
+	// Generate CSS for each file in parallel
+	timer.Begin("Print CSS files")
+	waitGroup := sync.WaitGroup{}
+	for i, entry := range chunkRepr.importsInChunkInOrder {
+		// Create a goroutine for this file
+		waitGroup.Add(1)
+		go func(i int, entry cssImportOrder, compileResult *compileResultCSS) {
+			cssOptions := css_printer.Options{
+				MinifyWhitespace:    c.options.MinifyWhitespace,
+				LineLimit:           c.options.LineLimit,
+				ASCIIOnly:           c.options.ASCIIOnly,
+				LegalComments:       c.options.LegalComments,
+				SourceMap:           c.options.SourceMap,
+				UnsupportedFeatures: c.options.UnsupportedCSSFeatures,
+				NeedsMetafile:       c.options.NeedsMetafile,
+				LocalNames:          c.mangledProps,
+			}
+
+			if entry.kind == cssImportSourceIndex {
+				defer c.recoverInternalError(&waitGroup, entry.sourceIndex)
+				file := &c.graph.Files[entry.sourceIndex]
+
+				// Only generate a source map if needed
+				if file.InputFile.Loader.CanHaveSourceMap() && c.options.SourceMap != config.SourceMapNone {
+					cssOptions.AddSourceMappings = true
+					cssOptions.InputSourceMap = file.InputFile.InputSourceMap
+					cssOptions.LineOffsetTables = dataForSourceMaps[entry.sourceIndex].LineOffsetTables
+				}
+
+				cssOptions.InputSourceIndex = entry.sourceIndex
+				compileResult.sourceIndex = ast.MakeIndex32(entry.sourceIndex)
+			}
+
+			compileResult.PrintResult = css_printer.Print(asts[i], c.graph.Symbols, cssOptions)
+			waitGroup.Done()
+		}(i, entry, &compileResults[i])
+	}
+
+	waitGroup.Wait()
+	timer.End("Print CSS files")
+	timer.Begin("Join CSS files")
+	j := helpers.Joiner{}
+	prevOffset := sourcemap.LineColumnOffset{}
+	newlineBeforeComment := false
+
+	if len(c.options.CSSBanner) > 0 {
+		prevOffset.AdvanceString(c.options.CSSBanner)
+		j.AddString(c.options.CSSBanner)
+		prevOffset.AdvanceString("\n")
+		j.AddString("\n")
+	}
+
+	// Generate any prefix rules now
+	var jsonMetadataImports []string
+	{
+		tree := css_ast.AST{}
+
+		// "@charset" is the only thing that comes before "@import"
+		for _, compileResult := range compileResults {
+			if compileResult.hasCharset {
+				tree.Rules = append(tree.Rules, css_ast.Rule{Data: &css_ast.RAtCharset{Encoding: "UTF-8"}})
+				break
+			}
+		}
+
+		if len(tree.Rules) > 0 {
+			result := css_printer.Print(tree, c.graph.Symbols, css_printer.Options{
+				MinifyWhitespace: c.options.MinifyWhitespace,
+				LineLimit:        c.options.LineLimit,
+				ASCIIOnly:        c.options.ASCIIOnly,
+				NeedsMetafile:    c.options.NeedsMetafile,
+			})
+			jsonMetadataImports = result.JSONMetadataImports
+			if len(result.CSS) > 0 {
+				prevOffset.AdvanceBytes(result.CSS)
+				j.AddBytes(result.CSS)
+				newlineBeforeComment = true
+			}
+		}
+	}
+
+	// Start the metadata
+	jMeta := helpers.Joiner{}
+	if c.options.NeedsMetafile {
+		isFirstMeta := true
+		jMeta.AddString("{\n      \"imports\": [")
+		for _, json := range jsonMetadataImports {
+			if isFirstMeta {
+				isFirstMeta = false
+			} else {
+				jMeta.AddString(",")
+			}
+			jMeta.AddString(json)
+		}
+		for _, compileResult := range compileResults {
+			for _, json := range compileResult.JSONMetadataImports {
+				if isFirstMeta {
+					isFirstMeta = false
+				} else {
+					jMeta.AddString(",")
+				}
+				jMeta.AddString(json)
+			}
+		}
+		if !isFirstMeta {
+			jMeta.AddString("\n      ")
+		}
+		if chunk.isEntryPoint {
+			file := &c.graph.Files[chunk.sourceIndex]
+
+			// Do not generate "entryPoint" for CSS files that are the result of
+			// importing CSS into JavaScript. We want this to be a 1:1 relationship
+			// and there is already an output file for the JavaScript entry point.
+			if _, ok := file.InputFile.Repr.(*graph.CSSRepr); ok {
+				jMeta.AddString(fmt.Sprintf("],\n      \"entryPoint\": %s,\n      \"inputs\": {",
+					helpers.QuoteForJSON(file.InputFile.Source.PrettyPath, c.options.ASCIIOnly)))
+			} else {
+				jMeta.AddString("],\n      \"inputs\": {")
+			}
+		} else {
+			jMeta.AddString("],\n      \"inputs\": {")
+		}
+	}
+
+	// Concatenate the generated CSS chunks together
+	var compileResultsForSourceMap []compileResultForSourceMap
+	var legalCommentList []legalCommentEntry
+	for _, compileResult := range compileResults {
+		if len(compileResult.ExtractedLegalComments) > 0 && compileResult.sourceIndex.IsValid() {
+			legalCommentList = append(legalCommentList, legalCommentEntry{
+				sourceIndex: compileResult.sourceIndex.GetIndex(),
+				comments:    compileResult.ExtractedLegalComments,
+			})
+		}
+
+		if c.options.Mode == config.ModeBundle && !c.options.MinifyWhitespace && compileResult.sourceIndex.IsValid() {
+			var newline string
+			if newlineBeforeComment {
+				newline = "\n"
+			}
+			comment := fmt.Sprintf("%s/* %s */\n", newline, c.graph.Files[compileResult.sourceIndex.GetIndex()].InputFile.Source.PrettyPath)
+			prevOffset.AdvanceString(comment)
+			j.AddString(comment)
+		}
+		if len(compileResult.CSS) > 0 {
+			newlineBeforeComment = true
+		}
+
+		// Save the offset to the start of the stored JavaScript
+		compileResult.generatedOffset = prevOffset
+		j.AddBytes(compileResult.CSS)
+
+		// Ignore empty source map chunks
+		if compileResult.SourceMapChunk.ShouldIgnore {
+			prevOffset.AdvanceBytes(compileResult.CSS)
+		} else {
+			prevOffset = sourcemap.LineColumnOffset{}
+
+			// Include this file in the source map
+			if c.options.SourceMap != config.SourceMapNone && compileResult.sourceIndex.IsValid() {
+				compileResultsForSourceMap = append(compileResultsForSourceMap, compileResultForSourceMap{
+					sourceMapChunk:  compileResult.SourceMapChunk,
+					generatedOffset: compileResult.generatedOffset,
+					sourceIndex:     compileResult.sourceIndex.GetIndex(),
+				})
+			}
+		}
+	}
+
+	// Make sure the file ends with a newline
+	j.EnsureNewlineAtEnd()
+	slashTag := "/style"
+	if c.options.UnsupportedCSSFeatures.Has(compat.InlineStyle) {
+		slashTag = ""
+	}
+	c.maybeAppendLegalComments(c.options.LegalComments, legalCommentList, chunk, &j, slashTag)
+
+	if len(c.options.CSSFooter) > 0 {
+		j.AddString(c.options.CSSFooter)
+		j.AddString("\n")
+	}
+
+	// The CSS contents are done now that the source map comment is in
+	chunk.intermediateOutput = c.breakJoinerIntoPieces(j)
+	timer.End("Join CSS files")
+
+	if c.options.SourceMap != config.SourceMapNone {
+		timer.Begin("Generate source map")
+		canHaveShifts := chunk.intermediateOutput.pieces != nil
+		chunk.outputSourceMap = c.generateSourceMapForChunk(compileResultsForSourceMap, chunkAbsDir, dataForSourceMaps, canHaveShifts)
+		timer.End("Generate source map")
+	}
+
+	// End the metadata lazily. The final output size is not known until the
+	// final import paths are substituted into the output pieces generated below.
+	if c.options.NeedsMetafile {
+		pieces := make([]intermediateOutput, len(compileResults))
+		for i, compileResult := range compileResults {
+			pieces[i] = c.breakOutputIntoPieces(compileResult.CSS)
+		}
+		chunk.jsonMetadataChunkCallback = func(finalOutputSize int) helpers.Joiner {
+			finalRelDir := c.fs.Dir(chunk.finalRelPath)
+			isFirst := true
+			for i, compileResult := range compileResults {
+				if !compileResult.sourceIndex.IsValid() {
+					continue
+				}
+				if isFirst {
+					isFirst = false
+				} else {
+					jMeta.AddString(",")
+				}
+				jMeta.AddString(fmt.Sprintf("\n        %s: {\n          \"bytesInOutput\": %d\n        }",
+					helpers.QuoteForJSON(c.graph.Files[compileResult.sourceIndex.GetIndex()].InputFile.Source.PrettyPath, c.options.ASCIIOnly),
+					c.accurateFinalByteCount(pieces[i], finalRelDir)))
+			}
+			if len(compileResults) > 0 {
+				jMeta.AddString("\n      ")
+			}
+			jMeta.AddString(fmt.Sprintf("},\n      \"bytes\": %d\n    }", finalOutputSize))
+			return jMeta
+		}
+	}
+
+	c.generateIsolatedHashInParallel(chunk)
+	chunkWaitGroup.Done()
+}
+
+func wrapRulesWithConditions(
+	rules []css_ast.Rule, importRecords []ast.ImportRecord,
+	conditions []css_ast.ImportConditions, conditionImportRecords []ast.ImportRecord,
+) ([]css_ast.Rule, []ast.ImportRecord) {
+	for i := len(conditions) - 1; i >= 0; i-- {
+		item := conditions[i]
+
+		// Generate "@layer" wrappers. Note that empty "@layer" rules still have
+		// a side effect (they set the layer order) so they cannot be removed.
+		for _, t := range item.Layers {
+			if len(rules) == 0 {
+				if t.Children == nil {
+					// Omit an empty "@layer {}" entirely
+					continue
+				} else {
+					// Generate "@layer foo;" instead of "@layer foo {}"
+					rules = nil
+				}
+			}
+			var prelude []css_ast.Token
+			if t.Children != nil {
+				prelude = *t.Children
+			}
+			prelude, importRecords = css_ast.CloneTokensWithImportRecords(prelude, conditionImportRecords, nil, importRecords)
+			rules = []css_ast.Rule{{Data: &css_ast.RKnownAt{
+				AtToken: "layer",
+				Prelude: prelude,
+				Rules:   rules,
+			}}}
+		}
+
+		// Generate "@supports" wrappers. This is not done if the rule block is
+		// empty because empty "@supports" rules have no effect.
+		if len(rules) > 0 {
+			for _, t := range item.Supports {
+				t.Kind = css_lexer.TOpenParen
+				t.Text = "("
+				var prelude []css_ast.Token
+				prelude, importRecords = css_ast.CloneTokensWithImportRecords([]css_ast.Token{t}, conditionImportRecords, nil, importRecords)
+				rules = []css_ast.Rule{{Data: &css_ast.RKnownAt{
+					AtToken: "supports",
+					Prelude: prelude,
+					Rules:   rules,
+				}}}
+			}
+		}
+
+		// Generate "@media" wrappers. This is not done if the rule block is
+		// empty because empty "@media" rules have no effect.
+		if len(rules) > 0 && len(item.Media) > 0 {
+			var prelude []css_ast.Token
+			prelude, importRecords = css_ast.CloneTokensWithImportRecords(item.Media, conditionImportRecords, nil, importRecords)
+			rules = []css_ast.Rule{{Data: &css_ast.RKnownAt{
+				AtToken: "media",
+				Prelude: prelude,
+				Rules:   rules,
+			}}}
+		}
+	}
+
+	return rules, importRecords
+}
+
+type legalCommentEntry struct {
+	sourceIndex uint32
+	comments    []string
+}
+
+// Add all unique legal comments to the end of the file. These are
+// deduplicated because some projects have thousands of files with the same
+// comment. The comment must be preserved in the output for legal reasons but
+// at the same time we want to generate a small bundle when minifying.
+func (c *linkerContext) maybeAppendLegalComments(
+	legalComments config.LegalComments,
+	legalCommentList []legalCommentEntry,
+	chunk *chunkInfo,
+	j *helpers.Joiner,
+	slashTag string,
+) {
+	switch legalComments {
+	case config.LegalCommentsNone, config.LegalCommentsInline:
+		return
+	}
+
+	type thirdPartyEntry struct {
+		packagePath string
+		comments    []string
+	}
+
+	var uniqueFirstPartyComments []string
+	var thirdPartyComments []thirdPartyEntry
+	hasFirstPartyComment := make(map[string]struct{})
+
+	for _, entry := range legalCommentList {
+		source := c.graph.Files[entry.sourceIndex].InputFile.Source
+		packagePath := ""
+
+		// Try to extract a package name from the source path. If we can find a
+		// "node_modules" path component in the path, then assume this is a legal
+		// comment in third-party code and that everything after "node_modules" is
+		// the package name and subpath. If we can't, then assume this is a legal
+		// comment in first-party code.
+		//
+		// The rationale for this behavior: If we just include third-party comments
+		// as-is and the third-party comments don't say what package they're from
+		// (which isn't uncommon), then it'll look like that comment applies to
+		// all code in the file which is very wrong. So we need to somehow say
+		// where the comment comes from. But we don't want to say where every
+		// comment comes from because people probably won't appreciate this for
+		// first-party comments. And we don't want to include the whole path to
+		// each third-part module because a) that could contain information about
+		// the local machine that people don't want in their bundle and b) that
+		// could differ depending on unimportant details like the package manager
+		// used to install the packages (npm vs. pnpm vs. yarn).
+		if source.KeyPath.Namespace != "dataurl" {
+			path := source.KeyPath.Text
+			previous := len(path)
+			for previous > 0 {
+				slash := strings.LastIndexAny(path[:previous], "\\/")
+				component := path[slash+1 : previous]
+				if component == "node_modules" {
+					if previous < len(path) {
+						packagePath = strings.ReplaceAll(path[previous+1:], "\\", "/")
+					}
+					break
+				}
+				previous = slash
+			}
+		}
+
+		if packagePath != "" {
+			thirdPartyComments = append(thirdPartyComments, thirdPartyEntry{
+				packagePath: packagePath,
+				comments:    entry.comments,
+			})
+		} else {
+			for _, comment := range entry.comments {
+				if _, ok := hasFirstPartyComment[comment]; !ok {
+					hasFirstPartyComment[comment] = struct{}{}
+					uniqueFirstPartyComments = append(uniqueFirstPartyComments, comment)
+				}
+			}
+		}
+	}
+
+	switch legalComments {
+	case config.LegalCommentsEndOfFile:
+		for _, comment := range uniqueFirstPartyComments {
+			j.AddString(helpers.EscapeClosingTag(comment, slashTag))
+			j.AddString("\n")
+		}
+
+		if len(thirdPartyComments) > 0 {
+			j.AddString("/*! Bundled license information:\n")
+			for _, entry := range thirdPartyComments {
+				j.AddString(fmt.Sprintf("\n%s:\n", helpers.EscapeClosingTag(entry.packagePath, slashTag)))
+				for _, comment := range entry.comments {
+					comment = helpers.EscapeClosingTag(comment, slashTag)
+					if strings.HasPrefix(comment, "//") {
+						j.AddString(fmt.Sprintf("  (*%s *)\n", comment[2:]))
+					} else if strings.HasPrefix(comment, "/*") && strings.HasSuffix(comment, "*/") {
+						j.AddString(fmt.Sprintf("  (%s)\n", strings.ReplaceAll(comment[1:len(comment)-1], "\n", "\n  ")))
+					}
+				}
+			}
+			j.AddString("*/\n")
+		}
+
+	case config.LegalCommentsLinkedWithComment, config.LegalCommentsExternalWithoutComment:
+		var jComments helpers.Joiner
+
+		for _, comment := range uniqueFirstPartyComments {
+			jComments.AddString(comment)
+			jComments.AddString("\n")
+		}
+
+		if len(thirdPartyComments) > 0 {
+			if len(uniqueFirstPartyComments) > 0 {
+				jComments.AddString("\n")
+			}
+			jComments.AddString("Bundled license information:\n")
+			for _, entry := range thirdPartyComments {
+				jComments.AddString(fmt.Sprintf("\n%s:\n", entry.packagePath))
+				for _, comment := range entry.comments {
+					jComments.AddString(fmt.Sprintf("  %s\n", strings.ReplaceAll(comment, "\n", "\n  ")))
+				}
+			}
+		}
+
+		chunk.externalLegalComments = jComments.Done()
+	}
+}
+
+func (c *linkerContext) appendIsolatedHashesForImportedChunks(
+	hash hash.Hash,
+	chunkIndex uint32,
+	visited []uint32,
+	visitedKey uint32,
+) {
+	// Only visit each chunk at most once. This is important because there may be
+	// cycles in the chunk import graph. If there's a cycle, we want to include
+	// the hash of every chunk involved in the cycle (along with all of their
+	// dependencies). This depth-first traversal will naturally do that.
+	if visited[chunkIndex] == visitedKey {
+		return
+	}
+	visited[chunkIndex] = visitedKey
+	chunk := &c.chunks[chunkIndex]
+
+	// Visit the other chunks that this chunk imports before visiting this chunk
+	for _, chunkImport := range chunk.crossChunkImports {
+		c.appendIsolatedHashesForImportedChunks(hash, chunkImport.chunkIndex, visited, visitedKey)
+	}
+
+	// Mix in hashes for referenced asset paths (i.e. the "file" loader)
+	for _, piece := range chunk.intermediateOutput.pieces {
+		if piece.kind == outputPieceAssetIndex {
+			file := c.graph.Files[piece.index]
+			if len(file.InputFile.AdditionalFiles) != 1 {
+				panic("Internal error")
+			}
+			relPath, _ := c.fs.Rel(c.options.AbsOutputDir, file.InputFile.AdditionalFiles[0].AbsPath)
+
+			// Make sure to always use forward slashes, even on Windows
+			relPath = strings.ReplaceAll(relPath, "\\", "/")
+
+			// Mix in the hash for the relative path, which ends up as a JS string
+			hashWriteLengthPrefixed(hash, []byte(relPath))
+		}
+	}
+
+	// Mix in the hash for this chunk
+	hash.Write(chunk.waitForIsolatedHash())
+}
+
+func (c *linkerContext) breakJoinerIntoPieces(j helpers.Joiner) intermediateOutput {
+	// Optimization: If there can be no substitutions, just reuse the initial
+	// joiner that was used when generating the intermediate chunk output
+	// instead of creating another one and copying the whole file into it.
+	if !j.Contains(c.uniqueKeyPrefix, c.uniqueKeyPrefixBytes) {
+		return intermediateOutput{joiner: j}
+	}
+	return c.breakOutputIntoPieces(j.Done())
+}
+
+func (c *linkerContext) breakOutputIntoPieces(output []byte) intermediateOutput {
+	var pieces []outputPiece
+	prefix := c.uniqueKeyPrefixBytes
+	for {
+		// Scan for the next piece boundary
+		boundary := bytes.Index(output, prefix)
+
+		// Try to parse the piece boundary
+		var kind outputPieceIndexKind
+		var index uint32
+		if boundary != -1 {
+			if start := boundary + len(prefix); start+9 > len(output) {
+				boundary = -1
+			} else {
+				switch output[start] {
+				case 'A':
+					kind = outputPieceAssetIndex
+				case 'C':
+					kind = outputPieceChunkIndex
+				}
+				for j := 1; j < 9; j++ {
+					c := output[start+j]
+					if c < '0' || c > '9' {
+						boundary = -1
+						break
+					}
+					index = index*10 + uint32(c) - '0'
+				}
+			}
+		}
+
+		// Validate the boundary
+		switch kind {
+		case outputPieceAssetIndex:
+			if index >= uint32(len(c.graph.Files)) {
+				boundary = -1
+			}
+
+		case outputPieceChunkIndex:
+			if index >= uint32(len(c.chunks)) {
+				boundary = -1
+			}
+
+		default:
+			boundary = -1
+		}
+
+		// If we're at the end, generate one final piece
+		if boundary == -1 {
+			pieces = append(pieces, outputPiece{
+				data: output,
+			})
+			break
+		}
+
+		// Otherwise, generate an interior piece and continue
+		pieces = append(pieces, outputPiece{
+			data:  output[:boundary],
+			index: index,
+			kind:  kind,
+		})
+		output = output[boundary+len(prefix)+9:]
+	}
+	return intermediateOutput{pieces: pieces}
+}
+
+func (c *linkerContext) generateIsolatedHashInParallel(chunk *chunkInfo) {
+	// Compute the hash in parallel. This is a speedup when it turns out the hash
+	// isn't needed (well, as long as there are threads to spare).
+	channel := make(chan []byte, 1)
+	chunk.waitForIsolatedHash = func() []byte {
+		data := <-channel
+		channel <- data
+		return data
+	}
+	go c.generateIsolatedHash(chunk, channel)
+}
+
+func (c *linkerContext) generateIsolatedHash(chunk *chunkInfo, channel chan []byte) {
+	hash := xxhash.New()
+
+	// Mix the file names and part ranges of all of the files in this chunk into
+	// the hash. Objects that appear identical but that live in separate files or
+	// that live in separate parts in the same file must not be merged. This only
+	// needs to be done for JavaScript files, not CSS files.
+	if chunkRepr, ok := chunk.chunkRepr.(*chunkReprJS); ok {
+		for _, partRange := range chunkRepr.partsInChunkInOrder {
+			var filePath string
+			file := &c.graph.Files[partRange.sourceIndex]
+			if file.InputFile.Source.KeyPath.Namespace == "file" {
+				// Use the pretty path as the file name since it should be platform-
+				// independent (relative paths and the "/" path separator)
+				filePath = file.InputFile.Source.PrettyPath
+			} else {
+				// If this isn't in the "file" namespace, just use the full path text
+				// verbatim. This could be a source of cross-platform differences if
+				// plugins are storing platform-specific information in here, but then
+				// that problem isn't caused by esbuild itself.
+				filePath = file.InputFile.Source.KeyPath.Text
+			}
+
+			// Include the path namespace in the hash
+			hashWriteLengthPrefixed(hash, []byte(file.InputFile.Source.KeyPath.Namespace))
+
+			// Then include the file path
+			hashWriteLengthPrefixed(hash, []byte(filePath))
+
+			// Also write the part range. These numbers are deterministic and allocated
+			// per-file so this should be a well-behaved base for a hash.
+			hashWriteUint32(hash, partRange.partIndexBegin)
+			hashWriteUint32(hash, partRange.partIndexEnd)
+		}
+	}
+
+	// Hash the output path template as part of the content hash because we want
+	// any import to be considered different if the import's output path has changed.
+	for _, part := range chunk.finalTemplate {
+		hashWriteLengthPrefixed(hash, []byte(part.Data))
+	}
+
+	// Also hash the public path. If provided, this is used whenever files
+	// reference each other such as cross-chunk imports, asset file references,
+	// and source map comments. We always include the hash in all chunks instead
+	// of trying to figure out which chunks will include the public path for
+	// simplicity and for robustness to code changes in the future.
+	if c.options.PublicPath != "" {
+		hashWriteLengthPrefixed(hash, []byte(c.options.PublicPath))
+	}
+
+	// Include the generated output content in the hash. This excludes the
+	// randomly-generated import paths (the unique keys) and only includes the
+	// data in the spans between them.
+	if chunk.intermediateOutput.pieces != nil {
+		for _, piece := range chunk.intermediateOutput.pieces {
+			hashWriteLengthPrefixed(hash, piece.data)
+		}
+	} else {
+		bytes := chunk.intermediateOutput.joiner.Done()
+		hashWriteLengthPrefixed(hash, bytes)
+	}
+
+	// Also include the source map data in the hash. The source map is named the
+	// same name as the chunk name for ease of discovery. So we want the hash to
+	// change if the source map data changes even if the chunk data doesn't change.
+	// Otherwise the output path for the source map wouldn't change and the source
+	// map wouldn't end up being updated.
+	//
+	// Note that this means the contents of all input files are included in the
+	// hash because of "sourcesContent", so changing a comment in an input file
+	// can now change the hash of the output file. This only happens when you
+	// have source maps enabled (and "sourcesContent", which is on by default).
+	//
+	// The generated positions in the mappings here are in the output content
+	// *before* the final paths have been substituted. This may seem weird.
+	// However, I think this shouldn't cause issues because a) the unique key
+	// values are all always the same length so the offsets are deterministic
+	// and b) the final paths will be folded into the final hash later.
+	hashWriteLengthPrefixed(hash, chunk.outputSourceMap.Prefix)
+	hashWriteLengthPrefixed(hash, chunk.outputSourceMap.Mappings)
+	hashWriteLengthPrefixed(hash, chunk.outputSourceMap.Suffix)
+
+	// Store the hash so far. All other chunks that import this chunk will mix
+	// this hash into their final hash to ensure that the import path changes
+	// if this chunk (or any dependencies of this chunk) is changed.
+	channel <- hash.Sum(nil)
+}
+
+func hashWriteUint32(hash hash.Hash, value uint32) {
+	var lengthBytes [4]byte
+	binary.LittleEndian.PutUint32(lengthBytes[:], value)
+	hash.Write(lengthBytes[:])
+}
+
+// Hash the data in length-prefixed form because boundary locations are
+// important. We don't want "a" + "bc" to hash the same as "ab" + "c".
+func hashWriteLengthPrefixed(hash hash.Hash, bytes []byte) {
+	hashWriteUint32(hash, uint32(len(bytes)))
+	hash.Write(bytes)
+}
+
+// Marking a symbol as unbound prevents it from being renamed or minified.
+// This is only used when a module is compiled independently. We use a very
+// different way of handling exports and renaming/minifying when bundling.
+func (c *linkerContext) preventExportsFromBeingRenamed(sourceIndex uint32) {
+	repr, ok := c.graph.Files[sourceIndex].InputFile.Repr.(*graph.JSRepr)
+	if !ok {
+		return
+	}
+	hasImportOrExport := false
+
+	for _, part := range repr.AST.Parts {
+		for _, stmt := range part.Stmts {
+			switch s := stmt.Data.(type) {
+			case *js_ast.SImport:
+				// Ignore imports from internal (i.e. non-external) code. Since this
+				// function is only called when we're not bundling, these imports are
+				// all for files that were generated automatically and aren't part of
+				// the original source code (e.g. the runtime or an injected file).
+				// We shouldn't consider the file a module if the only ESM imports or
+				// exports are automatically generated ones.
+				if repr.AST.ImportRecords[s.ImportRecordIndex].SourceIndex.IsValid() {
+					continue
+				}
+
+				hasImportOrExport = true
+
+			case *js_ast.SLocal:
+				if s.IsExport {
+					js_ast.ForEachIdentifierBindingInDecls(s.Decls, func(loc logger.Loc, b *js_ast.BIdentifier) {
+						c.graph.Symbols.Get(b.Ref).Flags |= ast.MustNotBeRenamed
+					})
+					hasImportOrExport = true
+				}
+
+			case *js_ast.SFunction:
+				if s.IsExport {
+					c.graph.Symbols.Get(s.Fn.Name.Ref).Kind = ast.SymbolUnbound
+					hasImportOrExport = true
+				}
+
+			case *js_ast.SClass:
+				if s.IsExport {
+					c.graph.Symbols.Get(s.Class.Name.Ref).Kind = ast.SymbolUnbound
+					hasImportOrExport = true
+				}
+
+			case *js_ast.SExportClause, *js_ast.SExportDefault, *js_ast.SExportStar:
+				hasImportOrExport = true
+
+			case *js_ast.SExportFrom:
+				hasImportOrExport = true
+			}
+		}
+	}
+
+	// Heuristic: If this module has top-level import or export statements, we
+	// consider this an ES6 module and only preserve the names of the exported
+	// symbols. Everything else is minified since the names are private.
+	//
+	// Otherwise, we consider this potentially a script-type file instead of an
+	// ES6 module. In that case, preserve the names of all top-level symbols
+	// since they are all potentially exported (e.g. if this is used in a
+	// <script> tag). All symbols in nested scopes are still minified.
+	if !hasImportOrExport {
+		for _, member := range repr.AST.ModuleScope.Members {
+			c.graph.Symbols.Get(member.Ref).Flags |= ast.MustNotBeRenamed
+		}
+	}
+}
+
+type compileResultForSourceMap struct {
+	sourceMapChunk  sourcemap.Chunk
+	generatedOffset sourcemap.LineColumnOffset
+	sourceIndex     uint32
+}
+
+func (c *linkerContext) generateSourceMapForChunk(
+	results []compileResultForSourceMap,
+	chunkAbsDir string,
+	dataForSourceMaps []bundler.DataForSourceMap,
+	canHaveShifts bool,
+) (pieces sourcemap.SourceMapPieces) {
+	j := helpers.Joiner{}
+	j.AddString("{\n  \"version\": 3")
+
+	// Only write out the sources for a given source index once
+	sourceIndexToSourcesIndex := make(map[uint32]int)
+
+	// Generate the "sources" and "sourcesContent" arrays
+	type item struct {
+		path           logger.Path
+		prettyPath     string
+		quotedContents []byte
+	}
+	items := make([]item, 0, len(results))
+	nextSourcesIndex := 0
+	for _, result := range results {
+		if _, ok := sourceIndexToSourcesIndex[result.sourceIndex]; ok {
+			continue
+		}
+		sourceIndexToSourcesIndex[result.sourceIndex] = nextSourcesIndex
+		file := &c.graph.Files[result.sourceIndex]
+
+		// Simple case: no nested source map
+		if file.InputFile.InputSourceMap == nil {
+			var quotedContents []byte
+			if !c.options.ExcludeSourcesContent {
+				quotedContents = dataForSourceMaps[result.sourceIndex].QuotedContents[0]
+			}
+			items = append(items, item{
+				path:           file.InputFile.Source.KeyPath,
+				prettyPath:     file.InputFile.Source.PrettyPath,
+				quotedContents: quotedContents,
+			})
+			nextSourcesIndex++
+			continue
+		}
+
+		// Complex case: nested source map
+		sm := file.InputFile.InputSourceMap
+		for i, source := range sm.Sources {
+			path := logger.Path{
+				Namespace: file.InputFile.Source.KeyPath.Namespace,
+				Text:      source,
+			}
+
+			// If this file is in the "file" namespace, change the relative path in
+			// the source map into an absolute path using the directory of this file
+			if path.Namespace == "file" {
+				path.Text = c.fs.Join(c.fs.Dir(file.InputFile.Source.KeyPath.Text), source)
+			}
+
+			var quotedContents []byte
+			if !c.options.ExcludeSourcesContent {
+				quotedContents = dataForSourceMaps[result.sourceIndex].QuotedContents[i]
+			}
+			items = append(items, item{
+				path:           path,
+				prettyPath:     source,
+				quotedContents: quotedContents,
+			})
+		}
+		nextSourcesIndex += len(sm.Sources)
+	}
+
+	// Write the sources
+	j.AddString(",\n  \"sources\": [")
+	for i, item := range items {
+		if i != 0 {
+			j.AddString(", ")
+		}
+
+		// Modify the absolute path to the original file to be relative to the
+		// directory that will contain the output file for this chunk
+		if item.path.Namespace == "file" {
+			if relPath, ok := c.fs.Rel(chunkAbsDir, item.path.Text); ok {
+				// Make sure to always use forward slashes, even on Windows
+				item.prettyPath = strings.ReplaceAll(relPath, "\\", "/")
+			}
+		}
+
+		j.AddBytes(helpers.QuoteForJSON(item.prettyPath, c.options.ASCIIOnly))
+	}
+	j.AddString("]")
+
+	if c.options.SourceRoot != "" {
+		j.AddString(",\n  \"sourceRoot\": ")
+		j.AddBytes(helpers.QuoteForJSON(c.options.SourceRoot, c.options.ASCIIOnly))
+	}
+
+	// Write the sourcesContent
+	if !c.options.ExcludeSourcesContent {
+		j.AddString(",\n  \"sourcesContent\": [")
+		for i, item := range items {
+			if i != 0 {
+				j.AddString(", ")
+			}
+			j.AddBytes(item.quotedContents)
+		}
+		j.AddString("]")
+	}
+
+	j.AddString(",\n  \"mappings\": \"")
+
+	// Write the mappings
+	mappingsStart := j.Length()
+	prevEndState := sourcemap.SourceMapState{}
+	prevColumnOffset := 0
+	totalQuotedNameLen := 0
+	for _, result := range results {
+		chunk := result.sourceMapChunk
+		offset := result.generatedOffset
+		sourcesIndex := sourceIndexToSourcesIndex[result.sourceIndex]
+
+		// This should have already been checked earlier
+		if chunk.ShouldIgnore {
+			panic("Internal error")
+		}
+
+		// Because each file for the bundle is converted to a source map once,
+		// the source maps are shared between all entry points in the bundle.
+		// The easiest way of getting this to work is to have all source maps
+		// generate as if their source index is 0. We then adjust the source
+		// index per entry point by modifying the first source mapping. This
+		// is done by AppendSourceMapChunk() using the source index passed
+		// here.
+		startState := sourcemap.SourceMapState{
+			SourceIndex:     sourcesIndex,
+			GeneratedLine:   offset.Lines,
+			GeneratedColumn: offset.Columns,
+			OriginalName:    totalQuotedNameLen,
+		}
+		if offset.Lines == 0 {
+			startState.GeneratedColumn += prevColumnOffset
+		}
+
+		// Append the precomputed source map chunk
+		sourcemap.AppendSourceMapChunk(&j, prevEndState, startState, chunk.Buffer)
+
+		// Generate the relative offset to start from next time
+		prevOriginalName := prevEndState.OriginalName
+		prevEndState = chunk.EndState
+		prevEndState.SourceIndex += sourcesIndex
+		if chunk.Buffer.FirstNameOffset.IsValid() {
+			prevEndState.OriginalName += totalQuotedNameLen
+		} else {
+			// It's possible for a chunk to have mappings but for none of those
+			// mappings to have an associated name. The name is optional and is
+			// omitted when the mapping is for a non-name token or if the final
+			// and original names are the same. In that case we need to restore
+			// the previous original name end state since it wasn't modified after
+			// all. If we don't do this, then files after this will adjust their
+			// name offsets assuming that the previous generated mapping has this
+			// file's offset, which is wrong.
+			prevEndState.OriginalName = prevOriginalName
+		}
+		prevColumnOffset = chunk.FinalGeneratedColumn
+		totalQuotedNameLen += len(chunk.QuotedNames)
+
+		// If this was all one line, include the column offset from the start
+		if prevEndState.GeneratedLine == 0 {
+			prevEndState.GeneratedColumn += startState.GeneratedColumn
+			prevColumnOffset += startState.GeneratedColumn
+		}
+	}
+	mappingsEnd := j.Length()
+
+	// Write the names
+	isFirstName := true
+	j.AddString("\",\n  \"names\": [")
+	for _, result := range results {
+		for _, quotedName := range result.sourceMapChunk.QuotedNames {
+			if isFirstName {
+				isFirstName = false
+			} else {
+				j.AddString(", ")
+			}
+			j.AddBytes(quotedName)
+		}
+	}
+	j.AddString("]")
+
+	// Finish the source map
+	j.AddString("\n}\n")
+	bytes := j.Done()
+
+	if !canHaveShifts {
+		// If there cannot be any shifts, then we can avoid doing extra work later
+		// on by preserving the source map as a single memory allocation throughout
+		// the pipeline. That way we won't need to reallocate it.
+		pieces.Prefix = bytes
+	} else {
+		// Otherwise if there can be shifts, then we need to split this into several
+		// slices so that the shifts in the mappings array can be processed. This is
+		// more expensive because everything will need to be recombined into a new
+		// memory allocation at the end.
+		pieces.Prefix = bytes[:mappingsStart]
+		pieces.Mappings = bytes[mappingsStart:mappingsEnd]
+		pieces.Suffix = bytes[mappingsEnd:]
+	}
+	return
+}
+
+// Recover from a panic by logging it as an internal error instead of crashing
+func (c *linkerContext) recoverInternalError(waitGroup *sync.WaitGroup, sourceIndex uint32) {
+	if r := recover(); r != nil {
+		text := fmt.Sprintf("panic: %v", r)
+		if sourceIndex != runtime.SourceIndex {
+			text = fmt.Sprintf("%s (while printing %q)", text, c.graph.Files[sourceIndex].InputFile.Source.PrettyPath)
+		}
+		c.log.AddErrorWithNotes(nil, logger.Range{}, text,
+			[]logger.MsgData{{Text: helpers.PrettyPrintedStack()}})
+		waitGroup.Done()
+	}
+}
+
+func joinWithPublicPath(publicPath string, relPath string) string {
+	if strings.HasPrefix(relPath, "./") {
+		relPath = relPath[2:]
+
+		// Strip any amount of further no-op slashes (i.e. ".///././/x/y" => "x/y")
+		for {
+			if strings.HasPrefix(relPath, "/") {
+				relPath = relPath[1:]
+			} else if strings.HasPrefix(relPath, "./") {
+				relPath = relPath[2:]
+			} else {
+				break
+			}
+		}
+	}
+
+	// Use a relative path if there is no public path
+	if publicPath == "" {
+		publicPath = "."
+	}
+
+	// Join with a slash
+	slash := "/"
+	if strings.HasSuffix(publicPath, "/") {
+		slash = ""
+	}
+	return fmt.Sprintf("%s%s%s", publicPath, slash, relPath)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/logger/logger.go b/source/vendor/github.com/evanw/esbuild/internal/logger/logger.go
new file mode 100644
index 0000000..8acb904
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/logger/logger.go
@@ -0,0 +1,2045 @@
+package logger
+
+// Logging is either done to stderr (via "NewStderrLog") or to an in-memory
+// array (via "NewDeferLog"). In-memory arrays are used to capture messages
+// from parsing individual files because during incremental builds, log
+// messages for a given file can be replayed from memory if the file ends up
+// not being reparsed.
+//
+// Errors are streamed asynchronously as they happen, each error contains the
+// contents of the line with the error, and the error count is limited by
+// default.
+
+import (
+	"encoding/binary"
+	"fmt"
+	"os"
+	"runtime"
+	"sort"
+	"strings"
+	"sync"
+	"time"
+	"unicode/utf8"
+)
+
+const defaultTerminalWidth = 80
+
+type Log struct {
+	AddMsg    func(Msg)
+	HasErrors func() bool
+	Peek      func() []Msg
+
+	Done func() []Msg
+
+	Level     LogLevel
+	Overrides map[MsgID]LogLevel
+}
+
+type LogLevel int8
+
+const (
+	LevelNone LogLevel = iota
+	LevelVerbose
+	LevelDebug
+	LevelInfo
+	LevelWarning
+	LevelError
+	LevelSilent
+)
+
+type MsgKind uint8
+
+const (
+	Error MsgKind = iota
+	Warning
+	Info
+	Note
+	Debug
+	Verbose
+)
+
+func (kind MsgKind) String() string {
+	switch kind {
+	case Error:
+		return "ERROR"
+	case Warning:
+		return "WARNING"
+	case Info:
+		return "INFO"
+	case Note:
+		return "NOTE"
+	case Debug:
+		return "DEBUG"
+	case Verbose:
+		return "VERBOSE"
+	default:
+		panic("Internal error")
+	}
+}
+
+func (kind MsgKind) Icon() string {
+	// Special-case Windows command prompt, which only supports a few characters
+	if isProbablyWindowsCommandPrompt() {
+		switch kind {
+		case Error:
+			return "X"
+		case Warning:
+			return "▲"
+		case Info:
+			return "►"
+		case Note:
+			return "→"
+		case Debug:
+			return "●"
+		case Verbose:
+			return "♦"
+		default:
+			panic("Internal error")
+		}
+	}
+
+	switch kind {
+	case Error:
+		return "✘"
+	case Warning:
+		return "▲"
+	case Info:
+		return "▶"
+	case Note:
+		return "→"
+	case Debug:
+		return "●"
+	case Verbose:
+		return "⬥"
+	default:
+		panic("Internal error")
+	}
+}
+
+var windowsCommandPrompt struct {
+	mutex         sync.Mutex
+	once          bool
+	isProbablyCMD bool
+}
+
+func isProbablyWindowsCommandPrompt() bool {
+	windowsCommandPrompt.mutex.Lock()
+	defer windowsCommandPrompt.mutex.Unlock()
+
+	if !windowsCommandPrompt.once {
+		windowsCommandPrompt.once = true
+
+		// Assume we are running in Windows Command Prompt if we're on Windows. If
+		// so, we can't use emoji because it won't be supported. Except we can
+		// still use emoji if the WT_SESSION environment variable is present
+		// because that means we're running in the new Windows Terminal instead.
+		if runtime.GOOS == "windows" {
+			windowsCommandPrompt.isProbablyCMD = true
+			if _, ok := os.LookupEnv("WT_SESSION"); ok {
+				windowsCommandPrompt.isProbablyCMD = false
+			}
+		}
+	}
+
+	return windowsCommandPrompt.isProbablyCMD
+}
+
+type Msg struct {
+	Notes      []MsgData
+	PluginName string
+	Data       MsgData
+	Kind       MsgKind
+	ID         MsgID
+}
+
+type MsgData struct {
+	// Optional user-specified data that is passed through unmodified
+	UserDetail interface{}
+
+	Location *MsgLocation
+	Text     string
+
+	DisableMaximumWidth bool
+}
+
+type MsgLocation struct {
+	File       string
+	Namespace  string
+	LineText   string
+	Suggestion string
+	Line       int // 1-based
+	Column     int // 0-based, in bytes
+	Length     int // in bytes
+}
+
+type Loc struct {
+	// This is the 0-based index of this location from the start of the file, in bytes
+	Start int32
+}
+
+type Range struct {
+	Loc Loc
+	Len int32
+}
+
+func (r Range) End() int32 {
+	return r.Loc.Start + r.Len
+}
+
+func (a *Range) ExpandBy(b Range) {
+	if a.Len == 0 {
+		*a = b
+	} else {
+		end := a.End()
+		if n := b.End(); n > end {
+			end = n
+		}
+		if b.Loc.Start < a.Loc.Start {
+			a.Loc.Start = b.Loc.Start
+		}
+		a.Len = end - a.Loc.Start
+	}
+}
+
+type Span struct {
+	Text  string
+	Range Range
+}
+
+// This type is just so we can use Go's native sort function
+type SortableMsgs []Msg
+
+func (a SortableMsgs) Len() int          { return len(a) }
+func (a SortableMsgs) Swap(i int, j int) { a[i], a[j] = a[j], a[i] }
+
+func (a SortableMsgs) Less(i int, j int) bool {
+	ai := a[i]
+	aj := a[j]
+	aiLoc := ai.Data.Location
+	ajLoc := aj.Data.Location
+	if aiLoc == nil || ajLoc == nil {
+		return aiLoc == nil && ajLoc != nil
+	}
+	if aiLoc.File != ajLoc.File {
+		return aiLoc.File < ajLoc.File
+	}
+	if aiLoc.Line != ajLoc.Line {
+		return aiLoc.Line < ajLoc.Line
+	}
+	if aiLoc.Column != ajLoc.Column {
+		return aiLoc.Column < ajLoc.Column
+	}
+	if ai.Kind != aj.Kind {
+		return ai.Kind < aj.Kind
+	}
+	return ai.Data.Text < aj.Data.Text
+}
+
+// This is used to represent both file system paths (Namespace == "file") and
+// abstract module paths (Namespace != "file"). Abstract module paths represent
+// "virtual modules" when used for an input file and "package paths" when used
+// to represent an external module.
+type Path struct {
+	Text      string
+	Namespace string
+
+	// This feature was added to support ancient CSS libraries that append things
+	// like "?#iefix" and "#icons" to some of their import paths as a hack for IE6.
+	// The intent is for these suffix parts to be ignored but passed through to
+	// the output. This is supported by other bundlers, so we also support this.
+	IgnoredSuffix string
+
+	// Import attributes (the "with" keyword after an import) can affect path
+	// resolution. In other words, two paths in the same file that are otherwise
+	// equal but that have different import attributes may resolve to different
+	// paths.
+	ImportAttributes ImportAttributes
+
+	Flags PathFlags
+}
+
+// We rely on paths as map keys. Go doesn't support custom hash codes and
+// only implements hash codes for certain types. In particular, hash codes
+// are implemented for strings but not for arrays of strings. So we have to
+// pack these import attributes into a string.
+type ImportAttributes struct {
+	packedData string
+}
+
+type ImportAttribute struct {
+	Key   string
+	Value string
+}
+
+// This returns a sorted array instead of a map to make determinism easier
+func (attrs ImportAttributes) DecodeIntoArray() (result []ImportAttribute) {
+	if attrs.packedData == "" {
+		return nil
+	}
+	bytes := []byte(attrs.packedData)
+	for len(bytes) > 0 {
+		kn := 4 + binary.LittleEndian.Uint32(bytes[:4])
+		k := string(bytes[4:kn])
+		bytes = bytes[kn:]
+		vn := 4 + binary.LittleEndian.Uint32(bytes[:4])
+		v := string(bytes[4:vn])
+		bytes = bytes[vn:]
+		result = append(result, ImportAttribute{Key: k, Value: v})
+	}
+	return result
+}
+
+func (attrs ImportAttributes) DecodeIntoMap() (result map[string]string) {
+	if array := attrs.DecodeIntoArray(); len(array) > 0 {
+		result = make(map[string]string, len(array))
+		for _, attr := range array {
+			result[attr.Key] = attr.Value
+		}
+	}
+	return
+}
+
+func EncodeImportAttributes(value map[string]string) ImportAttributes {
+	if len(value) == 0 {
+		return ImportAttributes{}
+	}
+	keys := make([]string, 0, len(value))
+	for k := range value {
+		keys = append(keys, k)
+	}
+	sort.Strings(keys)
+	var sb strings.Builder
+	var n [4]byte
+	for _, k := range keys {
+		v := value[k]
+		binary.LittleEndian.PutUint32(n[:], uint32(len(k)))
+		sb.Write(n[:])
+		sb.WriteString(k)
+		binary.LittleEndian.PutUint32(n[:], uint32(len(v)))
+		sb.Write(n[:])
+		sb.WriteString(v)
+	}
+	return ImportAttributes{packedData: sb.String()}
+}
+
+type PathFlags uint8
+
+const (
+	// This corresponds to a value of "false' in the "browser" package.json field
+	PathDisabled PathFlags = 1 << iota
+)
+
+func (p Path) IsDisabled() bool {
+	return (p.Flags & PathDisabled) != 0
+}
+
+var noColorResult bool
+var noColorOnce sync.Once
+
+func hasNoColorEnvironmentVariable() bool {
+	noColorOnce.Do(func() {
+		// Read "NO_COLOR" from the environment. This is a convention that some
+		// software follows. See https://no-color.org/ for more information.
+		if _, ok := os.LookupEnv("NO_COLOR"); ok {
+			noColorResult = true
+		}
+	})
+	return noColorResult
+}
+
+// This has a custom implementation instead of using "filepath.Dir/Base/Ext"
+// because it should work the same on Unix and Windows. These names end up in
+// the generated output and the generated output should not depend on the OS.
+func PlatformIndependentPathDirBaseExt(path string) (dir string, base string, ext string) {
+	absRootSlash := -1
+
+	// Make sure we don't strip off the slash for the root of the file system
+	if len(path) > 0 && (path[0] == '/' || path[0] == '\\') {
+		absRootSlash = 0 // Unix
+	} else if len(path) > 2 && path[1] == ':' && (path[2] == '/' || path[2] == '\\') {
+		if c := path[0]; (c >= 'a' && c < 'z') || (c >= 'A' && c <= 'Z') {
+			absRootSlash = 2 // Windows
+		}
+	}
+
+	for {
+		i := strings.LastIndexAny(path, "/\\")
+
+		// Stop if there are no more slashes
+		if i < 0 {
+			base = path
+			break
+		}
+
+		// Stop if we found a non-trailing slash
+		if i == absRootSlash {
+			dir, base = path[:i+1], path[i+1:]
+			break
+		}
+		if i+1 != len(path) {
+			dir, base = path[:i], path[i+1:]
+			break
+		}
+
+		// Ignore trailing slashes
+		path = path[:i]
+	}
+
+	// Strip off the extension
+	if dot := strings.LastIndexByte(base, '.'); dot >= 0 {
+		ext = base[dot:]
+
+		// We default to the "local-css" loader for ".module.css" files. Make sure
+		// the string names generated by this don't all have "_module_" in them.
+		if ext == ".css" {
+			if dot2 := strings.LastIndexByte(base[:dot], '.'); dot2 >= 0 && base[dot2:] == ".module.css" {
+				dot = dot2
+				ext = base[dot:]
+			}
+		}
+
+		base = base[:dot]
+	}
+	return
+}
+
+type Source struct {
+	// This is used for error messages and the metadata JSON file.
+	//
+	// This is a mostly platform-independent path. It's relative to the current
+	// working directory and always uses standard path separators. Use this for
+	// referencing a file in all output data. These paths still use the original
+	// case of the path so they may still work differently on file systems that
+	// are case-insensitive vs. case-sensitive.
+	PrettyPath string
+
+	// An identifier that is mixed in to automatically-generated symbol names to
+	// improve readability. For example, if the identifier is "util" then the
+	// symbol for an "export default" statement will be called "util_default".
+	IdentifierName string
+
+	Contents string
+
+	// This is used as a unique key to identify this source file. It should never
+	// be shown to the user (e.g. never print this to the terminal).
+	//
+	// If it's marked as an absolute path, it's a platform-dependent path that
+	// includes environment-specific things such as Windows backslash path
+	// separators and potentially the user's home directory. Only use this for
+	// passing to syscalls for reading and writing to the file system. Do not
+	// include this in any output data.
+	//
+	// If it's marked as not an absolute path, it's an opaque string that is used
+	// to refer to an automatically-generated module.
+	KeyPath Path
+
+	Index uint32
+}
+
+func (s *Source) TextForRange(r Range) string {
+	return s.Contents[r.Loc.Start : r.Loc.Start+r.Len]
+}
+
+func (s *Source) LocBeforeWhitespace(loc Loc) Loc {
+	for loc.Start > 0 {
+		c, width := utf8.DecodeLastRuneInString(s.Contents[:loc.Start])
+		if c != ' ' && c != '\t' && c != '\r' && c != '\n' {
+			break
+		}
+		loc.Start -= int32(width)
+	}
+	return loc
+}
+
+func (s *Source) RangeOfOperatorBefore(loc Loc, op string) Range {
+	text := s.Contents[:loc.Start]
+	index := strings.LastIndex(text, op)
+	if index >= 0 {
+		return Range{Loc: Loc{Start: int32(index)}, Len: int32(len(op))}
+	}
+	return Range{Loc: loc}
+}
+
+func (s *Source) RangeOfOperatorAfter(loc Loc, op string) Range {
+	text := s.Contents[loc.Start:]
+	index := strings.Index(text, op)
+	if index >= 0 {
+		return Range{Loc: Loc{Start: loc.Start + int32(index)}, Len: int32(len(op))}
+	}
+	return Range{Loc: loc}
+}
+
+func (s *Source) RangeOfString(loc Loc) Range {
+	text := s.Contents[loc.Start:]
+	if len(text) == 0 {
+		return Range{Loc: loc, Len: 0}
+	}
+
+	quote := text[0]
+	if quote == '"' || quote == '\'' {
+		// Search for the matching quote character
+		for i := 1; i < len(text); i++ {
+			c := text[i]
+			if c == quote {
+				return Range{Loc: loc, Len: int32(i + 1)}
+			} else if c == '\\' {
+				i += 1
+			}
+		}
+	}
+
+	if quote == '`' {
+		// Search for the matching quote character
+		for i := 1; i < len(text); i++ {
+			c := text[i]
+			if c == quote {
+				return Range{Loc: loc, Len: int32(i + 1)}
+			} else if c == '\\' {
+				i += 1
+			} else if c == '$' && i+1 < len(text) && text[i+1] == '{' {
+				break // Only return the range for no-substitution template literals
+			}
+		}
+	}
+
+	return Range{Loc: loc, Len: 0}
+}
+
+func (s *Source) RangeOfNumber(loc Loc) (r Range) {
+	text := s.Contents[loc.Start:]
+	r = Range{Loc: loc, Len: 0}
+
+	if len(text) > 0 {
+		if c := text[0]; c >= '0' && c <= '9' {
+			r.Len = 1
+			for int(r.Len) < len(text) {
+				c := text[r.Len]
+				if (c < '0' || c > '9') && (c < 'a' || c > 'z') && (c < 'A' || c > 'Z') && c != '.' && c != '_' {
+					break
+				}
+				r.Len++
+			}
+		}
+	}
+	return
+}
+
+func (s *Source) RangeOfLegacyOctalEscape(loc Loc) (r Range) {
+	text := s.Contents[loc.Start:]
+	r = Range{Loc: loc, Len: 0}
+
+	if len(text) >= 2 && text[0] == '\\' {
+		r.Len = 2
+		for r.Len < 4 && int(r.Len) < len(text) {
+			c := text[r.Len]
+			if c < '0' || c > '9' {
+				break
+			}
+			r.Len++
+		}
+	}
+	return
+}
+
+func (s *Source) CommentTextWithoutIndent(r Range) string {
+	text := s.Contents[r.Loc.Start:r.End()]
+	if len(text) < 2 || !strings.HasPrefix(text, "/*") {
+		return text
+	}
+	prefix := s.Contents[:r.Loc.Start]
+
+	// Figure out the initial indent
+	indent := 0
+seekBackwardToNewline:
+	for len(prefix) > 0 {
+		c, size := utf8.DecodeLastRuneInString(prefix)
+		switch c {
+		case '\r', '\n', '\u2028', '\u2029':
+			break seekBackwardToNewline
+		}
+		prefix = prefix[:len(prefix)-size]
+		indent++
+	}
+
+	// Split the comment into lines
+	var lines []string
+	start := 0
+	for i, c := range text {
+		switch c {
+		case '\r', '\n':
+			// Don't double-append for Windows style "\r\n" newlines
+			if start <= i {
+				lines = append(lines, text[start:i])
+			}
+
+			start = i + 1
+
+			// Ignore the second part of Windows style "\r\n" newlines
+			if c == '\r' && start < len(text) && text[start] == '\n' {
+				start++
+			}
+
+		case '\u2028', '\u2029':
+			lines = append(lines, text[start:i])
+			start = i + 3
+		}
+	}
+	lines = append(lines, text[start:])
+
+	// Find the minimum indent over all lines after the first line
+	for _, line := range lines[1:] {
+		lineIndent := 0
+		for _, c := range line {
+			if c != ' ' && c != '\t' {
+				break
+			}
+			lineIndent++
+		}
+		if indent > lineIndent {
+			indent = lineIndent
+		}
+	}
+
+	// Trim the indent off of all lines after the first line
+	for i, line := range lines {
+		if i > 0 {
+			lines[i] = line[indent:]
+		}
+	}
+	return strings.Join(lines, "\n")
+}
+
+func plural(prefix string, count int, shown int, someAreMissing bool) string {
+	var text string
+	if count == 1 {
+		text = fmt.Sprintf("%d %s", count, prefix)
+	} else {
+		text = fmt.Sprintf("%d %ss", count, prefix)
+	}
+	if shown < count {
+		text = fmt.Sprintf("%d of %s", shown, text)
+	} else if someAreMissing && count > 1 {
+		text = "all " + text
+	}
+	return text
+}
+
+func errorAndWarningSummary(errors int, warnings int, shownErrors int, shownWarnings int) string {
+	someAreMissing := shownWarnings < warnings || shownErrors < errors
+	switch {
+	case errors == 0:
+		return plural("warning", warnings, shownWarnings, someAreMissing)
+	case warnings == 0:
+		return plural("error", errors, shownErrors, someAreMissing)
+	default:
+		return fmt.Sprintf("%s and %s",
+			plural("warning", warnings, shownWarnings, someAreMissing),
+			plural("error", errors, shownErrors, someAreMissing))
+	}
+}
+
+type APIKind uint8
+
+const (
+	GoAPI APIKind = iota
+	CLIAPI
+	JSAPI
+)
+
+// This can be used to customize error messages for the current API kind
+var API APIKind
+
+type TerminalInfo struct {
+	IsTTY           bool
+	UseColorEscapes bool
+	Width           int
+	Height          int
+}
+
+func NewStderrLog(options OutputOptions) Log {
+	var mutex sync.Mutex
+	var msgs SortableMsgs
+	terminalInfo := GetTerminalInfo(os.Stderr)
+	errors := 0
+	warnings := 0
+	shownErrors := 0
+	shownWarnings := 0
+	hasErrors := false
+	remainingMessagesBeforeLimit := options.MessageLimit
+	if remainingMessagesBeforeLimit == 0 {
+		remainingMessagesBeforeLimit = 0x7FFFFFFF
+	}
+	var deferredWarnings []Msg
+
+	finalizeLog := func() {
+		// Print the deferred warning now if there was no error after all
+		for remainingMessagesBeforeLimit > 0 && len(deferredWarnings) > 0 {
+			shownWarnings++
+			writeStringWithColor(os.Stderr, deferredWarnings[0].String(options, terminalInfo))
+			deferredWarnings = deferredWarnings[1:]
+			remainingMessagesBeforeLimit--
+		}
+
+		// Print out a summary
+		if options.MessageLimit > 0 && errors+warnings > options.MessageLimit {
+			writeStringWithColor(os.Stderr, fmt.Sprintf("%s shown (disable the message limit with --log-limit=0)\n",
+				errorAndWarningSummary(errors, warnings, shownErrors, shownWarnings)))
+		} else if options.LogLevel <= LevelInfo && (warnings != 0 || errors != 0) {
+			writeStringWithColor(os.Stderr, fmt.Sprintf("%s\n",
+				errorAndWarningSummary(errors, warnings, shownErrors, shownWarnings)))
+		}
+	}
+
+	switch options.Color {
+	case ColorNever:
+		terminalInfo.UseColorEscapes = false
+	case ColorAlways:
+		terminalInfo.UseColorEscapes = SupportsColorEscapes
+	}
+
+	return Log{
+		Level:     options.LogLevel,
+		Overrides: options.Overrides,
+
+		AddMsg: func(msg Msg) {
+			mutex.Lock()
+			defer mutex.Unlock()
+			msgs = append(msgs, msg)
+
+			switch msg.Kind {
+			case Verbose:
+				if options.LogLevel <= LevelVerbose {
+					writeStringWithColor(os.Stderr, msg.String(options, terminalInfo))
+				}
+
+			case Debug:
+				if options.LogLevel <= LevelDebug {
+					writeStringWithColor(os.Stderr, msg.String(options, terminalInfo))
+				}
+
+			case Info:
+				if options.LogLevel <= LevelInfo {
+					writeStringWithColor(os.Stderr, msg.String(options, terminalInfo))
+				}
+
+			case Error:
+				hasErrors = true
+				if options.LogLevel <= LevelError {
+					errors++
+				}
+
+			case Warning:
+				if options.LogLevel <= LevelWarning {
+					warnings++
+				}
+			}
+
+			// Be silent if we're past the limit so we don't flood the terminal
+			if remainingMessagesBeforeLimit == 0 {
+				return
+			}
+
+			switch msg.Kind {
+			case Error:
+				if options.LogLevel <= LevelError {
+					shownErrors++
+					writeStringWithColor(os.Stderr, msg.String(options, terminalInfo))
+					remainingMessagesBeforeLimit--
+				}
+
+			case Warning:
+				if options.LogLevel <= LevelWarning {
+					if remainingMessagesBeforeLimit > (options.MessageLimit+1)/2 {
+						shownWarnings++
+						writeStringWithColor(os.Stderr, msg.String(options, terminalInfo))
+						remainingMessagesBeforeLimit--
+					} else {
+						// If we have less than half of the slots left, wait for potential
+						// future errors instead of using up all of the slots with warnings.
+						// We want the log for a failed build to always have at least one
+						// error in it.
+						deferredWarnings = append(deferredWarnings, msg)
+					}
+				}
+			}
+		},
+
+		HasErrors: func() bool {
+			mutex.Lock()
+			defer mutex.Unlock()
+			return hasErrors
+		},
+
+		Peek: func() []Msg {
+			mutex.Lock()
+			defer mutex.Unlock()
+			sort.Stable(msgs)
+			return append([]Msg{}, msgs...)
+		},
+
+		Done: func() []Msg {
+			mutex.Lock()
+			defer mutex.Unlock()
+			finalizeLog()
+			sort.Stable(msgs)
+			return msgs
+		},
+	}
+}
+
+func PrintErrorToStderr(osArgs []string, text string) {
+	PrintMessageToStderr(osArgs, Msg{Kind: Error, Data: MsgData{Text: text}})
+}
+
+func PrintErrorWithNoteToStderr(osArgs []string, text string, note string) {
+	msg := Msg{
+		Kind: Error,
+		Data: MsgData{Text: text},
+	}
+	if note != "" {
+		msg.Notes = []MsgData{{Text: note}}
+	}
+	PrintMessageToStderr(osArgs, msg)
+}
+
+func OutputOptionsForArgs(osArgs []string) OutputOptions {
+	options := OutputOptions{IncludeSource: true}
+
+	// Implement a mini argument parser so these options always work even if we
+	// haven't yet gotten to the general-purpose argument parsing code
+	for _, arg := range osArgs {
+		switch arg {
+		case "--color=false":
+			options.Color = ColorNever
+		case "--color=true", "--color":
+			options.Color = ColorAlways
+		case "--log-level=info":
+			options.LogLevel = LevelInfo
+		case "--log-level=warning":
+			options.LogLevel = LevelWarning
+		case "--log-level=error":
+			options.LogLevel = LevelError
+		case "--log-level=silent":
+			options.LogLevel = LevelSilent
+		}
+	}
+
+	return options
+}
+
+func PrintMessageToStderr(osArgs []string, msg Msg) {
+	log := NewStderrLog(OutputOptionsForArgs(osArgs))
+	log.AddMsg(msg)
+	log.Done()
+}
+
+type Colors struct {
+	Reset     string
+	Bold      string
+	Dim       string
+	Underline string
+
+	Red   string
+	Green string
+	Blue  string
+
+	Cyan    string
+	Magenta string
+	Yellow  string
+
+	RedBgRed     string
+	RedBgWhite   string
+	GreenBgGreen string
+	GreenBgWhite string
+	BlueBgBlue   string
+	BlueBgWhite  string
+
+	CyanBgCyan       string
+	CyanBgBlack      string
+	MagentaBgMagenta string
+	MagentaBgBlack   string
+	YellowBgYellow   string
+	YellowBgBlack    string
+}
+
+var TerminalColors = Colors{
+	Reset:     "\033[0m",
+	Bold:      "\033[1m",
+	Dim:       "\033[37m",
+	Underline: "\033[4m",
+
+	Red:   "\033[31m",
+	Green: "\033[32m",
+	Blue:  "\033[34m",
+
+	Cyan:    "\033[36m",
+	Magenta: "\033[35m",
+	Yellow:  "\033[33m",
+
+	RedBgRed:     "\033[41;31m",
+	RedBgWhite:   "\033[41;97m",
+	GreenBgGreen: "\033[42;32m",
+	GreenBgWhite: "\033[42;97m",
+	BlueBgBlue:   "\033[44;34m",
+	BlueBgWhite:  "\033[44;97m",
+
+	CyanBgCyan:       "\033[46;36m",
+	CyanBgBlack:      "\033[46;30m",
+	MagentaBgMagenta: "\033[45;35m",
+	MagentaBgBlack:   "\033[45;30m",
+	YellowBgYellow:   "\033[43;33m",
+	YellowBgBlack:    "\033[43;30m",
+}
+
+func PrintText(file *os.File, level LogLevel, osArgs []string, callback func(Colors) string) {
+	options := OutputOptionsForArgs(osArgs)
+
+	// Skip logging these if these logs are disabled
+	if options.LogLevel > level {
+		return
+	}
+
+	PrintTextWithColor(file, options.Color, callback)
+}
+
+func PrintTextWithColor(file *os.File, useColor UseColor, callback func(Colors) string) {
+	var useColorEscapes bool
+	switch useColor {
+	case ColorNever:
+		useColorEscapes = false
+	case ColorAlways:
+		useColorEscapes = SupportsColorEscapes
+	case ColorIfTerminal:
+		useColorEscapes = GetTerminalInfo(file).UseColorEscapes
+	}
+
+	var colors Colors
+	if useColorEscapes {
+		colors = TerminalColors
+	}
+	writeStringWithColor(file, callback(colors))
+}
+
+type SummaryTableEntry struct {
+	Dir         string
+	Base        string
+	Size        string
+	Bytes       int
+	IsSourceMap bool
+}
+
+// This type is just so we can use Go's native sort function
+type SummaryTable []SummaryTableEntry
+
+func (t SummaryTable) Len() int          { return len(t) }
+func (t SummaryTable) Swap(i int, j int) { t[i], t[j] = t[j], t[i] }
+
+func (t SummaryTable) Less(i int, j int) bool {
+	ti := t[i]
+	tj := t[j]
+
+	// Sort source maps last
+	if !ti.IsSourceMap && tj.IsSourceMap {
+		return true
+	}
+	if ti.IsSourceMap && !tj.IsSourceMap {
+		return false
+	}
+
+	// Sort by size first
+	if ti.Bytes > tj.Bytes {
+		return true
+	}
+	if ti.Bytes < tj.Bytes {
+		return false
+	}
+
+	// Sort alphabetically by directory first
+	if ti.Dir < tj.Dir {
+		return true
+	}
+	if ti.Dir > tj.Dir {
+		return false
+	}
+
+	// Then sort alphabetically by file name
+	return ti.Base < tj.Base
+}
+
+// Show a warning icon next to output files that are 1mb or larger
+const sizeWarningThreshold = 1024 * 1024
+
+func PrintSummary(useColor UseColor, table SummaryTable, start *time.Time) {
+	PrintTextWithColor(os.Stderr, useColor, func(colors Colors) string {
+		isProbablyWindowsCommandPrompt := isProbablyWindowsCommandPrompt()
+		sb := strings.Builder{}
+
+		if len(table) > 0 {
+			info := GetTerminalInfo(os.Stderr)
+
+			// Truncate the table in case it's really long
+			maxLength := info.Height / 2
+			if info.Height == 0 {
+				maxLength = 20
+			} else if maxLength < 5 {
+				maxLength = 5
+			}
+			length := len(table)
+			sort.Sort(table)
+			if length > maxLength {
+				table = table[:maxLength]
+			}
+
+			// Compute the maximum width of the size column
+			spacingBetweenColumns := 2
+			hasSizeWarning := false
+			maxPath := 0
+			maxSize := 0
+			for _, entry := range table {
+				path := len(entry.Dir) + len(entry.Base)
+				size := len(entry.Size) + spacingBetweenColumns
+				if path > maxPath {
+					maxPath = path
+				}
+				if size > maxSize {
+					maxSize = size
+				}
+				if !entry.IsSourceMap && entry.Bytes >= sizeWarningThreshold {
+					hasSizeWarning = true
+				}
+			}
+
+			margin := "  "
+			layoutWidth := info.Width
+			if layoutWidth < 1 {
+				layoutWidth = defaultTerminalWidth
+			}
+			layoutWidth -= 2 * len(margin)
+			if hasSizeWarning {
+				// Add space for the warning icon
+				layoutWidth -= 2
+			}
+			if layoutWidth > maxPath+maxSize {
+				layoutWidth = maxPath + maxSize
+			}
+			sb.WriteByte('\n')
+
+			for _, entry := range table {
+				dir, base := entry.Dir, entry.Base
+				pathWidth := layoutWidth - maxSize
+
+				// Truncate the path with "..." to fit on one line
+				if len(dir)+len(base) > pathWidth {
+					// Trim the directory from the front, leaving the trailing slash
+					if len(dir) > 0 {
+						n := pathWidth - len(base) - 3
+						if n < 1 {
+							n = 1
+						}
+						dir = "..." + dir[len(dir)-n:]
+					}
+
+					// Trim the file name from the back
+					if len(dir)+len(base) > pathWidth {
+						n := pathWidth - len(dir) - 3
+						if n < 0 {
+							n = 0
+						}
+						base = base[:n] + "..."
+					}
+				}
+
+				spacer := layoutWidth - len(entry.Size) - len(dir) - len(base)
+				if spacer < 0 {
+					spacer = 0
+				}
+
+				// Put a warning next to the size if it's above a certain threshold
+				sizeColor := colors.Cyan
+				sizeWarning := ""
+				if !entry.IsSourceMap && entry.Bytes >= sizeWarningThreshold {
+					sizeColor = colors.Yellow
+
+					// Emoji don't work in Windows Command Prompt
+					if !isProbablyWindowsCommandPrompt {
+						sizeWarning = " ⚠️"
+					}
+				}
+
+				sb.WriteString(fmt.Sprintf("%s%s%s%s%s%s%s%s%s%s%s%s\n",
+					margin,
+					colors.Dim,
+					dir,
+					colors.Reset,
+					colors.Bold,
+					base,
+					colors.Reset,
+					strings.Repeat(" ", spacer),
+					sizeColor,
+					entry.Size,
+					sizeWarning,
+					colors.Reset,
+				))
+			}
+
+			// Say how many remaining files are not shown
+			if length > maxLength {
+				plural := "s"
+				if length == maxLength+1 {
+					plural = ""
+				}
+				sb.WriteString(fmt.Sprintf("%s%s...and %d more output file%s...%s\n", margin, colors.Dim, length-maxLength, plural, colors.Reset))
+			}
+		}
+		sb.WriteByte('\n')
+
+		lightningSymbol := "⚡ "
+
+		// Emoji don't work in Windows Command Prompt
+		if isProbablyWindowsCommandPrompt {
+			lightningSymbol = ""
+		}
+
+		// Printing the time taken is optional
+		if start != nil {
+			sb.WriteString(fmt.Sprintf("%s%sDone in %dms%s\n",
+				lightningSymbol,
+				colors.Green,
+				time.Since(*start).Milliseconds(),
+				colors.Reset,
+			))
+		}
+
+		return sb.String()
+	})
+}
+
+type DeferLogKind uint8
+
+const (
+	DeferLogAll DeferLogKind = iota
+	DeferLogNoVerboseOrDebug
+)
+
+func NewDeferLog(kind DeferLogKind, overrides map[MsgID]LogLevel) Log {
+	var msgs SortableMsgs
+	var mutex sync.Mutex
+	var hasErrors bool
+
+	return Log{
+		Level:     LevelInfo,
+		Overrides: overrides,
+
+		AddMsg: func(msg Msg) {
+			if kind == DeferLogNoVerboseOrDebug && (msg.Kind == Verbose || msg.Kind == Debug) {
+				return
+			}
+			mutex.Lock()
+			defer mutex.Unlock()
+			if msg.Kind == Error {
+				hasErrors = true
+			}
+			msgs = append(msgs, msg)
+		},
+
+		HasErrors: func() bool {
+			mutex.Lock()
+			defer mutex.Unlock()
+			return hasErrors
+		},
+
+		Peek: func() []Msg {
+			mutex.Lock()
+			defer mutex.Unlock()
+			return append([]Msg{}, msgs...)
+		},
+
+		Done: func() []Msg {
+			mutex.Lock()
+			defer mutex.Unlock()
+			sort.Stable(msgs)
+			return msgs
+		},
+	}
+}
+
+type UseColor uint8
+
+const (
+	ColorIfTerminal UseColor = iota
+	ColorNever
+	ColorAlways
+)
+
+type OutputOptions struct {
+	MessageLimit  int
+	IncludeSource bool
+	Color         UseColor
+	LogLevel      LogLevel
+	Overrides     map[MsgID]LogLevel
+}
+
+func (msg Msg) String(options OutputOptions, terminalInfo TerminalInfo) string {
+	// Format the message
+	text := msgString(options.IncludeSource, terminalInfo, msg.ID, msg.Kind, msg.Data, msg.PluginName)
+
+	// Format the notes
+	var oldData MsgData
+	for i, note := range msg.Notes {
+		if options.IncludeSource && (i == 0 || strings.IndexByte(oldData.Text, '\n') >= 0 || oldData.Location != nil) {
+			text += "\n"
+		}
+		text += msgString(options.IncludeSource, terminalInfo, MsgID_None, Note, note, "")
+		oldData = note
+	}
+
+	// Add extra spacing between messages if source code is present
+	if options.IncludeSource {
+		text += "\n"
+	}
+	return text
+}
+
+// The number of margin characters in addition to the line number
+const extraMarginChars = 9
+
+func marginWithLineText(maxMargin int, line int) string {
+	number := fmt.Sprintf("%d", line)
+	return fmt.Sprintf("      %s%s │ ", strings.Repeat(" ", maxMargin-len(number)), number)
+}
+
+func emptyMarginText(maxMargin int, isLast bool) string {
+	space := strings.Repeat(" ", maxMargin)
+	if isLast {
+		return fmt.Sprintf("      %s ╵ ", space)
+	}
+	return fmt.Sprintf("      %s │ ", space)
+}
+
+func msgString(includeSource bool, terminalInfo TerminalInfo, id MsgID, kind MsgKind, data MsgData, pluginName string) string {
+	if !includeSource {
+		if loc := data.Location; loc != nil {
+			return fmt.Sprintf("%s: %s: %s\n", loc.File, kind.String(), data.Text)
+		}
+		return fmt.Sprintf("%s: %s\n", kind.String(), data.Text)
+	}
+
+	var colors Colors
+	if terminalInfo.UseColorEscapes {
+		colors = TerminalColors
+	}
+
+	var iconColor string
+	var kindColorBrackets string
+	var kindColorText string
+
+	location := ""
+
+	if data.Location != nil {
+		maxMargin := len(fmt.Sprintf("%d", data.Location.Line))
+		d := detailStruct(data, terminalInfo, maxMargin)
+
+		if d.Suggestion != "" {
+			location = fmt.Sprintf("\n    %s:%d:%d:\n%s%s%s%s%s%s\n%s%s%s%s%s\n%s%s%s%s%s\n%s",
+				d.Path, d.Line, d.Column,
+				colors.Dim, d.SourceBefore, colors.Green, d.SourceMarked, colors.Dim, d.SourceAfter,
+				emptyMarginText(maxMargin, false), d.Indent, colors.Green, d.Marker, colors.Dim,
+				emptyMarginText(maxMargin, true), d.Indent, colors.Green, d.Suggestion, colors.Reset,
+				d.ContentAfter,
+			)
+		} else {
+			location = fmt.Sprintf("\n    %s:%d:%d:\n%s%s%s%s%s%s\n%s%s%s%s%s\n%s",
+				d.Path, d.Line, d.Column,
+				colors.Dim, d.SourceBefore, colors.Green, d.SourceMarked, colors.Dim, d.SourceAfter,
+				emptyMarginText(maxMargin, true), d.Indent, colors.Green, d.Marker, colors.Reset,
+				d.ContentAfter,
+			)
+		}
+	}
+
+	switch kind {
+	case Verbose:
+		iconColor = colors.Cyan
+		kindColorBrackets = colors.CyanBgCyan
+		kindColorText = colors.CyanBgBlack
+
+	case Debug:
+		iconColor = colors.Green
+		kindColorBrackets = colors.GreenBgGreen
+		kindColorText = colors.GreenBgWhite
+
+	case Info:
+		iconColor = colors.Blue
+		kindColorBrackets = colors.BlueBgBlue
+		kindColorText = colors.BlueBgWhite
+
+	case Error:
+		iconColor = colors.Red
+		kindColorBrackets = colors.RedBgRed
+		kindColorText = colors.RedBgWhite
+
+	case Warning:
+		iconColor = colors.Yellow
+		kindColorBrackets = colors.YellowBgYellow
+		kindColorText = colors.YellowBgBlack
+
+	case Note:
+		sb := strings.Builder{}
+
+		for _, line := range strings.Split(data.Text, "\n") {
+			// Special-case word wrapping
+			if wrapWidth := terminalInfo.Width; wrapWidth > 2 {
+				if !data.DisableMaximumWidth && wrapWidth > 100 {
+					wrapWidth = 100 // Enforce a maximum paragraph width for readability
+				}
+				for _, run := range wrapWordsInString(line, wrapWidth-2) {
+					sb.WriteString("  ")
+					sb.WriteString(linkifyText(run, colors.Underline, colors.Reset))
+					sb.WriteByte('\n')
+				}
+				continue
+			}
+
+			// Otherwise, just write an indented line
+			sb.WriteString("  ")
+			sb.WriteString(linkifyText(line, colors.Underline, colors.Reset))
+			sb.WriteByte('\n')
+		}
+
+		sb.WriteString(location)
+		return sb.String()
+	}
+
+	if pluginName != "" {
+		pluginName = fmt.Sprintf(" %s%s[plugin %s]%s", colors.Bold, colors.Magenta, pluginName, colors.Reset)
+	}
+
+	msgID := MsgIDToString(id)
+	if msgID != "" {
+		msgID = fmt.Sprintf(" [%s]", msgID)
+	}
+
+	return fmt.Sprintf("%s%s %s[%s%s%s]%s %s%s%s%s%s\n%s",
+		iconColor, kind.Icon(),
+		kindColorBrackets, kindColorText, kind.String(), kindColorBrackets, colors.Reset,
+		colors.Bold, data.Text, colors.Reset, pluginName, msgID,
+		location,
+	)
+}
+
+func linkifyText(text string, underline string, reset string) string {
+	if underline == "" {
+		return text
+	}
+
+	https := strings.Index(text, "https://")
+	if https == -1 {
+		return text
+	}
+
+	sb := strings.Builder{}
+	for {
+		https := strings.Index(text, "https://")
+		if https == -1 {
+			break
+		}
+
+		end := strings.IndexByte(text[https:], ' ')
+		if end == -1 {
+			end = len(text)
+		} else {
+			end += https
+		}
+
+		// Remove trailing punctuation
+		if end > https {
+			switch text[end-1] {
+			case '.', ',', '?', '!', ')', ']', '}':
+				end--
+			}
+		}
+
+		sb.WriteString(text[:https])
+		sb.WriteString(underline)
+		sb.WriteString(text[https:end])
+		sb.WriteString(reset)
+		text = text[end:]
+	}
+
+	sb.WriteString(text)
+	return sb.String()
+}
+
+func wrapWordsInString(text string, width int) []string {
+	runs := []string{}
+
+outer:
+	for text != "" {
+		i := 0
+		x := 0
+		wordEndI := 0
+
+		// Skip over any leading spaces
+		for i < len(text) && text[i] == ' ' {
+			i++
+			x++
+		}
+
+		// Find out how many words will fit in this run
+		for i < len(text) {
+			oldWordEndI := wordEndI
+			wordStartI := i
+
+			// Find the end of the word
+			for i < len(text) {
+				c, width := utf8.DecodeRuneInString(text[i:])
+				if c == ' ' {
+					break
+				}
+				i += width
+				x += 1 // Naively assume that each unicode code point is a single column
+			}
+			wordEndI = i
+
+			// Split into a new run if this isn't the first word in the run and the end is past the width
+			if wordStartI > 0 && x > width {
+				runs = append(runs, text[:oldWordEndI])
+				text = text[wordStartI:]
+				continue outer
+			}
+
+			// Skip over any spaces after the word
+			for i < len(text) && text[i] == ' ' {
+				i++
+				x++
+			}
+		}
+
+		// If we get here, this is the last run (i.e. everything fits)
+		break
+	}
+
+	// Remove any trailing spaces on the last run
+	for len(text) > 0 && text[len(text)-1] == ' ' {
+		text = text[:len(text)-1]
+	}
+	runs = append(runs, text)
+	return runs
+}
+
+type MsgDetail struct {
+	SourceBefore string
+	SourceMarked string
+	SourceAfter  string
+
+	Indent     string
+	Marker     string
+	Suggestion string
+
+	ContentAfter string
+
+	Path   string
+	Line   int
+	Column int
+}
+
+// It's not common for large files to have many warnings. But when it happens,
+// we want to make sure that it's not too slow. Source code locations are
+// represented as byte offsets for compactness but transforming these to
+// line/column locations for warning messages requires scanning through the
+// file. A naive approach for this would cause O(n^2) scanning time for n
+// warnings distributed throughout the file.
+//
+// Warnings are typically generated sequentially as the file is scanned. So
+// one way of optimizing this is to just start scanning from where we left
+// off last time instead of always starting from the beginning of the file.
+// That's what this object does.
+//
+// Another option could be to eagerly populate an array of line/column offsets
+// and then use binary search for each query. This might slow down the common
+// case of a file with only at most a few warnings though, so think before
+// optimizing too much. Performance in the zero or one warning case is by far
+// the most important.
+type LineColumnTracker struct {
+	contents     string
+	prettyPath   string
+	offset       int32
+	line         int32
+	lineStart    int32
+	lineEnd      int32
+	hasLineStart bool
+	hasLineEnd   bool
+	hasSource    bool
+}
+
+func MakeLineColumnTracker(source *Source) LineColumnTracker {
+	if source == nil {
+		return LineColumnTracker{
+			hasSource: false,
+		}
+	}
+
+	return LineColumnTracker{
+		contents:     source.Contents,
+		prettyPath:   source.PrettyPath,
+		hasLineStart: true,
+		hasSource:    true,
+	}
+}
+
+func (tracker *LineColumnTracker) MsgData(r Range, text string) MsgData {
+	return MsgData{
+		Text:     text,
+		Location: tracker.MsgLocationOrNil(r),
+	}
+}
+
+func (t *LineColumnTracker) scanTo(offset int32) {
+	contents := t.contents
+	i := t.offset
+
+	// Scan forward
+	if i < offset {
+		for {
+			r, size := utf8.DecodeRuneInString(contents[i:])
+			i += int32(size)
+
+			switch r {
+			case '\n':
+				t.hasLineStart = true
+				t.hasLineEnd = false
+				t.lineStart = i
+				if i == int32(size) || contents[i-int32(size)-1] != '\r' {
+					t.line++
+				}
+
+			case '\r', '\u2028', '\u2029':
+				t.hasLineStart = true
+				t.hasLineEnd = false
+				t.lineStart = i
+				t.line++
+			}
+
+			if i >= offset {
+				t.offset = i
+				return
+			}
+		}
+	}
+
+	// Scan backward
+	if i > offset {
+		for {
+			r, size := utf8.DecodeLastRuneInString(contents[:i])
+			i -= int32(size)
+
+			switch r {
+			case '\n':
+				t.hasLineStart = false
+				t.hasLineEnd = true
+				t.lineEnd = i
+				if i == 0 || contents[i-1] != '\r' {
+					t.line--
+				}
+
+			case '\r', '\u2028', '\u2029':
+				t.hasLineStart = false
+				t.hasLineEnd = true
+				t.lineEnd = i
+				t.line--
+			}
+
+			if i <= offset {
+				t.offset = i
+				return
+			}
+		}
+	}
+}
+
+func (t *LineColumnTracker) computeLineAndColumn(offset int) (lineCount int, columnCount int, lineStart int, lineEnd int) {
+	t.scanTo(int32(offset))
+
+	// Scan for the start of the line
+	if !t.hasLineStart {
+		contents := t.contents
+		i := t.offset
+		for i > 0 {
+			r, size := utf8.DecodeLastRuneInString(contents[:i])
+			if r == '\n' || r == '\r' || r == '\u2028' || r == '\u2029' {
+				break
+			}
+			i -= int32(size)
+		}
+		t.hasLineStart = true
+		t.lineStart = i
+	}
+
+	// Scan for the end of the line
+	if !t.hasLineEnd {
+		contents := t.contents
+		i := t.offset
+		n := int32(len(contents))
+		for i < n {
+			r, size := utf8.DecodeRuneInString(contents[i:])
+			if r == '\n' || r == '\r' || r == '\u2028' || r == '\u2029' {
+				break
+			}
+			i += int32(size)
+		}
+		t.hasLineEnd = true
+		t.lineEnd = i
+	}
+
+	return int(t.line), offset - int(t.lineStart), int(t.lineStart), int(t.lineEnd)
+}
+
+func (tracker *LineColumnTracker) MsgLocationOrNil(r Range) *MsgLocation {
+	if tracker == nil || !tracker.hasSource {
+		return nil
+	}
+
+	// Convert the index into a line and column number
+	lineCount, columnCount, lineStart, lineEnd := tracker.computeLineAndColumn(int(r.Loc.Start))
+
+	return &MsgLocation{
+		File:     tracker.prettyPath,
+		Line:     lineCount + 1, // 0-based to 1-based
+		Column:   columnCount,
+		Length:   int(r.Len),
+		LineText: tracker.contents[lineStart:lineEnd],
+	}
+}
+
+func detailStruct(data MsgData, terminalInfo TerminalInfo, maxMargin int) MsgDetail {
+	// Only highlight the first line of the line text
+	loc := *data.Location
+	endOfFirstLine := len(loc.LineText)
+
+	// Note: This uses "IndexByte" because Go implements this with SIMD, which
+	// can matter a lot for really long lines. Some people pass huge >100mb
+	// minified files as line text for the log message.
+	if i := strings.IndexByte(loc.LineText, '\n'); i >= 0 {
+		endOfFirstLine = i
+	}
+
+	firstLine := loc.LineText[:endOfFirstLine]
+	afterFirstLine := loc.LineText[endOfFirstLine:]
+	if afterFirstLine != "" && !strings.HasSuffix(afterFirstLine, "\n") {
+		afterFirstLine += "\n"
+	}
+
+	// Clamp values in range
+	if loc.Line < 0 {
+		loc.Line = 0
+	}
+	if loc.Column < 0 {
+		loc.Column = 0
+	}
+	if loc.Length < 0 {
+		loc.Length = 0
+	}
+	if loc.Column > endOfFirstLine {
+		loc.Column = endOfFirstLine
+	}
+	if loc.Length > endOfFirstLine-loc.Column {
+		loc.Length = endOfFirstLine - loc.Column
+	}
+
+	spacesPerTab := 2
+	lineText := renderTabStops(firstLine, spacesPerTab)
+	textUpToLoc := renderTabStops(firstLine[:loc.Column], spacesPerTab)
+	markerStart := len(textUpToLoc)
+	markerEnd := markerStart
+	indent := strings.Repeat(" ", estimateWidthInTerminal(textUpToLoc))
+	marker := "^"
+
+	// Extend markers to cover the full range of the error
+	if loc.Length > 0 {
+		markerEnd = len(renderTabStops(firstLine[:loc.Column+loc.Length], spacesPerTab))
+	}
+
+	// Clip the marker to the bounds of the line
+	if markerStart > len(lineText) {
+		markerStart = len(lineText)
+	}
+	if markerEnd > len(lineText) {
+		markerEnd = len(lineText)
+	}
+	if markerEnd < markerStart {
+		markerEnd = markerStart
+	}
+
+	// Trim the line to fit the terminal width
+	width := terminalInfo.Width
+	if width < 1 {
+		width = defaultTerminalWidth
+	}
+	width -= maxMargin + extraMarginChars
+	if width < 1 {
+		width = 1
+	}
+	if loc.Column == endOfFirstLine {
+		// If the marker is at the very end of the line, the marker will be a "^"
+		// character that extends one column past the end of the line. In this case
+		// we should reserve a column at the end so the marker doesn't wrap.
+		width -= 1
+	}
+	if len(lineText) > width {
+		// Try to center the error
+		sliceStart := (markerStart + markerEnd - width) / 2
+		if sliceStart > markerStart-width/5 {
+			sliceStart = markerStart - width/5
+		}
+		if sliceStart < 0 {
+			sliceStart = 0
+		}
+		if sliceStart > len(lineText)-width {
+			sliceStart = len(lineText) - width
+		}
+		sliceEnd := sliceStart + width
+
+		// Slice the line
+		slicedLine := lineText[sliceStart:sliceEnd]
+		markerStart -= sliceStart
+		markerEnd -= sliceStart
+		if markerStart < 0 {
+			markerStart = 0
+		}
+		if markerEnd > len(slicedLine) {
+			markerEnd = len(slicedLine)
+		}
+
+		// Truncate the ends with "..."
+		if len(slicedLine) > 3 && sliceStart > 0 {
+			slicedLine = "..." + slicedLine[3:]
+			if markerStart < 3 {
+				markerStart = 3
+			}
+		}
+		if len(slicedLine) > 3 && sliceEnd < len(lineText) {
+			slicedLine = slicedLine[:len(slicedLine)-3] + "..."
+			if markerEnd > len(slicedLine)-3 {
+				markerEnd = len(slicedLine) - 3
+			}
+			if markerEnd < markerStart {
+				markerEnd = markerStart
+			}
+		}
+
+		// Now we can compute the indent
+		lineText = slicedLine
+		indent = strings.Repeat(" ", estimateWidthInTerminal(lineText[:markerStart]))
+	}
+
+	// If marker is still multi-character after clipping, make the marker wider
+	if markerEnd-markerStart > 1 {
+		marker = strings.Repeat("~", estimateWidthInTerminal(lineText[markerStart:markerEnd]))
+	}
+
+	// Put a margin before the marker indent
+	margin := marginWithLineText(maxMargin, loc.Line)
+
+	return MsgDetail{
+		Path:   loc.File,
+		Line:   loc.Line,
+		Column: loc.Column,
+
+		SourceBefore: margin + lineText[:markerStart],
+		SourceMarked: lineText[markerStart:markerEnd],
+		SourceAfter:  lineText[markerEnd:],
+
+		Indent:     indent,
+		Marker:     marker,
+		Suggestion: loc.Suggestion,
+
+		ContentAfter: afterFirstLine,
+	}
+}
+
+// Estimate the number of columns this string will take when printed
+func estimateWidthInTerminal(text string) int {
+	// For now just assume each code point is one column. This is wrong but is
+	// less wrong than assuming each code unit is one column.
+	width := 0
+	for text != "" {
+		c, size := utf8.DecodeRuneInString(text)
+		text = text[size:]
+
+		// Ignore the Zero Width No-Break Space character (UTF-8 BOM)
+		if c != 0xFEFF {
+			width++
+		}
+	}
+	return width
+}
+
+func renderTabStops(withTabs string, spacesPerTab int) string {
+	if !strings.ContainsRune(withTabs, '\t') {
+		return withTabs
+	}
+
+	withoutTabs := strings.Builder{}
+	count := 0
+
+	for _, c := range withTabs {
+		if c == '\t' {
+			spaces := spacesPerTab - count%spacesPerTab
+			for i := 0; i < spaces; i++ {
+				withoutTabs.WriteRune(' ')
+				count++
+			}
+		} else {
+			withoutTabs.WriteRune(c)
+			count++
+		}
+	}
+
+	return withoutTabs.String()
+}
+
+func (log Log) AddError(tracker *LineColumnTracker, r Range, text string) {
+	log.AddMsg(Msg{
+		Kind: Error,
+		Data: tracker.MsgData(r, text),
+	})
+}
+
+func (log Log) AddID(id MsgID, kind MsgKind, tracker *LineColumnTracker, r Range, text string) {
+	if override, ok := allowOverride(log.Overrides, id, kind); ok {
+		log.AddMsg(Msg{
+			ID:   id,
+			Kind: override,
+			Data: tracker.MsgData(r, text),
+		})
+	}
+}
+
+func (log Log) AddErrorWithNotes(tracker *LineColumnTracker, r Range, text string, notes []MsgData) {
+	log.AddMsg(Msg{
+		Kind:  Error,
+		Data:  tracker.MsgData(r, text),
+		Notes: notes,
+	})
+}
+
+func (log Log) AddIDWithNotes(id MsgID, kind MsgKind, tracker *LineColumnTracker, r Range, text string, notes []MsgData) {
+	if override, ok := allowOverride(log.Overrides, id, kind); ok {
+		log.AddMsg(Msg{
+			ID:    id,
+			Kind:  override,
+			Data:  tracker.MsgData(r, text),
+			Notes: notes,
+		})
+	}
+}
+
+func (log Log) AddMsgID(id MsgID, msg Msg) {
+	if override, ok := allowOverride(log.Overrides, id, msg.Kind); ok {
+		msg.ID = id
+		msg.Kind = override
+		log.AddMsg(msg)
+	}
+}
+
+func allowOverride(overrides map[MsgID]LogLevel, id MsgID, kind MsgKind) (MsgKind, bool) {
+	if logLevel, ok := overrides[id]; ok {
+		switch logLevel {
+		case LevelVerbose:
+			return Verbose, true
+		case LevelDebug:
+			return Debug, true
+		case LevelInfo:
+			return Info, true
+		case LevelWarning:
+			return Warning, true
+		case LevelError:
+			return Error, true
+		default:
+			// Setting the log level to "silent" silences this log message
+			return MsgKind(0), false
+		}
+	}
+	return kind, true
+}
+
+type StringInJSTableEntry struct {
+	innerLine   int32
+	innerColumn int32
+	innerLoc    Loc
+	outerLoc    Loc
+}
+
+// For Yarn PnP we sometimes parse JSON embedded in a JS string. This generates
+// a table that remaps locations inside the embedded JSON string literal into
+// locations in the actual JS file, which makes them easier to understand.
+func GenerateStringInJSTable(outerContents string, outerStringLiteralLoc Loc, innerContents string) (table []StringInJSTableEntry) {
+	i := int32(0)
+	n := int32(len(innerContents))
+	line := int32(1)
+	column := int32(0)
+	loc := Loc{Start: outerStringLiteralLoc.Start + 1}
+
+	for i < n {
+		// Ignore line continuations. A line continuation is not an escaped newline.
+		for {
+			if c, _ := utf8.DecodeRuneInString(outerContents[loc.Start:]); c != '\\' {
+				break
+			}
+			c, width := utf8.DecodeRuneInString(outerContents[loc.Start+1:])
+			switch c {
+			case '\n', '\r', '\u2028', '\u2029':
+				loc.Start += 1 + int32(width)
+				if c == '\r' && outerContents[loc.Start] == '\n' {
+					// Make sure Windows CRLF counts as a single newline
+					loc.Start++
+				}
+				continue
+			}
+			break
+		}
+
+		c, width := utf8.DecodeRuneInString(innerContents[i:])
+
+		// Compress the table using run-length encoding
+		table = append(table, StringInJSTableEntry{innerLine: line, innerColumn: column, innerLoc: Loc{Start: i}, outerLoc: loc})
+		if len(table) > 1 {
+			if last := table[len(table)-2]; line == last.innerLine && loc.Start-column == last.outerLoc.Start-last.innerColumn {
+				table = table[:len(table)-1]
+			}
+		}
+
+		// Advance the inner line/column
+		switch c {
+		case '\n', '\r', '\u2028', '\u2029':
+			line++
+			column = 0
+
+			// Handle newlines on Windows
+			if c == '\r' && i+1 < n && innerContents[i+1] == '\n' {
+				i++
+			}
+
+		default:
+			column += int32(width)
+		}
+		i += int32(width)
+
+		// Advance the outer loc, assuming the string syntax is already valid
+		c, width = utf8.DecodeRuneInString(outerContents[loc.Start:])
+		if c == '\r' && outerContents[loc.Start+1] == '\n' {
+			// Handle newlines on Windows in template literal strings
+			loc.Start += 2
+		} else if c != '\\' {
+			loc.Start += int32(width)
+		} else {
+			// Handle an escape sequence
+			c, width = utf8.DecodeRuneInString(outerContents[loc.Start+1:])
+			switch c {
+			case 'x':
+				// 2-digit hexadecimal
+				loc.Start += 1 + 2
+
+			case 'u':
+				loc.Start++
+				if outerContents[loc.Start] == '{' {
+					// Variable-length
+					for outerContents[loc.Start] != '}' {
+						loc.Start++
+					}
+					loc.Start++
+				} else {
+					// Fixed-length
+					loc.Start += 4
+				}
+
+			case '\n', '\r', '\u2028', '\u2029':
+				// This will be handled by the next iteration
+				break
+
+			default:
+				loc.Start += 1 + int32(width)
+			}
+		}
+	}
+
+	return
+}
+
+func RemapStringInJSLoc(table []StringInJSTableEntry, innerLoc Loc) Loc {
+	count := len(table)
+	index := 0
+
+	// Binary search to find the previous entry
+	for count > 0 {
+		step := count / 2
+		i := index + step
+		if i+1 < len(table) {
+			if entry := table[i+1]; entry.innerLoc.Start < innerLoc.Start {
+				index = i + 1
+				count -= step + 1
+				continue
+			}
+		}
+		count = step
+	}
+
+	entry := table[index]
+	entry.outerLoc.Start += innerLoc.Start - entry.innerLoc.Start // Undo run-length compression
+	return entry.outerLoc
+}
+
+func NewStringInJSLog(log Log, outerTracker *LineColumnTracker, table []StringInJSTableEntry) Log {
+	oldAddMsg := log.AddMsg
+
+	remapLineAndColumnToLoc := func(line int32, column int32) Loc {
+		count := len(table)
+		index := 0
+
+		// Binary search to find the previous entry
+		for count > 0 {
+			step := count / 2
+			i := index + step
+			if i+1 < len(table) {
+				if entry := table[i+1]; entry.innerLine < line || (entry.innerLine == line && entry.innerColumn < column) {
+					index = i + 1
+					count -= step + 1
+					continue
+				}
+			}
+			count = step
+		}
+
+		entry := table[index]
+		entry.outerLoc.Start += column - entry.innerColumn // Undo run-length compression
+		return entry.outerLoc
+	}
+
+	remapData := func(data MsgData) MsgData {
+		if data.Location == nil {
+			return data
+		}
+
+		// Generate a range in the outer source using the line/column/length in the inner source
+		r := Range{Loc: remapLineAndColumnToLoc(int32(data.Location.Line), int32(data.Location.Column))}
+		if data.Location.Length != 0 {
+			r.Len = remapLineAndColumnToLoc(int32(data.Location.Line), int32(data.Location.Column+data.Location.Length)).Start - r.Loc.Start
+		}
+
+		// Use that range to look up the line in the outer source
+		location := outerTracker.MsgData(r, data.Text).Location
+		location.Suggestion = data.Location.Suggestion
+		data.Location = location
+		return data
+	}
+
+	log.AddMsg = func(msg Msg) {
+		msg.Data = remapData(msg.Data)
+		for i, note := range msg.Notes {
+			msg.Notes[i] = remapData(note)
+		}
+		oldAddMsg(msg)
+	}
+
+	return log
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/logger/logger_darwin.go b/source/vendor/github.com/evanw/esbuild/internal/logger/logger_darwin.go
new file mode 100644
index 0000000..dac2a04
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/logger/logger_darwin.go
@@ -0,0 +1,34 @@
+//go:build darwin
+// +build darwin
+
+package logger
+
+import (
+	"os"
+
+	"golang.org/x/sys/unix"
+)
+
+const SupportsColorEscapes = true
+
+func GetTerminalInfo(file *os.File) (info TerminalInfo) {
+	fd := file.Fd()
+
+	// Is this file descriptor a terminal?
+	if _, err := unix.IoctlGetTermios(int(fd), unix.TIOCGETA); err == nil {
+		info.IsTTY = true
+		info.UseColorEscapes = !hasNoColorEnvironmentVariable()
+
+		// Get the width of the window
+		if w, err := unix.IoctlGetWinsize(int(fd), unix.TIOCGWINSZ); err == nil {
+			info.Width = int(w.Col)
+			info.Height = int(w.Row)
+		}
+	}
+
+	return
+}
+
+func writeStringWithColor(file *os.File, text string) {
+	file.WriteString(text)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/logger/logger_linux.go b/source/vendor/github.com/evanw/esbuild/internal/logger/logger_linux.go
new file mode 100644
index 0000000..b825cbb
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/logger/logger_linux.go
@@ -0,0 +1,34 @@
+//go:build linux
+// +build linux
+
+package logger
+
+import (
+	"os"
+
+	"golang.org/x/sys/unix"
+)
+
+const SupportsColorEscapes = true
+
+func GetTerminalInfo(file *os.File) (info TerminalInfo) {
+	fd := file.Fd()
+
+	// Is this file descriptor a terminal?
+	if _, err := unix.IoctlGetTermios(int(fd), unix.TCGETS); err == nil {
+		info.IsTTY = true
+		info.UseColorEscapes = !hasNoColorEnvironmentVariable()
+
+		// Get the width of the window
+		if w, err := unix.IoctlGetWinsize(int(fd), unix.TIOCGWINSZ); err == nil {
+			info.Width = int(w.Col)
+			info.Height = int(w.Row)
+		}
+	}
+
+	return
+}
+
+func writeStringWithColor(file *os.File, text string) {
+	file.WriteString(text)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/logger/logger_other.go b/source/vendor/github.com/evanw/esbuild/internal/logger/logger_other.go
new file mode 100644
index 0000000..7da8a85
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/logger/logger_other.go
@@ -0,0 +1,16 @@
+//go:build !darwin && !linux && !windows
+// +build !darwin,!linux,!windows
+
+package logger
+
+import "os"
+
+const SupportsColorEscapes = false
+
+func GetTerminalInfo(*os.File) TerminalInfo {
+	return TerminalInfo{}
+}
+
+func writeStringWithColor(file *os.File, text string) {
+	file.WriteString(text)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/logger/logger_windows.go b/source/vendor/github.com/evanw/esbuild/internal/logger/logger_windows.go
new file mode 100644
index 0000000..f2383ff
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/logger/logger_windows.go
@@ -0,0 +1,136 @@
+//go:build windows
+// +build windows
+
+package logger
+
+import (
+	"os"
+	"strings"
+	"syscall"
+	"unsafe"
+)
+
+const SupportsColorEscapes = true
+
+var kernel32 = syscall.NewLazyDLL("kernel32.dll")
+var getConsoleMode = kernel32.NewProc("GetConsoleMode")
+var setConsoleTextAttribute = kernel32.NewProc("SetConsoleTextAttribute")
+var getConsoleScreenBufferInfo = kernel32.NewProc("GetConsoleScreenBufferInfo")
+
+type consoleScreenBufferInfo struct {
+	dwSizeX              int16
+	dwSizeY              int16
+	dwCursorPositionX    int16
+	dwCursorPositionY    int16
+	wAttributes          uint16
+	srWindowLeft         int16
+	srWindowTop          int16
+	srWindowRight        int16
+	srWindowBottom       int16
+	dwMaximumWindowSizeX int16
+	dwMaximumWindowSizeY int16
+}
+
+func GetTerminalInfo(file *os.File) TerminalInfo {
+	fd := file.Fd()
+
+	// Is this file descriptor a terminal?
+	var unused uint32
+	isTTY, _, _ := syscall.Syscall(getConsoleMode.Addr(), 2, fd, uintptr(unsafe.Pointer(&unused)), 0)
+
+	// Get the width of the window
+	var info consoleScreenBufferInfo
+	syscall.Syscall(getConsoleScreenBufferInfo.Addr(), 2, fd, uintptr(unsafe.Pointer(&info)), 0)
+
+	return TerminalInfo{
+		IsTTY:           isTTY != 0,
+		Width:           int(info.dwSizeX) - 1,
+		Height:          int(info.dwSizeY) - 1,
+		UseColorEscapes: !hasNoColorEnvironmentVariable(),
+	}
+}
+
+const (
+	FOREGROUND_BLUE uint8 = 1 << iota
+	FOREGROUND_GREEN
+	FOREGROUND_RED
+	FOREGROUND_INTENSITY
+	BACKGROUND_BLUE
+	BACKGROUND_GREEN
+	BACKGROUND_RED
+	BACKGROUND_INTENSITY
+)
+
+var windowsEscapeSequenceMap = map[string]uint8{
+	TerminalColors.Reset: FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE,
+	TerminalColors.Dim:   FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE,
+	TerminalColors.Bold:  FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE | FOREGROUND_INTENSITY,
+
+	// Apparently underlines only work with the CJK locale on Windows :(
+	TerminalColors.Underline: FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE,
+
+	TerminalColors.Red:   FOREGROUND_RED,
+	TerminalColors.Green: FOREGROUND_GREEN,
+	TerminalColors.Blue:  FOREGROUND_BLUE,
+
+	TerminalColors.Cyan:    FOREGROUND_GREEN | FOREGROUND_BLUE,
+	TerminalColors.Magenta: FOREGROUND_RED | FOREGROUND_BLUE,
+	TerminalColors.Yellow:  FOREGROUND_RED | FOREGROUND_GREEN,
+
+	TerminalColors.RedBgRed:     FOREGROUND_RED | BACKGROUND_RED,
+	TerminalColors.RedBgWhite:   FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE | BACKGROUND_RED,
+	TerminalColors.GreenBgGreen: FOREGROUND_GREEN | BACKGROUND_GREEN,
+	TerminalColors.GreenBgWhite: FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE | BACKGROUND_GREEN,
+	TerminalColors.BlueBgBlue:   FOREGROUND_BLUE | BACKGROUND_BLUE,
+	TerminalColors.BlueBgWhite:  FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE | BACKGROUND_BLUE,
+
+	TerminalColors.CyanBgCyan:       FOREGROUND_GREEN | FOREGROUND_BLUE | BACKGROUND_GREEN | BACKGROUND_BLUE,
+	TerminalColors.CyanBgBlack:      BACKGROUND_GREEN | BACKGROUND_BLUE,
+	TerminalColors.MagentaBgMagenta: FOREGROUND_RED | FOREGROUND_BLUE | BACKGROUND_RED | BACKGROUND_BLUE,
+	TerminalColors.MagentaBgBlack:   BACKGROUND_RED | BACKGROUND_BLUE,
+	TerminalColors.YellowBgYellow:   FOREGROUND_RED | FOREGROUND_GREEN | BACKGROUND_RED | BACKGROUND_GREEN,
+	TerminalColors.YellowBgBlack:    BACKGROUND_RED | BACKGROUND_GREEN,
+}
+
+func writeStringWithColor(file *os.File, text string) {
+	fd := file.Fd()
+	i := 0
+
+	for i < len(text) {
+		// Find the escape
+		if text[i] != 033 {
+			i++
+			continue
+		}
+
+		// Find the 'm'
+		window := text[i:]
+		if len(window) > 8 {
+			window = window[:8]
+		}
+		m := strings.IndexByte(window, 'm')
+		if m == -1 {
+			i++
+			continue
+		}
+		m += i + 1
+
+		// Find the escape sequence
+		attributes, ok := windowsEscapeSequenceMap[text[i:m]]
+		if !ok {
+			i++
+			continue
+		}
+
+		// Write out the text before the escape sequence
+		file.WriteString(text[:i])
+
+		// Apply the escape sequence
+		text = text[m:]
+		i = 0
+		setConsoleTextAttribute.Call(fd, uintptr(attributes))
+	}
+
+	// Write out the remaining text
+	file.WriteString(text)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/logger/msg_ids.go b/source/vendor/github.com/evanw/esbuild/internal/logger/msg_ids.go
new file mode 100644
index 0000000..2e1e305
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/logger/msg_ids.go
@@ -0,0 +1,371 @@
+package logger
+
+// Most non-error log messages are given a message ID that can be used to set
+// the log level for that message. Errors do not get a message ID because you
+// cannot turn errors into non-errors (otherwise the build would incorrectly
+// succeed). Some internal log messages do not get a message ID because they
+// are part of verbose and/or internal debugging output. These messages use
+// "MsgID_None" instead.
+type MsgID = uint8
+
+const (
+	MsgID_None MsgID = iota
+
+	// JavaScript
+	MsgID_JS_AssertToWith
+	MsgID_JS_AssertTypeJSON
+	MsgID_JS_AssignToConstant
+	MsgID_JS_AssignToDefine
+	MsgID_JS_AssignToImport
+	MsgID_JS_CallImportNamespace
+	MsgID_JS_ClassNameWillThrow
+	MsgID_JS_CommonJSVariableInESM
+	MsgID_JS_DeleteSuperProperty
+	MsgID_JS_DirectEval
+	MsgID_JS_DuplicateCase
+	MsgID_JS_DuplicateClassMember
+	MsgID_JS_DuplicateObjectKey
+	MsgID_JS_EmptyImportMeta
+	MsgID_JS_EqualsNaN
+	MsgID_JS_EqualsNegativeZero
+	MsgID_JS_EqualsNewObject
+	MsgID_JS_HTMLCommentInJS
+	MsgID_JS_ImpossibleTypeof
+	MsgID_JS_IndirectRequire
+	MsgID_JS_PrivateNameWillThrow
+	MsgID_JS_SemicolonAfterReturn
+	MsgID_JS_SuspiciousBooleanNot
+	MsgID_JS_SuspiciousDefine
+	MsgID_JS_SuspiciousLogicalOperator
+	MsgID_JS_SuspiciousNullishCoalescing
+	MsgID_JS_ThisIsUndefinedInESM
+	MsgID_JS_UnsupportedDynamicImport
+	MsgID_JS_UnsupportedJSXComment
+	MsgID_JS_UnsupportedRegExp
+	MsgID_JS_UnsupportedRequireCall
+
+	// CSS
+	MsgID_CSS_CSSSyntaxError
+	MsgID_CSS_InvalidAtCharset
+	MsgID_CSS_InvalidAtImport
+	MsgID_CSS_InvalidAtLayer
+	MsgID_CSS_InvalidCalc
+	MsgID_CSS_JSCommentInCSS
+	MsgID_CSS_UndefinedComposesFrom
+	MsgID_CSS_UnsupportedAtCharset
+	MsgID_CSS_UnsupportedAtNamespace
+	MsgID_CSS_UnsupportedCSSProperty
+	MsgID_CSS_UnsupportedCSSNesting
+
+	// Bundler
+	MsgID_Bundler_AmbiguousReexport
+	MsgID_Bundler_DifferentPathCase
+	MsgID_Bundler_EmptyGlob
+	MsgID_Bundler_IgnoredBareImport
+	MsgID_Bundler_IgnoredDynamicImport
+	MsgID_Bundler_ImportIsUndefined
+	MsgID_Bundler_RequireResolveNotExternal
+
+	// Source maps
+	MsgID_SourceMap_InvalidSourceMappings
+	MsgID_SourceMap_SectionsInSourceMap
+	MsgID_SourceMap_MissingSourceMap
+	MsgID_SourceMap_UnsupportedSourceMapComment
+
+	// package.json
+	MsgID_PackageJSON_FIRST // Keep this first
+	MsgID_PackageJSON_DeadCondition
+	MsgID_PackageJSON_InvalidBrowser
+	MsgID_PackageJSON_InvalidImportsOrExports
+	MsgID_PackageJSON_InvalidSideEffects
+	MsgID_PackageJSON_InvalidType
+	MsgID_PackageJSON_LAST // Keep this last
+
+	// tsconfig.json
+	MsgID_TSConfigJSON_FIRST // Keep this first
+	MsgID_TSConfigJSON_Cycle
+	MsgID_TSConfigJSON_InvalidImportsNotUsedAsValues
+	MsgID_TSConfigJSON_InvalidJSX
+	MsgID_TSConfigJSON_InvalidPaths
+	MsgID_TSConfigJSON_InvalidTarget
+	MsgID_TSConfigJSON_InvalidTopLevelOption
+	MsgID_TSConfigJSON_Missing
+	MsgID_TSConfigJSON_LAST // Keep this last
+
+	MsgID_END // Keep this at the end (used only for tests)
+)
+
+func StringToMsgIDs(str string, logLevel LogLevel, overrides map[MsgID]LogLevel) {
+	switch str {
+	// JS
+	case "assert-to-with":
+		overrides[MsgID_JS_AssertToWith] = logLevel
+	case "assert-type-json":
+		overrides[MsgID_JS_AssertTypeJSON] = logLevel
+	case "assign-to-constant":
+		overrides[MsgID_JS_AssignToConstant] = logLevel
+	case "assign-to-define":
+		overrides[MsgID_JS_AssignToDefine] = logLevel
+	case "assign-to-import":
+		overrides[MsgID_JS_AssignToImport] = logLevel
+	case "call-import-namespace":
+		overrides[MsgID_JS_CallImportNamespace] = logLevel
+	case "class-name-will-throw":
+		overrides[MsgID_JS_ClassNameWillThrow] = logLevel
+	case "commonjs-variable-in-esm":
+		overrides[MsgID_JS_CommonJSVariableInESM] = logLevel
+	case "delete-super-property":
+		overrides[MsgID_JS_DeleteSuperProperty] = logLevel
+	case "direct-eval":
+		overrides[MsgID_JS_DirectEval] = logLevel
+	case "duplicate-case":
+		overrides[MsgID_JS_DuplicateCase] = logLevel
+	case "duplicate-class-member":
+		overrides[MsgID_JS_DuplicateClassMember] = logLevel
+	case "duplicate-object-key":
+		overrides[MsgID_JS_DuplicateObjectKey] = logLevel
+	case "empty-import-meta":
+		overrides[MsgID_JS_EmptyImportMeta] = logLevel
+	case "equals-nan":
+		overrides[MsgID_JS_EqualsNaN] = logLevel
+	case "equals-negative-zero":
+		overrides[MsgID_JS_EqualsNegativeZero] = logLevel
+	case "equals-new-object":
+		overrides[MsgID_JS_EqualsNewObject] = logLevel
+	case "html-comment-in-js":
+		overrides[MsgID_JS_HTMLCommentInJS] = logLevel
+	case "impossible-typeof":
+		overrides[MsgID_JS_ImpossibleTypeof] = logLevel
+	case "indirect-require":
+		overrides[MsgID_JS_IndirectRequire] = logLevel
+	case "private-name-will-throw":
+		overrides[MsgID_JS_PrivateNameWillThrow] = logLevel
+	case "semicolon-after-return":
+		overrides[MsgID_JS_SemicolonAfterReturn] = logLevel
+	case "suspicious-boolean-not":
+		overrides[MsgID_JS_SuspiciousBooleanNot] = logLevel
+	case "suspicious-define":
+		overrides[MsgID_JS_SuspiciousDefine] = logLevel
+	case "suspicious-logical-operator":
+		overrides[MsgID_JS_SuspiciousLogicalOperator] = logLevel
+	case "suspicious-nullish-coalescing":
+		overrides[MsgID_JS_SuspiciousNullishCoalescing] = logLevel
+	case "this-is-undefined-in-esm":
+		overrides[MsgID_JS_ThisIsUndefinedInESM] = logLevel
+	case "unsupported-dynamic-import":
+		overrides[MsgID_JS_UnsupportedDynamicImport] = logLevel
+	case "unsupported-jsx-comment":
+		overrides[MsgID_JS_UnsupportedJSXComment] = logLevel
+	case "unsupported-regexp":
+		overrides[MsgID_JS_UnsupportedRegExp] = logLevel
+	case "unsupported-require-call":
+		overrides[MsgID_JS_UnsupportedRequireCall] = logLevel
+
+	// CSS
+	case "css-syntax-error":
+		overrides[MsgID_CSS_CSSSyntaxError] = logLevel
+	case "invalid-@charset":
+		overrides[MsgID_CSS_InvalidAtCharset] = logLevel
+	case "invalid-@import":
+		overrides[MsgID_CSS_InvalidAtImport] = logLevel
+	case "invalid-@layer":
+		overrides[MsgID_CSS_InvalidAtLayer] = logLevel
+	case "invalid-calc":
+		overrides[MsgID_CSS_InvalidCalc] = logLevel
+	case "js-comment-in-css":
+		overrides[MsgID_CSS_JSCommentInCSS] = logLevel
+	case "undefined-composes-from":
+		overrides[MsgID_CSS_UndefinedComposesFrom] = logLevel
+	case "unsupported-@charset":
+		overrides[MsgID_CSS_UnsupportedAtCharset] = logLevel
+	case "unsupported-@namespace":
+		overrides[MsgID_CSS_UnsupportedAtNamespace] = logLevel
+	case "unsupported-css-property":
+		overrides[MsgID_CSS_UnsupportedCSSProperty] = logLevel
+	case "unsupported-css-nesting":
+		overrides[MsgID_CSS_UnsupportedCSSNesting] = logLevel
+
+	// Bundler
+	case "ambiguous-reexport":
+		overrides[MsgID_Bundler_AmbiguousReexport] = logLevel
+	case "different-path-case":
+		overrides[MsgID_Bundler_DifferentPathCase] = logLevel
+	case "empty-glob":
+		overrides[MsgID_Bundler_EmptyGlob] = logLevel
+	case "ignored-bare-import":
+		overrides[MsgID_Bundler_IgnoredBareImport] = logLevel
+	case "ignored-dynamic-import":
+		overrides[MsgID_Bundler_IgnoredDynamicImport] = logLevel
+	case "import-is-undefined":
+		overrides[MsgID_Bundler_ImportIsUndefined] = logLevel
+	case "require-resolve-not-external":
+		overrides[MsgID_Bundler_RequireResolveNotExternal] = logLevel
+
+	// Source maps
+	case "invalid-source-mappings":
+		overrides[MsgID_SourceMap_InvalidSourceMappings] = logLevel
+	case "sections-in-source-map":
+		overrides[MsgID_SourceMap_SectionsInSourceMap] = logLevel
+	case "missing-source-map":
+		overrides[MsgID_SourceMap_MissingSourceMap] = logLevel
+	case "unsupported-source-map-comment":
+		overrides[MsgID_SourceMap_UnsupportedSourceMapComment] = logLevel
+
+	case "package.json":
+		for i := MsgID_PackageJSON_FIRST; i <= MsgID_PackageJSON_LAST; i++ {
+			overrides[i] = logLevel
+		}
+
+	case "tsconfig.json":
+		for i := MsgID_TSConfigJSON_FIRST; i <= MsgID_TSConfigJSON_LAST; i++ {
+			overrides[i] = logLevel
+		}
+
+	default:
+		// Ignore invalid entries since this message id may have
+		// been renamed/removed since when this code was written
+	}
+}
+
+func MsgIDToString(id MsgID) string {
+	switch id {
+	// JS
+	case MsgID_JS_AssertToWith:
+		return "assert-to-with"
+	case MsgID_JS_AssertTypeJSON:
+		return "assert-type-json"
+	case MsgID_JS_AssignToConstant:
+		return "assign-to-constant"
+	case MsgID_JS_AssignToDefine:
+		return "assign-to-define"
+	case MsgID_JS_AssignToImport:
+		return "assign-to-import"
+	case MsgID_JS_CallImportNamespace:
+		return "call-import-namespace"
+	case MsgID_JS_ClassNameWillThrow:
+		return "class-name-will-throw"
+	case MsgID_JS_CommonJSVariableInESM:
+		return "commonjs-variable-in-esm"
+	case MsgID_JS_DeleteSuperProperty:
+		return "delete-super-property"
+	case MsgID_JS_DirectEval:
+		return "direct-eval"
+	case MsgID_JS_DuplicateCase:
+		return "duplicate-case"
+	case MsgID_JS_DuplicateClassMember:
+		return "duplicate-class-member"
+	case MsgID_JS_DuplicateObjectKey:
+		return "duplicate-object-key"
+	case MsgID_JS_EmptyImportMeta:
+		return "empty-import-meta"
+	case MsgID_JS_EqualsNaN:
+		return "equals-nan"
+	case MsgID_JS_EqualsNegativeZero:
+		return "equals-negative-zero"
+	case MsgID_JS_EqualsNewObject:
+		return "equals-new-object"
+	case MsgID_JS_HTMLCommentInJS:
+		return "html-comment-in-js"
+	case MsgID_JS_ImpossibleTypeof:
+		return "impossible-typeof"
+	case MsgID_JS_IndirectRequire:
+		return "indirect-require"
+	case MsgID_JS_PrivateNameWillThrow:
+		return "private-name-will-throw"
+	case MsgID_JS_SemicolonAfterReturn:
+		return "semicolon-after-return"
+	case MsgID_JS_SuspiciousBooleanNot:
+		return "suspicious-boolean-not"
+	case MsgID_JS_SuspiciousDefine:
+		return "suspicious-define"
+	case MsgID_JS_SuspiciousLogicalOperator:
+		return "suspicious-logical-operator"
+	case MsgID_JS_SuspiciousNullishCoalescing:
+		return "suspicious-nullish-coalescing"
+	case MsgID_JS_ThisIsUndefinedInESM:
+		return "this-is-undefined-in-esm"
+	case MsgID_JS_UnsupportedDynamicImport:
+		return "unsupported-dynamic-import"
+	case MsgID_JS_UnsupportedJSXComment:
+		return "unsupported-jsx-comment"
+	case MsgID_JS_UnsupportedRegExp:
+		return "unsupported-regexp"
+	case MsgID_JS_UnsupportedRequireCall:
+		return "unsupported-require-call"
+
+	// CSS
+	case MsgID_CSS_CSSSyntaxError:
+		return "css-syntax-error"
+	case MsgID_CSS_InvalidAtCharset:
+		return "invalid-@charset"
+	case MsgID_CSS_InvalidAtImport:
+		return "invalid-@import"
+	case MsgID_CSS_InvalidAtLayer:
+		return "invalid-@layer"
+	case MsgID_CSS_InvalidCalc:
+		return "invalid-calc"
+	case MsgID_CSS_JSCommentInCSS:
+		return "js-comment-in-css"
+	case MsgID_CSS_UndefinedComposesFrom:
+		return "undefined-composes-from"
+	case MsgID_CSS_UnsupportedAtCharset:
+		return "unsupported-@charset"
+	case MsgID_CSS_UnsupportedAtNamespace:
+		return "unsupported-@namespace"
+	case MsgID_CSS_UnsupportedCSSProperty:
+		return "unsupported-css-property"
+	case MsgID_CSS_UnsupportedCSSNesting:
+		return "unsupported-css-nesting"
+
+	// Bundler
+	case MsgID_Bundler_AmbiguousReexport:
+		return "ambiguous-reexport"
+	case MsgID_Bundler_DifferentPathCase:
+		return "different-path-case"
+	case MsgID_Bundler_EmptyGlob:
+		return "empty-glob"
+	case MsgID_Bundler_IgnoredBareImport:
+		return "ignored-bare-import"
+	case MsgID_Bundler_IgnoredDynamicImport:
+		return "ignored-dynamic-import"
+	case MsgID_Bundler_ImportIsUndefined:
+		return "import-is-undefined"
+	case MsgID_Bundler_RequireResolveNotExternal:
+		return "require-resolve-not-external"
+
+	// Source maps
+	case MsgID_SourceMap_InvalidSourceMappings:
+		return "invalid-source-mappings"
+	case MsgID_SourceMap_SectionsInSourceMap:
+		return "sections-in-source-map"
+	case MsgID_SourceMap_MissingSourceMap:
+		return "missing-source-map"
+	case MsgID_SourceMap_UnsupportedSourceMapComment:
+		return "unsupported-source-map-comment"
+
+	default:
+		if id >= MsgID_PackageJSON_FIRST && id <= MsgID_PackageJSON_LAST {
+			return "package.json"
+		}
+		if id >= MsgID_TSConfigJSON_FIRST && id <= MsgID_TSConfigJSON_LAST {
+			return "tsconfig.json"
+		}
+	}
+
+	return ""
+}
+
+// Some message IDs are more diverse internally than externally (in case we
+// want to expand the set of them later on). So just map these to the largest
+// one arbitrarily since you can't tell the difference externally anyway.
+func StringToMaximumMsgID(id string) MsgID {
+	overrides := make(map[MsgID]LogLevel)
+	maxID := MsgID_None
+	StringToMsgIDs(id, LevelInfo, overrides)
+	for id := range overrides {
+		if id > maxID {
+			maxID = id
+		}
+	}
+	return maxID
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/renamer/renamer.go b/source/vendor/github.com/evanw/esbuild/internal/renamer/renamer.go
new file mode 100644
index 0000000..15a9bf6
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/renamer/renamer.go
@@ -0,0 +1,662 @@
+package renamer
+
+import (
+	"fmt"
+	"sort"
+	"strconv"
+	"sync"
+	"sync/atomic"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/js_lexer"
+)
+
+func ComputeReservedNames(moduleScopes []*js_ast.Scope, symbols ast.SymbolMap) map[string]uint32 {
+	names := make(map[string]uint32)
+
+	// All keywords and strict mode reserved words are reserved names
+	for k := range js_lexer.Keywords {
+		names[k] = 1
+	}
+	for k := range js_lexer.StrictModeReservedWords {
+		names[k] = 1
+	}
+
+	// All unbound symbols must be reserved names
+	for _, scope := range moduleScopes {
+		computeReservedNamesForScope(scope, symbols, names)
+	}
+
+	return names
+}
+
+func computeReservedNamesForScope(scope *js_ast.Scope, symbols ast.SymbolMap, names map[string]uint32) {
+	for _, member := range scope.Members {
+		symbol := symbols.Get(member.Ref)
+		if symbol.Kind == ast.SymbolUnbound || symbol.Flags.Has(ast.MustNotBeRenamed) {
+			names[symbol.OriginalName] = 1
+		}
+	}
+	for _, ref := range scope.Generated {
+		symbol := symbols.Get(ref)
+		if symbol.Kind == ast.SymbolUnbound || symbol.Flags.Has(ast.MustNotBeRenamed) {
+			names[symbol.OriginalName] = 1
+		}
+	}
+
+	// If there's a direct "eval" somewhere inside the current scope, continue
+	// traversing down the scope tree until we find it to get all reserved names
+	if scope.ContainsDirectEval {
+		for _, child := range scope.Children {
+			if child.ContainsDirectEval {
+				computeReservedNamesForScope(child, symbols, names)
+			}
+		}
+	}
+}
+
+type Renamer interface {
+	NameForSymbol(ref ast.Ref) string
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// noOpRenamer
+
+type noOpRenamer struct {
+	symbols ast.SymbolMap
+}
+
+func NewNoOpRenamer(symbols ast.SymbolMap) Renamer {
+	return &noOpRenamer{
+		symbols: symbols,
+	}
+}
+
+func (r *noOpRenamer) NameForSymbol(ref ast.Ref) string {
+	ref = ast.FollowSymbols(r.symbols, ref)
+	return r.symbols.Get(ref).OriginalName
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// MinifyRenamer
+
+type symbolSlot struct {
+	name               string
+	count              uint32
+	needsCapitalForJSX uint32 // This is really a bool but needs to be atomic
+}
+
+type MinifyRenamer struct {
+	reservedNames        map[string]uint32
+	slots                [4][]symbolSlot
+	topLevelSymbolToSlot map[ast.Ref]uint32
+	symbols              ast.SymbolMap
+}
+
+func NewMinifyRenamer(symbols ast.SymbolMap, firstTopLevelSlots ast.SlotCounts, reservedNames map[string]uint32) *MinifyRenamer {
+	return &MinifyRenamer{
+		symbols:       symbols,
+		reservedNames: reservedNames,
+		slots: [4][]symbolSlot{
+			make([]symbolSlot, firstTopLevelSlots[0]),
+			make([]symbolSlot, firstTopLevelSlots[1]),
+			make([]symbolSlot, firstTopLevelSlots[2]),
+			make([]symbolSlot, firstTopLevelSlots[3]),
+		},
+		topLevelSymbolToSlot: make(map[ast.Ref]uint32),
+	}
+}
+
+func (r *MinifyRenamer) NameForSymbol(ref ast.Ref) string {
+	// Follow links to get to the underlying symbol
+	ref = ast.FollowSymbols(r.symbols, ref)
+	symbol := r.symbols.Get(ref)
+
+	// Skip this symbol if the name is pinned
+	ns := symbol.SlotNamespace()
+	if ns == ast.SlotMustNotBeRenamed {
+		return symbol.OriginalName
+	}
+
+	// Check if it's a nested scope symbol
+	i := symbol.NestedScopeSlot
+
+	// If it's not (i.e. it's in a top-level scope), look up the slot
+	if !i.IsValid() {
+		index, ok := r.topLevelSymbolToSlot[ref]
+		if !ok {
+			// If we get here, then we're printing a symbol that never had any
+			// recorded uses. This is odd but can happen in certain scenarios.
+			// For example, code in a branch with dead control flow won't mark
+			// any uses but may still be printed. In that case it doesn't matter
+			// what name we use since it's dead code.
+			return symbol.OriginalName
+		}
+		i = ast.MakeIndex32(index)
+	}
+
+	return r.slots[ns][i.GetIndex()].name
+}
+
+// The InnerIndex should be stable because the parser for a single file is
+// single-threaded and deterministically assigns out InnerIndex values
+// sequentially. But the SourceIndex should be unstable because the main thread
+// assigns out source index values sequentially to newly-discovered dependencies
+// in a multi-threaded producer/consumer relationship. So instead we use the
+// index of the source in the DFS order over all entry points for stability.
+type StableSymbolCount struct {
+	StableSourceIndex uint32
+	Ref               ast.Ref
+	Count             uint32
+}
+
+// This type is just so we can use Go's native sort function
+type StableSymbolCountArray []StableSymbolCount
+
+func (a StableSymbolCountArray) Len() int          { return len(a) }
+func (a StableSymbolCountArray) Swap(i int, j int) { a[i], a[j] = a[j], a[i] }
+
+func (a StableSymbolCountArray) Less(i int, j int) bool {
+	ai, aj := a[i], a[j]
+	if ai.Count > aj.Count {
+		return true
+	}
+	if ai.Count < aj.Count {
+		return false
+	}
+	if ai.StableSourceIndex < aj.StableSourceIndex {
+		return true
+	}
+	if ai.StableSourceIndex > aj.StableSourceIndex {
+		return false
+	}
+	return ai.Ref.InnerIndex < aj.Ref.InnerIndex
+}
+
+func (r *MinifyRenamer) AccumulateSymbolUseCounts(
+	topLevelSymbols *StableSymbolCountArray,
+	symbolUses map[ast.Ref]js_ast.SymbolUse,
+	stableSourceIndices []uint32,
+) {
+	// NOTE: This function is run in parallel. Make sure to avoid data races.
+
+	for ref, use := range symbolUses {
+		r.AccumulateSymbolCount(topLevelSymbols, ref, use.CountEstimate, stableSourceIndices)
+	}
+}
+
+func (r *MinifyRenamer) AccumulateSymbolCount(
+	topLevelSymbols *StableSymbolCountArray,
+	ref ast.Ref,
+	count uint32,
+	stableSourceIndices []uint32,
+) {
+	// NOTE: This function is run in parallel. Make sure to avoid data races.
+
+	// Follow links to get to the underlying symbol
+	ref = ast.FollowSymbols(r.symbols, ref)
+	symbol := r.symbols.Get(ref)
+	for symbol.NamespaceAlias != nil {
+		ref = ast.FollowSymbols(r.symbols, symbol.NamespaceAlias.NamespaceRef)
+		symbol = r.symbols.Get(ref)
+	}
+
+	// Skip this symbol if the name is pinned
+	ns := symbol.SlotNamespace()
+	if ns == ast.SlotMustNotBeRenamed {
+		return
+	}
+
+	// Check if it's a nested scope symbol
+	if i := symbol.NestedScopeSlot; i.IsValid() {
+		// If it is, accumulate the count using a parallel-safe atomic increment
+		slot := &r.slots[ns][i.GetIndex()]
+		atomic.AddUint32(&slot.count, count)
+		if symbol.Flags.Has(ast.MustStartWithCapitalLetterForJSX) {
+			atomic.StoreUint32(&slot.needsCapitalForJSX, 1)
+		}
+		return
+	}
+
+	// If it's a top-level symbol, defer it to later since we have
+	// to allocate slots for these in serial instead of in parallel
+	*topLevelSymbols = append(*topLevelSymbols, StableSymbolCount{
+		StableSourceIndex: stableSourceIndices[ref.SourceIndex],
+		Ref:               ref,
+		Count:             count,
+	})
+}
+
+// The parallel part of the symbol count accumulation algorithm above processes
+// nested symbols and generates an array of top-level symbols to process later.
+// After the parallel part has finished, that array of top-level symbols is passed
+// to this function which processes them in serial.
+func (r *MinifyRenamer) AllocateTopLevelSymbolSlots(topLevelSymbols StableSymbolCountArray) {
+	for _, stable := range topLevelSymbols {
+		symbol := r.symbols.Get(stable.Ref)
+		slots := &r.slots[symbol.SlotNamespace()]
+		if i, ok := r.topLevelSymbolToSlot[stable.Ref]; ok {
+			slot := &(*slots)[i]
+			slot.count += stable.Count
+			if symbol.Flags.Has(ast.MustStartWithCapitalLetterForJSX) {
+				slot.needsCapitalForJSX = 1
+			}
+		} else {
+			needsCapitalForJSX := uint32(0)
+			if symbol.Flags.Has(ast.MustStartWithCapitalLetterForJSX) {
+				needsCapitalForJSX = 1
+			}
+			i = uint32(len(*slots))
+			*slots = append(*slots, symbolSlot{
+				count:              stable.Count,
+				needsCapitalForJSX: needsCapitalForJSX,
+			})
+			r.topLevelSymbolToSlot[stable.Ref] = i
+		}
+	}
+}
+
+func (r *MinifyRenamer) AssignNamesByFrequency(minifier *ast.NameMinifier) {
+	for ns, slots := range r.slots {
+		// Sort symbols by count
+		sorted := make(slotAndCountArray, len(slots))
+		for i, item := range slots {
+			sorted[i] = slotAndCount{slot: uint32(i), count: item.count}
+		}
+		sort.Sort(sorted)
+
+		// Assign names to symbols
+		nextName := 0
+		for _, data := range sorted {
+			slot := &slots[data.slot]
+			name := minifier.NumberToMinifiedName(nextName)
+			nextName++
+
+			// Make sure we never generate a reserved name. We only have to worry
+			// about collisions with reserved identifiers for normal symbols, and we
+			// only have to worry about collisions with keywords for labels. We do
+			// not have to worry about either for private names because they start
+			// with a "#" character.
+			switch ast.SlotNamespace(ns) {
+			case ast.SlotDefault:
+				for r.reservedNames[name] != 0 {
+					name = minifier.NumberToMinifiedName(nextName)
+					nextName++
+				}
+
+				// Make sure names of symbols used in JSX elements start with a capital letter
+				if slot.needsCapitalForJSX != 0 {
+					for name[0] >= 'a' && name[0] <= 'z' {
+						name = minifier.NumberToMinifiedName(nextName)
+						nextName++
+					}
+				}
+
+			case ast.SlotLabel:
+				for js_lexer.Keywords[name] != 0 {
+					name = minifier.NumberToMinifiedName(nextName)
+					nextName++
+				}
+			}
+
+			// Private names must be prefixed with "#"
+			if ast.SlotNamespace(ns) == ast.SlotPrivateName {
+				name = "#" + name
+			}
+
+			slot.name = name
+		}
+	}
+}
+
+// Returns the number of nested slots
+func AssignNestedScopeSlots(moduleScope *js_ast.Scope, symbols []ast.Symbol) (slotCounts ast.SlotCounts) {
+	// Temporarily set the nested scope slots of top-level symbols to valid so
+	// they aren't renamed in nested scopes. This prevents us from accidentally
+	// assigning nested scope slots to variables declared using "var" in a nested
+	// scope that are actually hoisted up to the module scope to become a top-
+	// level symbol.
+	validSlot := ast.MakeIndex32(1)
+	for _, member := range moduleScope.Members {
+		symbols[member.Ref.InnerIndex].NestedScopeSlot = validSlot
+	}
+	for _, ref := range moduleScope.Generated {
+		symbols[ref.InnerIndex].NestedScopeSlot = validSlot
+	}
+
+	// Assign nested scope slots independently for each nested scope
+	for _, child := range moduleScope.Children {
+		slotCounts.UnionMax(assignNestedScopeSlotsHelper(child, symbols, ast.SlotCounts{}))
+	}
+
+	// Then set the nested scope slots of top-level symbols back to zero. Top-
+	// level symbols are not supposed to have nested scope slots.
+	for _, member := range moduleScope.Members {
+		symbols[member.Ref.InnerIndex].NestedScopeSlot = ast.Index32{}
+	}
+	for _, ref := range moduleScope.Generated {
+		symbols[ref.InnerIndex].NestedScopeSlot = ast.Index32{}
+	}
+	return
+}
+
+func assignNestedScopeSlotsHelper(scope *js_ast.Scope, symbols []ast.Symbol, slot ast.SlotCounts) ast.SlotCounts {
+	// Sort member map keys for determinism
+	sortedMembers := make([]int, 0, len(scope.Members))
+	for _, member := range scope.Members {
+		sortedMembers = append(sortedMembers, int(member.Ref.InnerIndex))
+	}
+	sort.Ints(sortedMembers)
+
+	// Assign slots for this scope's symbols. Only do this if the slot is
+	// not already assigned. Nested scopes have copies of symbols from parent
+	// scopes and we want to use the slot from the parent scope, not child scopes.
+	for _, innerIndex := range sortedMembers {
+		symbol := &symbols[innerIndex]
+		if ns := symbol.SlotNamespace(); ns != ast.SlotMustNotBeRenamed && !symbol.NestedScopeSlot.IsValid() {
+			symbol.NestedScopeSlot = ast.MakeIndex32(slot[ns])
+			slot[ns]++
+		}
+	}
+	for _, ref := range scope.Generated {
+		symbol := &symbols[ref.InnerIndex]
+		if ns := symbol.SlotNamespace(); ns != ast.SlotMustNotBeRenamed && !symbol.NestedScopeSlot.IsValid() {
+			symbol.NestedScopeSlot = ast.MakeIndex32(slot[ns])
+			slot[ns]++
+		}
+	}
+
+	// Labels are always declared in a nested scope, so we don't need to check.
+	if scope.Label.Ref != ast.InvalidRef {
+		symbol := &symbols[scope.Label.Ref.InnerIndex]
+		symbol.NestedScopeSlot = ast.MakeIndex32(slot[ast.SlotLabel])
+		slot[ast.SlotLabel]++
+	}
+
+	// Assign slots for the symbols of child scopes
+	slotCounts := slot
+	for _, child := range scope.Children {
+		slotCounts.UnionMax(assignNestedScopeSlotsHelper(child, symbols, slot))
+	}
+	return slotCounts
+}
+
+type slotAndCount struct {
+	slot  uint32
+	count uint32
+}
+
+// This type is just so we can use Go's native sort function
+type slotAndCountArray []slotAndCount
+
+func (a slotAndCountArray) Len() int          { return len(a) }
+func (a slotAndCountArray) Swap(i int, j int) { a[i], a[j] = a[j], a[i] }
+func (a slotAndCountArray) Less(i int, j int) bool {
+	ai, aj := a[i], a[j]
+	return ai.count > aj.count || (ai.count == aj.count && ai.slot < aj.slot)
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// NumberRenamer
+
+type NumberRenamer struct {
+	symbols ast.SymbolMap
+	root    numberScope
+	names   [][]string
+}
+
+func NewNumberRenamer(symbols ast.SymbolMap, reservedNames map[string]uint32) *NumberRenamer {
+	return &NumberRenamer{
+		symbols: symbols,
+		names:   make([][]string, len(symbols.SymbolsForSource)),
+		root:    numberScope{nameCounts: reservedNames},
+	}
+}
+
+func (r *NumberRenamer) NameForSymbol(ref ast.Ref) string {
+	ref = ast.FollowSymbols(r.symbols, ref)
+	if inner := r.names[ref.SourceIndex]; inner != nil {
+		if name := inner[ref.InnerIndex]; name != "" {
+			return name
+		}
+	}
+	return r.symbols.Get(ref).OriginalName
+}
+
+func (r *NumberRenamer) AddTopLevelSymbol(ref ast.Ref) {
+	r.assignName(&r.root, ref)
+}
+
+func (r *NumberRenamer) assignName(scope *numberScope, ref ast.Ref) {
+	ref = ast.FollowSymbols(r.symbols, ref)
+
+	// Don't rename the same symbol more than once
+	inner := r.names[ref.SourceIndex]
+	if inner != nil && inner[ref.InnerIndex] != "" {
+		return
+	}
+
+	// Don't rename unbound symbols, symbols marked as reserved names, labels, or private names
+	symbol := r.symbols.Get(ref)
+	ns := symbol.SlotNamespace()
+	if ns != ast.SlotDefault && ns != ast.SlotPrivateName {
+		return
+	}
+
+	// Make sure names of symbols used in JSX elements start with a capital letter
+	originalName := symbol.OriginalName
+	if symbol.Flags.Has(ast.MustStartWithCapitalLetterForJSX) {
+		if first := rune(originalName[0]); first >= 'a' && first <= 'z' {
+			originalName = fmt.Sprintf("%c%s", first+('A'-'a'), originalName[1:])
+		}
+	}
+
+	// Compute a new name
+	name := scope.findUnusedName(originalName, ns)
+
+	// Store the new name
+	if inner == nil {
+		// Note: This should not be a data race even though this method is run from
+		// multiple threads. The parallel part only looks at symbols defined in
+		// nested scopes, and those can only ever be accessed from within the file.
+		// References to those symbols should never spread across files.
+		//
+		// While we could avoid the data race by densely preallocating the entire
+		// "names" array ahead of time, that will waste a lot more memory for
+		// builds that make heavy use of code splitting and have many chunks. Doing
+		// things lazily like this means we use less memory but still stay safe.
+		inner = make([]string, len(r.symbols.SymbolsForSource[ref.SourceIndex]))
+		r.names[ref.SourceIndex] = inner
+	}
+	inner[ref.InnerIndex] = name
+}
+
+func (r *NumberRenamer) assignNamesInScope(scope *js_ast.Scope, sourceIndex uint32, parent *numberScope, sorted *[]int) *numberScope {
+	s := &numberScope{parent: parent, nameCounts: make(map[string]uint32)}
+
+	if len(scope.Members) > 0 {
+		// Sort member map keys for determinism, reusing a shared memory buffer
+		*sorted = (*sorted)[:0]
+		for _, member := range scope.Members {
+			*sorted = append(*sorted, int(member.Ref.InnerIndex))
+		}
+		sort.Ints(*sorted)
+
+		// Rename all user-defined symbols in this scope
+		for _, innerIndex := range *sorted {
+			r.assignName(s, ast.Ref{SourceIndex: sourceIndex, InnerIndex: uint32(innerIndex)})
+		}
+	}
+
+	// Also rename all generated symbols in this scope
+	for _, ref := range scope.Generated {
+		r.assignName(s, ref)
+	}
+
+	return s
+}
+
+func (r *NumberRenamer) assignNamesRecursive(scope *js_ast.Scope, sourceIndex uint32, parent *numberScope, sorted *[]int) {
+	// For performance in extreme cases (e.g. 10,000 nested scopes), traversing
+	// through singly-nested scopes uses iteration instead of recursion
+	for {
+		if len(scope.Members) > 0 || len(scope.Generated) > 0 {
+			// For performance in extreme cases (e.g. 10,000 nested scopes), only
+			// allocate a scope when it's necessary. I'm not quite sure why allocating
+			// one scope per level is so much overhead. It's not that many objects.
+			// Or at least there are already that many objects for the AST that we're
+			// traversing, so I don't know why 80% of the time in these extreme cases
+			// is taken by this function (if we don't avoid this allocation).
+			parent = r.assignNamesInScope(scope, sourceIndex, parent, sorted)
+		}
+		if children := scope.Children; len(children) == 1 {
+			scope = children[0]
+		} else {
+			break
+		}
+	}
+
+	// Symbols in child scopes may also have to be renamed to avoid conflicts
+	for _, child := range scope.Children {
+		r.assignNamesRecursive(child, sourceIndex, parent, sorted)
+	}
+}
+
+func (r *NumberRenamer) AssignNamesByScope(nestedScopes map[uint32][]*js_ast.Scope) {
+	waitGroup := sync.WaitGroup{}
+	waitGroup.Add(len(nestedScopes))
+
+	// Rename nested scopes from separate files in parallel
+	for sourceIndex, scopes := range nestedScopes {
+		go func(sourceIndex uint32, scopes []*js_ast.Scope) {
+			var sorted []int
+			for _, scope := range scopes {
+				r.assignNamesRecursive(scope, sourceIndex, &r.root, &sorted)
+			}
+			waitGroup.Done()
+		}(sourceIndex, scopes)
+	}
+
+	waitGroup.Wait()
+}
+
+type numberScope struct {
+	parent *numberScope
+
+	// This is used as a set of used names in this scope. This also maps the name
+	// to the number of times the name has experienced a collision. When a name
+	// collides with an already-used name, we need to rename it. This is done by
+	// incrementing a number at the end until the name is unused. We save the
+	// count here so that subsequent collisions can start counting from where the
+	// previous collision ended instead of having to start counting from 1.
+	nameCounts map[string]uint32
+}
+
+type nameUse uint8
+
+const (
+	nameUnused nameUse = iota
+	nameUsed
+	nameUsedInSameScope
+)
+
+func (s *numberScope) findNameUse(name string) nameUse {
+	original := s
+	for {
+		if _, ok := s.nameCounts[name]; ok {
+			if s == original {
+				return nameUsedInSameScope
+			}
+			return nameUsed
+		}
+		s = s.parent
+		if s == nil {
+			return nameUnused
+		}
+	}
+}
+
+func (s *numberScope) findUnusedName(name string, ns ast.SlotNamespace) string {
+	// We may not have a valid identifier if this is an internally-constructed name
+	if ns == ast.SlotPrivateName {
+		if id := name[1:]; !js_ast.IsIdentifier(id) {
+			name = js_ast.ForceValidIdentifier("#", id)
+		}
+	} else {
+		if !js_ast.IsIdentifier(name) {
+			name = js_ast.ForceValidIdentifier("", name)
+		}
+	}
+
+	if use := s.findNameUse(name); use != nameUnused {
+		// If the name is already in use, generate a new name by appending a number
+		tries := uint32(1)
+		if use == nameUsedInSameScope {
+			// To avoid O(n^2) behavior, the number must start off being the number
+			// that we used last time there was a collision with this name. Otherwise
+			// if there are many collisions with the same name, each name collision
+			// would have to increment the counter past all previous name collisions
+			// which is a O(n^2) time algorithm. Only do this if this symbol comes
+			// from the same scope as the previous one since sibling scopes can reuse
+			// the same name without problems.
+			tries = s.nameCounts[name]
+		}
+		prefix := name
+
+		// Keep incrementing the number until the name is unused
+		for {
+			tries++
+			name = prefix + strconv.Itoa(int(tries))
+
+			// Make sure this new name is unused
+			if s.findNameUse(name) == nameUnused {
+				// Store the count so we can start here next time instead of starting
+				// from 1. This means we avoid O(n^2) behavior.
+				if use == nameUsedInSameScope {
+					s.nameCounts[prefix] = tries
+				}
+				break
+			}
+		}
+	}
+
+	// Each name starts off with a count of 1 so that the first collision with
+	// "name" is called "name2"
+	s.nameCounts[name] = 1
+	return name
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// ExportRenamer
+
+type ExportRenamer struct {
+	used  map[string]uint32
+	count int
+}
+
+func (r *ExportRenamer) NextRenamedName(name string) string {
+	if r.used == nil {
+		r.used = make(map[string]uint32)
+	}
+	if tries, ok := r.used[name]; ok {
+		prefix := name
+		for {
+			tries++
+			name = prefix + strconv.Itoa(int(tries))
+			if _, ok := r.used[name]; !ok {
+				break
+			}
+		}
+		r.used[name] = tries
+	} else {
+		r.used[name] = 1
+	}
+	return name
+}
+
+func (r *ExportRenamer) NextMinifiedName() string {
+	name := ast.DefaultNameMinifierJS.NumberToMinifiedName(r.count)
+	r.count++
+	return name
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/resolver/dataurl.go b/source/vendor/github.com/evanw/esbuild/internal/resolver/dataurl.go
new file mode 100644
index 0000000..c46e607
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/resolver/dataurl.go
@@ -0,0 +1,76 @@
+package resolver
+
+import (
+	"encoding/base64"
+	"fmt"
+	"net/url"
+	"strings"
+)
+
+type DataURL struct {
+	mimeType string
+	data     string
+	isBase64 bool
+}
+
+func ParseDataURL(url string) (parsed DataURL, ok bool) {
+	if strings.HasPrefix(url, "data:") {
+		if comma := strings.IndexByte(url, ','); comma != -1 {
+			parsed.mimeType = url[len("data:"):comma]
+			parsed.data = url[comma+1:]
+			if strings.HasSuffix(parsed.mimeType, ";base64") {
+				parsed.mimeType = parsed.mimeType[:len(parsed.mimeType)-len(";base64")]
+				parsed.isBase64 = true
+			}
+			ok = true
+		}
+	}
+	return
+}
+
+type MIMEType uint8
+
+const (
+	MIMETypeUnsupported MIMEType = iota
+	MIMETypeTextCSS
+	MIMETypeTextJavaScript
+	MIMETypeApplicationJSON
+)
+
+func (parsed DataURL) DecodeMIMEType() MIMEType {
+	// Remove things like ";charset=utf-8"
+	mimeType := parsed.mimeType
+	if semicolon := strings.IndexByte(mimeType, ';'); semicolon != -1 {
+		mimeType = mimeType[:semicolon]
+	}
+
+	// Hard-code a few supported types
+	switch mimeType {
+	case "text/css":
+		return MIMETypeTextCSS
+	case "text/javascript":
+		return MIMETypeTextJavaScript
+	case "application/json":
+		return MIMETypeApplicationJSON
+	default:
+		return MIMETypeUnsupported
+	}
+}
+
+func (parsed DataURL) DecodeData() (string, error) {
+	// Try to read base64 data
+	if parsed.isBase64 {
+		bytes, err := base64.StdEncoding.DecodeString(parsed.data)
+		if err != nil {
+			return "", fmt.Errorf("could not decode base64 data: %s", err.Error())
+		}
+		return string(bytes), nil
+	}
+
+	// Try to read percent-escaped data
+	content, err := url.PathUnescape(parsed.data)
+	if err != nil {
+		return "", fmt.Errorf("could not decode percent-escaped data: %s", err.Error())
+	}
+	return content, nil
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/resolver/package_json.go b/source/vendor/github.com/evanw/esbuild/internal/resolver/package_json.go
new file mode 100644
index 0000000..068acde
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/resolver/package_json.go
@@ -0,0 +1,1462 @@
+package resolver
+
+import (
+	"fmt"
+	"net/url"
+	"path"
+	"regexp"
+	"sort"
+	"strings"
+
+	"github.com/evanw/esbuild/internal/config"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/js_lexer"
+	"github.com/evanw/esbuild/internal/js_parser"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type packageJSON struct {
+	name           string
+	mainFields     map[string]mainField
+	moduleTypeData js_ast.ModuleTypeData
+
+	// "TypeScript will first check whether package.json contains a "tsconfig"
+	// field, and if it does, TypeScript will try to load a configuration file
+	// from that field. If neither exists, TypeScript will try to read from a
+	// tsconfig.json at the root."
+	//
+	// See: https://www.typescriptlang.org/docs/handbook/release-notes/typescript-3-2.html#tsconfigjson-inheritance-via-nodejs-packages
+	tsconfig string
+
+	// Present if the "browser" field is present. This field is intended to be
+	// used by bundlers and lets you redirect the paths of certain 3rd-party
+	// modules that don't work in the browser to other modules that shim that
+	// functionality. That way you don't have to rewrite the code for those 3rd-
+	// party modules. For example, you might remap the native "util" node module
+	// to something like https://www.npmjs.com/package/util so it works in the
+	// browser.
+	//
+	// This field contains the original mapping object in "package.json". Mapping
+	// to a nil path indicates that the module is disabled. As far as I can
+	// tell, the official spec is an abandoned GitHub repo hosted by a user account:
+	// https://github.com/defunctzombie/package-browser-field-spec. The npm docs
+	// say almost nothing: https://docs.npmjs.com/files/package.json.
+	//
+	// Note that the non-package "browser" map has to be checked twice to match
+	// Webpack's behavior: once before resolution and once after resolution. It
+	// leads to some unintuitive failure cases that we must emulate around missing
+	// file extensions:
+	//
+	// * Given the mapping "./no-ext": "./no-ext-browser.js" the query "./no-ext"
+	//   should match but the query "./no-ext.js" should NOT match.
+	//
+	// * Given the mapping "./ext.js": "./ext-browser.js" the query "./ext.js"
+	//   should match and the query "./ext" should ALSO match.
+	//
+	browserMap map[string]*string
+
+	// If this is non-nil, each entry in this map is the absolute path of a file
+	// with side effects. Any entry not in this map should be considered to have
+	// no side effects, which means import statements for these files can be
+	// removed if none of the imports are used. This is a convention from Webpack:
+	// https://webpack.js.org/guides/tree-shaking/.
+	//
+	// Note that if a file is included, all statements that can't be proven to be
+	// free of side effects must be included. This convention does not say
+	// anything about whether any statements within the file have side effects or
+	// not.
+	sideEffectsMap     map[string]bool
+	sideEffectsRegexps []*regexp.Regexp
+	sideEffectsData    *SideEffectsData
+
+	// This represents the "imports" field in this package.json file.
+	importsMap *pjMap
+
+	// This represents the "exports" field in this package.json file.
+	exportsMap *pjMap
+
+	source logger.Source
+}
+
+type mainField struct {
+	relPath string
+	keyLoc  logger.Loc
+}
+
+type browserPathKind uint8
+
+const (
+	absolutePathKind browserPathKind = iota
+	packagePathKind
+)
+
+func (r resolverQuery) checkBrowserMap(resolveDirInfo *dirInfo, inputPath string, kind browserPathKind) (remapped *string, ok bool) {
+	// This only applies if the current platform is "browser"
+	if r.options.Platform != config.PlatformBrowser {
+		return nil, false
+	}
+
+	// There must be an enclosing directory with a "package.json" file with a "browser" map
+	if resolveDirInfo.enclosingBrowserScope == nil {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("No \"browser\" map found in directory %q", resolveDirInfo.absPath))
+		}
+		return nil, false
+	}
+
+	packageJSON := resolveDirInfo.enclosingBrowserScope.packageJSON
+	browserMap := packageJSON.browserMap
+
+	type implicitExtensions uint8
+
+	const (
+		includeImplicitExtensions implicitExtensions = iota
+		skipImplicitExtensions
+	)
+
+	checkPath := func(pathToCheck string, implicitExtensions implicitExtensions) bool {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("Checking for %q in the \"browser\" map in %q",
+				pathToCheck, packageJSON.source.KeyPath.Text))
+		}
+
+		// Check for equality
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("  Checking for %q", pathToCheck))
+		}
+		remapped, ok = browserMap[pathToCheck]
+		if ok {
+			inputPath = pathToCheck
+			return true
+		}
+
+		// If that failed, try adding implicit extensions
+		if implicitExtensions == includeImplicitExtensions {
+			for _, ext := range r.options.ExtensionOrder {
+				extPath := pathToCheck + ext
+				if r.debugLogs != nil {
+					r.debugLogs.addNote(fmt.Sprintf("  Checking for %q", extPath))
+				}
+				remapped, ok = browserMap[extPath]
+				if ok {
+					inputPath = extPath
+					return true
+				}
+			}
+		}
+
+		// If that failed, try assuming this is a directory and looking for an "index" file
+		indexPath := path.Join(pathToCheck, "index")
+		if IsPackagePath(indexPath) && !IsPackagePath(pathToCheck) {
+			indexPath = "./" + indexPath
+		}
+
+		// Check for equality
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("  Checking for %q", indexPath))
+		}
+		remapped, ok = browserMap[indexPath]
+		if ok {
+			inputPath = indexPath
+			return true
+		}
+
+		// If that failed, try adding implicit extensions
+		if implicitExtensions == includeImplicitExtensions {
+			for _, ext := range r.options.ExtensionOrder {
+				extPath := indexPath + ext
+				if r.debugLogs != nil {
+					r.debugLogs.addNote(fmt.Sprintf("  Checking for %q", extPath))
+				}
+				remapped, ok = browserMap[extPath]
+				if ok {
+					inputPath = extPath
+					return true
+				}
+			}
+		}
+
+		return false
+	}
+
+	// Turn absolute paths into paths relative to the "browser" map location
+	if kind == absolutePathKind {
+		relPath, ok := r.fs.Rel(resolveDirInfo.enclosingBrowserScope.absPath, inputPath)
+		if !ok {
+			return nil, false
+		}
+		inputPath = strings.ReplaceAll(relPath, "\\", "/")
+	}
+
+	if inputPath == "." {
+		// No bundler supports remapping ".", so we don't either
+		return nil, false
+	}
+
+	// First try the import path as a package path
+	if !checkPath(inputPath, includeImplicitExtensions) && IsPackagePath(inputPath) {
+		// If a package path didn't work, try the import path as a relative path
+		switch kind {
+		case absolutePathKind:
+			checkPath("./"+inputPath, includeImplicitExtensions)
+
+		case packagePathKind:
+			// Browserify allows a browser map entry of "./pkg" to override a package
+			// path of "require('pkg')". This is weird, and arguably a bug. But we
+			// replicate this bug for compatibility. However, Browserify only allows
+			// this within the same package. It does not allow such an entry in a
+			// parent package to override this in a child package. So this behavior
+			// is disallowed if there is a "node_modules" folder in between the child
+			// package and the parent package.
+			isInSamePackage := true
+			for info := resolveDirInfo; info != nil && info != resolveDirInfo.enclosingBrowserScope; info = info.parent {
+				if info.isNodeModules {
+					isInSamePackage = false
+					break
+				}
+			}
+			if isInSamePackage {
+				relativePathPrefix := "./"
+
+				// Use the relative path from the file containing the import path to the
+				// enclosing package.json file. This includes any subdirectories within the
+				// package if there are any.
+				if relPath, ok := r.fs.Rel(resolveDirInfo.enclosingBrowserScope.absPath, resolveDirInfo.absPath); ok && relPath != "." {
+					relativePathPrefix += strings.ReplaceAll(relPath, "\\", "/") + "/"
+				}
+
+				// Browserify lets "require('pkg')" match "./pkg" but not "./pkg.js".
+				// So don't add implicit extensions specifically in this place so we
+				// match Browserify's behavior.
+				checkPath(relativePathPrefix+inputPath, skipImplicitExtensions)
+			}
+		}
+	}
+
+	if r.debugLogs != nil {
+		if ok {
+			if remapped == nil {
+				r.debugLogs.addNote(fmt.Sprintf("Found %q marked as disabled", inputPath))
+			} else {
+				r.debugLogs.addNote(fmt.Sprintf("Found %q mapping to %q", inputPath, *remapped))
+			}
+		} else {
+			r.debugLogs.addNote(fmt.Sprintf("Failed to find %q", inputPath))
+		}
+	}
+	return
+}
+
+func (r resolverQuery) parsePackageJSON(inputPath string) *packageJSON {
+	packageJSONPath := r.fs.Join(inputPath, "package.json")
+	contents, err, originalError := r.caches.FSCache.ReadFile(r.fs, packageJSONPath)
+	if r.debugLogs != nil && originalError != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Failed to read file %q: %s", packageJSONPath, originalError.Error()))
+	}
+	if err != nil {
+		r.log.AddError(nil, logger.Range{},
+			fmt.Sprintf("Cannot read file %q: %s",
+				PrettyPath(r.fs, logger.Path{Text: packageJSONPath, Namespace: "file"}), err.Error()))
+		return nil
+	}
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("The file %q exists", packageJSONPath))
+	}
+
+	keyPath := logger.Path{Text: packageJSONPath, Namespace: "file"}
+	jsonSource := logger.Source{
+		KeyPath:    keyPath,
+		PrettyPath: PrettyPath(r.fs, keyPath),
+		Contents:   contents,
+	}
+	tracker := logger.MakeLineColumnTracker(&jsonSource)
+
+	json, ok := r.caches.JSONCache.Parse(r.log, jsonSource, js_parser.JSONOptions{})
+	if !ok {
+		return nil
+	}
+
+	packageJSON := &packageJSON{
+		source:     jsonSource,
+		mainFields: make(map[string]mainField),
+	}
+
+	// Read the "name" field
+	if nameJSON, _, ok := getProperty(json, "name"); ok {
+		if nameValue, ok := getString(nameJSON); ok {
+			packageJSON.name = nameValue
+		}
+	}
+
+	// Read the "type" field
+	if typeJSON, typeKeyLoc, ok := getProperty(json, "type"); ok {
+		if typeValue, ok := getString(typeJSON); ok {
+			switch typeValue {
+			case "commonjs":
+				packageJSON.moduleTypeData = js_ast.ModuleTypeData{
+					Type:   js_ast.ModuleCommonJS_PackageJSON,
+					Source: &packageJSON.source,
+					Range:  jsonSource.RangeOfString(typeJSON.Loc),
+				}
+			case "module":
+				packageJSON.moduleTypeData = js_ast.ModuleTypeData{
+					Type:   js_ast.ModuleESM_PackageJSON,
+					Source: &packageJSON.source,
+					Range:  jsonSource.RangeOfString(typeJSON.Loc),
+				}
+			default:
+				notes := []logger.MsgData{{Text: "The \"type\" field must be set to either \"commonjs\" or \"module\"."}}
+				kind := logger.Warning
+
+				// If someone does something like "type": "./index.d.ts" then they
+				// likely meant "types" instead of "type". Customize the message
+				// for this and hide it if it's inside a published npm package.
+				if strings.HasSuffix(typeValue, ".d.ts") {
+					notes[0] = tracker.MsgData(jsonSource.RangeOfString(typeKeyLoc),
+						"TypeScript type declarations use the \"types\" field, not the \"type\" field:")
+					notes[0].Location.Suggestion = "\"types\""
+					if helpers.IsInsideNodeModules(jsonSource.KeyPath.Text) {
+						kind = logger.Debug
+					}
+				}
+
+				r.log.AddIDWithNotes(logger.MsgID_PackageJSON_InvalidType, kind, &tracker, jsonSource.RangeOfString(typeJSON.Loc),
+					fmt.Sprintf("%q is not a valid value for the \"type\" field", typeValue),
+					notes)
+			}
+		} else {
+			r.log.AddID(logger.MsgID_PackageJSON_InvalidType, logger.Warning, &tracker, logger.Range{Loc: typeJSON.Loc},
+				"The value for \"type\" must be a string")
+		}
+	}
+
+	// Read the "tsconfig" field
+	if tsconfigJSON, _, ok := getProperty(json, "tsconfig"); ok {
+		if tsconfigValue, ok := getString(tsconfigJSON); ok {
+			packageJSON.tsconfig = tsconfigValue
+		}
+	}
+
+	// Read the "main" fields
+	mainFields := r.options.MainFields
+	if mainFields == nil {
+		mainFields = defaultMainFields[r.options.Platform]
+	}
+	for _, field := range mainFields {
+		if mainJSON, mainLoc, ok := getProperty(json, field); ok {
+			if main, ok := getString(mainJSON); ok && main != "" {
+				packageJSON.mainFields[field] = mainField{keyLoc: mainLoc, relPath: main}
+			}
+		}
+	}
+	for _, field := range mainFieldsForFailure {
+		if _, ok := packageJSON.mainFields[field]; !ok {
+			if mainJSON, mainLoc, ok := getProperty(json, field); ok {
+				if main, ok := getString(mainJSON); ok && main != "" {
+					packageJSON.mainFields[field] = mainField{keyLoc: mainLoc, relPath: main}
+				}
+			}
+		}
+	}
+
+	// Read the "browser" property, but only when targeting the browser
+	if browserJSON, _, ok := getProperty(json, "browser"); ok && r.options.Platform == config.PlatformBrowser {
+		// We both want the ability to have the option of CJS vs. ESM and the
+		// option of having node vs. browser. The way to do this is to use the
+		// object literal form of the "browser" field like this:
+		//
+		//   "main": "dist/index.node.cjs.js",
+		//   "module": "dist/index.node.esm.js",
+		//   "browser": {
+		//     "./dist/index.node.cjs.js": "./dist/index.browser.cjs.js",
+		//     "./dist/index.node.esm.js": "./dist/index.browser.esm.js"
+		//   },
+		//
+		if browser, ok := browserJSON.Data.(*js_ast.EObject); ok {
+			// The value is an object
+			browserMap := make(map[string]*string)
+
+			// Remap all files in the browser field
+			for _, prop := range browser.Properties {
+				if key, ok := getString(prop.Key); ok && prop.ValueOrNil.Data != nil {
+					if value, ok := getString(prop.ValueOrNil); ok {
+						// If this is a string, it's a replacement package
+						browserMap[key] = &value
+					} else if value, ok := getBool(prop.ValueOrNil); ok {
+						// If this is false, it means the package is disabled
+						if !value {
+							browserMap[key] = nil
+						}
+					} else {
+						r.log.AddID(logger.MsgID_PackageJSON_InvalidBrowser, logger.Warning, &tracker, logger.Range{Loc: prop.ValueOrNil.Loc},
+							"Each \"browser\" mapping must be a string or a boolean")
+					}
+				}
+			}
+
+			packageJSON.browserMap = browserMap
+		}
+	}
+
+	// Read the "sideEffects" property
+	if sideEffectsJSON, sideEffectsLoc, ok := getProperty(json, "sideEffects"); ok {
+		switch data := sideEffectsJSON.Data.(type) {
+		case *js_ast.EBoolean:
+			if !data.Value {
+				// Make an empty map for "sideEffects: false", which indicates all
+				// files in this module can be considered to not have side effects.
+				packageJSON.sideEffectsMap = make(map[string]bool)
+				packageJSON.sideEffectsData = &SideEffectsData{
+					IsSideEffectsArrayInJSON: false,
+					Source:                   &jsonSource,
+					Range:                    jsonSource.RangeOfString(sideEffectsLoc),
+				}
+			}
+
+		case *js_ast.EArray:
+			// The "sideEffects: []" format means all files in this module but not in
+			// the array can be considered to not have side effects.
+			packageJSON.sideEffectsMap = make(map[string]bool)
+			packageJSON.sideEffectsData = &SideEffectsData{
+				IsSideEffectsArrayInJSON: true,
+				Source:                   &jsonSource,
+				Range:                    jsonSource.RangeOfString(sideEffectsLoc),
+			}
+			for _, itemJSON := range data.Items {
+				item, ok := itemJSON.Data.(*js_ast.EString)
+				if !ok || item.Value == nil {
+					r.log.AddID(logger.MsgID_PackageJSON_InvalidSideEffects, logger.Warning, &tracker, logger.Range{Loc: itemJSON.Loc},
+						"Expected string in array for \"sideEffects\"")
+					continue
+				}
+
+				// Reference: https://github.com/webpack/webpack/blob/ed175cd22f89eb9fecd0a70572a3fd0be028e77c/lib/optimize/SideEffectsFlagPlugin.js
+				pattern := helpers.UTF16ToString(item.Value)
+				if !strings.ContainsRune(pattern, '/') {
+					pattern = "**/" + pattern
+				}
+				absPattern := r.fs.Join(inputPath, pattern)
+				absPattern = strings.ReplaceAll(absPattern, "\\", "/") // Avoid problems with Windows-style slashes
+				re, hadWildcard := globstarToEscapedRegexp(absPattern)
+
+				// Wildcard patterns require more expensive matching
+				if hadWildcard {
+					packageJSON.sideEffectsRegexps = append(packageJSON.sideEffectsRegexps, regexp.MustCompile(re))
+					continue
+				}
+
+				// Normal strings can be matched with a map lookup
+				packageJSON.sideEffectsMap[absPattern] = true
+			}
+
+		default:
+			r.log.AddID(logger.MsgID_PackageJSON_InvalidSideEffects, logger.Warning, &tracker, logger.Range{Loc: sideEffectsJSON.Loc},
+				"The value for \"sideEffects\" must be a boolean or an array")
+		}
+	}
+
+	// Read the "imports" map
+	if importsJSON, importsLoc, ok := getProperty(json, "imports"); ok {
+		if importsMap := parseImportsExportsMap(jsonSource, r.log, importsJSON, "imports", importsLoc); importsMap != nil {
+			if importsMap.root.kind != pjObject {
+				r.log.AddID(logger.MsgID_PackageJSON_InvalidImportsOrExports, logger.Warning, &tracker, importsMap.root.firstToken,
+					"The value for \"imports\" must be an object")
+			}
+			packageJSON.importsMap = importsMap
+		}
+	}
+
+	// Read the "exports" map
+	if exportsJSON, exportsLoc, ok := getProperty(json, "exports"); ok {
+		if exportsMap := parseImportsExportsMap(jsonSource, r.log, exportsJSON, "exports", exportsLoc); exportsMap != nil {
+			packageJSON.exportsMap = exportsMap
+		}
+	}
+
+	return packageJSON
+}
+
+// Reference: https://github.com/fitzgen/glob-to-regexp/blob/2abf65a834259c6504ed3b80e85f893f8cd99127/index.js
+func globstarToEscapedRegexp(glob string) (string, bool) {
+	sb := strings.Builder{}
+	sb.WriteByte('^')
+	hadWildcard := false
+	n := len(glob)
+
+	for i := 0; i < n; i++ {
+		c := glob[i]
+		switch c {
+		case '\\', '^', '$', '.', '+', '|', '(', ')', '[', ']', '{', '}':
+			sb.WriteByte('\\')
+			sb.WriteByte(c)
+
+		case '?':
+			sb.WriteByte('.')
+			hadWildcard = true
+
+		case '*':
+			// Move over all consecutive "*"'s.
+			// Also store the previous and next characters
+			prevChar := -1
+			if i > 0 {
+				prevChar = int(glob[i-1])
+			}
+			starCount := 1
+			for i+1 < n && glob[i+1] == '*' {
+				starCount++
+				i++
+			}
+			nextChar := -1
+			if i+1 < n {
+				nextChar = int(glob[i+1])
+			}
+
+			// Determine if this is a globstar segment
+			isGlobstar := starCount > 1 && // multiple "*"'s
+				(prevChar == '/' || prevChar == -1) && // from the start of the segment
+				(nextChar == '/' || nextChar == -1) // to the end of the segment
+
+			if isGlobstar {
+				// It's a globstar, so match zero or more path segments
+				sb.WriteString("(?:[^/]*(?:/|$))*")
+				i++ // Move over the "/"
+			} else {
+				// It's not a globstar, so only match one path segment
+				sb.WriteString("[^/]*")
+			}
+
+			hadWildcard = true
+
+		default:
+			sb.WriteByte(c)
+		}
+	}
+
+	sb.WriteByte('$')
+	return sb.String(), hadWildcard
+}
+
+// Reference: https://nodejs.org/api/esm.html#esm_resolver_algorithm_specification
+type pjMap struct {
+	root           pjEntry
+	propertyKey    string
+	propertyKeyLoc logger.Loc
+}
+
+type pjKind uint8
+
+const (
+	pjNull pjKind = iota
+	pjString
+	pjArray
+	pjObject
+	pjInvalid
+)
+
+type pjEntry struct {
+	strData       string
+	arrData       []pjEntry
+	mapData       []pjMapEntry // Can't be a "map" because order matters
+	expansionKeys expansionKeysArray
+	firstToken    logger.Range
+	kind          pjKind
+}
+
+type pjMapEntry struct {
+	key      string
+	value    pjEntry
+	keyRange logger.Range
+}
+
+// This type is just so we can use Go's native sort function
+type expansionKeysArray []pjMapEntry
+
+func (a expansionKeysArray) Len() int          { return len(a) }
+func (a expansionKeysArray) Swap(i int, j int) { a[i], a[j] = a[j], a[i] }
+
+func (a expansionKeysArray) Less(i int, j int) bool {
+	// Assert: keyA ends with "/" or contains only a single "*".
+	// Assert: keyB ends with "/" or contains only a single "*".
+	keyA := a[i].key
+	keyB := a[j].key
+
+	// Let baseLengthA be the index of "*" in keyA plus one, if keyA contains "*", or the length of keyA otherwise.
+	// Let baseLengthB be the index of "*" in keyB plus one, if keyB contains "*", or the length of keyB otherwise.
+	starA := strings.IndexByte(keyA, '*')
+	starB := strings.IndexByte(keyB, '*')
+	var baseLengthA int
+	var baseLengthB int
+	if starA >= 0 {
+		baseLengthA = starA
+	} else {
+		baseLengthA = len(keyA)
+	}
+	if starB >= 0 {
+		baseLengthB = starB
+	} else {
+		baseLengthB = len(keyB)
+	}
+
+	// If baseLengthA is greater than baseLengthB, return -1.
+	// If baseLengthB is greater than baseLengthA, return 1.
+	if baseLengthA > baseLengthB {
+		return true
+	}
+	if baseLengthB > baseLengthA {
+		return false
+	}
+
+	// If keyA does not contain "*", return 1.
+	// If keyB does not contain "*", return -1.
+	if starA < 0 {
+		return false
+	}
+	if starB < 0 {
+		return true
+	}
+
+	// If the length of keyA is greater than the length of keyB, return -1.
+	// If the length of keyB is greater than the length of keyA, return 1.
+	if len(keyA) > len(keyB) {
+		return true
+	}
+	if len(keyB) > len(keyA) {
+		return false
+	}
+
+	return false
+}
+
+func (entry pjEntry) valueForKey(key string) (pjEntry, bool) {
+	for _, item := range entry.mapData {
+		if item.key == key {
+			return item.value, true
+		}
+	}
+	return pjEntry{}, false
+}
+
+func parseImportsExportsMap(source logger.Source, log logger.Log, json js_ast.Expr, propertyKey string, propertyKeyLoc logger.Loc) *pjMap {
+	var visit func(expr js_ast.Expr) pjEntry
+	tracker := logger.MakeLineColumnTracker(&source)
+
+	visit = func(expr js_ast.Expr) pjEntry {
+		var firstToken logger.Range
+
+		switch e := expr.Data.(type) {
+		case *js_ast.ENull:
+			return pjEntry{
+				kind:       pjNull,
+				firstToken: js_lexer.RangeOfIdentifier(source, expr.Loc),
+			}
+
+		case *js_ast.EString:
+			return pjEntry{
+				kind:       pjString,
+				firstToken: source.RangeOfString(expr.Loc),
+				strData:    helpers.UTF16ToString(e.Value),
+			}
+
+		case *js_ast.EArray:
+			arrData := make([]pjEntry, len(e.Items))
+			for i, item := range e.Items {
+				arrData[i] = visit(item)
+			}
+			return pjEntry{
+				kind:       pjArray,
+				firstToken: logger.Range{Loc: expr.Loc, Len: 1},
+				arrData:    arrData,
+			}
+
+		case *js_ast.EObject:
+			mapData := make([]pjMapEntry, len(e.Properties))
+			expansionKeys := make(expansionKeysArray, 0, len(e.Properties))
+			firstToken := logger.Range{Loc: expr.Loc, Len: 1}
+			isConditionalSugar := false
+
+			type DeadCondition struct {
+				reason string
+				ranges []logger.Range
+				notes  []logger.MsgData
+			}
+			var foundDefault logger.Range
+			var foundImport logger.Range
+			var foundRequire logger.Range
+			var deadCondition DeadCondition
+
+			for i, property := range e.Properties {
+				keyStr, _ := property.Key.Data.(*js_ast.EString)
+				key := helpers.UTF16ToString(keyStr.Value)
+				keyRange := source.RangeOfString(property.Key.Loc)
+
+				// If exports is an Object with both a key starting with "." and a key
+				// not starting with ".", throw an Invalid Package Configuration error.
+				curIsConditionalSugar := !strings.HasPrefix(key, ".")
+				if i == 0 {
+					isConditionalSugar = curIsConditionalSugar
+				} else if isConditionalSugar != curIsConditionalSugar {
+					prevEntry := mapData[i-1]
+					log.AddIDWithNotes(logger.MsgID_PackageJSON_InvalidImportsOrExports, logger.Warning, &tracker, keyRange,
+						"This object cannot contain keys that both start with \".\" and don't start with \".\"",
+						[]logger.MsgData{tracker.MsgData(prevEntry.keyRange,
+							fmt.Sprintf("The key %q is incompatible with the previous key %q:", key, prevEntry.key))})
+					return pjEntry{
+						kind:       pjInvalid,
+						firstToken: firstToken,
+					}
+				}
+
+				// Track "dead" conditional branches that can never be reached
+				if foundDefault.Len != 0 || (foundImport.Len != 0 && foundRequire.Len != 0) {
+					deadCondition.ranges = append(deadCondition.ranges, keyRange)
+					// Note: Don't warn about the "default" condition as it's supposed to be a catch-all condition
+					if deadCondition.reason == "" && key != "default" {
+						if foundDefault.Len != 0 {
+							deadCondition.reason = "\"default\""
+							deadCondition.notes = []logger.MsgData{
+								tracker.MsgData(foundDefault, "The \"default\" condition comes earlier and will always be chosen:"),
+							}
+						} else {
+							deadCondition.reason = "both \"import\" and \"require\""
+							deadCondition.notes = []logger.MsgData{
+								tracker.MsgData(foundImport, "The \"import\" condition comes earlier and will be used for all \"import\" statements:"),
+								tracker.MsgData(foundRequire, "The \"require\" condition comes earlier and will be used for all \"require\" calls:"),
+							}
+						}
+					}
+				} else {
+					switch key {
+					case "default":
+						foundDefault = keyRange
+					case "import":
+						foundImport = keyRange
+					case "require":
+						foundRequire = keyRange
+					}
+				}
+
+				entry := pjMapEntry{
+					key:      key,
+					keyRange: keyRange,
+					value:    visit(property.ValueOrNil),
+				}
+
+				if strings.HasSuffix(key, "/") || strings.IndexByte(key, '*') >= 0 {
+					expansionKeys = append(expansionKeys, entry)
+				}
+
+				mapData[i] = entry
+			}
+
+			// Let expansionKeys be the list of keys of matchObj either ending in "/"
+			// or containing only a single "*", sorted by the sorting function
+			// PATTERN_KEY_COMPARE which orders in descending order of specificity.
+			sort.Stable(expansionKeys)
+
+			// Warn about "dead" conditional branches that can never be reached
+			if deadCondition.reason != "" {
+				kind := logger.Warning
+				if helpers.IsInsideNodeModules(source.KeyPath.Text) {
+					kind = logger.Debug
+				}
+				var conditions string
+				conditionWord := "condition"
+				itComesWord := "it comes"
+				if len(deadCondition.ranges) > 1 {
+					conditionWord = "conditions"
+					itComesWord = "they come"
+				}
+				for i, r := range deadCondition.ranges {
+					if i > 0 {
+						conditions += " and "
+					}
+					conditions += source.TextForRange(r)
+				}
+				log.AddIDWithNotes(logger.MsgID_PackageJSON_DeadCondition, kind, &tracker, deadCondition.ranges[0],
+					fmt.Sprintf("The %s %s here will never be used as %s after %s", conditionWord, conditions, itComesWord, deadCondition.reason),
+					deadCondition.notes)
+			}
+
+			return pjEntry{
+				kind:          pjObject,
+				firstToken:    firstToken,
+				mapData:       mapData,
+				expansionKeys: expansionKeys,
+			}
+
+		case *js_ast.EBoolean:
+			firstToken = js_lexer.RangeOfIdentifier(source, expr.Loc)
+
+		case *js_ast.ENumber:
+			firstToken = source.RangeOfNumber(expr.Loc)
+
+		default:
+			firstToken.Loc = expr.Loc
+		}
+
+		log.AddID(logger.MsgID_PackageJSON_InvalidImportsOrExports, logger.Warning, &tracker, firstToken,
+			"This value must be a string, an object, an array, or null")
+		return pjEntry{
+			kind:       pjInvalid,
+			firstToken: firstToken,
+		}
+	}
+
+	root := visit(json)
+
+	if root.kind == pjNull {
+		return nil
+	}
+
+	return &pjMap{
+		root:           root,
+		propertyKey:    propertyKey,
+		propertyKeyLoc: propertyKeyLoc,
+	}
+}
+
+func (entry pjEntry) keysStartWithDot() bool {
+	return len(entry.mapData) > 0 && strings.HasPrefix(entry.mapData[0].key, ".")
+}
+
+type pjStatus uint8
+
+const (
+	pjStatusUndefined                  pjStatus = iota
+	pjStatusUndefinedNoConditionsMatch          // A more friendly error message for when no conditions are matched
+	pjStatusNull
+	pjStatusExact
+	pjStatusExactEndsWithStar
+	pjStatusInexact        // This means we may need to try CommonJS-style extension suffixes
+	pjStatusPackageResolve // Need to re-run package resolution on the result
+
+	// Module specifier is an invalid URL, package name or package subpath specifier.
+	pjStatusInvalidModuleSpecifier
+
+	// package.json configuration is invalid or contains an invalid configuration.
+	pjStatusInvalidPackageConfiguration
+
+	// Package exports or imports define a target module for the package that is an invalid type or string target.
+	pjStatusInvalidPackageTarget
+
+	// Package exports do not define or permit a target subpath in the package for the given module.
+	pjStatusPackagePathNotExported
+
+	// Package imports do not define the specifiespecifier
+	pjStatusPackageImportNotDefined
+
+	// The package or module requested does not exist.
+	pjStatusModuleNotFound
+	pjStatusModuleNotFoundMissingExtension // The user just needs to add the missing extension
+
+	// The resolved path corresponds to a directory, which is not a supported target for module imports.
+	pjStatusUnsupportedDirectoryImport
+	pjStatusUnsupportedDirectoryImportMissingIndex // The user just needs to add the missing "/index.js" suffix
+)
+
+func (status pjStatus) isUndefined() bool {
+	return status == pjStatusUndefined || status == pjStatusUndefinedNoConditionsMatch
+}
+
+type pjDebug struct {
+	// If the status is "pjStatusInvalidPackageTarget" or "pjStatusInvalidModuleSpecifier",
+	// then this is the reason. It always starts with " because".
+	invalidBecause string
+
+	// If the status is "pjStatusUndefinedNoConditionsMatch", this is the set of
+	// conditions that didn't match, in the order that they were found in the file.
+	// This information is used for error messages.
+	unmatchedConditions []logger.Span
+
+	// This is the range of the token to use for error messages
+	token logger.Range
+
+	// If true, the token is a "null" literal
+	isBecauseOfNullLiteral bool
+}
+
+func (r resolverQuery) esmHandlePostConditions(
+	resolved string,
+	status pjStatus,
+	debug pjDebug,
+) (string, pjStatus, pjDebug) {
+	if status != pjStatusExact && status != pjStatusExactEndsWithStar && status != pjStatusInexact {
+		return resolved, status, debug
+	}
+
+	// If resolved contains any percent encodings of "/" or "\" ("%2f" and "%5C"
+	// respectively), then throw an Invalid Module Specifier error.
+	resolvedPath, err := url.PathUnescape(resolved)
+	if err != nil {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("The path %q contains invalid URL escapes: %s", resolved, err.Error()))
+		}
+		return resolved, pjStatusInvalidModuleSpecifier, debug
+	}
+	var found string
+	if strings.Contains(resolved, "%2f") {
+		found = "%2f"
+	} else if strings.Contains(resolved, "%2F") {
+		found = "%2F"
+	} else if strings.Contains(resolved, "%5c") {
+		found = "%5c"
+	} else if strings.Contains(resolved, "%5C") {
+		found = "%5C"
+	}
+	if found != "" {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("The path %q is not allowed to contain %q", resolved, found))
+		}
+		return resolved, pjStatusInvalidModuleSpecifier, debug
+	}
+
+	// If the file at resolved is a directory, then throw an Unsupported Directory
+	// Import error.
+	if strings.HasSuffix(resolvedPath, "/") || strings.HasSuffix(resolvedPath, "\\") {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("The path %q is not allowed to end with a slash", resolved))
+		}
+		return resolved, pjStatusUnsupportedDirectoryImport, debug
+	}
+
+	// Set resolved to the real path of resolved.
+	return resolvedPath, status, debug
+}
+
+func (r resolverQuery) esmPackageImportsResolve(
+	specifier string,
+	imports pjEntry,
+	conditions map[string]bool,
+) (string, pjStatus, pjDebug) {
+	// ALGORITHM DEVIATION: Provide a friendly error message if "imports" is not an object
+	if imports.kind != pjObject {
+		return "", pjStatusInvalidPackageConfiguration, pjDebug{token: imports.firstToken}
+	}
+
+	resolved, status, debug := r.esmPackageImportsExportsResolve(specifier, imports, "/", true, conditions)
+	if status != pjStatusNull && status != pjStatusUndefined {
+		return resolved, status, debug
+	}
+
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("The package import %q is not defined", specifier))
+	}
+	return specifier, pjStatusPackageImportNotDefined, pjDebug{token: imports.firstToken}
+}
+
+func (r resolverQuery) esmPackageExportsResolve(
+	packageURL string,
+	subpath string,
+	exports pjEntry,
+	conditions map[string]bool,
+) (string, pjStatus, pjDebug) {
+	if exports.kind == pjInvalid {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote("Invalid package configuration")
+		}
+		return "", pjStatusInvalidPackageConfiguration, pjDebug{token: exports.firstToken}
+	}
+
+	debugToReturn := pjDebug{token: exports.firstToken}
+	if subpath == "." {
+		mainExport := pjEntry{kind: pjNull}
+		if exports.kind == pjString || exports.kind == pjArray || (exports.kind == pjObject && !exports.keysStartWithDot()) {
+			mainExport = exports
+		} else if exports.kind == pjObject {
+			if dot, ok := exports.valueForKey("."); ok {
+				if r.debugLogs != nil {
+					r.debugLogs.addNote("Using the entry for \".\"")
+				}
+				mainExport = dot
+			}
+		}
+		if mainExport.kind != pjNull {
+			resolved, status, debug := r.esmPackageTargetResolve(packageURL, mainExport, "", false, false, conditions)
+			if status != pjStatusNull && status != pjStatusUndefined {
+				return resolved, status, debug
+			} else {
+				debugToReturn = debug
+			}
+		}
+	} else if exports.kind == pjObject && exports.keysStartWithDot() {
+		resolved, status, debug := r.esmPackageImportsExportsResolve(subpath, exports, packageURL, false, conditions)
+		if status != pjStatusNull && status != pjStatusUndefined {
+			return resolved, status, debug
+		} else {
+			debugToReturn = debug
+		}
+	}
+
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("The path %q is not exported", subpath))
+	}
+	return "", pjStatusPackagePathNotExported, debugToReturn
+}
+
+func (r resolverQuery) esmPackageImportsExportsResolve(
+	matchKey string,
+	matchObj pjEntry,
+	packageURL string,
+	isImports bool,
+	conditions map[string]bool,
+) (string, pjStatus, pjDebug) {
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Checking object path map for %q", matchKey))
+	}
+
+	// If matchKey is a key of matchObj and does not end in "/" or contain "*", then
+	if !strings.HasSuffix(matchKey, "/") && strings.IndexByte(matchKey, '*') < 0 {
+		if target, ok := matchObj.valueForKey(matchKey); ok {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("Found exact match for %q", matchKey))
+			}
+			return r.esmPackageTargetResolve(packageURL, target, "", false, isImports, conditions)
+		}
+	}
+
+	for _, expansion := range matchObj.expansionKeys {
+		// If expansionKey contains "*", set patternBase to the substring of
+		// expansionKey up to but excluding the first "*" character
+		if star := strings.IndexByte(expansion.key, '*'); star >= 0 {
+			patternBase := expansion.key[:star]
+
+			// If patternBase is not null and matchKey starts with but is not equal
+			// to patternBase, then
+			if strings.HasPrefix(matchKey, patternBase) {
+				// Let patternTrailer be the substring of expansionKey from the index
+				// after the first "*" character.
+				patternTrailer := expansion.key[star+1:]
+
+				// If patternTrailer has zero length, or if matchKey ends with
+				// patternTrailer and the length of matchKey is greater than or
+				// equal to the length of expansionKey, then
+				if patternTrailer == "" || (strings.HasSuffix(matchKey, patternTrailer) && len(matchKey) >= len(expansion.key)) {
+					target := expansion.value
+					subpath := matchKey[len(patternBase) : len(matchKey)-len(patternTrailer)]
+					if r.debugLogs != nil {
+						r.debugLogs.addNote(fmt.Sprintf("The key %q matched with %q left over", expansion.key, subpath))
+					}
+					return r.esmPackageTargetResolve(packageURL, target, subpath, true, isImports, conditions)
+				}
+			}
+		} else {
+			// Otherwise if patternBase is null and matchKey starts with
+			// expansionKey, then
+			if strings.HasPrefix(matchKey, expansion.key) {
+				target := expansion.value
+				subpath := matchKey[len(expansion.key):]
+				if r.debugLogs != nil {
+					r.debugLogs.addNote(fmt.Sprintf("The key %q matched with %q left over", expansion.key, subpath))
+				}
+				result, status, debug := r.esmPackageTargetResolve(packageURL, target, subpath, false, isImports, conditions)
+				if status == pjStatusExact || status == pjStatusExactEndsWithStar {
+					// Return the object { resolved, exact: false }.
+					status = pjStatusInexact
+				}
+				return result, status, debug
+			}
+		}
+
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("The key %q did not match", expansion.key))
+		}
+	}
+
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("No keys matched %q", matchKey))
+	}
+	return "", pjStatusNull, pjDebug{token: matchObj.firstToken}
+}
+
+// If path split on "/" or "\" contains any ".", ".." or "node_modules"
+// segments after the first segment, throw an Invalid Package Target error.
+func findInvalidSegment(path string) string {
+	slash := strings.IndexAny(path, "/\\")
+	if slash == -1 {
+		return ""
+	}
+	path = path[slash+1:]
+	for path != "" {
+		slash := strings.IndexAny(path, "/\\")
+		segment := path
+		if slash != -1 {
+			segment = path[:slash]
+			path = path[slash+1:]
+		} else {
+			path = ""
+		}
+		if segment == "." || segment == ".." || segment == "node_modules" {
+			return segment
+		}
+	}
+	return ""
+}
+
+func (r resolverQuery) esmPackageTargetResolve(
+	packageURL string,
+	target pjEntry,
+	subpath string,
+	pattern bool,
+	internal bool,
+	conditions map[string]bool,
+) (string, pjStatus, pjDebug) {
+	switch target.kind {
+	case pjString:
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("Checking path %q against target %q", subpath, target.strData))
+			r.debugLogs.increaseIndent()
+			defer r.debugLogs.decreaseIndent()
+		}
+
+		// If pattern is false, subpath has non-zero length and target
+		// does not end with "/", throw an Invalid Module Specifier error.
+		if !pattern && subpath != "" && !strings.HasSuffix(target.strData, "/") {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("The target %q is invalid because it doesn't end in \"/\"", target.strData))
+			}
+			return target.strData, pjStatusInvalidModuleSpecifier, pjDebug{
+				token:          target.firstToken,
+				invalidBecause: " because it doesn't end in \"/\"",
+			}
+		}
+
+		// If target does not start with "./", then...
+		if !strings.HasPrefix(target.strData, "./") {
+			if internal && !strings.HasPrefix(target.strData, "../") && !strings.HasPrefix(target.strData, "/") {
+				if pattern {
+					result := strings.ReplaceAll(target.strData, "*", subpath)
+					if r.debugLogs != nil {
+						r.debugLogs.addNote(fmt.Sprintf("Substituted %q for \"*\" in %q to get %q", subpath, target.strData, result))
+					}
+					return result, pjStatusPackageResolve, pjDebug{token: target.firstToken}
+				}
+				result := target.strData + subpath
+				if r.debugLogs != nil {
+					r.debugLogs.addNote(fmt.Sprintf("Joined %q to %q to get %q", target.strData, subpath, result))
+				}
+				return result, pjStatusPackageResolve, pjDebug{token: target.firstToken}
+			}
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("The target %q is invalid because it doesn't start with \"./\"", target.strData))
+			}
+			return target.strData, pjStatusInvalidPackageTarget, pjDebug{
+				token:          target.firstToken,
+				invalidBecause: " because it doesn't start with \"./\"",
+			}
+		}
+
+		// If target split on "/" or "\" contains any ".", ".." or "node_modules"
+		// segments after the first segment, throw an Invalid Package Target error.
+		if invalidSegment := findInvalidSegment(target.strData); invalidSegment != "" {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("The target %q is invalid because it contains invalid segment %q", target.strData, invalidSegment))
+			}
+			return target.strData, pjStatusInvalidPackageTarget, pjDebug{
+				token:          target.firstToken,
+				invalidBecause: fmt.Sprintf(" because it contains invalid segment %q", invalidSegment),
+			}
+		}
+
+		// Let resolvedTarget be the URL resolution of the concatenation of packageURL and target.
+		resolvedTarget := path.Join(packageURL, target.strData)
+
+		// If subpath split on "/" or "\" contains any ".", ".." or "node_modules"
+		// segments, throw an Invalid Module Specifier error.
+		if invalidSegment := findInvalidSegment(subpath); invalidSegment != "" {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("The path %q is invalid because it contains invalid segment %q", subpath, invalidSegment))
+			}
+			return subpath, pjStatusInvalidModuleSpecifier, pjDebug{
+				token:          target.firstToken,
+				invalidBecause: fmt.Sprintf(" because it contains invalid segment %q", invalidSegment),
+			}
+		}
+
+		if pattern {
+			// Return the URL resolution of resolvedTarget with every instance of "*" replaced with subpath.
+			result := strings.ReplaceAll(resolvedTarget, "*", subpath)
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("Substituted %q for \"*\" in %q to get %q", subpath, "."+resolvedTarget, "."+result))
+			}
+			status := pjStatusExact
+			if strings.HasSuffix(resolvedTarget, "*") && strings.IndexByte(resolvedTarget, '*') == len(resolvedTarget)-1 {
+				status = pjStatusExactEndsWithStar
+			}
+			return result, status, pjDebug{token: target.firstToken}
+		} else {
+			// Return the URL resolution of the concatenation of subpath and resolvedTarget.
+			result := path.Join(resolvedTarget, subpath)
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("Joined %q to %q to get %q", subpath, "."+resolvedTarget, "."+result))
+			}
+			return result, pjStatusExact, pjDebug{token: target.firstToken}
+		}
+
+	case pjObject:
+		if r.debugLogs != nil {
+			keys := make([]string, 0, len(conditions))
+			for key := range conditions {
+				keys = append(keys, fmt.Sprintf("%q", key))
+			}
+			sort.Strings(keys)
+			r.debugLogs.addNote(fmt.Sprintf("Checking condition map for one of [%s]", strings.Join(keys, ", ")))
+			r.debugLogs.increaseIndent()
+			defer r.debugLogs.decreaseIndent()
+		}
+
+		var didFindMapEntry bool
+		var lastMapEntry pjMapEntry
+
+		for _, p := range target.mapData {
+			if p.key == "default" || conditions[p.key] {
+				if r.debugLogs != nil {
+					r.debugLogs.addNote(fmt.Sprintf("The key %q applies", p.key))
+				}
+				resolved, status, debug := r.esmPackageTargetResolve(packageURL, p.value, subpath, pattern, internal, conditions)
+				if status.isUndefined() {
+					didFindMapEntry = true
+					lastMapEntry = p
+					continue
+				}
+				return resolved, status, debug
+			}
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("The key %q does not apply", p.key))
+			}
+		}
+
+		if r.debugLogs != nil {
+			r.debugLogs.addNote("No keys in the map were applicable")
+		}
+
+		// ALGORITHM DEVIATION: Provide a friendly error message if no conditions matched
+		if len(target.mapData) > 0 && !target.keysStartWithDot() {
+			if didFindMapEntry && lastMapEntry.value.kind == pjObject &&
+				len(lastMapEntry.value.mapData) > 0 && !lastMapEntry.value.keysStartWithDot() {
+				// If a top-level condition did match but no sub-condition matched,
+				// complain about the sub-condition instead of the top-level condition.
+				// This leads to a less confusing error message. For example:
+				//
+				//   "exports": {
+				//     "node": {
+				//       "require": "./dist/bwip-js-node.js"
+				//     }
+				//   },
+				//
+				// We want the warning to say this:
+				//
+				//   note: None of the conditions in the package definition ("require") match any of the
+				//         currently active conditions ("default", "import", "node")
+				//   14 |       "node": {
+				//      |               ^
+				//
+				// We don't want the warning to say this:
+				//
+				//   note: None of the conditions in the package definition ("browser", "electron", "node")
+				//         match any of the currently active conditions ("default", "import", "node")
+				//   7 |   "exports": {
+				//     |              ^
+				//
+				// More information: https://github.com/evanw/esbuild/issues/1484
+				target = lastMapEntry.value
+			}
+			keys := make([]logger.Span, len(target.mapData))
+			for i, p := range target.mapData {
+				keys[i] = logger.Span{Text: p.key, Range: p.keyRange}
+			}
+			return "", pjStatusUndefinedNoConditionsMatch, pjDebug{
+				token:               target.firstToken,
+				unmatchedConditions: keys,
+			}
+		}
+
+		return "", pjStatusUndefined, pjDebug{token: target.firstToken}
+
+	case pjArray:
+		if len(target.arrData) == 0 {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("The path %q is set to an empty array", subpath))
+			}
+			return "", pjStatusNull, pjDebug{token: target.firstToken}
+		}
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("Checking for %q in an array", subpath))
+			r.debugLogs.increaseIndent()
+			defer r.debugLogs.decreaseIndent()
+		}
+		lastException := pjStatusUndefined
+		lastDebug := pjDebug{token: target.firstToken}
+		for _, targetValue := range target.arrData {
+			// Let resolved be the result, continuing the loop on any Invalid Package Target error.
+			resolved, status, debug := r.esmPackageTargetResolve(packageURL, targetValue, subpath, pattern, internal, conditions)
+			if status == pjStatusInvalidPackageTarget || status == pjStatusNull {
+				lastException = status
+				lastDebug = debug
+				continue
+			}
+			if status.isUndefined() {
+				continue
+			}
+			return resolved, status, debug
+		}
+
+		// Return or throw the last fallback resolution null return or error.
+		return "", lastException, lastDebug
+
+	case pjNull:
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("The path %q is set to null", subpath))
+		}
+		return "", pjStatusNull, pjDebug{token: target.firstToken, isBecauseOfNullLiteral: true}
+	}
+
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Invalid package target for path %q", subpath))
+	}
+	return "", pjStatusInvalidPackageTarget, pjDebug{token: target.firstToken}
+}
+
+func esmParsePackageName(packageSpecifier string) (packageName string, packageSubpath string, ok bool) {
+	if packageSpecifier == "" {
+		return
+	}
+
+	slash := strings.IndexByte(packageSpecifier, '/')
+	if !strings.HasPrefix(packageSpecifier, "@") {
+		if slash == -1 {
+			slash = len(packageSpecifier)
+		}
+		packageName = packageSpecifier[:slash]
+	} else {
+		if slash == -1 {
+			return
+		}
+		slash2 := strings.IndexByte(packageSpecifier[slash+1:], '/')
+		if slash2 == -1 {
+			slash2 = len(packageSpecifier[slash+1:])
+		}
+		packageName = packageSpecifier[:slash+1+slash2]
+	}
+
+	if strings.HasPrefix(packageName, ".") || strings.ContainsAny(packageName, "\\%") {
+		return
+	}
+
+	packageSubpath = "." + packageSpecifier[len(packageName):]
+	ok = true
+	return
+}
+
+func (r resolverQuery) esmPackageExportsReverseResolve(
+	query string,
+	root pjEntry,
+	conditions map[string]bool,
+) (bool, string, logger.Range) {
+	if root.kind == pjObject && root.keysStartWithDot() {
+		if ok, subpath, token := r.esmPackageImportsExportsReverseResolve(query, root, conditions); ok {
+			return true, subpath, token
+		}
+	}
+
+	return false, "", logger.Range{}
+}
+
+func (r resolverQuery) esmPackageImportsExportsReverseResolve(
+	query string,
+	matchObj pjEntry,
+	conditions map[string]bool,
+) (bool, string, logger.Range) {
+	if !strings.HasSuffix(query, "*") {
+		for _, entry := range matchObj.mapData {
+			if ok, subpath, token := r.esmPackageTargetReverseResolve(query, entry.key, entry.value, esmReverseExact, conditions); ok {
+				return true, subpath, token
+			}
+		}
+	}
+
+	for _, expansion := range matchObj.expansionKeys {
+		if strings.HasSuffix(expansion.key, "*") {
+			if ok, subpath, token := r.esmPackageTargetReverseResolve(query, expansion.key, expansion.value, esmReversePattern, conditions); ok {
+				return true, subpath, token
+			}
+		}
+
+		if ok, subpath, token := r.esmPackageTargetReverseResolve(query, expansion.key, expansion.value, esmReversePrefix, conditions); ok {
+			return true, subpath, token
+		}
+	}
+
+	return false, "", logger.Range{}
+}
+
+type esmReverseKind uint8
+
+const (
+	esmReverseExact esmReverseKind = iota
+	esmReversePattern
+	esmReversePrefix
+)
+
+func (r resolverQuery) esmPackageTargetReverseResolve(
+	query string,
+	key string,
+	target pjEntry,
+	kind esmReverseKind,
+	conditions map[string]bool,
+) (bool, string, logger.Range) {
+	switch target.kind {
+	case pjString:
+		switch kind {
+		case esmReverseExact:
+			if query == target.strData {
+				return true, key, target.firstToken
+			}
+
+		case esmReversePrefix:
+			if strings.HasPrefix(query, target.strData) {
+				return true, key + query[len(target.strData):], target.firstToken
+			}
+
+		case esmReversePattern:
+			star := strings.IndexByte(target.strData, '*')
+			keyWithoutTrailingStar := strings.TrimSuffix(key, "*")
+
+			// Handle the case of no "*"
+			if star == -1 {
+				if query == target.strData {
+					return true, keyWithoutTrailingStar, target.firstToken
+				}
+				break
+			}
+
+			// Only support tracing through a single "*"
+			prefix := target.strData[0:star]
+			suffix := target.strData[star+1:]
+			if !strings.ContainsRune(suffix, '*') && strings.HasPrefix(query, prefix) {
+				if afterPrefix := query[len(prefix):]; strings.HasSuffix(afterPrefix, suffix) {
+					starData := afterPrefix[:len(afterPrefix)-len(suffix)]
+					return true, keyWithoutTrailingStar + starData, target.firstToken
+				}
+			}
+		}
+
+	case pjObject:
+		for _, p := range target.mapData {
+			if p.key == "default" || conditions[p.key] {
+				if ok, subpath, token := r.esmPackageTargetReverseResolve(query, key, p.value, kind, conditions); ok {
+					return true, subpath, token
+				}
+			}
+		}
+
+	case pjArray:
+		for _, targetValue := range target.arrData {
+			if ok, subpath, token := r.esmPackageTargetReverseResolve(query, key, targetValue, kind, conditions); ok {
+				return true, subpath, token
+			}
+		}
+	}
+
+	return false, "", logger.Range{}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/resolver/resolver.go b/source/vendor/github.com/evanw/esbuild/internal/resolver/resolver.go
new file mode 100644
index 0000000..b3f6c8b
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/resolver/resolver.go
@@ -0,0 +1,2923 @@
+package resolver
+
+import (
+	"errors"
+	"fmt"
+	"path"
+	"regexp"
+	"sort"
+	"strings"
+	"sync"
+	"syscall"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/cache"
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/config"
+	"github.com/evanw/esbuild/internal/fs"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+var defaultMainFields = map[config.Platform][]string{
+	// Note that this means if a package specifies "main", "module", and
+	// "browser" then "browser" will win out over "module". This is the
+	// same behavior as webpack: https://github.com/webpack/webpack/issues/4674.
+	//
+	// This is deliberate because the presence of the "browser" field is a
+	// good signal that the "module" field may have non-browser stuff in it,
+	// which will crash or fail to be bundled when targeting the browser.
+	config.PlatformBrowser: {"browser", "module", "main"},
+
+	// Note that this means if a package specifies "module" and "main", the ES6
+	// module will not be selected. This means tree shaking will not work when
+	// targeting node environments.
+	//
+	// This is unfortunately necessary for compatibility. Some packages
+	// incorrectly treat the "module" field as "code for the browser". It
+	// actually means "code for ES6 environments" which includes both node
+	// and the browser.
+	//
+	// For example, the package "@firebase/app" prints a warning on startup about
+	// the bundler incorrectly using code meant for the browser if the bundler
+	// selects the "module" field instead of the "main" field.
+	//
+	// If you want to enable tree shaking when targeting node, you will have to
+	// configure the main fields to be "module" and then "main". Keep in mind
+	// that some packages may break if you do this.
+	config.PlatformNode: {"main", "module"},
+
+	// The neutral platform is for people that don't want esbuild to try to
+	// pick good defaults for their platform. In that case, the list of main
+	// fields is empty by default. You must explicitly configure it yourself.
+	config.PlatformNeutral: {},
+}
+
+// These are the main fields to use when the "main fields" setting is configured
+// to something unusual, such as something without the "main" field.
+var mainFieldsForFailure = []string{"main", "module"}
+
+// Path resolution is a mess. One tricky issue is the "module" override for the
+// "main" field in "package.json" files. Bundlers generally prefer "module" over
+// "main" but that breaks packages that export a function in "main" for use with
+// "require()", since resolving to "module" means an object will be returned. We
+// attempt to handle this automatically by having import statements resolve to
+// "module" but switch that out later for "main" if "require()" is used too.
+type PathPair struct {
+	// Either secondary will be empty, or primary will be "module" and secondary
+	// will be "main"
+	Primary   logger.Path
+	Secondary logger.Path
+
+	IsExternal bool
+}
+
+func (pp *PathPair) iter() []*logger.Path {
+	result := []*logger.Path{&pp.Primary, &pp.Secondary}
+	if !pp.HasSecondary() {
+		result = result[:1]
+	}
+	return result
+}
+
+func (pp *PathPair) HasSecondary() bool {
+	return pp.Secondary.Text != ""
+}
+
+type SideEffectsData struct {
+	Source *logger.Source
+
+	// If non-empty, this false value came from a plugin
+	PluginName string
+
+	Range logger.Range
+
+	// If true, "sideEffects" was an array. If false, "sideEffects" was false.
+	IsSideEffectsArrayInJSON bool
+}
+
+type ResolveResult struct {
+	PathPair PathPair
+
+	// If this was resolved by a plugin, the plugin gets to store its data here
+	PluginData interface{}
+
+	DifferentCase *fs.DifferentCase
+
+	// If present, any ES6 imports to this file can be considered to have no side
+	// effects. This means they should be removed if unused.
+	PrimarySideEffectsData *SideEffectsData
+
+	// These are from "tsconfig.json"
+	TSConfigJSX    config.TSConfigJSX
+	TSConfig       *config.TSConfig
+	TSAlwaysStrict *config.TSAlwaysStrict
+
+	// This is the "type" field from "package.json"
+	ModuleTypeData js_ast.ModuleTypeData
+}
+
+type suggestionRange uint8
+
+const (
+	suggestionRangeFull suggestionRange = iota
+	suggestionRangeEnd
+)
+
+type DebugMeta struct {
+	notes              []logger.MsgData
+	suggestionText     string
+	suggestionMessage  string
+	suggestionRange    suggestionRange
+	ModifiedImportPath string
+}
+
+func (dm DebugMeta) LogErrorMsg(log logger.Log, source *logger.Source, r logger.Range, text string, suggestion string, notes []logger.MsgData) {
+	tracker := logger.MakeLineColumnTracker(source)
+
+	if source != nil && dm.suggestionMessage != "" {
+		suggestionRange := r
+		if dm.suggestionRange == suggestionRangeEnd {
+			suggestionRange = logger.Range{Loc: logger.Loc{Start: r.End() - 1}}
+		}
+		data := tracker.MsgData(suggestionRange, dm.suggestionMessage)
+		data.Location.Suggestion = dm.suggestionText
+		dm.notes = append(dm.notes, data)
+	}
+
+	msg := logger.Msg{
+		Kind:  logger.Error,
+		Data:  tracker.MsgData(r, text),
+		Notes: append(dm.notes, notes...),
+	}
+
+	if msg.Data.Location != nil && suggestion != "" {
+		msg.Data.Location.Suggestion = suggestion
+	}
+
+	log.AddMsg(msg)
+}
+
+type Resolver struct {
+	fs     fs.FS
+	log    logger.Log
+	caches *cache.CacheSet
+
+	tsConfigOverride *TSConfigJSON
+
+	// These are sets that represent various conditions for the "exports" field
+	// in package.json.
+	esmConditionsDefault map[string]bool
+	esmConditionsImport  map[string]bool
+	esmConditionsRequire map[string]bool
+
+	// A special filtered import order for CSS "@import" imports.
+	//
+	// The "resolve extensions" setting determines the order of implicit
+	// extensions to try when resolving imports with the extension omitted.
+	// Sometimes people create a JavaScript/TypeScript file and a CSS file with
+	// the same name when they create a component. At a high level, users expect
+	// implicit extensions to resolve to the JS file when being imported from JS
+	// and to resolve to the CSS file when being imported from CSS.
+	//
+	// Different bundlers handle this in different ways. Parcel handles this by
+	// having the resolver prefer the same extension as the importing file in
+	// front of the configured "resolve extensions" order. Webpack's "css-loader"
+	// plugin just explicitly configures a special "resolve extensions" order
+	// consisting of only ".css" for CSS files.
+	//
+	// It's unclear what behavior is best here. What we currently do is to create
+	// a special filtered version of the configured "resolve extensions" order
+	// for CSS files that filters out any extension that has been explicitly
+	// configured with a non-CSS loader. This still gives users control over the
+	// order but avoids the scenario where we match an import in a CSS file to a
+	// JavaScript-related file. It's probably not perfect with plugins in the
+	// picture but it's better than some alternatives and probably pretty good.
+	cssExtensionOrder []string
+
+	// A special sorted import order for imports inside packages.
+	//
+	// The "resolve extensions" setting determines the order of implicit
+	// extensions to try when resolving imports with the extension omitted.
+	// Sometimes people author a package using TypeScript and publish both the
+	// compiled JavaScript and the original TypeScript. The compiled JavaScript
+	// depends on the "tsconfig.json" settings that were passed to "tsc" when
+	// it was compiled, and we don't know what they are (they may even be
+	// unknowable if the "tsconfig.json" file wasn't published).
+	//
+	// To work around this, we sort TypeScript file extensions after JavaScript
+	// file extensions (but only within packages) so that esbuild doesn't load
+	// the original source code in these scenarios. Instead we should load the
+	// compiled code, which is what will be loaded by node at run-time.
+	nodeModulesExtensionOrder []string
+
+	// This cache maps a directory path to information about that directory and
+	// all parent directories
+	dirCache map[string]*dirInfo
+
+	pnpManifestWasChecked bool
+	pnpManifest           *pnpData
+
+	options config.Options
+
+	// This mutex serves two purposes. First of all, it guards access to "dirCache"
+	// which is potentially mutated during path resolution. But this mutex is also
+	// necessary for performance. The "React admin" benchmark mysteriously runs
+	// twice as fast when this mutex is locked around the whole resolve operation
+	// instead of around individual accesses to "dirCache". For some reason,
+	// reducing parallelism in the resolver helps the rest of the bundler go
+	// faster. I'm not sure why this is but please don't change this unless you
+	// do a lot of testing with various benchmarks and there aren't any regressions.
+	mutex sync.Mutex
+}
+
+type resolverQuery struct {
+	*Resolver
+	debugMeta *DebugMeta
+	debugLogs *debugLogs
+	kind      ast.ImportKind
+}
+
+func NewResolver(call config.APICall, fs fs.FS, log logger.Log, caches *cache.CacheSet, options *config.Options) *Resolver {
+	// Filter out non-CSS extensions for CSS "@import" imports
+	cssExtensionOrder := make([]string, 0, len(options.ExtensionOrder))
+	for _, ext := range options.ExtensionOrder {
+		if loader, ok := options.ExtensionToLoader[ext]; !ok || loader.IsCSS() {
+			cssExtensionOrder = append(cssExtensionOrder, ext)
+		}
+	}
+
+	// Sort all TypeScript file extensions after all JavaScript file extensions
+	// for imports of files inside of "node_modules" directories. But insert
+	// the TypeScript file extensions right after the last JavaScript file
+	// extension instead of at the end so that they might come before the
+	// first CSS file extension, which is important to people that publish
+	// TypeScript and CSS code to npm with the same file names for both.
+	nodeModulesExtensionOrder := make([]string, 0, len(options.ExtensionOrder))
+	split := 0
+	for i, ext := range options.ExtensionOrder {
+		if loader, ok := options.ExtensionToLoader[ext]; ok && loader == config.LoaderJS || loader == config.LoaderJSX {
+			split = i + 1 // Split after the last JavaScript extension
+		}
+	}
+	if split != 0 { // Only do this if there are any JavaScript extensions
+		for _, ext := range options.ExtensionOrder[:split] { // Non-TypeScript extensions before the split
+			if loader, ok := options.ExtensionToLoader[ext]; !ok || !loader.IsTypeScript() {
+				nodeModulesExtensionOrder = append(nodeModulesExtensionOrder, ext)
+			}
+		}
+		for _, ext := range options.ExtensionOrder { // All TypeScript extensions
+			if loader, ok := options.ExtensionToLoader[ext]; ok && loader.IsTypeScript() {
+				nodeModulesExtensionOrder = append(nodeModulesExtensionOrder, ext)
+			}
+		}
+		for _, ext := range options.ExtensionOrder[split:] { // Non-TypeScript extensions after the split
+			if loader, ok := options.ExtensionToLoader[ext]; !ok || !loader.IsTypeScript() {
+				nodeModulesExtensionOrder = append(nodeModulesExtensionOrder, ext)
+			}
+		}
+	}
+
+	// Generate the condition sets for interpreting the "exports" field
+	esmConditionsDefault := map[string]bool{"default": true}
+	esmConditionsImport := map[string]bool{"import": true}
+	esmConditionsRequire := map[string]bool{"require": true}
+	for _, condition := range options.Conditions {
+		esmConditionsDefault[condition] = true
+	}
+	switch options.Platform {
+	case config.PlatformBrowser:
+		esmConditionsDefault["browser"] = true
+	case config.PlatformNode:
+		esmConditionsDefault["node"] = true
+	}
+	for key := range esmConditionsDefault {
+		esmConditionsImport[key] = true
+		esmConditionsRequire[key] = true
+	}
+
+	fs.Cwd()
+
+	res := &Resolver{
+		fs:                        fs,
+		log:                       log,
+		options:                   *options,
+		caches:                    caches,
+		dirCache:                  make(map[string]*dirInfo),
+		cssExtensionOrder:         cssExtensionOrder,
+		nodeModulesExtensionOrder: nodeModulesExtensionOrder,
+		esmConditionsDefault:      esmConditionsDefault,
+		esmConditionsImport:       esmConditionsImport,
+		esmConditionsRequire:      esmConditionsRequire,
+	}
+
+	// Handle the "tsconfig.json" override when the resolver is created. This
+	// isn't done when we validate the build options both because the code for
+	// "tsconfig.json" handling is already in the resolver, and because we want
+	// watch mode to pick up changes to "tsconfig.json" and rebuild.
+	var debugMeta DebugMeta
+	if options.TSConfigPath != "" || options.TSConfigRaw != "" {
+		r := resolverQuery{
+			Resolver:  res,
+			debugMeta: &debugMeta,
+		}
+		var visited map[string]bool
+		var err error
+		if call == config.BuildCall {
+			visited = make(map[string]bool)
+		}
+		if options.TSConfigPath != "" {
+			if r.log.Level <= logger.LevelDebug {
+				r.debugLogs = &debugLogs{what: fmt.Sprintf("Resolving tsconfig file %q", options.TSConfigPath)}
+			}
+			res.tsConfigOverride, err = r.parseTSConfig(options.TSConfigPath, visited, fs.Dir(options.TSConfigPath))
+		} else {
+			source := logger.Source{
+				KeyPath:    logger.Path{Text: fs.Join(fs.Cwd(), "<tsconfig.json>"), Namespace: "file"},
+				PrettyPath: "<tsconfig.json>",
+				Contents:   options.TSConfigRaw,
+			}
+			res.tsConfigOverride, err = r.parseTSConfigFromSource(source, visited, fs.Cwd())
+		}
+		if err != nil {
+			if err == syscall.ENOENT {
+				r.log.AddError(nil, logger.Range{}, fmt.Sprintf("Cannot find tsconfig file %q",
+					PrettyPath(r.fs, logger.Path{Text: options.TSConfigPath, Namespace: "file"})))
+			} else if err != errParseErrorAlreadyLogged {
+				r.log.AddError(nil, logger.Range{}, fmt.Sprintf("Cannot read file %q: %s",
+					PrettyPath(r.fs, logger.Path{Text: options.TSConfigPath, Namespace: "file"}), err.Error()))
+			}
+		} else {
+			r.flushDebugLogs(flushDueToSuccess)
+		}
+	}
+
+	// Mutate the provided options by settings from "tsconfig.json" if present
+	if res.tsConfigOverride != nil {
+		options.TS.Config = res.tsConfigOverride.Settings
+		res.tsConfigOverride.JSXSettings.ApplyTo(&options.JSX)
+		options.TSAlwaysStrict = res.tsConfigOverride.TSAlwaysStrictOrStrict()
+	}
+
+	return res
+}
+
+func (res *Resolver) Resolve(sourceDir string, importPath string, kind ast.ImportKind) (*ResolveResult, DebugMeta) {
+	var debugMeta DebugMeta
+	r := resolverQuery{
+		Resolver:  res,
+		debugMeta: &debugMeta,
+		kind:      kind,
+	}
+	if r.log.Level <= logger.LevelDebug {
+		r.debugLogs = &debugLogs{what: fmt.Sprintf(
+			"Resolving import %q in directory %q of type %q",
+			importPath, sourceDir, kind.StringForMetafile())}
+	}
+
+	// Apply package alias substitutions first
+	if r.options.PackageAliases != nil && IsPackagePath(importPath) {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote("Checking for package alias matches")
+		}
+		longestKey := ""
+		longestValue := ""
+
+		for key, value := range r.options.PackageAliases {
+			if len(key) > len(longestKey) && strings.HasPrefix(importPath, key) && (len(importPath) == len(key) || importPath[len(key)] == '/') {
+				longestKey = key
+				longestValue = value
+			}
+		}
+
+		if longestKey != "" {
+			debugMeta.ModifiedImportPath = longestValue
+			if tail := importPath[len(longestKey):]; tail != "/" {
+				// Don't include the trailing characters if they are equal to a
+				// single slash. This comes up because you can abuse this quirk of
+				// node's path resolution to force node to load the package from the
+				// file system instead of as a built-in module. For example, "util"
+				// is node's built-in module while "util/" is one on the file system.
+				// Leaving the trailing slash in place causes problems for people:
+				// https://github.com/evanw/esbuild/issues/2730. It should be ok to
+				// always strip the trailing slash even when using the alias feature
+				// to swap one package for another (except when you swap a reference
+				// to one built-in node module with another but really why would you
+				// do that).
+				debugMeta.ModifiedImportPath += tail
+			}
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("  Matched with alias from %q to %q", longestKey, longestValue))
+				r.debugLogs.addNote(fmt.Sprintf("  Modified import path from %q to %q", importPath, debugMeta.ModifiedImportPath))
+			}
+			importPath = debugMeta.ModifiedImportPath
+
+			// Resolve the package using the current path instead of the original
+			// path. This is trying to resolve the substitute in the top-level
+			// package instead of the nested package, which lets the top-level
+			// package control the version of the substitution. It's also critical
+			// when using Yarn PnP because Yarn PnP doesn't allow nested packages
+			// to "reach outside" of their normal dependency lists.
+			sourceDir = r.fs.Cwd()
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("  Changed resolve directory to %q", sourceDir))
+			}
+		} else if r.debugLogs != nil {
+			r.debugLogs.addNote("  Failed to find any package alias matches")
+		}
+	}
+
+	// Certain types of URLs default to being external for convenience
+	if isExplicitlyExternal := r.isExternal(r.options.ExternalSettings.PreResolve, importPath, kind); isExplicitlyExternal ||
+
+		// "fill: url(#filter);"
+		(kind == ast.ImportURL && strings.HasPrefix(importPath, "#")) ||
+
+		// "background: url(http://example.com/images/image.png);"
+		strings.HasPrefix(importPath, "http://") ||
+
+		// "background: url(https://example.com/images/image.png);"
+		strings.HasPrefix(importPath, "https://") ||
+
+		// "background: url(//example.com/images/image.png);"
+		strings.HasPrefix(importPath, "//") {
+
+		if r.debugLogs != nil {
+			if isExplicitlyExternal {
+				r.debugLogs.addNote(fmt.Sprintf("The path %q was marked as external by the user", importPath))
+			} else {
+				r.debugLogs.addNote("Marking this path as implicitly external")
+			}
+		}
+
+		r.flushDebugLogs(flushDueToSuccess)
+		return &ResolveResult{
+			PathPair: PathPair{Primary: logger.Path{Text: importPath}, IsExternal: true},
+		}, debugMeta
+	}
+
+	if pathPair, ok, sideEffects := r.checkForBuiltInNodeModules(importPath); ok {
+		r.flushDebugLogs(flushDueToSuccess)
+		return &ResolveResult{
+			PathPair:               pathPair,
+			PrimarySideEffectsData: sideEffects,
+		}, debugMeta
+	}
+
+	if parsed, ok := ParseDataURL(importPath); ok {
+		// "import 'data:text/javascript,console.log(123)';"
+		// "@import 'data:text/css,body{background:white}';"
+		if parsed.DecodeMIMEType() != MIMETypeUnsupported {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote("Putting this path in the \"dataurl\" namespace")
+			}
+			r.flushDebugLogs(flushDueToSuccess)
+			return &ResolveResult{
+				PathPair: PathPair{Primary: logger.Path{Text: importPath, Namespace: "dataurl"}},
+			}, debugMeta
+		}
+
+		// "background: url();"
+		if r.debugLogs != nil {
+			r.debugLogs.addNote("Marking this data URL as external")
+		}
+		r.flushDebugLogs(flushDueToSuccess)
+		return &ResolveResult{
+			PathPair: PathPair{Primary: logger.Path{Text: importPath}, IsExternal: true},
+		}, debugMeta
+	}
+
+	// Fail now if there is no directory to resolve in. This can happen for
+	// virtual modules (e.g. stdin) if a resolve directory is not specified.
+	if sourceDir == "" {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote("Cannot resolve this path without a directory")
+		}
+		r.flushDebugLogs(flushDueToFailure)
+		return nil, debugMeta
+	}
+
+	// Glob imports only work in a multi-path context
+	if strings.ContainsRune(importPath, '*') {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote("Cannot resolve a path containing a wildcard character in a single-path context")
+		}
+		r.flushDebugLogs(flushDueToFailure)
+		return nil, debugMeta
+	}
+
+	r.mutex.Lock()
+	defer r.mutex.Unlock()
+
+	// Check for the Yarn PnP manifest if it hasn't already been checked for
+	if !r.pnpManifestWasChecked {
+		r.pnpManifestWasChecked = true
+
+		// Use the current working directory to find the Yarn PnP manifest. We
+		// can't necessarily use the entry point locations because the entry
+		// point locations aren't necessarily file paths. For example, they could
+		// be HTTP URLs that will be handled by a plugin.
+		for dirInfo := r.dirInfoCached(r.fs.Cwd()); dirInfo != nil; dirInfo = dirInfo.parent {
+			if absPath := dirInfo.pnpManifestAbsPath; absPath != "" {
+				if strings.HasSuffix(absPath, ".json") {
+					if json, source := r.extractYarnPnPDataFromJSON(absPath, pnpReportErrorsAboutMissingFiles); json.Data != nil {
+						r.pnpManifest = compileYarnPnPData(absPath, r.fs.Dir(absPath), json, source)
+					}
+				} else {
+					if json, source := r.tryToExtractYarnPnPDataFromJS(absPath, pnpReportErrorsAboutMissingFiles); json.Data != nil {
+						r.pnpManifest = compileYarnPnPData(absPath, r.fs.Dir(absPath), json, source)
+					}
+				}
+				if r.debugLogs != nil && r.pnpManifest != nil && r.pnpManifest.invalidIgnorePatternData != "" {
+					r.debugLogs.addNote("  Invalid Go regular expression for \"ignorePatternData\": " + r.pnpManifest.invalidIgnorePatternData)
+				}
+				break
+			}
+		}
+	}
+
+	sourceDirInfo := r.dirInfoCached(sourceDir)
+	if sourceDirInfo == nil {
+		// Bail if the directory is missing for some reason
+		return nil, debugMeta
+	}
+
+	result := r.resolveWithoutSymlinks(sourceDir, sourceDirInfo, importPath)
+	if result == nil {
+		// If resolution failed, try again with the URL query and/or hash removed
+		suffix := strings.IndexAny(importPath, "?#")
+		if suffix < 1 {
+			r.flushDebugLogs(flushDueToFailure)
+			return nil, debugMeta
+		}
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("Retrying resolution after removing the suffix %q", importPath[suffix:]))
+		}
+		if result2 := r.resolveWithoutSymlinks(sourceDir, sourceDirInfo, importPath[:suffix]); result2 == nil {
+			r.flushDebugLogs(flushDueToFailure)
+			return nil, debugMeta
+		} else {
+			result = result2
+			result.PathPair.Primary.IgnoredSuffix = importPath[suffix:]
+			if result.PathPair.HasSecondary() {
+				result.PathPair.Secondary.IgnoredSuffix = importPath[suffix:]
+			}
+		}
+	}
+
+	// If successful, resolve symlinks using the directory info cache
+	r.finalizeResolve(result)
+	r.flushDebugLogs(flushDueToSuccess)
+	return result, debugMeta
+}
+
+// This returns nil on failure and non-nil on success. Note that this may
+// return an empty array to indicate a successful search that returned zero
+// results.
+func (res *Resolver) ResolveGlob(sourceDir string, importPathPattern []helpers.GlobPart, kind ast.ImportKind, prettyPattern string) (map[string]ResolveResult, *logger.Msg) {
+	var debugMeta DebugMeta
+	r := resolverQuery{
+		Resolver:  res,
+		debugMeta: &debugMeta,
+		kind:      kind,
+	}
+
+	if r.log.Level <= logger.LevelDebug {
+		r.debugLogs = &debugLogs{what: fmt.Sprintf(
+			"Resolving glob import %s in directory %q of type %q",
+			prettyPattern, sourceDir, kind.StringForMetafile())}
+	}
+
+	if len(importPathPattern) == 0 {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote("Ignoring empty glob pattern")
+		}
+		r.flushDebugLogs(flushDueToFailure)
+		return nil, nil
+	}
+	firstPrefix := importPathPattern[0].Prefix
+
+	// Glob patterns only work for relative URLs
+	if !strings.HasPrefix(firstPrefix, "./") && !strings.HasPrefix(firstPrefix, "../") &&
+		!strings.HasPrefix(firstPrefix, ".\\") && !strings.HasPrefix(firstPrefix, "..\\") {
+		if kind == ast.ImportEntryPoint {
+			// Be permissive about forgetting "./" for entry points since it's common
+			// to omit "./" on the command line. But don't accidentally treat absolute
+			// paths as relative (even on Windows).
+			if !r.fs.IsAbs(firstPrefix) {
+				firstPrefix = "./" + firstPrefix
+			}
+		} else {
+			// Don't allow omitting "./" for other imports since node doesn't let you do this either
+			if r.debugLogs != nil {
+				r.debugLogs.addNote("Ignoring glob import that doesn't start with \"./\" or \"../\"")
+			}
+			r.flushDebugLogs(flushDueToFailure)
+			return nil, nil
+		}
+	}
+
+	// Handle leading directories in the pattern (including "../")
+	dirPrefix := 0
+	for {
+		slash := strings.IndexAny(firstPrefix[dirPrefix:], "/\\")
+		if slash == -1 {
+			break
+		}
+		if star := strings.IndexByte(firstPrefix[dirPrefix:], '*'); star != -1 && slash > star {
+			break
+		}
+		dirPrefix += slash + 1
+	}
+
+	// If the pattern is an absolute path, then just replace source directory.
+	// Otherwise join the source directory with the prefix from the pattern.
+	if suffix := firstPrefix[:dirPrefix]; r.fs.IsAbs(suffix) {
+		sourceDir = suffix
+	} else {
+		sourceDir = r.fs.Join(sourceDir, suffix)
+	}
+
+	r.mutex.Lock()
+	defer r.mutex.Unlock()
+
+	// Look up the directory to start from
+	sourceDirInfo := r.dirInfoCached(sourceDir)
+	if sourceDirInfo == nil {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("Failed to find the directory %q", sourceDir))
+		}
+		r.flushDebugLogs(flushDueToFailure)
+		return nil, nil
+	}
+
+	// Turn the glob pattern into a regular expression
+	canMatchOnSlash := false
+	wasGlobStar := false
+	sb := strings.Builder{}
+	sb.WriteByte('^')
+	for i, part := range importPathPattern {
+		prefix := part.Prefix
+		if i == 0 {
+			prefix = firstPrefix
+		}
+		if wasGlobStar && len(prefix) > 0 && (prefix[0] == '/' || prefix[0] == '\\') {
+			prefix = prefix[1:] // Move over the "/" after a globstar
+		}
+		sb.WriteString(regexp.QuoteMeta(prefix))
+		switch part.Wildcard {
+		case helpers.GlobAllIncludingSlash:
+			// It's a globstar, so match zero or more path segments
+			sb.WriteString("(?:[^/]*(?:/|$))*")
+			canMatchOnSlash = true
+			wasGlobStar = true
+		case helpers.GlobAllExceptSlash:
+			// It's not a globstar, so only match one path segment
+			sb.WriteString("[^/]*")
+			wasGlobStar = false
+		}
+	}
+	sb.WriteByte('$')
+	re := regexp.MustCompile(sb.String())
+
+	// Initialize "results" to a non-nil value to indicate that the glob is valid
+	results := make(map[string]ResolveResult)
+
+	var visit func(dirInfo *dirInfo, dir string)
+	visit = func(dirInfo *dirInfo, dir string) {
+		for _, key := range dirInfo.entries.SortedKeys() {
+			entry, _ := dirInfo.entries.Get(key)
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("Considering entry %q", r.fs.Join(dirInfo.absPath, key)))
+				r.debugLogs.increaseIndent()
+			}
+
+			switch entry.Kind(r.fs) {
+			case fs.DirEntry:
+				// To avoid infinite loops, don't follow any symlinks
+				if canMatchOnSlash && entry.Symlink(r.fs) == "" {
+					if childDirInfo := r.dirInfoCached(r.fs.Join(dirInfo.absPath, key)); childDirInfo != nil {
+						visit(childDirInfo, fmt.Sprintf("%s%s/", dir, key))
+					}
+				}
+
+			case fs.FileEntry:
+				if relPath := dir + key; re.MatchString(relPath) {
+					var result ResolveResult
+
+					if r.isExternal(r.options.ExternalSettings.PreResolve, relPath, kind) {
+						result.PathPair = PathPair{Primary: logger.Path{Text: relPath}, IsExternal: true}
+
+						if r.debugLogs != nil {
+							r.debugLogs.addNote(fmt.Sprintf("The path %q was marked as external by the user", result.PathPair.Primary.Text))
+						}
+					} else {
+						absPath := r.fs.Join(dirInfo.absPath, key)
+						result.PathPair = PathPair{Primary: logger.Path{Text: absPath, Namespace: "file"}}
+					}
+
+					r.finalizeResolve(&result)
+					results[relPath] = result
+				}
+			}
+
+			if r.debugLogs != nil {
+				r.debugLogs.decreaseIndent()
+			}
+		}
+	}
+
+	visit(sourceDirInfo, firstPrefix[:dirPrefix])
+
+	var warning *logger.Msg
+	if len(results) == 0 {
+		warning = &logger.Msg{
+			ID:   logger.MsgID_Bundler_EmptyGlob,
+			Kind: logger.Warning,
+			Data: logger.MsgData{Text: fmt.Sprintf("The glob pattern %s did not match any files", prettyPattern)},
+		}
+	}
+
+	r.flushDebugLogs(flushDueToSuccess)
+	return results, warning
+}
+
+func (r resolverQuery) isExternal(matchers config.ExternalMatchers, path string, kind ast.ImportKind) bool {
+	if kind == ast.ImportEntryPoint {
+		// Never mark an entry point as external. This is not useful.
+		return false
+	}
+	if _, ok := matchers.Exact[path]; ok {
+		return true
+	}
+	for _, pattern := range matchers.Patterns {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("Checking %q against the external pattern %q", path, pattern.Prefix+"*"+pattern.Suffix))
+		}
+		if len(path) >= len(pattern.Prefix)+len(pattern.Suffix) &&
+			strings.HasPrefix(path, pattern.Prefix) &&
+			strings.HasSuffix(path, pattern.Suffix) {
+			return true
+		}
+	}
+	return false
+}
+
+// This tries to run "Resolve" on a package path as a relative path. If
+// successful, the user just forgot a leading "./" in front of the path.
+func (res *Resolver) ProbeResolvePackageAsRelative(sourceDir string, importPath string, kind ast.ImportKind) (*ResolveResult, DebugMeta) {
+	var debugMeta DebugMeta
+	r := resolverQuery{
+		Resolver:  res,
+		debugMeta: &debugMeta,
+		kind:      kind,
+	}
+	absPath := r.fs.Join(sourceDir, importPath)
+
+	r.mutex.Lock()
+	defer r.mutex.Unlock()
+
+	if pair, ok, diffCase := r.loadAsFileOrDirectory(absPath); ok {
+		result := &ResolveResult{PathPair: pair, DifferentCase: diffCase}
+		r.finalizeResolve(result)
+		r.flushDebugLogs(flushDueToSuccess)
+		return result, debugMeta
+	}
+
+	return nil, debugMeta
+}
+
+type debugLogs struct {
+	what   string
+	indent string
+	notes  []logger.MsgData
+}
+
+func (d *debugLogs) addNote(text string) {
+	if d.indent != "" {
+		text = d.indent + text
+	}
+	d.notes = append(d.notes, logger.MsgData{Text: text, DisableMaximumWidth: true})
+}
+
+func (d *debugLogs) increaseIndent() {
+	d.indent += "  "
+}
+
+func (d *debugLogs) decreaseIndent() {
+	d.indent = d.indent[2:]
+}
+
+type flushMode uint8
+
+const (
+	flushDueToFailure flushMode = iota
+	flushDueToSuccess
+)
+
+func (r resolverQuery) flushDebugLogs(mode flushMode) {
+	if r.debugLogs != nil {
+		if mode == flushDueToFailure {
+			r.log.AddIDWithNotes(logger.MsgID_None, logger.Debug, nil, logger.Range{}, r.debugLogs.what, r.debugLogs.notes)
+		} else if r.log.Level <= logger.LevelVerbose {
+			r.log.AddIDWithNotes(logger.MsgID_None, logger.Verbose, nil, logger.Range{}, r.debugLogs.what, r.debugLogs.notes)
+		}
+	}
+}
+
+func (r resolverQuery) finalizeResolve(result *ResolveResult) {
+	if !result.PathPair.IsExternal && r.isExternal(r.options.ExternalSettings.PostResolve, result.PathPair.Primary.Text, r.kind) {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("The path %q was marked as external by the user", result.PathPair.Primary.Text))
+		}
+		result.PathPair.IsExternal = true
+	} else {
+		for i, path := range result.PathPair.iter() {
+			if path.Namespace != "file" {
+				continue
+			}
+			dirInfo := r.dirInfoCached(r.fs.Dir(path.Text))
+			if dirInfo == nil {
+				continue
+			}
+			base := r.fs.Base(path.Text)
+
+			// If the path contains symlinks, rewrite the path to the real path
+			if !r.options.PreserveSymlinks {
+				if entry, _ := dirInfo.entries.Get(base); entry != nil {
+					symlink := entry.Symlink(r.fs)
+					if symlink != "" {
+						// This means the entry itself is a symlink
+					} else if dirInfo.absRealPath != "" {
+						// There is at least one parent directory with a symlink
+						symlink = r.fs.Join(dirInfo.absRealPath, base)
+					}
+					if symlink != "" {
+						if r.debugLogs != nil {
+							r.debugLogs.addNote(fmt.Sprintf("Resolved symlink %q to %q", path.Text, symlink))
+						}
+						path.Text = symlink
+
+						// Look up the directory over again if it was changed
+						dirInfo = r.dirInfoCached(r.fs.Dir(path.Text))
+						if dirInfo == nil {
+							continue
+						}
+						base = r.fs.Base(path.Text)
+					}
+				}
+			}
+
+			// Path attributes are only taken from the primary path
+			if i > 0 {
+				continue
+			}
+
+			// Path attributes are not taken from disabled files
+			if path.IsDisabled() {
+				continue
+			}
+
+			// Look up this file in the "sideEffects" map in the nearest enclosing
+			// directory with a "package.json" file.
+			//
+			// Only do this for the primary path. Some packages have the primary
+			// path marked as having side effects and the secondary path marked
+			// as not having side effects. This is likely a bug in the package
+			// definition but we don't want to consider the primary path as not
+			// having side effects just because the secondary path is marked as
+			// not having side effects.
+			if pkgJSON := dirInfo.enclosingPackageJSON; pkgJSON != nil {
+				if pkgJSON.sideEffectsMap != nil {
+					hasSideEffects := false
+					pathLookup := strings.ReplaceAll(path.Text, "\\", "/") // Avoid problems with Windows-style slashes
+					if pkgJSON.sideEffectsMap[pathLookup] {
+						// Fast path: map lookup
+						hasSideEffects = true
+					} else {
+						// Slow path: glob tests
+						for _, re := range pkgJSON.sideEffectsRegexps {
+							if re.MatchString(pathLookup) {
+								hasSideEffects = true
+								break
+							}
+						}
+					}
+					if !hasSideEffects {
+						if r.debugLogs != nil {
+							r.debugLogs.addNote(fmt.Sprintf("Marking this file as having no side effects due to %q",
+								pkgJSON.source.KeyPath.Text))
+						}
+						result.PrimarySideEffectsData = pkgJSON.sideEffectsData
+					}
+				}
+
+				// Also copy over the "type" field
+				result.ModuleTypeData = pkgJSON.moduleTypeData
+			}
+
+			// Copy various fields from the nearest enclosing "tsconfig.json" file if present
+			if tsConfigJSON := r.tsConfigForDir(dirInfo); tsConfigJSON != nil {
+				result.TSConfig = &tsConfigJSON.Settings
+				result.TSConfigJSX = tsConfigJSON.JSXSettings
+				result.TSAlwaysStrict = tsConfigJSON.TSAlwaysStrictOrStrict()
+
+				if r.debugLogs != nil {
+					r.debugLogs.addNote(fmt.Sprintf("This import is under the effect of %q",
+						tsConfigJSON.AbsPath))
+					if result.TSConfigJSX.JSXFactory != nil {
+						r.debugLogs.addNote(fmt.Sprintf("\"jsxFactory\" is %q due to %q",
+							strings.Join(result.TSConfigJSX.JSXFactory, "."),
+							tsConfigJSON.AbsPath))
+					}
+					if result.TSConfigJSX.JSXFragmentFactory != nil {
+						r.debugLogs.addNote(fmt.Sprintf("\"jsxFragment\" is %q due to %q",
+							strings.Join(result.TSConfigJSX.JSXFragmentFactory, "."),
+							tsConfigJSON.AbsPath))
+					}
+				}
+			}
+		}
+	}
+
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Primary path is %q in namespace %q", result.PathPair.Primary.Text, result.PathPair.Primary.Namespace))
+		if result.PathPair.HasSecondary() {
+			r.debugLogs.addNote(fmt.Sprintf("Secondary path is %q in namespace %q", result.PathPair.Secondary.Text, result.PathPair.Secondary.Namespace))
+		}
+	}
+}
+
+func (r resolverQuery) resolveWithoutSymlinks(sourceDir string, sourceDirInfo *dirInfo, importPath string) *ResolveResult {
+	// This implements the module resolution algorithm from node.js, which is
+	// described here: https://nodejs.org/api/modules.html#modules_all_together
+	var result ResolveResult
+
+	// Return early if this is already an absolute path. In addition to asking
+	// the file system whether this is an absolute path, we also explicitly check
+	// whether it starts with a "/" and consider that an absolute path too. This
+	// is because relative paths can technically start with a "/" on Windows
+	// because it's not an absolute path on Windows. Then people might write code
+	// with imports that start with a "/" that works fine on Windows only to
+	// experience unexpected build failures later on other operating systems.
+	// Treating these paths as absolute paths on all platforms means Windows
+	// users will not be able to accidentally make use of these paths.
+	if strings.HasPrefix(importPath, "/") || r.fs.IsAbs(importPath) {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("The import %q is being treated as an absolute path", importPath))
+		}
+
+		// First, check path overrides from the nearest enclosing TypeScript "tsconfig.json" file
+		if tsConfigJSON := r.tsConfigForDir(sourceDirInfo); tsConfigJSON != nil && tsConfigJSON.Paths != nil {
+			if absolute, ok, diffCase := r.matchTSConfigPaths(tsConfigJSON, importPath); ok {
+				return &ResolveResult{PathPair: absolute, DifferentCase: diffCase}
+			}
+		}
+
+		// Run node's resolution rules (e.g. adding ".js")
+		if absolute, ok, diffCase := r.loadAsFileOrDirectory(importPath); ok {
+			return &ResolveResult{PathPair: absolute, DifferentCase: diffCase}
+		} else {
+			return nil
+		}
+	}
+
+	// Check both relative and package paths for CSS URL tokens, with relative
+	// paths taking precedence over package paths to match Webpack behavior.
+	isPackagePath := IsPackagePath(importPath)
+	checkRelative := !isPackagePath || r.kind.IsFromCSS()
+	checkPackage := isPackagePath
+
+	if checkRelative {
+		absPath := r.fs.Join(sourceDir, importPath)
+
+		// Check for external packages first
+		if r.isExternal(r.options.ExternalSettings.PostResolve, absPath, r.kind) {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("The path %q was marked as external by the user", absPath))
+			}
+			return &ResolveResult{PathPair: PathPair{Primary: logger.Path{Text: absPath, Namespace: "file"}, IsExternal: true}}
+		}
+
+		// Check the "browser" map
+		if importDirInfo := r.dirInfoCached(r.fs.Dir(absPath)); importDirInfo != nil {
+			if remapped, ok := r.checkBrowserMap(importDirInfo, absPath, absolutePathKind); ok {
+				if remapped == nil {
+					return &ResolveResult{PathPair: PathPair{Primary: logger.Path{Text: absPath, Namespace: "file", Flags: logger.PathDisabled}}}
+				}
+				if remappedResult, ok, diffCase, sideEffects := r.resolveWithoutRemapping(importDirInfo.enclosingBrowserScope, *remapped); ok {
+					result = ResolveResult{PathPair: remappedResult, DifferentCase: diffCase, PrimarySideEffectsData: sideEffects}
+					checkRelative = false
+					checkPackage = false
+				}
+			}
+		}
+
+		if checkRelative {
+			if absolute, ok, diffCase := r.loadAsFileOrDirectory(absPath); ok {
+				checkPackage = false
+				result = ResolveResult{PathPair: absolute, DifferentCase: diffCase}
+			} else if !checkPackage {
+				return nil
+			}
+		}
+	}
+
+	if checkPackage {
+		// Support remapping one package path to another via the "browser" field
+		if remapped, ok := r.checkBrowserMap(sourceDirInfo, importPath, packagePathKind); ok {
+			if remapped == nil {
+				// "browser": {"module": false}
+				if absolute, ok, diffCase, sideEffects := r.loadNodeModules(importPath, sourceDirInfo, false /* forbidImports */); ok {
+					absolute.Primary = logger.Path{Text: absolute.Primary.Text, Namespace: "file", Flags: logger.PathDisabled}
+					if absolute.HasSecondary() {
+						absolute.Secondary = logger.Path{Text: absolute.Secondary.Text, Namespace: "file", Flags: logger.PathDisabled}
+					}
+					return &ResolveResult{PathPair: absolute, DifferentCase: diffCase, PrimarySideEffectsData: sideEffects}
+				} else {
+					return &ResolveResult{PathPair: PathPair{Primary: logger.Path{Text: importPath, Flags: logger.PathDisabled}}, DifferentCase: diffCase}
+				}
+			}
+
+			// "browser": {"module": "./some-file"}
+			// "browser": {"module": "another-module"}
+			importPath = *remapped
+			sourceDirInfo = sourceDirInfo.enclosingBrowserScope
+		}
+
+		if absolute, ok, diffCase, sideEffects := r.resolveWithoutRemapping(sourceDirInfo, importPath); ok {
+			result = ResolveResult{PathPair: absolute, DifferentCase: diffCase, PrimarySideEffectsData: sideEffects}
+		} else {
+			// Note: node's "self references" are not currently supported
+			return nil
+		}
+	}
+
+	return &result
+}
+
+func (r resolverQuery) resolveWithoutRemapping(sourceDirInfo *dirInfo, importPath string) (PathPair, bool, *fs.DifferentCase, *SideEffectsData) {
+	if IsPackagePath(importPath) {
+		return r.loadNodeModules(importPath, sourceDirInfo, false /* forbidImports */)
+	} else {
+		absolute, ok, diffCase := r.loadAsFileOrDirectory(r.fs.Join(sourceDirInfo.absPath, importPath))
+		return absolute, ok, diffCase, nil
+	}
+}
+
+func PrettyPath(fs fs.FS, path logger.Path) string {
+	if path.Namespace == "file" {
+		if rel, ok := fs.Rel(fs.Cwd(), path.Text); ok {
+			path.Text = rel
+		}
+
+		// These human-readable paths are used in error messages, comments in output
+		// files, source names in source maps, and paths in the metadata JSON file.
+		// These should be platform-independent so our output doesn't depend on which
+		// operating system it was run. Replace Windows backward slashes with standard
+		// forward slashes.
+		path.Text = strings.ReplaceAll(path.Text, "\\", "/")
+	} else if path.Namespace != "" {
+		path.Text = fmt.Sprintf("%s:%s", path.Namespace, path.Text)
+	}
+
+	if path.IsDisabled() {
+		path.Text = "(disabled):" + path.Text
+	}
+
+	return path.Text + path.IgnoredSuffix
+}
+
+////////////////////////////////////////////////////////////////////////////////
+
+type dirInfo struct {
+	// These objects are immutable, so we can just point to the parent directory
+	// and avoid having to lock the cache again
+	parent *dirInfo
+
+	// A pointer to the enclosing dirInfo with a valid "browser" field in
+	// package.json. We need this to remap paths after they have been resolved.
+	enclosingBrowserScope *dirInfo
+
+	// All relevant information about this directory
+	absPath               string
+	pnpManifestAbsPath    string
+	entries               fs.DirEntries
+	packageJSON           *packageJSON  // Is there a "package.json" file in this directory?
+	enclosingPackageJSON  *packageJSON  // Is there a "package.json" file in this directory or a parent directory?
+	enclosingTSConfigJSON *TSConfigJSON // Is there a "tsconfig.json" file in this directory or a parent directory?
+	absRealPath           string        // If non-empty, this is the real absolute path resolving any symlinks
+	isNodeModules         bool          // Is the base name "node_modules"?
+	hasNodeModules        bool          // Is there a "node_modules" subdirectory?
+	isInsideNodeModules   bool          // Is this within a  "node_modules" subtree?
+}
+
+func (r resolverQuery) tsConfigForDir(dirInfo *dirInfo) *TSConfigJSON {
+	if dirInfo.isInsideNodeModules {
+		return nil
+	}
+	if r.tsConfigOverride != nil {
+		return r.tsConfigOverride
+	}
+	if dirInfo != nil {
+		return dirInfo.enclosingTSConfigJSON
+	}
+	return nil
+}
+
+func (r resolverQuery) dirInfoCached(path string) *dirInfo {
+	// First, check the cache
+	cached, ok := r.dirCache[path]
+
+	// Cache hit: stop now
+	if !ok {
+		// Update the cache to indicate failure. Even if the read failed, we don't
+		// want to retry again later. The directory is inaccessible so trying again
+		// is wasted. Doing this before calling "dirInfoUncached" prevents stack
+		// overflow in case this directory is recursively encountered again.
+		r.dirCache[path] = nil
+
+		// Cache miss: read the info
+		cached = r.dirInfoUncached(path)
+
+		// Only update the cache again on success
+		if cached != nil {
+			r.dirCache[path] = cached
+		}
+	}
+
+	if r.debugLogs != nil {
+		if cached == nil {
+			r.debugLogs.addNote(fmt.Sprintf("Failed to read directory %q", path))
+		} else {
+			count := cached.entries.PeekEntryCount()
+			entries := "entries"
+			if count == 1 {
+				entries = "entry"
+			}
+			r.debugLogs.addNote(fmt.Sprintf("Read %d %s for directory %q", count, entries, path))
+		}
+	}
+
+	return cached
+}
+
+var errParseErrorImportCycle = errors.New("(import cycle)")
+var errParseErrorAlreadyLogged = errors.New("(error already logged)")
+
+// This may return "parseErrorAlreadyLogged" in which case there was a syntax
+// error, but it's already been reported. No further errors should be logged.
+//
+// Nested calls may also return "parseErrorImportCycle". In that case the
+// caller is responsible for logging an appropriate error message.
+func (r resolverQuery) parseTSConfig(file string, visited map[string]bool, configDir string) (*TSConfigJSON, error) {
+	// Resolve any symlinks first before parsing the file
+	if !r.options.PreserveSymlinks {
+		if real, ok := r.fs.EvalSymlinks(file); ok {
+			file = real
+		}
+	}
+
+	// Don't infinite loop if a series of "extends" links forms a cycle
+	if visited[file] {
+		return nil, errParseErrorImportCycle
+	}
+
+	contents, err, originalError := r.caches.FSCache.ReadFile(r.fs, file)
+	if r.debugLogs != nil && originalError != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Failed to read file %q: %s", file, originalError.Error()))
+	}
+	if err != nil {
+		return nil, err
+	}
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("The file %q exists", file))
+	}
+
+	keyPath := logger.Path{Text: file, Namespace: "file"}
+	source := logger.Source{
+		KeyPath:    keyPath,
+		PrettyPath: PrettyPath(r.fs, keyPath),
+		Contents:   contents,
+	}
+	if visited != nil {
+		// This is only non-nil for "build" API calls. This is nil for "transform"
+		// API calls, which tells us to not process "extends" fields.
+		visited[file] = true
+	}
+	result, err := r.parseTSConfigFromSource(source, visited, configDir)
+	if visited != nil {
+		// Reset this to back false in case something uses TypeScript 5.0's multiple
+		// inheritance feature for "tsconfig.json" files. It should be valid to visit
+		// the same base "tsconfig.json" file multiple times from different multiple
+		// inheritance subtrees.
+		visited[file] = false
+	}
+	return result, err
+}
+
+func (r resolverQuery) parseTSConfigFromSource(source logger.Source, visited map[string]bool, configDir string) (*TSConfigJSON, error) {
+	tracker := logger.MakeLineColumnTracker(&source)
+	fileDir := r.fs.Dir(source.KeyPath.Text)
+	isExtends := len(visited) > 1
+
+	result := ParseTSConfigJSON(r.log, source, &r.caches.JSONCache, r.fs, fileDir, configDir, func(extends string, extendsRange logger.Range) *TSConfigJSON {
+		if visited == nil {
+			// If this is nil, then we're in a "transform" API call. In that case we
+			// deliberately skip processing "extends" fields. This is because the
+			// "transform" API is supposed to be without a file system.
+			return nil
+		}
+
+		// Note: This doesn't use the normal node module resolution algorithm
+		// both because it's different (e.g. we don't want to match a directory)
+		// and because it would deadlock since we're currently in the middle of
+		// populating the directory info cache.
+
+		maybeFinishOurSearch := func(base *TSConfigJSON, err error, extendsFile string) (*TSConfigJSON, bool) {
+			if err == nil {
+				return base, true
+			}
+
+			if err == syscall.ENOENT {
+				// Return false to indicate that we should continue searching
+				return nil, false
+			}
+
+			if err == errParseErrorImportCycle {
+				r.log.AddID(logger.MsgID_TSConfigJSON_Cycle, logger.Warning, &tracker, extendsRange,
+					fmt.Sprintf("Base config file %q forms cycle", extends))
+			} else if err != errParseErrorAlreadyLogged {
+				r.log.AddError(&tracker, extendsRange,
+					fmt.Sprintf("Cannot read file %q: %s",
+						PrettyPath(r.fs, logger.Path{Text: extendsFile, Namespace: "file"}), err.Error()))
+			}
+			return nil, true
+		}
+
+		// Check for a Yarn PnP manifest and use that to rewrite the path
+		if IsPackagePath(extends) {
+			pnpData := r.pnpManifest
+
+			// If we haven't loaded the Yarn PnP manifest yet, try to find one
+			if pnpData == nil {
+				current := fileDir
+				for {
+					if _, _, ok := fs.ParseYarnPnPVirtualPath(current); !ok {
+						absPath := r.fs.Join(current, ".pnp.data.json")
+						if json, source := r.extractYarnPnPDataFromJSON(absPath, pnpIgnoreErrorsAboutMissingFiles); json.Data != nil {
+							pnpData = compileYarnPnPData(absPath, current, json, source)
+							break
+						}
+
+						absPath = r.fs.Join(current, ".pnp.cjs")
+						if json, source := r.tryToExtractYarnPnPDataFromJS(absPath, pnpIgnoreErrorsAboutMissingFiles); json.Data != nil {
+							pnpData = compileYarnPnPData(absPath, current, json, source)
+							break
+						}
+
+						absPath = r.fs.Join(current, ".pnp.js")
+						if json, source := r.tryToExtractYarnPnPDataFromJS(absPath, pnpIgnoreErrorsAboutMissingFiles); json.Data != nil {
+							pnpData = compileYarnPnPData(absPath, current, json, source)
+							break
+						}
+					}
+
+					// Go to the parent directory, stopping at the file system root
+					next := r.fs.Dir(current)
+					if current == next {
+						break
+					}
+					current = next
+				}
+			}
+
+			if pnpData != nil {
+				if result := r.resolveToUnqualified(extends, fileDir, pnpData); result.status == pnpErrorGeneric {
+					if r.debugLogs != nil {
+						r.debugLogs.addNote("The Yarn PnP path resolution algorithm returned an error")
+					}
+					goto pnpError
+				} else if result.status == pnpSuccess {
+					// If Yarn PnP path resolution succeeded, run a custom abbreviated
+					// version of node's module resolution algorithm. The Yarn PnP
+					// specification says to use node's module resolution algorithm verbatim
+					// but that isn't what Yarn actually does. See this for more info:
+					// https://github.com/evanw/esbuild/issues/2473#issuecomment-1216774461
+					if entries, _, dirErr := r.fs.ReadDirectory(result.pkgDirPath); dirErr == nil {
+						if entry, _ := entries.Get("package.json"); entry != nil && entry.Kind(r.fs) == fs.FileEntry {
+							// Check the "exports" map
+							if packageJSON := r.parsePackageJSON(result.pkgDirPath); packageJSON != nil && packageJSON.exportsMap != nil {
+								if absolute, ok, _ := r.esmResolveAlgorithm(finalizeImportsExportsYarnPnPTSConfigExtends,
+									result.pkgIdent, "."+result.pkgSubpath, packageJSON, result.pkgDirPath, source.KeyPath.Text); ok {
+									base, err := r.parseTSConfig(absolute.Primary.Text, visited, configDir)
+									if result, shouldReturn := maybeFinishOurSearch(base, err, absolute.Primary.Text); shouldReturn {
+										return result
+									}
+								}
+								goto pnpError
+							}
+						}
+					}
+
+					// Continue with the module resolution algorithm from node.js
+					extends = r.fs.Join(result.pkgDirPath, result.pkgSubpath)
+				}
+			}
+		}
+
+		if IsPackagePath(extends) && !r.fs.IsAbs(extends) {
+			esmPackageName, esmPackageSubpath, esmOK := esmParsePackageName(extends)
+			if r.debugLogs != nil && esmOK {
+				r.debugLogs.addNote(fmt.Sprintf("Parsed tsconfig package name %q and package subpath %q", esmPackageName, esmPackageSubpath))
+			}
+
+			// If this is still a package path, try to resolve it to a "node_modules" directory
+			current := fileDir
+			for {
+				// Skip "node_modules" folders
+				if r.fs.Base(current) != "node_modules" {
+					join := r.fs.Join(current, "node_modules", extends)
+
+					// Check to see if "package.json" exists
+					pkgDir := r.fs.Join(current, "node_modules", esmPackageName)
+					pjFile := r.fs.Join(pkgDir, "package.json")
+					if _, err, originalError := r.fs.ReadFile(pjFile); err == nil {
+						if packageJSON := r.parsePackageJSON(pkgDir); packageJSON != nil {
+							// Try checking the "tsconfig" field of "package.json". The ability to use "extends" like this was added in TypeScript 3.2:
+							// https://www.typescriptlang.org/docs/handbook/release-notes/typescript-3-2.html#tsconfigjson-inheritance-via-nodejs-packages
+							if packageJSON.tsconfig != "" {
+								join = packageJSON.tsconfig
+								if !r.fs.IsAbs(join) {
+									join = r.fs.Join(pkgDir, join)
+								}
+							}
+
+							// Try checking the "exports" map. The ability to use "extends" like this was added in TypeScript 5.0:
+							// https://devblogs.microsoft.com/typescript/announcing-typescript-5-0/
+							if packageJSON.exportsMap != nil {
+								if r.debugLogs != nil {
+									r.debugLogs.addNote(fmt.Sprintf("Looking for %q in \"exports\" map in %q", esmPackageSubpath, packageJSON.source.KeyPath.Text))
+									r.debugLogs.increaseIndent()
+									defer r.debugLogs.decreaseIndent()
+								}
+
+								// Note: TypeScript appears to always treat this as a "require" import
+								conditions := r.esmConditionsRequire
+								resolvedPath, status, debug := r.esmPackageExportsResolve("/", esmPackageSubpath, packageJSON.exportsMap.root, conditions)
+								resolvedPath, status, debug = r.esmHandlePostConditions(resolvedPath, status, debug)
+
+								// This is a very abbreviated version of our ESM resolution
+								if status == pjStatusExact || status == pjStatusExactEndsWithStar {
+									fileToCheck := r.fs.Join(pkgDir, resolvedPath)
+									base, err := r.parseTSConfig(fileToCheck, visited, configDir)
+
+									if result, shouldReturn := maybeFinishOurSearch(base, err, fileToCheck); shouldReturn {
+										return result
+									}
+								}
+							}
+						}
+					} else if r.debugLogs != nil && originalError != nil {
+						r.debugLogs.addNote(fmt.Sprintf("Failed to read file %q: %s", pjFile, originalError.Error()))
+					}
+
+					filesToCheck := []string{r.fs.Join(join, "tsconfig.json"), join, join + ".json"}
+					for _, fileToCheck := range filesToCheck {
+						base, err := r.parseTSConfig(fileToCheck, visited, configDir)
+
+						// Explicitly ignore matches if they are directories instead of files
+						if err != nil && err != syscall.ENOENT {
+							if entries, _, dirErr := r.fs.ReadDirectory(r.fs.Dir(fileToCheck)); dirErr == nil {
+								if entry, _ := entries.Get(r.fs.Base(fileToCheck)); entry != nil && entry.Kind(r.fs) == fs.DirEntry {
+									continue
+								}
+							}
+						}
+
+						if result, shouldReturn := maybeFinishOurSearch(base, err, fileToCheck); shouldReturn {
+							return result
+						}
+					}
+				}
+
+				// Go to the parent directory, stopping at the file system root
+				next := r.fs.Dir(current)
+				if current == next {
+					break
+				}
+				current = next
+			}
+		} else {
+			extendsFile := extends
+
+			// The TypeScript compiler has a strange behavior that seems like a bug
+			// where "." and ".." behave differently than other forms such as "./."
+			// or "../." and are interpreted as having an implicit "tsconfig.json"
+			// suffix.
+			//
+			// I believe their bug is caused by some parts of their code checking for
+			// relative paths using the literal "./" and "../" prefixes (requiring
+			// the slash) and other parts checking using the regular expression
+			// /^\.\.?($|[\\/])/ (with the slash optional).
+			//
+			// In any case, people are now relying on this behavior. One example is
+			// this: https://github.com/esbuild-kit/tsx/pull/158. So we replicate this
+			// bug in esbuild as well.
+			if extendsFile == "." || extendsFile == ".." {
+				extendsFile += "/tsconfig.json"
+			}
+
+			// If this is a regular path, search relative to the enclosing directory
+			if !r.fs.IsAbs(extendsFile) {
+				extendsFile = r.fs.Join(fileDir, extendsFile)
+			}
+			base, err := r.parseTSConfig(extendsFile, visited, configDir)
+
+			// TypeScript's handling of "extends" has some specific edge cases. We
+			// must only try adding ".json" if it's not already present, which is
+			// unlike how node path resolution works. We also need to explicitly
+			// ignore matches if they are directories instead of files. Some users
+			// name directories the same name as their config files.
+			if err != nil && !strings.HasSuffix(extendsFile, ".json") {
+				if entries, _, dirErr := r.fs.ReadDirectory(r.fs.Dir(extendsFile)); dirErr == nil {
+					extendsBase := r.fs.Base(extendsFile)
+					if entry, _ := entries.Get(extendsBase); entry == nil || entry.Kind(r.fs) != fs.FileEntry {
+						if entry, _ := entries.Get(extendsBase + ".json"); entry != nil && entry.Kind(r.fs) == fs.FileEntry {
+							base, err = r.parseTSConfig(extendsFile+".json", visited, configDir)
+						}
+					}
+				}
+			}
+
+			if result, shouldReturn := maybeFinishOurSearch(base, err, extendsFile); shouldReturn {
+				return result
+			}
+		}
+
+		// Suppress warnings about missing base config files inside "node_modules"
+	pnpError:
+		if !helpers.IsInsideNodeModules(source.KeyPath.Text) {
+			var notes []logger.MsgData
+			if r.debugLogs != nil {
+				notes = r.debugLogs.notes
+			}
+			r.log.AddIDWithNotes(logger.MsgID_TSConfigJSON_Missing, logger.Warning, &tracker, extendsRange,
+				fmt.Sprintf("Cannot find base config file %q", extends), notes)
+		}
+
+		return nil
+	})
+
+	if result == nil {
+		return nil, errParseErrorAlreadyLogged
+	}
+
+	// Now that we have parsed the entire "tsconfig.json" file, filter out any
+	// paths that are invalid due to being a package-style path without a base
+	// URL specified. This must be done here instead of when we're parsing the
+	// original file because TypeScript allows one "tsconfig.json" file to
+	// specify "baseUrl" and inherit a "paths" from another file via "extends".
+	if !isExtends && result.Paths != nil && result.BaseURL == nil {
+		var tracker *logger.LineColumnTracker
+		for key, paths := range result.Paths.Map {
+			end := 0
+			for _, path := range paths {
+				if isValidTSConfigPathNoBaseURLPattern(path.Text, r.log, &result.Paths.Source, &tracker, path.Loc) {
+					paths[end] = path
+					end++
+				}
+			}
+			if end < len(paths) {
+				result.Paths.Map[key] = paths[:end]
+			}
+		}
+	}
+
+	return result, nil
+}
+
+func (r resolverQuery) dirInfoUncached(path string) *dirInfo {
+	// Get the info for the parent directory
+	var parentInfo *dirInfo
+	parentDir := r.fs.Dir(path)
+	if parentDir != path {
+		parentInfo = r.dirInfoCached(parentDir)
+
+		// Stop now if the parent directory doesn't exist
+		if parentInfo == nil {
+			return nil
+		}
+	}
+
+	// List the directories
+	entries, err, originalError := r.fs.ReadDirectory(path)
+	if err == syscall.EACCES || err == syscall.EPERM {
+		// Just pretend this directory is empty if we can't access it. This is the
+		// case on Unix for directories that only have the execute permission bit
+		// set. It means we will just pass through the empty directory and
+		// continue to check the directories above it, which is now node behaves.
+		entries = fs.MakeEmptyDirEntries(path)
+		err = nil
+	}
+	if r.debugLogs != nil && originalError != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Failed to read directory %q: %s", path, originalError.Error()))
+	}
+	if err != nil {
+		// Ignore "ENOTDIR" here so that calling "ReadDirectory" on a file behaves
+		// as if there is nothing there at all instead of causing an error due to
+		// the directory actually being a file. This is a workaround for situations
+		// where people try to import from a path containing a file as a parent
+		// directory. The "pnpm" package manager generates a faulty "NODE_PATH"
+		// list which contains such paths and treating them as missing means we just
+		// ignore them during path resolution.
+		if err != syscall.ENOENT && err != syscall.ENOTDIR {
+			r.log.AddError(nil, logger.Range{},
+				fmt.Sprintf("Cannot read directory %q: %s",
+					PrettyPath(r.fs, logger.Path{Text: path, Namespace: "file"}), err.Error()))
+		}
+		return nil
+	}
+	info := &dirInfo{
+		absPath: path,
+		parent:  parentInfo,
+		entries: entries,
+	}
+
+	// A "node_modules" directory isn't allowed to directly contain another "node_modules" directory
+	base := r.fs.Base(path)
+	if base == "node_modules" {
+		info.isNodeModules = true
+		info.isInsideNodeModules = true
+	} else if entry, _ := entries.Get("node_modules"); entry != nil {
+		info.hasNodeModules = entry.Kind(r.fs) == fs.DirEntry
+	}
+
+	// Propagate the browser scope into child directories
+	if parentInfo != nil {
+		info.enclosingPackageJSON = parentInfo.enclosingPackageJSON
+		info.enclosingBrowserScope = parentInfo.enclosingBrowserScope
+		info.enclosingTSConfigJSON = parentInfo.enclosingTSConfigJSON
+		if parentInfo.isInsideNodeModules {
+			info.isInsideNodeModules = true
+		}
+
+		// Make sure "absRealPath" is the real path of the directory (resolving any symlinks)
+		if !r.options.PreserveSymlinks {
+			if entry, _ := parentInfo.entries.Get(base); entry != nil {
+				if symlink := entry.Symlink(r.fs); symlink != "" {
+					if r.debugLogs != nil {
+						r.debugLogs.addNote(fmt.Sprintf("Resolved symlink %q to %q", path, symlink))
+					}
+					info.absRealPath = symlink
+				} else if parentInfo.absRealPath != "" {
+					symlink := r.fs.Join(parentInfo.absRealPath, base)
+					if r.debugLogs != nil {
+						r.debugLogs.addNote(fmt.Sprintf("Resolved symlink %q to %q", path, symlink))
+					}
+					info.absRealPath = symlink
+				}
+			}
+		}
+	}
+
+	// Record if this directory has a package.json file
+	if entry, _ := entries.Get("package.json"); entry != nil && entry.Kind(r.fs) == fs.FileEntry {
+		info.packageJSON = r.parsePackageJSON(path)
+
+		// Propagate this "package.json" file into child directories
+		if info.packageJSON != nil {
+			info.enclosingPackageJSON = info.packageJSON
+			if info.packageJSON.browserMap != nil {
+				info.enclosingBrowserScope = info
+			}
+		}
+	}
+
+	// Record if this directory has a tsconfig.json or jsconfig.json file
+	if r.tsConfigOverride == nil {
+		var tsConfigPath string
+		if entry, _ := entries.Get("tsconfig.json"); entry != nil && entry.Kind(r.fs) == fs.FileEntry {
+			tsConfigPath = r.fs.Join(path, "tsconfig.json")
+		} else if entry, _ := entries.Get("jsconfig.json"); entry != nil && entry.Kind(r.fs) == fs.FileEntry {
+			tsConfigPath = r.fs.Join(path, "jsconfig.json")
+		}
+
+		// Except don't do this if we're inside a "node_modules" directory. Package
+		// authors often publish their "tsconfig.json" files to npm because of
+		// npm's default-include publishing model and because these authors
+		// probably don't know about ".npmignore" files.
+		//
+		// People trying to use these packages with esbuild have historically
+		// complained that esbuild is respecting "tsconfig.json" in these cases.
+		// The assumption is that the package author published these files by
+		// accident.
+		//
+		// Ignoring "tsconfig.json" files inside "node_modules" directories breaks
+		// the use case of publishing TypeScript code and having it be transpiled
+		// for you, but that's the uncommon case and likely doesn't work with
+		// many other tools anyway. So now these files are ignored.
+		if tsConfigPath != "" && !info.isInsideNodeModules {
+			var err error
+			info.enclosingTSConfigJSON, err = r.parseTSConfig(tsConfigPath, make(map[string]bool), r.fs.Dir(tsConfigPath))
+			if err != nil {
+				if err == syscall.ENOENT {
+					r.log.AddError(nil, logger.Range{}, fmt.Sprintf("Cannot find tsconfig file %q",
+						PrettyPath(r.fs, logger.Path{Text: tsConfigPath, Namespace: "file"})))
+				} else if err != errParseErrorAlreadyLogged {
+					r.log.AddID(logger.MsgID_TSConfigJSON_Missing, logger.Debug, nil, logger.Range{},
+						fmt.Sprintf("Cannot read file %q: %s",
+							PrettyPath(r.fs, logger.Path{Text: tsConfigPath, Namespace: "file"}), err.Error()))
+				}
+			}
+		}
+	}
+
+	// Record if this directory has a Yarn PnP manifest. This must not be done
+	// for Yarn virtual paths because that will result in duplicate copies of
+	// the same manifest which will result in multiple copies of the same virtual
+	// directory in the same path, which we don't handle (and which also doesn't
+	// match Yarn's behavior).
+	//
+	// For example, imagine a project with a manifest here:
+	//
+	//   /project/.pnp.cjs
+	//
+	// and a source file with an import of "bar" here:
+	//
+	//   /project/.yarn/__virtual__/pkg/1/foo.js
+	//
+	// If we didn't ignore Yarn PnP manifests in virtual folders, then we would
+	// pick up on the one here:
+	//
+	//   /project/.yarn/__virtual__/pkg/1/.pnp.cjs
+	//
+	// which means we would potentially resolve the import to something like this:
+	//
+	//   /project/.yarn/__virtual__/pkg/1/.yarn/__virtual__/pkg/1/bar
+	//
+	if r.pnpManifest == nil {
+		if _, _, ok := fs.ParseYarnPnPVirtualPath(path); !ok {
+			if pnp, _ := entries.Get(".pnp.data.json"); pnp != nil && pnp.Kind(r.fs) == fs.FileEntry {
+				info.pnpManifestAbsPath = r.fs.Join(path, ".pnp.data.json")
+			} else if pnp, _ := entries.Get(".pnp.cjs"); pnp != nil && pnp.Kind(r.fs) == fs.FileEntry {
+				info.pnpManifestAbsPath = r.fs.Join(path, ".pnp.cjs")
+			} else if pnp, _ := entries.Get(".pnp.js"); pnp != nil && pnp.Kind(r.fs) == fs.FileEntry {
+				info.pnpManifestAbsPath = r.fs.Join(path, ".pnp.js")
+			}
+		}
+	}
+
+	return info
+}
+
+// TypeScript-specific behavior: if the extension is ".js" or ".jsx", try
+// replacing it with ".ts" or ".tsx". At the time of writing this specific
+// behavior comes from the function "loadModuleFromFile()" in the file
+// "moduleNameResolver.ts" in the TypeScript compiler source code. It
+// contains this comment:
+//
+//	If that didn't work, try stripping a ".js" or ".jsx" extension and
+//	replacing it with a TypeScript one; e.g. "./foo.js" can be matched
+//	by "./foo.ts" or "./foo.d.ts"
+//
+// We don't care about ".d.ts" files because we can't do anything with
+// those, so we ignore that part of the behavior.
+//
+// See the discussion here for more historical context:
+// https://github.com/microsoft/TypeScript/issues/4595
+var rewrittenFileExtensions = map[string][]string{
+	// Note that the official compiler code always tries ".ts" before
+	// ".tsx" even if the original extension was ".jsx".
+	".js":  {".ts", ".tsx"},
+	".jsx": {".ts", ".tsx"},
+	".mjs": {".mts"},
+	".cjs": {".cts"},
+}
+
+func (r resolverQuery) loadAsFile(path string, extensionOrder []string) (string, bool, *fs.DifferentCase) {
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Attempting to load %q as a file", path))
+		r.debugLogs.increaseIndent()
+		defer r.debugLogs.decreaseIndent()
+	}
+
+	// Read the directory entries once to minimize locking
+	dirPath := r.fs.Dir(path)
+	entries, err, originalError := r.fs.ReadDirectory(dirPath)
+	if r.debugLogs != nil && originalError != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Failed to read directory %q: %s", dirPath, originalError.Error()))
+	}
+	if err != nil {
+		if err != syscall.ENOENT {
+			r.log.AddError(nil, logger.Range{},
+				fmt.Sprintf("Cannot read directory %q: %s",
+					PrettyPath(r.fs, logger.Path{Text: dirPath, Namespace: "file"}), err.Error()))
+		}
+		return "", false, nil
+	}
+
+	tryFile := func(base string) (string, bool, *fs.DifferentCase) {
+		baseWithSuffix := base
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("Checking for file %q", baseWithSuffix))
+		}
+		if entry, diffCase := entries.Get(baseWithSuffix); entry != nil && entry.Kind(r.fs) == fs.FileEntry {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("Found file %q", baseWithSuffix))
+			}
+			return r.fs.Join(dirPath, baseWithSuffix), true, diffCase
+		}
+		return "", false, nil
+	}
+
+	base := r.fs.Base(path)
+
+	// Given "./x.js", node's algorithm tries things in the following order:
+	//
+	//   ./x.js
+	//   ./x.js.js
+	//   ./x.js.json
+	//   ./x.js.node
+	//   ./x.js/index.js
+	//   ./x.js/index.json
+	//   ./x.js/index.node
+	//
+	// Given "./x.js", TypeScript's algorithm tries things in the following order:
+	//
+	//   ./x.js.ts
+	//   ./x.js.tsx
+	//   ./x.js.d.ts
+	//   ./x.ts
+	//   ./x.tsx
+	//   ./x.d.ts
+	//   ./x.js/index.ts
+	//   ./x.js/index.tsx
+	//   ./x.js/index.d.ts
+	//   ./x.js.js
+	//   ./x.js.jsx
+	//   ./x.js
+	//   ./x.jsx
+	//   ./x.js/index.js
+	//   ./x.js/index.jsx
+	//
+	// Our order below is a blend of both. We try to follow node's algorithm but
+	// with the features of TypeScript's algorithm (omitting ".d.ts" files, which
+	// don't contain code). This means we should end up checking the same files
+	// as TypeScript, but in a different order.
+	//
+	// One reason we use a different order is because we support a customizable
+	// extension resolution order, which doesn't fit well into TypeScript's
+	// algorithm. For example, you can configure esbuild to check for extensions
+	// in the order ".js,.ts,.jsx,.tsx" but TypeScript always checks TypeScript
+	// extensions before JavaScript extensions, so we can't obey the user's
+	// intent if we follow TypeScript's algorithm exactly.
+	//
+	// Another reason we deviate from TypeScript's order is because our code is
+	// structured to handle node's algorithm and TypeScript's algorithm has a
+	// different structure. It intermixes multiple calls to LOAD_AS_FILE and
+	// LOAD_INDEX together while node always does one LOAD_AS_FILE before one
+	// LOAD_INDEX.
+
+	// Try the plain path without any extensions
+	if absolute, ok, diffCase := tryFile(base); ok {
+		return absolute, ok, diffCase
+	}
+
+	// Try the path with extensions
+	for _, ext := range extensionOrder {
+		if absolute, ok, diffCase := tryFile(base + ext); ok {
+			return absolute, ok, diffCase
+		}
+	}
+
+	// TypeScript-specific behavior: try rewriting ".js" to ".ts"
+	for old, exts := range rewrittenFileExtensions {
+		if !strings.HasSuffix(base, old) {
+			continue
+		}
+		lastDot := strings.LastIndexByte(base, '.')
+		for _, ext := range exts {
+			if absolute, ok, diffCase := tryFile(base[:lastDot] + ext); ok {
+				return absolute, ok, diffCase
+			}
+		}
+		break
+	}
+
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Failed to find file %q", base))
+	}
+	return "", false, nil
+}
+
+func (r resolverQuery) loadAsIndex(dirInfo *dirInfo, extensionOrder []string) (PathPair, bool, *fs.DifferentCase) {
+	// Try the "index" file with extensions
+	for _, ext := range extensionOrder {
+		base := "index" + ext
+		if entry, diffCase := dirInfo.entries.Get(base); entry != nil && entry.Kind(r.fs) == fs.FileEntry {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("Found file %q", r.fs.Join(dirInfo.absPath, base)))
+			}
+			return PathPair{Primary: logger.Path{Text: r.fs.Join(dirInfo.absPath, base), Namespace: "file"}}, true, diffCase
+		}
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("Failed to find file %q", r.fs.Join(dirInfo.absPath, base)))
+		}
+	}
+
+	return PathPair{}, false, nil
+}
+
+func (r resolverQuery) loadAsIndexWithBrowserRemapping(dirInfo *dirInfo, path string, extensionOrder []string) (PathPair, bool, *fs.DifferentCase) {
+	// Potentially remap using the "browser" field
+	absPath := r.fs.Join(path, "index")
+	if remapped, ok := r.checkBrowserMap(dirInfo, absPath, absolutePathKind); ok {
+		if remapped == nil {
+			return PathPair{Primary: logger.Path{Text: absPath, Namespace: "file", Flags: logger.PathDisabled}}, true, nil
+		}
+		remappedAbs := r.fs.Join(path, *remapped)
+
+		// Is this a file?
+		absolute, ok, diffCase := r.loadAsFile(remappedAbs, extensionOrder)
+		if ok {
+			return PathPair{Primary: logger.Path{Text: absolute, Namespace: "file"}}, true, diffCase
+		}
+
+		// Is it a directory with an index?
+		if fieldDirInfo := r.dirInfoCached(remappedAbs); fieldDirInfo != nil {
+			if absolute, ok, _ := r.loadAsIndex(fieldDirInfo, extensionOrder); ok {
+				return absolute, true, nil
+			}
+		}
+
+		return PathPair{}, false, nil
+	}
+
+	return r.loadAsIndex(dirInfo, extensionOrder)
+}
+
+func getProperty(json js_ast.Expr, name string) (js_ast.Expr, logger.Loc, bool) {
+	if obj, ok := json.Data.(*js_ast.EObject); ok {
+		for _, prop := range obj.Properties {
+			if key, ok := prop.Key.Data.(*js_ast.EString); ok && key.Value != nil && helpers.UTF16EqualsString(key.Value, name) {
+				return prop.ValueOrNil, prop.Key.Loc, true
+			}
+		}
+	}
+	return js_ast.Expr{}, logger.Loc{}, false
+}
+
+func getString(json js_ast.Expr) (string, bool) {
+	if value, ok := json.Data.(*js_ast.EString); ok {
+		return helpers.UTF16ToString(value.Value), true
+	}
+	return "", false
+}
+
+func getBool(json js_ast.Expr) (bool, bool) {
+	if value, ok := json.Data.(*js_ast.EBoolean); ok {
+		return value.Value, true
+	}
+	return false, false
+}
+
+func (r resolverQuery) loadAsFileOrDirectory(path string) (PathPair, bool, *fs.DifferentCase) {
+	extensionOrder := r.options.ExtensionOrder
+	if r.kind.MustResolveToCSS() {
+		// Use a special import order for CSS "@import" imports
+		extensionOrder = r.cssExtensionOrder
+	} else if helpers.IsInsideNodeModules(path) {
+		// Use a special import order for imports inside "node_modules"
+		extensionOrder = r.nodeModulesExtensionOrder
+	}
+
+	// Is this a file?
+	absolute, ok, diffCase := r.loadAsFile(path, extensionOrder)
+	if ok {
+		return PathPair{Primary: logger.Path{Text: absolute, Namespace: "file"}}, true, diffCase
+	}
+
+	// Is this a directory?
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Attempting to load %q as a directory", path))
+		r.debugLogs.increaseIndent()
+		defer r.debugLogs.decreaseIndent()
+	}
+	dirInfo := r.dirInfoCached(path)
+	if dirInfo == nil {
+		return PathPair{}, false, nil
+	}
+
+	// Try using the main field(s) from "package.json"
+	if absolute, ok, diffCase := r.loadAsMainField(dirInfo, path, extensionOrder); ok {
+		return absolute, true, diffCase
+	}
+
+	// Look for an "index" file with known extensions
+	if absolute, ok, diffCase := r.loadAsIndexWithBrowserRemapping(dirInfo, path, extensionOrder); ok {
+		return absolute, true, diffCase
+	}
+
+	return PathPair{}, false, nil
+}
+
+func (r resolverQuery) loadAsMainField(dirInfo *dirInfo, path string, extensionOrder []string) (PathPair, bool, *fs.DifferentCase) {
+	if dirInfo.packageJSON == nil {
+		return PathPair{}, false, nil
+	}
+
+	mainFieldValues := dirInfo.packageJSON.mainFields
+	mainFieldKeys := r.options.MainFields
+	autoMain := false
+
+	// If the user has not explicitly specified a "main" field order,
+	// use a default one determined by the current platform target
+	if mainFieldKeys == nil {
+		mainFieldKeys = defaultMainFields[r.options.Platform]
+		autoMain = true
+	}
+
+	loadMainField := func(fieldRelPath string, field string) (PathPair, bool, *fs.DifferentCase) {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("Found main field %q with path %q", field, fieldRelPath))
+			r.debugLogs.increaseIndent()
+			defer r.debugLogs.decreaseIndent()
+		}
+
+		// Potentially remap using the "browser" field
+		fieldAbsPath := r.fs.Join(path, fieldRelPath)
+		if remapped, ok := r.checkBrowserMap(dirInfo, fieldAbsPath, absolutePathKind); ok {
+			if remapped == nil {
+				return PathPair{Primary: logger.Path{Text: fieldAbsPath, Namespace: "file", Flags: logger.PathDisabled}}, true, nil
+			}
+			fieldAbsPath = r.fs.Join(path, *remapped)
+		}
+
+		// Is this a file?
+		absolute, ok, diffCase := r.loadAsFile(fieldAbsPath, extensionOrder)
+		if ok {
+			return PathPair{Primary: logger.Path{Text: absolute, Namespace: "file"}}, true, diffCase
+		}
+
+		// Is it a directory with an index?
+		if fieldDirInfo := r.dirInfoCached(fieldAbsPath); fieldDirInfo != nil {
+			if absolute, ok, _ := r.loadAsIndexWithBrowserRemapping(fieldDirInfo, fieldAbsPath, extensionOrder); ok {
+				return absolute, true, nil
+			}
+		}
+
+		return PathPair{}, false, nil
+	}
+
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Searching for main fields in %q", dirInfo.packageJSON.source.KeyPath.Text))
+		r.debugLogs.increaseIndent()
+		defer r.debugLogs.decreaseIndent()
+	}
+
+	foundSomething := false
+
+	for _, key := range mainFieldKeys {
+		value, ok := mainFieldValues[key]
+		if !ok {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("Did not find main field %q", key))
+			}
+			continue
+		}
+		foundSomething = true
+
+		absolute, ok, diffCase := loadMainField(value.relPath, key)
+		if !ok {
+			continue
+		}
+
+		// If the user did not manually configure a "main" field order, then
+		// use a special per-module automatic algorithm to decide whether to
+		// use "module" or "main" based on whether the package is imported
+		// using "import" or "require".
+		if autoMain && key == "module" {
+			var absoluteMain PathPair
+			var okMain bool
+			var diffCaseMain *fs.DifferentCase
+
+			if main, ok := mainFieldValues["main"]; ok {
+				if absolute, ok, diffCase := loadMainField(main.relPath, "main"); ok {
+					absoluteMain = absolute
+					okMain = true
+					diffCaseMain = diffCase
+				}
+			} else {
+				// Some packages have a "module" field without a "main" field but
+				// still have an implicit "index.js" file. In that case, treat that
+				// as the value for "main".
+				if absolute, ok, diffCase := r.loadAsIndexWithBrowserRemapping(dirInfo, path, extensionOrder); ok {
+					absoluteMain = absolute
+					okMain = true
+					diffCaseMain = diffCase
+				}
+			}
+
+			if okMain {
+				// If both the "main" and "module" fields exist, use "main" if the
+				// path is for "require" and "module" if the path is for "import".
+				// If we're using "module", return enough information to be able to
+				// fall back to "main" later if something ended up using "require()"
+				// with this same path. The goal of this code is to avoid having
+				// both the "module" file and the "main" file in the bundle at the
+				// same time.
+				if r.kind != ast.ImportRequire {
+					if r.debugLogs != nil {
+						r.debugLogs.addNote(fmt.Sprintf("Resolved to %q using the \"module\" field in %q",
+							absolute.Primary.Text, dirInfo.packageJSON.source.KeyPath.Text))
+						r.debugLogs.addNote(fmt.Sprintf("The fallback path in case of \"require\" is %q",
+							absoluteMain.Primary.Text))
+					}
+					return PathPair{
+						// This is the whole point of the path pair
+						Primary:   absolute.Primary,
+						Secondary: absoluteMain.Primary,
+					}, true, diffCase
+				} else {
+					if r.debugLogs != nil {
+						r.debugLogs.addNote(fmt.Sprintf("Resolved to %q because of \"require\"", absoluteMain.Primary.Text))
+					}
+					return absoluteMain, true, diffCaseMain
+				}
+			}
+		}
+
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("Resolved to %q using the %q field in %q",
+				absolute.Primary.Text, key, dirInfo.packageJSON.source.KeyPath.Text))
+		}
+		return absolute, true, diffCase
+	}
+
+	// Let the user know if "main" exists but was skipped due to mis-configuration
+	if !foundSomething {
+		for _, field := range mainFieldsForFailure {
+			if main, ok := mainFieldValues[field]; ok {
+				tracker := logger.MakeLineColumnTracker(&dirInfo.packageJSON.source)
+				keyRange := dirInfo.packageJSON.source.RangeOfString(main.keyLoc)
+				if len(mainFieldKeys) == 0 && r.options.Platform == config.PlatformNeutral {
+					r.debugMeta.notes = append(r.debugMeta.notes, tracker.MsgData(keyRange,
+						fmt.Sprintf("The %q field here was ignored. Main fields must be configured explicitly when using the \"neutral\" platform.",
+							field)))
+				} else {
+					r.debugMeta.notes = append(r.debugMeta.notes, tracker.MsgData(keyRange,
+						fmt.Sprintf("The %q field here was ignored because the list of main fields to use is currently set to [%s].",
+							field, helpers.StringArrayToQuotedCommaSeparatedString(mainFieldKeys))))
+				}
+				break
+			}
+		}
+	}
+
+	return PathPair{}, false, nil
+}
+
+func hasCaseInsensitiveSuffix(s string, suffix string) bool {
+	return len(s) >= len(suffix) && strings.EqualFold(s[len(s)-len(suffix):], suffix)
+}
+
+// This closely follows the behavior of "tryLoadModuleUsingPaths()" in the
+// official TypeScript compiler
+func (r resolverQuery) matchTSConfigPaths(tsConfigJSON *TSConfigJSON, path string) (PathPair, bool, *fs.DifferentCase) {
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Matching %q against \"paths\" in %q", path, tsConfigJSON.AbsPath))
+		r.debugLogs.increaseIndent()
+		defer r.debugLogs.decreaseIndent()
+	}
+
+	absBaseURL := tsConfigJSON.BaseURLForPaths
+
+	// The explicit base URL should take precedence over the implicit base URL
+	// if present. This matters when a tsconfig.json file overrides "baseUrl"
+	// from another extended tsconfig.json file but doesn't override "paths".
+	if tsConfigJSON.BaseURL != nil {
+		absBaseURL = *tsConfigJSON.BaseURL
+	}
+
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Using %q as \"baseUrl\"", absBaseURL))
+	}
+
+	// Check for exact matches first
+	for key, originalPaths := range tsConfigJSON.Paths.Map {
+		if key == path {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("Found an exact match for %q in \"paths\"", key))
+			}
+			for _, originalPath := range originalPaths {
+				// Ignore ".d.ts" files because this rule is obviously only here for type checking
+				if hasCaseInsensitiveSuffix(originalPath.Text, ".d.ts") {
+					if r.debugLogs != nil {
+						r.debugLogs.addNote(fmt.Sprintf("Ignoring substitution %q because it ends in \".d.ts\"", originalPath.Text))
+					}
+					continue
+				}
+
+				// Load the original path relative to the "baseUrl" from tsconfig.json
+				absoluteOriginalPath := originalPath.Text
+				if !r.fs.IsAbs(absoluteOriginalPath) {
+					absoluteOriginalPath = r.fs.Join(absBaseURL, absoluteOriginalPath)
+				}
+				if absolute, ok, diffCase := r.loadAsFileOrDirectory(absoluteOriginalPath); ok {
+					return absolute, true, diffCase
+				}
+			}
+			return PathPair{}, false, nil
+		}
+	}
+
+	type match struct {
+		prefix        string
+		suffix        string
+		originalPaths []TSConfigPath
+	}
+
+	// Check for pattern matches next
+	longestMatchPrefixLength := -1
+	longestMatchSuffixLength := -1
+	var longestMatch match
+	for key, originalPaths := range tsConfigJSON.Paths.Map {
+		if starIndex := strings.IndexByte(key, '*'); starIndex != -1 {
+			prefix, suffix := key[:starIndex], key[starIndex+1:]
+
+			// Find the match with the longest prefix. If two matches have the same
+			// prefix length, pick the one with the longest suffix. This second edge
+			// case isn't handled by the TypeScript compiler, but we handle it
+			// because we want the output to always be deterministic and Go map
+			// iteration order is deliberately non-deterministic.
+			if strings.HasPrefix(path, prefix) && strings.HasSuffix(path, suffix) && (len(prefix) > longestMatchPrefixLength ||
+				(len(prefix) == longestMatchPrefixLength && len(suffix) > longestMatchSuffixLength)) {
+				longestMatchPrefixLength = len(prefix)
+				longestMatchSuffixLength = len(suffix)
+				longestMatch = match{
+					prefix:        prefix,
+					suffix:        suffix,
+					originalPaths: originalPaths,
+				}
+			}
+		}
+	}
+
+	// If there is at least one match, only consider the one with the longest
+	// prefix. This matches the behavior of the TypeScript compiler.
+	if longestMatchPrefixLength != -1 {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("Found a fuzzy match for %q in \"paths\"", longestMatch.prefix+"*"+longestMatch.suffix))
+		}
+
+		for _, originalPath := range longestMatch.originalPaths {
+			// Swap out the "*" in the original path for whatever the "*" matched
+			matchedText := path[len(longestMatch.prefix) : len(path)-len(longestMatch.suffix)]
+			originalPath := strings.Replace(originalPath.Text, "*", matchedText, 1)
+
+			// Ignore ".d.ts" files because this rule is obviously only here for type checking
+			if hasCaseInsensitiveSuffix(originalPath, ".d.ts") {
+				if r.debugLogs != nil {
+					r.debugLogs.addNote(fmt.Sprintf("Ignoring substitution %q because it ends in \".d.ts\"", originalPath))
+				}
+				continue
+			}
+
+			// Load the original path relative to the "baseUrl" from tsconfig.json
+			absoluteOriginalPath := originalPath
+			if !r.fs.IsAbs(originalPath) {
+				absoluteOriginalPath = r.fs.Join(absBaseURL, originalPath)
+			}
+			if absolute, ok, diffCase := r.loadAsFileOrDirectory(absoluteOriginalPath); ok {
+				return absolute, true, diffCase
+			}
+		}
+	}
+
+	return PathPair{}, false, nil
+}
+
+func (r resolverQuery) loadPackageImports(importPath string, dirInfoPackageJSON *dirInfo) (PathPair, bool, *fs.DifferentCase, *SideEffectsData) {
+	packageJSON := dirInfoPackageJSON.packageJSON
+
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Looking for %q in \"imports\" map in %q", importPath, packageJSON.source.KeyPath.Text))
+		r.debugLogs.increaseIndent()
+		defer r.debugLogs.decreaseIndent()
+	}
+
+	// Filter out invalid module specifiers now where we have more information for
+	// a better error message instead of later when we're inside the algorithm
+	if importPath == "#" || strings.HasPrefix(importPath, "#/") {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("The path %q must not equal \"#\" and must not start with \"#/\".", importPath))
+		}
+		tracker := logger.MakeLineColumnTracker(&packageJSON.source)
+		r.debugMeta.notes = append(r.debugMeta.notes, tracker.MsgData(packageJSON.importsMap.root.firstToken,
+			fmt.Sprintf("This \"imports\" map was ignored because the module specifier %q is invalid:", importPath)))
+		return PathPair{}, false, nil, nil
+	}
+
+	// The condition set is determined by the kind of import
+	conditions := r.esmConditionsDefault
+	switch r.kind {
+	case ast.ImportStmt, ast.ImportDynamic:
+		conditions = r.esmConditionsImport
+	case ast.ImportRequire, ast.ImportRequireResolve:
+		conditions = r.esmConditionsRequire
+	}
+
+	resolvedPath, status, debug := r.esmPackageImportsResolve(importPath, packageJSON.importsMap.root, conditions)
+	resolvedPath, status, debug = r.esmHandlePostConditions(resolvedPath, status, debug)
+
+	if status == pjStatusPackageResolve {
+		if pathPair, ok, sideEffects := r.checkForBuiltInNodeModules(resolvedPath); ok {
+			return pathPair, true, nil, sideEffects
+		}
+
+		// The import path was remapped via "imports" to another import path
+		// that now needs to be resolved too. Set "forbidImports" to true
+		// so we don't try to resolve "imports" again and end up in a loop.
+		absolute, ok, diffCase, sideEffects := r.loadNodeModules(resolvedPath, dirInfoPackageJSON, true /* forbidImports */)
+		if !ok {
+			tracker := logger.MakeLineColumnTracker(&packageJSON.source)
+			r.debugMeta.notes = append(
+				[]logger.MsgData{tracker.MsgData(debug.token,
+					fmt.Sprintf("The remapped path %q could not be resolved:", resolvedPath))},
+				r.debugMeta.notes...)
+		}
+		return absolute, ok, diffCase, sideEffects
+	}
+
+	absolute, ok, diffCase := r.finalizeImportsExportsResult(
+		finalizeImportsExportsNormal,
+		dirInfoPackageJSON.absPath, conditions, *packageJSON.importsMap, packageJSON,
+		resolvedPath, status, debug,
+		"", "", "",
+	)
+	return absolute, ok, diffCase, nil
+}
+
+func (r resolverQuery) esmResolveAlgorithm(
+	kind finalizeImportsExportsKind,
+	esmPackageName string,
+	esmPackageSubpath string,
+	packageJSON *packageJSON,
+	absPkgPath string,
+	absPath string,
+) (PathPair, bool, *fs.DifferentCase) {
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Looking for %q in \"exports\" map in %q", esmPackageSubpath, packageJSON.source.KeyPath.Text))
+		r.debugLogs.increaseIndent()
+		defer r.debugLogs.decreaseIndent()
+	}
+
+	// The condition set is determined by the kind of import
+	conditions := r.esmConditionsDefault
+	switch r.kind {
+	case ast.ImportStmt, ast.ImportDynamic:
+		conditions = r.esmConditionsImport
+	case ast.ImportRequire, ast.ImportRequireResolve:
+		conditions = r.esmConditionsRequire
+	case ast.ImportEntryPoint:
+		// Treat entry points as imports instead of requires for consistency with
+		// Webpack and Rollup. More information:
+		//
+		// * https://github.com/evanw/esbuild/issues/1956
+		// * https://github.com/nodejs/node/issues/41686
+		// * https://github.com/evanw/entry-point-resolve-test
+		//
+		conditions = r.esmConditionsImport
+	}
+
+	// Resolve against the path "/", then join it with the absolute
+	// directory path. This is done because ESM package resolution uses
+	// URLs while our path resolution uses file system paths. We don't
+	// want problems due to Windows paths, which are very unlike URL
+	// paths. We also want to avoid any "%" characters in the absolute
+	// directory path accidentally being interpreted as URL escapes.
+	resolvedPath, status, debug := r.esmPackageExportsResolve("/", esmPackageSubpath, packageJSON.exportsMap.root, conditions)
+	resolvedPath, status, debug = r.esmHandlePostConditions(resolvedPath, status, debug)
+
+	return r.finalizeImportsExportsResult(
+		kind,
+		absPkgPath, conditions, *packageJSON.exportsMap, packageJSON,
+		resolvedPath, status, debug,
+		esmPackageName, esmPackageSubpath, absPath,
+	)
+}
+
+func (r resolverQuery) loadNodeModules(importPath string, dirInfo *dirInfo, forbidImports bool) (PathPair, bool, *fs.DifferentCase, *SideEffectsData) {
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Searching for %q in \"node_modules\" directories starting from %q", importPath, dirInfo.absPath))
+		r.debugLogs.increaseIndent()
+		defer r.debugLogs.decreaseIndent()
+	}
+
+	// First, check path overrides from the nearest enclosing TypeScript "tsconfig.json" file
+	if tsConfigJSON := r.tsConfigForDir(dirInfo); tsConfigJSON != nil {
+		// Try path substitutions first
+		if tsConfigJSON.Paths != nil {
+			if absolute, ok, diffCase := r.matchTSConfigPaths(tsConfigJSON, importPath); ok {
+				return absolute, true, diffCase, nil
+			}
+		}
+
+		// Try looking up the path relative to the base URL
+		if tsConfigJSON.BaseURL != nil {
+			basePath := r.fs.Join(*tsConfigJSON.BaseURL, importPath)
+			if absolute, ok, diffCase := r.loadAsFileOrDirectory(basePath); ok {
+				return absolute, true, diffCase, nil
+			}
+		}
+	}
+
+	// Find the parent directory with the "package.json" file
+	dirInfoPackageJSON := dirInfo
+	for dirInfoPackageJSON != nil && dirInfoPackageJSON.packageJSON == nil {
+		dirInfoPackageJSON = dirInfoPackageJSON.parent
+	}
+
+	// Check for subpath imports: https://nodejs.org/api/packages.html#subpath-imports
+	if dirInfoPackageJSON != nil && strings.HasPrefix(importPath, "#") && !forbidImports && dirInfoPackageJSON.packageJSON.importsMap != nil {
+		return r.loadPackageImports(importPath, dirInfoPackageJSON)
+	}
+
+	// "import 'pkg'" when all packages are external (vs. "import './pkg'")
+	if r.options.ExternalPackages && IsPackagePath(importPath) {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote("Marking this path as external because it's a package path")
+		}
+		return PathPair{Primary: logger.Path{Text: importPath}, IsExternal: true}, true, nil, nil
+	}
+
+	// If Yarn PnP is active, use it to find the package
+	if r.pnpManifest != nil {
+		if result := r.resolveToUnqualified(importPath, dirInfo.absPath, r.pnpManifest); result.status.isError() {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote("The Yarn PnP path resolution algorithm returned an error")
+			}
+
+			// Try to provide more information about this error if it's available
+			switch result.status {
+			case pnpErrorDependencyNotFound:
+				r.debugMeta.notes = []logger.MsgData{r.pnpManifest.tracker.MsgData(result.errorRange,
+					fmt.Sprintf("The Yarn Plug'n'Play manifest forbids importing %q here because it's not listed as a dependency of this package:", result.errorIdent))}
+
+			case pnpErrorUnfulfilledPeerDependency:
+				r.debugMeta.notes = []logger.MsgData{r.pnpManifest.tracker.MsgData(result.errorRange,
+					fmt.Sprintf("The Yarn Plug'n'Play manifest says this package has a peer dependency on %q, but the package %q has not been installed:", result.errorIdent, result.errorIdent))}
+			}
+
+			return PathPair{}, false, nil, nil
+		} else if result.status == pnpSuccess {
+			absPath := r.fs.Join(result.pkgDirPath, result.pkgSubpath)
+
+			// If Yarn PnP path resolution succeeded, run a custom abbreviated
+			// version of node's module resolution algorithm. The Yarn PnP
+			// specification says to use node's module resolution algorithm verbatim
+			// but that isn't what Yarn actually does. See this for more info:
+			// https://github.com/evanw/esbuild/issues/2473#issuecomment-1216774461
+			if pkgDirInfo := r.dirInfoCached(result.pkgDirPath); pkgDirInfo != nil {
+				// Check the "exports" map
+				if packageJSON := pkgDirInfo.packageJSON; packageJSON != nil && packageJSON.exportsMap != nil {
+					absolute, ok, diffCase := r.esmResolveAlgorithm(finalizeImportsExportsNormal, result.pkgIdent, "."+result.pkgSubpath, packageJSON, pkgDirInfo.absPath, absPath)
+					return absolute, ok, diffCase, nil
+				}
+
+				// Check the "browser" map
+				if remapped, ok := r.checkBrowserMap(pkgDirInfo, absPath, absolutePathKind); ok {
+					if remapped == nil {
+						return PathPair{Primary: logger.Path{Text: absPath, Namespace: "file", Flags: logger.PathDisabled}}, true, nil, nil
+					}
+					if remappedResult, ok, diffCase, sideEffects := r.resolveWithoutRemapping(pkgDirInfo.enclosingBrowserScope, *remapped); ok {
+						return remappedResult, true, diffCase, sideEffects
+					}
+				}
+
+				if absolute, ok, diffCase := r.loadAsFileOrDirectory(absPath); ok {
+					return absolute, true, diffCase, nil
+				}
+			}
+
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("Failed to resolve %q to a file", absPath))
+			}
+			return PathPair{}, false, nil, nil
+		}
+	}
+
+	// Try to parse the package name using node's ESM-specific rules
+	esmPackageName, esmPackageSubpath, esmOK := esmParsePackageName(importPath)
+	if r.debugLogs != nil && esmOK {
+		r.debugLogs.addNote(fmt.Sprintf("Parsed package name %q and package subpath %q", esmPackageName, esmPackageSubpath))
+	}
+
+	// Check for self-references
+	if dirInfoPackageJSON != nil {
+		if packageJSON := dirInfoPackageJSON.packageJSON; packageJSON.name == esmPackageName && packageJSON.exportsMap != nil {
+			absolute, ok, diffCase := r.esmResolveAlgorithm(finalizeImportsExportsNormal, esmPackageName, esmPackageSubpath, packageJSON,
+				dirInfoPackageJSON.absPath, r.fs.Join(dirInfoPackageJSON.absPath, esmPackageSubpath))
+			return absolute, ok, diffCase, nil
+		}
+	}
+
+	// Common package resolution logic shared between "node_modules" and "NODE_PATHS"
+	tryToResolvePackage := func(absDir string) (PathPair, bool, *fs.DifferentCase, *SideEffectsData, bool) {
+		absPath := r.fs.Join(absDir, importPath)
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("Checking for a package in the directory %q", absPath))
+		}
+
+		// Try node's new package resolution rules
+		if esmOK {
+			absPkgPath := r.fs.Join(absDir, esmPackageName)
+			if pkgDirInfo := r.dirInfoCached(absPkgPath); pkgDirInfo != nil {
+				// Check the "exports" map
+				if packageJSON := pkgDirInfo.packageJSON; packageJSON != nil && packageJSON.exportsMap != nil {
+					absolute, ok, diffCase := r.esmResolveAlgorithm(finalizeImportsExportsNormal, esmPackageName, esmPackageSubpath, packageJSON, absPkgPath, absPath)
+					return absolute, ok, diffCase, nil, true
+				}
+
+				// Check the "browser" map
+				if remapped, ok := r.checkBrowserMap(pkgDirInfo, absPath, absolutePathKind); ok {
+					if remapped == nil {
+						return PathPair{Primary: logger.Path{Text: absPath, Namespace: "file", Flags: logger.PathDisabled}}, true, nil, nil, true
+					}
+					if remappedResult, ok, diffCase, sideEffects := r.resolveWithoutRemapping(pkgDirInfo.enclosingBrowserScope, *remapped); ok {
+						return remappedResult, true, diffCase, sideEffects, true
+					}
+				}
+			}
+		}
+
+		// Try node's old package resolution rules
+		if absolute, ok, diffCase := r.loadAsFileOrDirectory(absPath); ok {
+			return absolute, true, diffCase, nil, true
+		}
+
+		return PathPair{}, false, nil, nil, false
+	}
+
+	// Then check for the package in any enclosing "node_modules" directories
+	for {
+		// Skip directories that are themselves called "node_modules", since we
+		// don't ever want to search for "node_modules/node_modules"
+		if dirInfo.hasNodeModules {
+			if absolute, ok, diffCase, sideEffects, shouldStop := tryToResolvePackage(r.fs.Join(dirInfo.absPath, "node_modules")); shouldStop {
+				return absolute, ok, diffCase, sideEffects
+			}
+		}
+
+		// Go to the parent directory, stopping at the file system root
+		dirInfo = dirInfo.parent
+		if dirInfo == nil {
+			break
+		}
+	}
+
+	// Then check the global "NODE_PATH" environment variable. It has been
+	// clarified that this step comes last after searching for "node_modules"
+	// directories: https://github.com/nodejs/node/issues/38128.
+	for _, absDir := range r.options.AbsNodePaths {
+		if absolute, ok, diffCase, sideEffects, shouldStop := tryToResolvePackage(absDir); shouldStop {
+			return absolute, ok, diffCase, sideEffects
+		}
+	}
+
+	return PathPair{}, false, nil, nil
+}
+
+func (r resolverQuery) checkForBuiltInNodeModules(importPath string) (PathPair, bool, *SideEffectsData) {
+	// "import fs from 'fs'"
+	if r.options.Platform == config.PlatformNode && BuiltInNodeModules[importPath] {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote("Marking this path as implicitly external due to it being a node built-in")
+		}
+
+		r.flushDebugLogs(flushDueToSuccess)
+		return PathPair{Primary: logger.Path{Text: importPath}, IsExternal: true},
+			true,
+			&SideEffectsData{} // Mark this with "sideEffects: false"
+	}
+
+	// "import fs from 'node:fs'"
+	// "require('node:fs')"
+	if r.options.Platform == config.PlatformNode && strings.HasPrefix(importPath, "node:") {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote("Marking this path as implicitly external due to the \"node:\" prefix")
+		}
+
+		// If this is a known node built-in module, mark it with "sideEffects: false"
+		var sideEffects *SideEffectsData
+		if BuiltInNodeModules[strings.TrimPrefix(importPath, "node:")] {
+			sideEffects = &SideEffectsData{}
+		}
+
+		// Check whether the path will end up as "import" or "require"
+		convertImportToRequire := !r.options.OutputFormat.KeepESMImportExportSyntax()
+		isImport := !convertImportToRequire && (r.kind == ast.ImportStmt || r.kind == ast.ImportDynamic)
+		isRequire := r.kind == ast.ImportRequire || r.kind == ast.ImportRequireResolve ||
+			(convertImportToRequire && (r.kind == ast.ImportStmt || r.kind == ast.ImportDynamic))
+
+		// Check for support with "import"
+		if isImport && r.options.UnsupportedJSFeatures.Has(compat.NodeColonPrefixImport) {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote("Removing the \"node:\" prefix because the target environment doesn't support it with \"import\" statements")
+			}
+
+			// Automatically strip the prefix if it's not supported
+			importPath = importPath[5:]
+		}
+
+		// Check for support with "require"
+		if isRequire && r.options.UnsupportedJSFeatures.Has(compat.NodeColonPrefixRequire) {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote("Removing the \"node:\" prefix because the target environment doesn't support it with \"require\" calls")
+			}
+
+			// Automatically strip the prefix if it's not supported
+			importPath = importPath[5:]
+		}
+
+		r.flushDebugLogs(flushDueToSuccess)
+		return PathPair{Primary: logger.Path{Text: importPath}, IsExternal: true}, true, sideEffects
+	}
+
+	return PathPair{}, false, nil
+}
+
+type finalizeImportsExportsKind uint8
+
+const (
+	finalizeImportsExportsNormal finalizeImportsExportsKind = iota
+	finalizeImportsExportsYarnPnPTSConfigExtends
+)
+
+func (r resolverQuery) finalizeImportsExportsResult(
+	kind finalizeImportsExportsKind,
+	absDirPath string,
+	conditions map[string]bool,
+	importExportMap pjMap,
+	packageJSON *packageJSON,
+
+	// Resolution results
+	resolvedPath string,
+	status pjStatus,
+	debug pjDebug,
+
+	// Only for exports
+	esmPackageName string,
+	esmPackageSubpath string,
+	absImportPath string,
+) (PathPair, bool, *fs.DifferentCase) {
+	missingSuffix := ""
+
+	if (status == pjStatusExact || status == pjStatusExactEndsWithStar || status == pjStatusInexact) && strings.HasPrefix(resolvedPath, "/") {
+		absResolvedPath := r.fs.Join(absDirPath, resolvedPath)
+
+		switch status {
+		case pjStatusExact, pjStatusExactEndsWithStar:
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("The resolved path %q is exact", absResolvedPath))
+			}
+
+			// Avoid calling "dirInfoCached" recursively for "tsconfig.json" extends with Yarn PnP
+			if kind == finalizeImportsExportsYarnPnPTSConfigExtends {
+				if r.debugLogs != nil {
+					r.debugLogs.addNote(fmt.Sprintf("Resolved to %q", absResolvedPath))
+				}
+				return PathPair{Primary: logger.Path{Text: absResolvedPath, Namespace: "file"}}, true, nil
+			}
+
+			resolvedDirInfo := r.dirInfoCached(r.fs.Dir(absResolvedPath))
+			base := r.fs.Base(absResolvedPath)
+			extensionOrder := r.options.ExtensionOrder
+			if r.kind.MustResolveToCSS() {
+				extensionOrder = r.cssExtensionOrder
+			}
+
+			if resolvedDirInfo == nil {
+				status = pjStatusModuleNotFound
+			} else {
+				entry, diffCase := resolvedDirInfo.entries.Get(base)
+
+				// TypeScript-specific behavior: try rewriting ".js" to ".ts"
+				if entry == nil {
+					for old, exts := range rewrittenFileExtensions {
+						if !strings.HasSuffix(base, old) {
+							continue
+						}
+						lastDot := strings.LastIndexByte(base, '.')
+						for _, ext := range exts {
+							baseWithExt := base[:lastDot] + ext
+							entry, diffCase = resolvedDirInfo.entries.Get(baseWithExt)
+							if entry != nil {
+								absResolvedPath = r.fs.Join(resolvedDirInfo.absPath, baseWithExt)
+								break
+							}
+						}
+						break
+					}
+				}
+
+				if entry == nil {
+					endsWithStar := status == pjStatusExactEndsWithStar
+					status = pjStatusModuleNotFound
+
+					// Try to have a friendly error message if people forget the extension
+					if endsWithStar {
+						for _, ext := range extensionOrder {
+							if entry, _ := resolvedDirInfo.entries.Get(base + ext); entry != nil {
+								if r.debugLogs != nil {
+									r.debugLogs.addNote(fmt.Sprintf("The import %q is missing the extension %q", path.Join(esmPackageName, esmPackageSubpath), ext))
+								}
+								status = pjStatusModuleNotFoundMissingExtension
+								missingSuffix = ext
+								break
+							}
+						}
+					}
+				} else if kind := entry.Kind(r.fs); kind == fs.DirEntry {
+					if r.debugLogs != nil {
+						r.debugLogs.addNote(fmt.Sprintf("The path %q is a directory, which is not allowed", absResolvedPath))
+					}
+					endsWithStar := status == pjStatusExactEndsWithStar
+					status = pjStatusUnsupportedDirectoryImport
+
+					// Try to have a friendly error message if people forget the "/index.js" suffix
+					if endsWithStar {
+						if resolvedDirInfo := r.dirInfoCached(absResolvedPath); resolvedDirInfo != nil {
+							for _, ext := range extensionOrder {
+								base := "index" + ext
+								if entry, _ := resolvedDirInfo.entries.Get(base); entry != nil && entry.Kind(r.fs) == fs.FileEntry {
+									status = pjStatusUnsupportedDirectoryImportMissingIndex
+									missingSuffix = "/" + base
+									if r.debugLogs != nil {
+										r.debugLogs.addNote(fmt.Sprintf("The import %q is missing the suffix %q", path.Join(esmPackageName, esmPackageSubpath), missingSuffix))
+									}
+									break
+								}
+							}
+						}
+					}
+				} else if kind != fs.FileEntry {
+					status = pjStatusModuleNotFound
+				} else {
+					if r.debugLogs != nil {
+						r.debugLogs.addNote(fmt.Sprintf("Resolved to %q", absResolvedPath))
+					}
+					return PathPair{Primary: logger.Path{Text: absResolvedPath, Namespace: "file"}}, true, diffCase
+				}
+			}
+
+		case pjStatusInexact:
+			// If this was resolved against an expansion key ending in a "/"
+			// instead of a "*", we need to try CommonJS-style implicit
+			// extension and/or directory detection.
+			if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("The resolved path %q is inexact", absResolvedPath))
+			}
+			if absolute, ok, diffCase := r.loadAsFileOrDirectory(absResolvedPath); ok {
+				return absolute, true, diffCase
+			}
+			status = pjStatusModuleNotFound
+		}
+	}
+
+	if strings.HasPrefix(resolvedPath, "/") {
+		resolvedPath = "." + resolvedPath
+	}
+
+	// Provide additional details about the failure to help with debugging
+	tracker := logger.MakeLineColumnTracker(&packageJSON.source)
+	switch status {
+	case pjStatusInvalidModuleSpecifier:
+		r.debugMeta.notes = []logger.MsgData{tracker.MsgData(debug.token,
+			fmt.Sprintf("The module specifier %q is invalid%s:", resolvedPath, debug.invalidBecause))}
+
+	case pjStatusInvalidPackageConfiguration:
+		r.debugMeta.notes = []logger.MsgData{tracker.MsgData(debug.token,
+			"The package configuration has an invalid value here:")}
+
+	case pjStatusInvalidPackageTarget:
+		why := fmt.Sprintf("The package target %q is invalid%s:", resolvedPath, debug.invalidBecause)
+		if resolvedPath == "" {
+			// "PACKAGE_TARGET_RESOLVE" is specified to throw an "Invalid
+			// Package Target" error for what is actually an invalid package
+			// configuration error
+			why = "The package configuration has an invalid value here:"
+		}
+		r.debugMeta.notes = []logger.MsgData{tracker.MsgData(debug.token, why)}
+
+	case pjStatusPackagePathNotExported:
+		if debug.isBecauseOfNullLiteral {
+			r.debugMeta.notes = []logger.MsgData{tracker.MsgData(debug.token,
+				fmt.Sprintf("The path %q cannot be imported from package %q because it was explicitly disabled by the package author here:", esmPackageSubpath, esmPackageName))}
+			break
+		}
+
+		r.debugMeta.notes = []logger.MsgData{tracker.MsgData(debug.token,
+			fmt.Sprintf("The path %q is not exported by package %q:", esmPackageSubpath, esmPackageName))}
+
+		// If this fails, try to resolve it using the old algorithm
+		if absolute, ok, _ := r.loadAsFileOrDirectory(absImportPath); ok && absolute.Primary.Namespace == "file" {
+			if relPath, ok := r.fs.Rel(absDirPath, absolute.Primary.Text); ok {
+				query := "." + path.Join("/", strings.ReplaceAll(relPath, "\\", "/"))
+
+				// If that succeeds, try to do a reverse lookup using the
+				// "exports" map for the currently-active set of conditions
+				if ok, subpath, token := r.esmPackageExportsReverseResolve(
+					query, importExportMap.root, conditions); ok {
+					r.debugMeta.notes = append(r.debugMeta.notes, tracker.MsgData(token,
+						fmt.Sprintf("The file %q is exported at path %q:", query, subpath)))
+
+					// Provide an inline suggestion message with the correct import path
+					actualImportPath := path.Join(esmPackageName, subpath)
+					r.debugMeta.suggestionText = string(helpers.QuoteForJSON(actualImportPath, false))
+					r.debugMeta.suggestionMessage = fmt.Sprintf("Import from %q to get the file %q:",
+						actualImportPath, PrettyPath(r.fs, absolute.Primary))
+				}
+			}
+		}
+
+	case pjStatusPackageImportNotDefined:
+		r.debugMeta.notes = []logger.MsgData{tracker.MsgData(debug.token,
+			fmt.Sprintf("The package import %q is not defined in this \"imports\" map:", resolvedPath))}
+
+	case pjStatusModuleNotFound, pjStatusModuleNotFoundMissingExtension:
+		r.debugMeta.notes = []logger.MsgData{tracker.MsgData(debug.token,
+			fmt.Sprintf("The module %q was not found on the file system:", resolvedPath))}
+
+		// Provide an inline suggestion message with the correct import path
+		if status == pjStatusModuleNotFoundMissingExtension {
+			actualImportPath := path.Join(esmPackageName, esmPackageSubpath+missingSuffix)
+			r.debugMeta.suggestionRange = suggestionRangeEnd
+			r.debugMeta.suggestionText = missingSuffix
+			r.debugMeta.suggestionMessage = fmt.Sprintf("Import from %q to get the file %q:",
+				actualImportPath, PrettyPath(r.fs, logger.Path{Text: r.fs.Join(absDirPath, resolvedPath+missingSuffix), Namespace: "file"}))
+		}
+
+	case pjStatusUnsupportedDirectoryImport, pjStatusUnsupportedDirectoryImportMissingIndex:
+		r.debugMeta.notes = []logger.MsgData{
+			tracker.MsgData(debug.token, fmt.Sprintf("Importing the directory %q is forbidden by this package:", resolvedPath)),
+			tracker.MsgData(packageJSON.source.RangeOfString(importExportMap.propertyKeyLoc),
+				fmt.Sprintf("The presence of %q here makes importing a directory forbidden:", importExportMap.propertyKey)),
+		}
+
+		// Provide an inline suggestion message with the correct import path
+		if status == pjStatusUnsupportedDirectoryImportMissingIndex {
+			actualImportPath := path.Join(esmPackageName, esmPackageSubpath+missingSuffix)
+			r.debugMeta.suggestionRange = suggestionRangeEnd
+			r.debugMeta.suggestionText = missingSuffix
+			r.debugMeta.suggestionMessage = fmt.Sprintf("Import from %q to get the file %q:",
+				actualImportPath, PrettyPath(r.fs, logger.Path{Text: r.fs.Join(absDirPath, resolvedPath+missingSuffix), Namespace: "file"}))
+		}
+
+	case pjStatusUndefinedNoConditionsMatch:
+		keys := make([]string, 0, len(conditions))
+		for key := range conditions {
+			keys = append(keys, key)
+		}
+		sort.Strings(keys)
+
+		unmatchedConditions := make([]string, len(debug.unmatchedConditions))
+		for i, key := range debug.unmatchedConditions {
+			unmatchedConditions[i] = key.Text
+		}
+
+		r.debugMeta.notes = []logger.MsgData{
+			tracker.MsgData(importExportMap.root.firstToken,
+				fmt.Sprintf("The path %q is not currently exported by package %q:",
+					esmPackageSubpath, esmPackageName)),
+
+			tracker.MsgData(debug.token,
+				fmt.Sprintf("None of the conditions in the package definition (%s) match any of the currently active conditions (%s):",
+					helpers.StringArrayToQuotedCommaSeparatedString(unmatchedConditions),
+					helpers.StringArrayToQuotedCommaSeparatedString(keys),
+				)),
+		}
+
+		didSuggestEnablingCondition := false
+		for _, key := range debug.unmatchedConditions {
+			switch key.Text {
+			case "import":
+				if r.kind == ast.ImportRequire || r.kind == ast.ImportRequireResolve {
+					r.debugMeta.suggestionMessage = "Consider using an \"import\" statement to import this file, " +
+						"which will work because the \"import\" condition is supported by this package:"
+				}
+
+			case "require":
+				if r.kind == ast.ImportStmt || r.kind == ast.ImportDynamic {
+					r.debugMeta.suggestionMessage = "Consider using a \"require()\" call to import this file, " +
+						"which will work because the \"require\" condition is supported by this package:"
+				}
+
+			default:
+				// Note: Don't suggest the adding the "types" condition because
+				// TypeScript uses that for type definitions, which are not
+				// intended to be included in a bundle as executable code
+				if !didSuggestEnablingCondition && key.Text != "types" {
+					var how string
+					switch logger.API {
+					case logger.CLIAPI:
+						how = fmt.Sprintf("\"--conditions=%s\"", key.Text)
+					case logger.JSAPI:
+						how = fmt.Sprintf("\"conditions: ['%s']\"", key.Text)
+					case logger.GoAPI:
+						how = fmt.Sprintf("'Conditions: []string{%q}'", key.Text)
+					}
+					r.debugMeta.notes = append(r.debugMeta.notes, tracker.MsgData(key.Range,
+						fmt.Sprintf("Consider enabling the %q condition if this package expects it to be enabled. "+
+							"You can use %s to do that:", key.Text, how)))
+					didSuggestEnablingCondition = true
+				}
+			}
+		}
+	}
+
+	return PathPair{}, false, nil
+}
+
+// Package paths are loaded from a "node_modules" directory. Non-package paths
+// are relative or absolute paths.
+func IsPackagePath(path string) bool {
+	return !strings.HasPrefix(path, "/") && !strings.HasPrefix(path, "./") &&
+		!strings.HasPrefix(path, "../") && path != "." && path != ".."
+}
+
+// This list can be obtained with the following command:
+//
+//	node --experimental-wasi-unstable-preview1 -p "[...require('module').builtinModules].join('\n')"
+//
+// Be sure to use the *LATEST* version of node when updating this list!
+var BuiltInNodeModules = map[string]bool{
+	"_http_agent":         true,
+	"_http_client":        true,
+	"_http_common":        true,
+	"_http_incoming":      true,
+	"_http_outgoing":      true,
+	"_http_server":        true,
+	"_stream_duplex":      true,
+	"_stream_passthrough": true,
+	"_stream_readable":    true,
+	"_stream_transform":   true,
+	"_stream_wrap":        true,
+	"_stream_writable":    true,
+	"_tls_common":         true,
+	"_tls_wrap":           true,
+	"assert":              true,
+	"assert/strict":       true,
+	"async_hooks":         true,
+	"buffer":              true,
+	"child_process":       true,
+	"cluster":             true,
+	"console":             true,
+	"constants":           true,
+	"crypto":              true,
+	"dgram":               true,
+	"diagnostics_channel": true,
+	"dns":                 true,
+	"dns/promises":        true,
+	"domain":              true,
+	"events":              true,
+	"fs":                  true,
+	"fs/promises":         true,
+	"http":                true,
+	"http2":               true,
+	"https":               true,
+	"inspector":           true,
+	"module":              true,
+	"net":                 true,
+	"os":                  true,
+	"path":                true,
+	"path/posix":          true,
+	"path/win32":          true,
+	"perf_hooks":          true,
+	"process":             true,
+	"punycode":            true,
+	"querystring":         true,
+	"readline":            true,
+	"repl":                true,
+	"stream":              true,
+	"stream/consumers":    true,
+	"stream/promises":     true,
+	"stream/web":          true,
+	"string_decoder":      true,
+	"sys":                 true,
+	"timers":              true,
+	"timers/promises":     true,
+	"tls":                 true,
+	"trace_events":        true,
+	"tty":                 true,
+	"url":                 true,
+	"util":                true,
+	"util/types":          true,
+	"v8":                  true,
+	"vm":                  true,
+	"wasi":                true,
+	"worker_threads":      true,
+	"zlib":                true,
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/resolver/testExpectations.json b/source/vendor/github.com/evanw/esbuild/internal/resolver/testExpectations.json
new file mode 100644
index 0000000..56cd4a1
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/resolver/testExpectations.json
@@ -0,0 +1,311 @@
+[{
+  "manifest": {
+    "__info": [],
+    "dependencyTreeRoots": [{
+      "name": "root",
+      "reference": "workspace:."
+    }],
+    "ignorePatternData": null,
+    "enableTopLevelFallback": false,
+    "fallbackPool": [],
+    "fallbackExclusionList": [],
+    "packageRegistryData": [
+      [null, [
+        [null, {
+          "packageLocation": "./",
+          "packageDependencies": [["test", "npm:1.0.0"]],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["root", [
+        ["workspace:.", {
+          "packageLocation": "./",
+          "packageDependencies": [["test", "npm:1.0.0"]],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["workspace-alias-dependency", [
+        ["workspace:workspace-alias-dependency", {
+          "packageLocation": "./workspace-alias-dependency/",
+          "packageDependencies": [["alias", ["test", "npm:1.0.0"]]],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["workspace-self-dependency", [
+        ["workspace:workspace-self-dependency", {
+          "packageLocation": "./workspace-self-dependency/",
+          "packageDependencies": [["workspace-self-dependency", "workspace:workspace-self-dependency"]],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["workspace-unfulfilled-peer-dependency", [
+        ["workspace:workspace-unfulfilled-peer-dependency", {
+          "packageLocation": "./workspace-unfulfilled-peer-dependency/",
+          "packageDependencies": [["test", null]],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["longer", [
+        ["workspace:longer", {
+          "packageLocation": "./longer/",
+          "packageDependencies": [["test", "npm:2.0.0"]],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["long", [
+        ["workspace:long", {
+          "packageLocation": "./long/",
+          "packageDependencies": [["test", "npm:1.0.0"]],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["longerer", [
+        ["workspace:longerer", {
+          "packageLocation": "./longerer/",
+          "packageDependencies": [["test", "npm:3.0.0"]],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["test", [
+        ["npm:1.0.0", {
+          "packageLocation": "./test-1.0.0/",
+          "packageDependencies": [],
+          "linkType": "HARD"
+        }],
+        ["npm:2.0.0", {
+          "packageLocation": "./test-2.0.0/",
+          "packageDependencies": [],
+          "linkType": "HARD"
+        }],
+        ["npm:3.0.0", {
+          "packageLocation": "./test-3.0.0/",
+          "packageDependencies": [],
+          "linkType": "HARD"
+        }]
+      ]]
+    ]
+  },
+  "tests": [{
+    "it": "should allow a package to import one of its dependencies",
+    "imported": "test",
+    "importer": "/path/to/project/",
+    "expected": "/path/to/project/test-1.0.0/"
+  }, {
+    "it": "should allow a package to import itself, if specified in its own dependencies",
+    "imported": "workspace-self-dependency",
+    "importer": "/path/to/project/workspace-self-dependency/",
+    "expected": "/path/to/project/workspace-self-dependency/"
+  }, {
+    "it": "should allow a package to import an aliased dependency",
+    "imported": "alias",
+    "importer": "/path/to/project/workspace-alias-dependency/",
+    "expected": "/path/to/project/test-1.0.0/"
+  }, {
+    "it": "shouldn't allow a package to import something that isn't one of its dependencies",
+    "imported": "missing-dependency",
+    "importer": "/path/to/project/",
+    "expected": "error!"
+  }, {
+    "it": "shouldn't accidentally discard the trailing slash from the package locations",
+    "imported": "test",
+    "importer": "/path/to/project/long/",
+    "expected": "/path/to/project/test-1.0.0/"
+  }, {
+    "it": "should throw an exception when trying to access an unfulfilled peer dependency",
+    "imported": "test",
+    "importer": "/path/to/project/workspace-unfulfilled-peer-dependency/",
+    "expected": "error!"
+  }]
+}, {
+  "manifest": {
+    "__info": [],
+    "dependencyTreeRoots": [{
+      "name": "root",
+      "reference": "workspace:."
+    }],
+    "ignorePatternData": null,
+    "enableTopLevelFallback": true,
+    "fallbackPool": [
+      ["test-2", "npm:1.0.0"],
+      ["alias", ["test-1", "npm:1.0.0"]]
+    ],
+    "fallbackExclusionList": [[
+      "workspace-no-fallbacks",
+      ["workspace:workspace-no-fallbacks"]
+    ]],
+    "packageRegistryData": [
+      [null, [
+        [null, {
+          "packageLocation": "./",
+          "packageDependencies": [["test-1", "npm:1.0.0"]],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["root", [
+        ["workspace:.", {
+          "packageLocation": "./",
+          "packageDependencies": [["test-1", "npm:1.0.0"]],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["workspace-no-fallbacks", [
+        ["workspace:workspace-no-fallbacks", {
+          "packageLocation": "./workspace-no-fallbacks/",
+          "packageDependencies": [],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["workspace-with-fallbacks", [
+        ["workspace:workspace-with-fallbacks", {
+          "packageLocation": "./workspace-with-fallbacks/",
+          "packageDependencies": [],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["workspace-unfulfilled-peer-dependency", [
+        ["workspace:workspace-unfulfilled-peer-dependency", {
+          "packageLocation": "./workspace-unfulfilled-peer-dependency/",
+          "packageDependencies": [
+            ["test-1", null],
+            ["test-2", null]
+          ],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["test-1", [
+        ["npm:1.0.0", {
+          "packageLocation": "./test-1/",
+          "packageDependencies": [],
+          "linkType": "HARD"
+        }]
+      ]],
+      ["test-2", [
+        ["npm:1.0.0", {
+          "packageLocation": "./test-2/",
+          "packageDependencies": [],
+          "linkType": "HARD"
+        }]
+      ]]
+    ]
+  },
+  "tests": [{
+    "it": "should allow resolution coming from the fallback pool if enableTopLevelFallback is set to true",
+    "imported": "test-1",
+    "importer": "/path/to/project/",
+    "expected": "/path/to/project/test-1/"
+  }, {
+    "it": "should allow the fallback pool to contain aliases",
+    "imported": "alias",
+    "importer": "/path/to/project/",
+    "expected": "/path/to/project/test-1/"
+  }, {
+    "it": "shouldn't use the fallback pool when the importer package is listed in fallbackExclusionList",
+    "imported": "test-1",
+    "importer": "/path/to/project/workspace-no-fallbacks/",
+    "expected": "error!"
+  }, {
+    "it": "should implicitly use the top-level package dependencies as part of the fallback pool",
+    "imported": "test-2",
+    "importer": "/path/to/project/workspace-with-fallbacks/",
+    "expected": "/path/to/project/test-2/"
+  }, {
+    "it": "should throw an error if a resolution isn't in in the package dependencies, nor inside the fallback pool",
+    "imported": "test-3",
+    "importer": "/path/to/project/workspace-with-fallbacks/",
+    "expected": "error!"
+  }, {
+    "it": "should use the top-level fallback if a dependency is missing because of an unfulfilled peer dependency",
+    "imported": "test-1",
+    "importer": "/path/to/project/workspace-unfulfilled-peer-dependency/",
+    "expected": "/path/to/project/test-1/"
+  }, {
+    "it": "should use the fallback pool if a dependency is missing because of an unfulfilled peer dependency",
+    "imported": "test-2",
+    "importer": "/path/to/project/workspace-unfulfilled-peer-dependency/",
+    "expected": "/path/to/project/test-2/"
+  }]
+}, {
+  "manifest": {
+    "__info": [],
+    "dependencyTreeRoots": [{
+      "name": "root",
+      "reference": "workspace:."
+    }],
+    "ignorePatternData": null,
+    "enableTopLevelFallback": false,
+    "fallbackPool": [
+      ["test", "npm:1.0.0"]
+    ],
+    "fallbackExclusionList": [],
+    "packageRegistryData": [
+      [null, [
+        [null, {
+          "packageLocation": "./",
+          "packageDependencies": [],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["root", [
+        ["workspace:.", {
+          "packageLocation": "./",
+          "packageDependencies": [],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["test", [
+        ["npm:1.0.0", {
+          "packageLocation": "./test-1/",
+          "packageDependencies": [],
+          "linkType": "HARD"
+        }]
+      ]]
+    ]
+  },
+  "tests": [{
+    "it": "should ignore the fallback pool if enableTopLevelFallback is set to false",
+    "imported": "test",
+    "importer": "/path/to/project/",
+    "expected": "error!"
+  }]
+}, {
+  "manifest": {
+    "__info": [],
+    "dependencyTreeRoots": [{
+      "name": "root",
+      "reference": "workspace:."
+    }],
+    "ignorePatternData": "^not-a-workspace(/|$)",
+    "enableTopLevelFallback": false,
+    "fallbackPool": [],
+    "fallbackExclusionList": [],
+    "packageRegistryData": [
+      [null, [
+        [null, {
+          "packageLocation": "./",
+          "packageDependencies": [],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["root", [
+        ["workspace:.", {
+          "packageLocation": "./",
+          "packageDependencies": [["test", "npm:1.0.0"]],
+          "linkType": "SOFT"
+        }]
+      ]],
+      ["test", [
+        ["npm:1.0.0", {
+          "packageLocation": "./test/",
+          "packageDependencies": [],
+          "linkType": "HARD"
+        }]
+      ]]
+    ]
+  },
+  "tests": [{
+    "it": "shouldn't go through PnP when trying to resolve dependencies from packages covered by ignorePatternData",
+    "imported": "test",
+    "importer": "/path/to/project/not-a-workspace/",
+    "expected": "error!"
+  }]
+}]
diff --git a/source/vendor/github.com/evanw/esbuild/internal/resolver/tsconfig_json.go b/source/vendor/github.com/evanw/esbuild/internal/resolver/tsconfig_json.go
new file mode 100644
index 0000000..edfc775
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/resolver/tsconfig_json.go
@@ -0,0 +1,481 @@
+package resolver
+
+import (
+	"fmt"
+	"strings"
+
+	"github.com/evanw/esbuild/internal/cache"
+	"github.com/evanw/esbuild/internal/config"
+	"github.com/evanw/esbuild/internal/fs"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/js_lexer"
+	"github.com/evanw/esbuild/internal/js_parser"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type TSConfigJSON struct {
+	AbsPath string
+
+	// The absolute path of "compilerOptions.baseUrl"
+	BaseURL *string
+
+	// This is used if "Paths" is non-nil. It's equal to "BaseURL" except if
+	// "BaseURL" is missing, in which case it is as if "BaseURL" was ".". This
+	// is to implement the "paths without baseUrl" feature from TypeScript 4.1.
+	// More info: https://github.com/microsoft/TypeScript/issues/31869
+	BaseURLForPaths string
+
+	// The verbatim values of "compilerOptions.paths". The keys are patterns to
+	// match and the values are arrays of fallback paths to search. Each key and
+	// each fallback path can optionally have a single "*" wildcard character.
+	// If both the key and the value have a wildcard, the substring matched by
+	// the wildcard is substituted into the fallback path. The keys represent
+	// module-style path names and the fallback paths are relative to the
+	// "baseUrl" value in the "tsconfig.json" file.
+	Paths *TSConfigPaths
+
+	tsTargetKey    tsTargetKey
+	TSStrict       *config.TSAlwaysStrict
+	TSAlwaysStrict *config.TSAlwaysStrict
+	JSXSettings    config.TSConfigJSX
+	Settings       config.TSConfig
+}
+
+func (derived *TSConfigJSON) applyExtendedConfig(base TSConfigJSON) {
+	if base.tsTargetKey.Range.Len > 0 {
+		derived.tsTargetKey = base.tsTargetKey
+	}
+	if base.TSStrict != nil {
+		derived.TSStrict = base.TSStrict
+	}
+	if base.TSAlwaysStrict != nil {
+		derived.TSAlwaysStrict = base.TSAlwaysStrict
+	}
+	if base.BaseURL != nil {
+		derived.BaseURL = base.BaseURL
+	}
+	if base.Paths != nil {
+		derived.Paths = base.Paths
+		derived.BaseURLForPaths = base.BaseURLForPaths
+	}
+	derived.JSXSettings.ApplyExtendedConfig(base.JSXSettings)
+	derived.Settings.ApplyExtendedConfig(base.Settings)
+}
+
+func (config *TSConfigJSON) TSAlwaysStrictOrStrict() *config.TSAlwaysStrict {
+	if config.TSAlwaysStrict != nil {
+		return config.TSAlwaysStrict
+	}
+
+	// If "alwaysStrict" is absent, it defaults to "strict" instead
+	return config.TSStrict
+}
+
+// This information is only used for error messages
+type tsTargetKey struct {
+	LowerValue string
+	Source     logger.Source
+	Range      logger.Range
+}
+
+type TSConfigPath struct {
+	Text string
+	Loc  logger.Loc
+}
+
+type TSConfigPaths struct {
+	Map map[string][]TSConfigPath
+
+	// This may be different from the original "tsconfig.json" source if the
+	// "paths" value is from another file via an "extends" clause.
+	Source logger.Source
+}
+
+func ParseTSConfigJSON(
+	log logger.Log,
+	source logger.Source,
+	jsonCache *cache.JSONCache,
+	fs fs.FS,
+	fileDir string,
+	configDir string,
+	extends func(string, logger.Range) *TSConfigJSON,
+) *TSConfigJSON {
+	// Unfortunately "tsconfig.json" isn't actually JSON. It's some other
+	// format that appears to be defined by the implementation details of the
+	// TypeScript compiler.
+	//
+	// Attempt to parse it anyway by modifying the JSON parser, but just for
+	// these particular files. This is likely not a completely accurate
+	// emulation of what the TypeScript compiler does (e.g. string escape
+	// behavior may also be different).
+	json, ok := jsonCache.Parse(log, source, js_parser.JSONOptions{Flavor: js_lexer.TSConfigJSON})
+	if !ok {
+		return nil
+	}
+
+	var result TSConfigJSON
+	result.AbsPath = source.KeyPath.Text
+	tracker := logger.MakeLineColumnTracker(&source)
+
+	// Parse "extends"
+	if extends != nil {
+		if valueJSON, _, ok := getProperty(json, "extends"); ok {
+			if value, ok := getString(valueJSON); ok {
+				if base := extends(value, source.RangeOfString(valueJSON.Loc)); base != nil {
+					result.applyExtendedConfig(*base)
+				}
+			} else if array, ok := valueJSON.Data.(*js_ast.EArray); ok {
+				for _, item := range array.Items {
+					if str, ok := getString(item); ok {
+						if base := extends(str, source.RangeOfString(item.Loc)); base != nil {
+							result.applyExtendedConfig(*base)
+						}
+					}
+				}
+			}
+		}
+	}
+
+	// Parse "compilerOptions"
+	if compilerOptionsJSON, _, ok := getProperty(json, "compilerOptions"); ok {
+		// Parse "baseUrl"
+		if valueJSON, _, ok := getProperty(compilerOptionsJSON, "baseUrl"); ok {
+			if value, ok := getString(valueJSON); ok {
+				value = getSubstitutedPathWithConfigDirTemplate(fs, value, configDir)
+				if !fs.IsAbs(value) {
+					value = fs.Join(fileDir, value)
+				}
+				result.BaseURL = &value
+			}
+		}
+
+		// Parse "jsx"
+		if valueJSON, _, ok := getProperty(compilerOptionsJSON, "jsx"); ok {
+			if value, ok := getString(valueJSON); ok {
+				switch strings.ToLower(value) {
+				case "preserve":
+					result.JSXSettings.JSX = config.TSJSXPreserve
+				case "react-native":
+					result.JSXSettings.JSX = config.TSJSXReactNative
+				case "react":
+					result.JSXSettings.JSX = config.TSJSXReact
+				case "react-jsx":
+					result.JSXSettings.JSX = config.TSJSXReactJSX
+				case "react-jsxdev":
+					result.JSXSettings.JSX = config.TSJSXReactJSXDev
+				}
+			}
+		}
+
+		// Parse "jsxFactory"
+		if valueJSON, _, ok := getProperty(compilerOptionsJSON, "jsxFactory"); ok {
+			if value, ok := getString(valueJSON); ok {
+				result.JSXSettings.JSXFactory = parseMemberExpressionForJSX(log, &source, &tracker, valueJSON.Loc, value)
+			}
+		}
+
+		// Parse "jsxFragmentFactory"
+		if valueJSON, _, ok := getProperty(compilerOptionsJSON, "jsxFragmentFactory"); ok {
+			if value, ok := getString(valueJSON); ok {
+				result.JSXSettings.JSXFragmentFactory = parseMemberExpressionForJSX(log, &source, &tracker, valueJSON.Loc, value)
+			}
+		}
+
+		// Parse "jsxImportSource"
+		if valueJSON, _, ok := getProperty(compilerOptionsJSON, "jsxImportSource"); ok {
+			if value, ok := getString(valueJSON); ok {
+				result.JSXSettings.JSXImportSource = &value
+			}
+		}
+
+		// Parse "experimentalDecorators"
+		if valueJSON, _, ok := getProperty(compilerOptionsJSON, "experimentalDecorators"); ok {
+			if value, ok := getBool(valueJSON); ok {
+				if value {
+					result.Settings.ExperimentalDecorators = config.True
+				} else {
+					result.Settings.ExperimentalDecorators = config.False
+				}
+			}
+		}
+
+		// Parse "useDefineForClassFields"
+		if valueJSON, _, ok := getProperty(compilerOptionsJSON, "useDefineForClassFields"); ok {
+			if value, ok := getBool(valueJSON); ok {
+				if value {
+					result.Settings.UseDefineForClassFields = config.True
+				} else {
+					result.Settings.UseDefineForClassFields = config.False
+				}
+			}
+		}
+
+		// Parse "target"
+		if valueJSON, keyLoc, ok := getProperty(compilerOptionsJSON, "target"); ok {
+			if value, ok := getString(valueJSON); ok {
+				lowerValue := strings.ToLower(value)
+				ok := true
+
+				// See https://www.typescriptlang.org/tsconfig#target
+				switch lowerValue {
+				case "es3", "es5", "es6", "es2015", "es2016", "es2017", "es2018", "es2019", "es2020", "es2021":
+					result.Settings.Target = config.TSTargetBelowES2022
+				case "es2022", "es2023", "esnext":
+					result.Settings.Target = config.TSTargetAtOrAboveES2022
+				default:
+					ok = false
+					if !helpers.IsInsideNodeModules(source.KeyPath.Text) {
+						log.AddID(logger.MsgID_TSConfigJSON_InvalidTarget, logger.Warning, &tracker, source.RangeOfString(valueJSON.Loc),
+							fmt.Sprintf("Unrecognized target environment %q", value))
+					}
+				}
+
+				if ok {
+					result.tsTargetKey = tsTargetKey{
+						Source:     source,
+						Range:      source.RangeOfString(keyLoc),
+						LowerValue: lowerValue,
+					}
+				}
+			}
+		}
+
+		// Parse "strict"
+		if valueJSON, keyLoc, ok := getProperty(compilerOptionsJSON, "strict"); ok {
+			if value, ok := getBool(valueJSON); ok {
+				valueRange := js_lexer.RangeOfIdentifier(source, valueJSON.Loc)
+				result.TSStrict = &config.TSAlwaysStrict{
+					Name:   "strict",
+					Value:  value,
+					Source: source,
+					Range:  logger.Range{Loc: keyLoc, Len: valueRange.End() - keyLoc.Start},
+				}
+			}
+		}
+
+		// Parse "alwaysStrict"
+		if valueJSON, keyLoc, ok := getProperty(compilerOptionsJSON, "alwaysStrict"); ok {
+			if value, ok := getBool(valueJSON); ok {
+				valueRange := js_lexer.RangeOfIdentifier(source, valueJSON.Loc)
+				result.TSAlwaysStrict = &config.TSAlwaysStrict{
+					Name:   "alwaysStrict",
+					Value:  value,
+					Source: source,
+					Range:  logger.Range{Loc: keyLoc, Len: valueRange.End() - keyLoc.Start},
+				}
+			}
+		}
+
+		// Parse "importsNotUsedAsValues"
+		if valueJSON, _, ok := getProperty(compilerOptionsJSON, "importsNotUsedAsValues"); ok {
+			if value, ok := getString(valueJSON); ok {
+				switch value {
+				case "remove":
+					result.Settings.ImportsNotUsedAsValues = config.TSImportsNotUsedAsValues_Remove
+				case "preserve":
+					result.Settings.ImportsNotUsedAsValues = config.TSImportsNotUsedAsValues_Preserve
+				case "error":
+					result.Settings.ImportsNotUsedAsValues = config.TSImportsNotUsedAsValues_Error
+				default:
+					log.AddID(logger.MsgID_TSConfigJSON_InvalidImportsNotUsedAsValues, logger.Warning, &tracker, source.RangeOfString(valueJSON.Loc),
+						fmt.Sprintf("Invalid value %q for \"importsNotUsedAsValues\"", value))
+				}
+			}
+		}
+
+		// Parse "preserveValueImports"
+		if valueJSON, _, ok := getProperty(compilerOptionsJSON, "preserveValueImports"); ok {
+			if value, ok := getBool(valueJSON); ok {
+				if value {
+					result.Settings.PreserveValueImports = config.True
+				} else {
+					result.Settings.PreserveValueImports = config.False
+				}
+			}
+		}
+
+		// Parse "verbatimModuleSyntax"
+		if valueJSON, _, ok := getProperty(compilerOptionsJSON, "verbatimModuleSyntax"); ok {
+			if value, ok := getBool(valueJSON); ok {
+				if value {
+					result.Settings.VerbatimModuleSyntax = config.True
+				} else {
+					result.Settings.VerbatimModuleSyntax = config.False
+				}
+			}
+		}
+
+		// Parse "paths"
+		if valueJSON, _, ok := getProperty(compilerOptionsJSON, "paths"); ok {
+			if paths, ok := valueJSON.Data.(*js_ast.EObject); ok {
+				result.BaseURLForPaths = fileDir
+				result.Paths = &TSConfigPaths{Source: source, Map: make(map[string][]TSConfigPath)}
+				for _, prop := range paths.Properties {
+					if key, ok := getString(prop.Key); ok {
+						if !isValidTSConfigPathPattern(key, log, &source, &tracker, prop.Key.Loc) {
+							continue
+						}
+
+						// The "paths" field is an object which maps a pattern to an
+						// array of remapping patterns to try, in priority order. See
+						// the documentation for examples of how this is used:
+						// https://www.typescriptlang.org/docs/handbook/module-resolution.html#path-mapping.
+						//
+						// One particular example:
+						//
+						//   {
+						//     "compilerOptions": {
+						//       "baseUrl": "projectRoot",
+						//       "paths": {
+						//         "*": [
+						//           "*",
+						//           "generated/*"
+						//         ]
+						//       }
+						//     }
+						//   }
+						//
+						// Matching "folder1/file2" should first check "projectRoot/folder1/file2"
+						// and then, if that didn't work, also check "projectRoot/generated/folder1/file2".
+						if array, ok := prop.ValueOrNil.Data.(*js_ast.EArray); ok {
+							for _, item := range array.Items {
+								if str, ok := getString(item); ok {
+									if isValidTSConfigPathPattern(str, log, &source, &tracker, item.Loc) {
+										str = getSubstitutedPathWithConfigDirTemplate(fs, str, configDir)
+										result.Paths.Map[key] = append(result.Paths.Map[key], TSConfigPath{Text: str, Loc: item.Loc})
+									}
+								}
+							}
+						} else {
+							log.AddID(logger.MsgID_TSConfigJSON_InvalidPaths, logger.Warning, &tracker, source.RangeOfString(prop.ValueOrNil.Loc), fmt.Sprintf(
+								"Substitutions for pattern %q should be an array", key))
+						}
+					}
+				}
+			}
+		}
+	}
+
+	// Warn about compiler options not wrapped in "compilerOptions".
+	// For example: https://github.com/evanw/esbuild/issues/3301
+	if obj, ok := json.Data.(*js_ast.EObject); ok {
+	loop:
+		for _, prop := range obj.Properties {
+			if key, ok := prop.Key.Data.(*js_ast.EString); ok && key.Value != nil {
+				key := helpers.UTF16ToString(key.Value)
+				switch key {
+				case "alwaysStrict",
+					"baseUrl",
+					"experimentalDecorators",
+					"importsNotUsedAsValues",
+					"jsx",
+					"jsxFactory",
+					"jsxFragmentFactory",
+					"jsxImportSource",
+					"paths",
+					"preserveValueImports",
+					"strict",
+					"target",
+					"useDefineForClassFields",
+					"verbatimModuleSyntax":
+					log.AddIDWithNotes(logger.MsgID_TSConfigJSON_InvalidTopLevelOption, logger.Warning, &tracker, source.RangeOfString(prop.Key.Loc),
+						fmt.Sprintf("Expected the %q option to be nested inside a \"compilerOptions\" object", key),
+						[]logger.MsgData{})
+					break loop
+				}
+			}
+		}
+	}
+
+	return &result
+}
+
+// See: https://github.com/microsoft/TypeScript/pull/58042
+func getSubstitutedPathWithConfigDirTemplate(fs fs.FS, value string, basePath string) string {
+	if strings.HasPrefix(value, "${configDir}") {
+		return fs.Join(basePath, "./"+value[12:])
+	}
+	return value
+}
+
+func parseMemberExpressionForJSX(log logger.Log, source *logger.Source, tracker *logger.LineColumnTracker, loc logger.Loc, text string) []string {
+	if text == "" {
+		return nil
+	}
+	parts := strings.Split(text, ".")
+	for _, part := range parts {
+		if !js_ast.IsIdentifier(part) {
+			warnRange := source.RangeOfString(loc)
+			log.AddID(logger.MsgID_TSConfigJSON_InvalidJSX, logger.Warning, tracker, warnRange, fmt.Sprintf("Invalid JSX member expression: %q", text))
+			return nil
+		}
+	}
+	return parts
+}
+
+func isValidTSConfigPathPattern(text string, log logger.Log, source *logger.Source, tracker *logger.LineColumnTracker, loc logger.Loc) bool {
+	foundAsterisk := false
+	for i := 0; i < len(text); i++ {
+		if text[i] == '*' {
+			if foundAsterisk {
+				r := source.RangeOfString(loc)
+				log.AddID(logger.MsgID_TSConfigJSON_InvalidPaths, logger.Warning, tracker, r, fmt.Sprintf(
+					"Invalid pattern %q, must have at most one \"*\" character", text))
+				return false
+			}
+			foundAsterisk = true
+		}
+	}
+	return true
+}
+
+func isSlash(c byte) bool {
+	return c == '/' || c == '\\'
+}
+
+func isValidTSConfigPathNoBaseURLPattern(text string, log logger.Log, source *logger.Source, tracker **logger.LineColumnTracker, loc logger.Loc) bool {
+	var c0 byte
+	var c1 byte
+	var c2 byte
+	n := len(text)
+
+	if n > 0 {
+		c0 = text[0]
+		if n > 1 {
+			c1 = text[1]
+			if n > 2 {
+				c2 = text[2]
+			}
+		}
+	}
+
+	// Relative "." or ".."
+	if c0 == '.' && (n == 1 || (n == 2 && c1 == '.')) {
+		return true
+	}
+
+	// Relative "./" or "../" or ".\\" or "..\\"
+	if c0 == '.' && (isSlash(c1) || (c1 == '.' && isSlash(c2))) {
+		return true
+	}
+
+	// Absolute POSIX "/" or UNC "\\"
+	if isSlash(c0) {
+		return true
+	}
+
+	// Absolute DOS "c:/" or "c:\\"
+	if ((c0 >= 'a' && c0 <= 'z') || (c0 >= 'A' && c0 <= 'Z')) && c1 == ':' && isSlash(c2) {
+		return true
+	}
+
+	r := source.RangeOfString(loc)
+	if *tracker == nil {
+		t := logger.MakeLineColumnTracker(source)
+		*tracker = &t
+	}
+	log.AddID(logger.MsgID_TSConfigJSON_InvalidPaths, logger.Warning, *tracker, r, fmt.Sprintf(
+		"Non-relative path %q is not allowed when \"baseUrl\" is not set (did you forget a leading \"./\"?)", text))
+	return false
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/resolver/yarnpnp.go b/source/vendor/github.com/evanw/esbuild/internal/resolver/yarnpnp.go
new file mode 100644
index 0000000..5a6a68d
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/resolver/yarnpnp.go
@@ -0,0 +1,665 @@
+package resolver
+
+// This file implements the Yarn PnP specification: https://yarnpkg.com/advanced/pnp-spec/
+
+import (
+	"fmt"
+	"regexp"
+	"strings"
+	"syscall"
+
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/js_parser"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type pnpData struct {
+	// Keys are the package idents, values are sets of references. Combining the
+	// ident with each individual reference yields the set of affected locators.
+	fallbackExclusionList map[string]map[string]bool
+
+	// A map of locators that all packages are allowed to access, regardless
+	// whether they list them in their dependencies or not.
+	fallbackPool map[string]pnpIdentAndReference
+
+	// A nullable regexp. If set, all project-relative importer paths should be
+	// matched against it. If the match succeeds, the resolution should follow
+	// the classic Node.js resolution algorithm rather than the Plug'n'Play one.
+	// Note that unlike other paths in the manifest, the one checked against this
+	// regexp won't begin by `./`.
+	ignorePatternData        *regexp.Regexp
+	invalidIgnorePatternData string
+
+	// This is the main part of the PnP data file. This table contains the list
+	// of all packages, first keyed by package ident then by package reference.
+	// One entry will have `null` in both fields and represents the absolute
+	// top-level package.
+	packageRegistryData map[string]map[string]pnpPackage
+
+	packageLocatorsByLocations map[string]pnpPackageLocatorByLocation
+
+	// If true, should a dependency resolution fail for an importer that isn't
+	// explicitly listed in `fallbackExclusionList`, the runtime must first check
+	// whether the resolution would succeed for any of the packages in
+	// `fallbackPool`; if it would, transparently return this resolution. Note
+	// that all dependencies from the top-level package are implicitly part of
+	// the fallback pool, even if not listed here.
+	enableTopLevelFallback bool
+
+	tracker    logger.LineColumnTracker
+	absPath    string
+	absDirPath string
+}
+
+// This is called both a "locator" and a "dependency target" in the specification.
+// When it's used as a dependency target, it can only be in one of three states:
+//
+//  1. A reference, to link with the dependency name
+//     In this case ident is "".
+//
+//  2. An aliased package
+//     In this case neither ident nor reference are "".
+//
+//  3. A missing peer dependency
+//     In this case ident and reference are "".
+type pnpIdentAndReference struct {
+	ident     string // Empty if null
+	reference string // Empty if null
+	span      logger.Range
+}
+
+type pnpPackage struct {
+	packageDependencies      map[string]pnpIdentAndReference
+	packageLocation          string
+	packageDependenciesRange logger.Range
+	discardFromLookup        bool
+}
+
+type pnpPackageLocatorByLocation struct {
+	locator           pnpIdentAndReference
+	discardFromLookup bool
+}
+
+func parseBareIdentifier(specifier string) (ident string, modulePath string, ok bool) {
+	slash := strings.IndexByte(specifier, '/')
+
+	// If specifier starts with "@", then
+	if strings.HasPrefix(specifier, "@") {
+		// If specifier doesn't contain a "/" separator, then
+		if slash == -1 {
+			// Throw an error
+			return
+		}
+
+		// Otherwise,
+		// Set ident to the substring of specifier until the second "/" separator or the end of string, whatever happens first
+		if slash2 := strings.IndexByte(specifier[slash+1:], '/'); slash2 != -1 {
+			ident = specifier[:slash+1+slash2]
+		} else {
+			ident = specifier
+		}
+	} else {
+		// Otherwise,
+		// Set ident to the substring of specifier until the first "/" separator or the end of string, whatever happens first
+		if slash != -1 {
+			ident = specifier[:slash]
+		} else {
+			ident = specifier
+		}
+	}
+
+	// Set modulePath to the substring of specifier starting from ident.length
+	modulePath = specifier[len(ident):]
+
+	// Return {ident, modulePath}
+	ok = true
+	return
+}
+
+type pnpStatus uint8
+
+const (
+	pnpErrorGeneric pnpStatus = iota
+	pnpErrorDependencyNotFound
+	pnpErrorUnfulfilledPeerDependency
+	pnpSuccess
+	pnpSkipped
+)
+
+func (status pnpStatus) isError() bool {
+	return status < pnpSuccess
+}
+
+type pnpResult struct {
+	status     pnpStatus
+	pkgDirPath string
+	pkgIdent   string
+	pkgSubpath string
+
+	// This is for error messages
+	errorIdent string
+	errorRange logger.Range
+}
+
+// Note: If this returns successfully then the node module resolution algorithm
+// (i.e. NM_RESOLVE in the Yarn PnP specification) is always run afterward
+func (r resolverQuery) resolveToUnqualified(specifier string, parentURL string, manifest *pnpData) pnpResult {
+	// Let resolved be undefined
+
+	// Let manifest be FIND_PNP_MANIFEST(parentURL)
+	// (this is already done by the time we get here)
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Using Yarn PnP manifest from %q", manifest.absPath))
+		r.debugLogs.addNote(fmt.Sprintf("  Resolving %q in %q", specifier, parentURL))
+	}
+
+	// Let ident and modulePath be the result of PARSE_BARE_IDENTIFIER(specifier)
+	ident, modulePath, ok := parseBareIdentifier(specifier)
+	if !ok {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("  Failed to parse specifier %q into a bare identifier", specifier))
+		}
+		return pnpResult{status: pnpErrorGeneric}
+	}
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("  Parsed bare identifier %q and module path %q", ident, modulePath))
+	}
+
+	// Let parentLocator be FIND_LOCATOR(manifest, parentURL)
+	parentLocator, ok := r.findLocator(manifest, parentURL)
+
+	// If parentLocator is null, then
+	// Set resolved to NM_RESOLVE(specifier, parentURL) and return it
+	if !ok {
+		return pnpResult{status: pnpSkipped}
+	}
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("  Found parent locator: [%s, %s]", quoteOrNullIfEmpty(parentLocator.ident), quoteOrNullIfEmpty(parentLocator.reference)))
+	}
+
+	// Let parentPkg be GET_PACKAGE(manifest, parentLocator)
+	parentPkg, ok := r.getPackage(manifest, parentLocator.ident, parentLocator.reference)
+	if !ok {
+		// We aren't supposed to get here according to the Yarn PnP specification
+		return pnpResult{status: pnpErrorGeneric}
+	}
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("  Found parent package at %q", parentPkg.packageLocation))
+	}
+
+	// Let referenceOrAlias be the entry from parentPkg.packageDependencies referenced by ident
+	referenceOrAlias, ok := parentPkg.packageDependencies[ident]
+
+	// If referenceOrAlias is null or undefined, then
+	if !ok || referenceOrAlias.reference == "" {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("  Failed to find %q in \"packageDependencies\" of parent package", ident))
+		}
+
+		// If manifest.enableTopLevelFallback is true, then
+		if manifest.enableTopLevelFallback {
+			if r.debugLogs != nil {
+				r.debugLogs.addNote("  Searching for a fallback because \"enableTopLevelFallback\" is true")
+			}
+
+			// If parentLocator isn't in manifest.fallbackExclusionList, then
+			if set := manifest.fallbackExclusionList[parentLocator.ident]; !set[parentLocator.reference] {
+				// Let fallback be RESOLVE_VIA_FALLBACK(manifest, ident)
+				fallback, _ := r.resolveViaFallback(manifest, ident)
+
+				// If fallback is neither null nor undefined
+				if fallback.reference != "" {
+					// Set referenceOrAlias to fallback
+					referenceOrAlias = fallback
+					ok = true
+				}
+			} else if r.debugLogs != nil {
+				r.debugLogs.addNote(fmt.Sprintf("    Stopping because [%s, %s] is in \"fallbackExclusionList\"",
+					quoteOrNullIfEmpty(parentLocator.ident), quoteOrNullIfEmpty(parentLocator.reference)))
+			}
+		}
+	}
+
+	// If referenceOrAlias is still undefined, then
+	if !ok {
+		// Throw a resolution error
+		return pnpResult{
+			status:     pnpErrorDependencyNotFound,
+			errorIdent: ident,
+			errorRange: parentPkg.packageDependenciesRange,
+		}
+	}
+
+	// If referenceOrAlias is still null, then
+	if referenceOrAlias.reference == "" {
+		// Note: It means that parentPkg has an unfulfilled peer dependency on ident
+		// Throw a resolution error
+		return pnpResult{
+			status:     pnpErrorUnfulfilledPeerDependency,
+			errorIdent: ident,
+			errorRange: referenceOrAlias.span,
+		}
+	}
+
+	if r.debugLogs != nil {
+		var referenceOrAliasStr string
+		if referenceOrAlias.ident != "" {
+			referenceOrAliasStr = fmt.Sprintf("[%q, %q]", referenceOrAlias.ident, referenceOrAlias.reference)
+		} else {
+			referenceOrAliasStr = quoteOrNullIfEmpty(referenceOrAlias.reference)
+		}
+		r.debugLogs.addNote(fmt.Sprintf("  Found dependency locator: [%s, %s]", quoteOrNullIfEmpty(ident), referenceOrAliasStr))
+	}
+
+	// Otherwise, if referenceOrAlias is an array, then
+	var dependencyPkg pnpPackage
+	if referenceOrAlias.ident != "" {
+		// Let alias be referenceOrAlias
+		alias := referenceOrAlias
+
+		// Let dependencyPkg be GET_PACKAGE(manifest, alias)
+		dependencyPkg, ok = r.getPackage(manifest, alias.ident, alias.reference)
+		if !ok {
+			// We aren't supposed to get here according to the Yarn PnP specification
+			return pnpResult{status: pnpErrorGeneric}
+		}
+	} else {
+		// Otherwise,
+		// Let dependencyPkg be GET_PACKAGE(manifest, {ident, reference})
+		dependencyPkg, ok = r.getPackage(manifest, ident, referenceOrAlias.reference)
+		if !ok {
+			// We aren't supposed to get here according to the Yarn PnP specification
+			return pnpResult{status: pnpErrorGeneric}
+		}
+	}
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("  Found package %q at %q", ident, dependencyPkg.packageLocation))
+	}
+
+	// Return path.resolve(manifest.dirPath, dependencyPkg.packageLocation, modulePath)
+	pkgDirPath := r.fs.Join(manifest.absDirPath, dependencyPkg.packageLocation)
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("  Resolved %q via Yarn PnP to %q with subpath %q", specifier, pkgDirPath, modulePath))
+	}
+	return pnpResult{
+		status:     pnpSuccess,
+		pkgDirPath: pkgDirPath,
+		pkgIdent:   ident,
+		pkgSubpath: modulePath,
+	}
+}
+
+func (r resolverQuery) findLocator(manifest *pnpData, moduleUrl string) (pnpIdentAndReference, bool) {
+	// Let relativeUrl be the relative path between manifest and moduleUrl
+	relativeUrl, ok := r.fs.Rel(manifest.absDirPath, moduleUrl)
+	if !ok {
+		return pnpIdentAndReference{}, false
+	} else {
+		// Relative URLs on Windows will use \ instead of /, which will break
+		// everything we do below. Use normal slashes to keep things working.
+		relativeUrl = strings.ReplaceAll(relativeUrl, "\\", "/")
+	}
+
+	// The relative path must not start with ./; trim it if needed
+	relativeUrl = strings.TrimPrefix(relativeUrl, "./")
+
+	// If relativeUrl matches manifest.ignorePatternData, then
+	if manifest.ignorePatternData != nil && manifest.ignorePatternData.MatchString(relativeUrl) {
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("  Ignoring %q because it matches \"ignorePatternData\"", relativeUrl))
+		}
+
+		// Return null
+		return pnpIdentAndReference{}, false
+	}
+
+	// Note: Make sure relativeUrl always starts with a ./ or ../
+	if !strings.HasSuffix(relativeUrl, "/") {
+		relativeUrl += "/"
+	}
+	if !strings.HasPrefix(relativeUrl, "./") && !strings.HasPrefix(relativeUrl, "../") {
+		relativeUrl = "./" + relativeUrl
+	}
+
+	// This is the inner loop from Yarn's PnP resolver implementation. This is
+	// different from the specification, which contains a hypothetical slow
+	// algorithm instead. The algorithm from the specification can sometimes
+	// produce different results from the one used by the implementation, so
+	// we follow the implementation.
+	for {
+		entry, ok := manifest.packageLocatorsByLocations[relativeUrl]
+		if !ok || entry.discardFromLookup {
+			// Remove the last path component and try again
+			relativeUrl = relativeUrl[:strings.LastIndexByte(relativeUrl[:len(relativeUrl)-1], '/')+1]
+			if relativeUrl == "" {
+				break
+			}
+			continue
+		}
+		return entry.locator, true
+	}
+
+	return pnpIdentAndReference{}, false
+}
+
+func (r resolverQuery) resolveViaFallback(manifest *pnpData, ident string) (pnpIdentAndReference, bool) {
+	// Let topLevelPkg be GET_PACKAGE(manifest, {null, null})
+	topLevelPkg, ok := r.getPackage(manifest, "", "")
+	if !ok {
+		// We aren't supposed to get here according to the Yarn PnP specification
+		return pnpIdentAndReference{}, false
+	}
+
+	// Let referenceOrAlias be the entry from topLevelPkg.packageDependencies referenced by ident
+	referenceOrAlias, ok := topLevelPkg.packageDependencies[ident]
+
+	// If referenceOrAlias is defined, then
+	if ok {
+		// Return it immediately
+		if r.debugLogs != nil {
+			r.debugLogs.addNote(fmt.Sprintf("    Found fallback for %q in \"packageDependencies\" of top-level package: [%s, %s]", ident,
+				quoteOrNullIfEmpty(referenceOrAlias.ident), quoteOrNullIfEmpty(referenceOrAlias.reference)))
+		}
+		return referenceOrAlias, true
+	}
+
+	// Otherwise,
+	// Let referenceOrAlias be the entry from manifest.fallbackPool referenced by ident
+	referenceOrAlias, ok = manifest.fallbackPool[ident]
+
+	// Return it immediately, whether it's defined or not
+	if r.debugLogs != nil {
+		if ok {
+			r.debugLogs.addNote(fmt.Sprintf("    Found fallback for %q in \"fallbackPool\": [%s, %s]", ident,
+				quoteOrNullIfEmpty(referenceOrAlias.ident), quoteOrNullIfEmpty(referenceOrAlias.reference)))
+		} else {
+			r.debugLogs.addNote(fmt.Sprintf("    Failed to find fallback for %q in \"fallbackPool\"", ident))
+		}
+	}
+	return referenceOrAlias, ok
+}
+
+func (r resolverQuery) getPackage(manifest *pnpData, ident string, reference string) (pnpPackage, bool) {
+	if inner, ok := manifest.packageRegistryData[ident]; ok {
+		if pkg, ok := inner[reference]; ok {
+			return pkg, true
+		}
+	}
+
+	if r.debugLogs != nil {
+		// We aren't supposed to get here according to the Yarn PnP specification:
+		// "Note: pkg cannot be undefined here; all packages referenced in any of the
+		// Plug'n'Play data tables MUST have a corresponding entry inside packageRegistryData."
+		r.debugLogs.addNote(fmt.Sprintf("  Yarn PnP invariant violation: GET_PACKAGE failed to find a package: [%s, %s]",
+			quoteOrNullIfEmpty(ident), quoteOrNullIfEmpty(reference)))
+	}
+	return pnpPackage{}, false
+}
+
+func quoteOrNullIfEmpty(str string) string {
+	if str != "" {
+		return fmt.Sprintf("%q", str)
+	}
+	return "null"
+}
+
+func compileYarnPnPData(absPath string, absDirPath string, json js_ast.Expr, source logger.Source) *pnpData {
+	data := pnpData{
+		absPath:    absPath,
+		absDirPath: absDirPath,
+		tracker:    logger.MakeLineColumnTracker(&source),
+	}
+
+	if value, _, ok := getProperty(json, "enableTopLevelFallback"); ok {
+		if enableTopLevelFallback, ok := getBool(value); ok {
+			data.enableTopLevelFallback = enableTopLevelFallback
+		}
+	}
+
+	if value, _, ok := getProperty(json, "fallbackExclusionList"); ok {
+		if array, ok := value.Data.(*js_ast.EArray); ok {
+			data.fallbackExclusionList = make(map[string]map[string]bool, len(array.Items))
+
+			for _, item := range array.Items {
+				if tuple, ok := item.Data.(*js_ast.EArray); ok && len(tuple.Items) == 2 {
+					if ident, ok := getStringOrNull(tuple.Items[0]); ok {
+						if array2, ok := tuple.Items[1].Data.(*js_ast.EArray); ok {
+							references := make(map[string]bool, len(array2.Items))
+
+							for _, item2 := range array2.Items {
+								if reference, ok := getString(item2); ok {
+									references[reference] = true
+								}
+							}
+
+							data.fallbackExclusionList[ident] = references
+						}
+					}
+				}
+			}
+		}
+	}
+
+	if value, _, ok := getProperty(json, "fallbackPool"); ok {
+		if array, ok := value.Data.(*js_ast.EArray); ok {
+			data.fallbackPool = make(map[string]pnpIdentAndReference, len(array.Items))
+
+			for _, item := range array.Items {
+				if array2, ok := item.Data.(*js_ast.EArray); ok && len(array2.Items) == 2 {
+					if ident, ok := getString(array2.Items[0]); ok {
+						if dependencyTarget, ok := getDependencyTarget(array2.Items[1]); ok {
+							data.fallbackPool[ident] = dependencyTarget
+						}
+					}
+				}
+			}
+		}
+	}
+
+	if value, _, ok := getProperty(json, "ignorePatternData"); ok {
+		if ignorePatternData, ok := getString(value); ok {
+			// The Go regular expression engine doesn't support some of the features
+			// that JavaScript regular expressions support, including "(?!" negative
+			// lookaheads which Yarn uses. This is deliberate on Go's part. See this:
+			// https://github.com/golang/go/issues/18868.
+			//
+			// Yarn uses this feature to exclude the "." and ".." path segments in
+			// the middle of a relative path. However, we shouldn't ever generate
+			// such path segments in the first place. So as a hack, we just remove
+			// the specific character sequences used by Yarn for this so that the
+			// regular expression is more likely to be able to be compiled.
+			ignorePatternData = strings.ReplaceAll(ignorePatternData, `(?!\.)`, "")
+			ignorePatternData = strings.ReplaceAll(ignorePatternData, `(?!(?:^|\/)\.)`, "")
+			ignorePatternData = strings.ReplaceAll(ignorePatternData, `(?!\.{1,2}(?:\/|$))`, "")
+			ignorePatternData = strings.ReplaceAll(ignorePatternData, `(?!(?:^|\/)\.{1,2}(?:\/|$))`, "")
+
+			if reg, err := regexp.Compile(ignorePatternData); err == nil {
+				data.ignorePatternData = reg
+			} else {
+				data.invalidIgnorePatternData = ignorePatternData
+			}
+		}
+	}
+
+	if value, _, ok := getProperty(json, "packageRegistryData"); ok {
+		if array, ok := value.Data.(*js_ast.EArray); ok {
+			data.packageRegistryData = make(map[string]map[string]pnpPackage, len(array.Items))
+			data.packageLocatorsByLocations = make(map[string]pnpPackageLocatorByLocation)
+
+			for _, item := range array.Items {
+				if tuple, ok := item.Data.(*js_ast.EArray); ok && len(tuple.Items) == 2 {
+					if packageIdent, ok := getStringOrNull(tuple.Items[0]); ok {
+						if array2, ok := tuple.Items[1].Data.(*js_ast.EArray); ok {
+							references := make(map[string]pnpPackage, len(array2.Items))
+							data.packageRegistryData[packageIdent] = references
+
+							for _, item2 := range array2.Items {
+								if tuple2, ok := item2.Data.(*js_ast.EArray); ok && len(tuple2.Items) == 2 {
+									if packageReference, ok := getStringOrNull(tuple2.Items[0]); ok {
+										pkg := tuple2.Items[1]
+
+										if packageLocation, _, ok := getProperty(pkg, "packageLocation"); ok {
+											if packageDependencies, _, ok := getProperty(pkg, "packageDependencies"); ok {
+												if packageLocation, ok := getString(packageLocation); ok {
+													if array3, ok := packageDependencies.Data.(*js_ast.EArray); ok {
+														deps := make(map[string]pnpIdentAndReference, len(array3.Items))
+														discardFromLookup := false
+
+														for _, dep := range array3.Items {
+															if array4, ok := dep.Data.(*js_ast.EArray); ok && len(array4.Items) == 2 {
+																if ident, ok := getString(array4.Items[0]); ok {
+																	if dependencyTarget, ok := getDependencyTarget(array4.Items[1]); ok {
+																		deps[ident] = dependencyTarget
+																	}
+																}
+															}
+														}
+
+														if value, _, ok := getProperty(pkg, "discardFromLookup"); ok {
+															if value, ok := getBool(value); ok {
+																discardFromLookup = value
+															}
+														}
+
+														references[packageReference] = pnpPackage{
+															packageLocation:     packageLocation,
+															packageDependencies: deps,
+															packageDependenciesRange: logger.Range{
+																Loc: packageDependencies.Loc,
+																Len: array3.CloseBracketLoc.Start + 1 - packageDependencies.Loc.Start,
+															},
+															discardFromLookup: discardFromLookup,
+														}
+
+														// This is what Yarn's PnP implementation does (specifically in
+														// "hydrateRuntimeState"), so we replicate that behavior here:
+														if entry, ok := data.packageLocatorsByLocations[packageLocation]; !ok {
+															data.packageLocatorsByLocations[packageLocation] = pnpPackageLocatorByLocation{
+																locator:           pnpIdentAndReference{ident: packageIdent, reference: packageReference},
+																discardFromLookup: discardFromLookup,
+															}
+														} else {
+															entry.discardFromLookup = entry.discardFromLookup && discardFromLookup
+															if !discardFromLookup {
+																entry.locator = pnpIdentAndReference{ident: packageIdent, reference: packageReference}
+															}
+															data.packageLocatorsByLocations[packageLocation] = entry
+														}
+													}
+												}
+											}
+										}
+									}
+								}
+							}
+						}
+					}
+				}
+			}
+		}
+	}
+
+	return &data
+}
+
+func getStringOrNull(json js_ast.Expr) (string, bool) {
+	switch value := json.Data.(type) {
+	case *js_ast.EString:
+		return helpers.UTF16ToString(value.Value), true
+
+	case *js_ast.ENull:
+		return "", true
+	}
+
+	return "", false
+}
+
+func getDependencyTarget(json js_ast.Expr) (pnpIdentAndReference, bool) {
+	switch d := json.Data.(type) {
+	case *js_ast.ENull:
+		return pnpIdentAndReference{span: logger.Range{Loc: json.Loc, Len: 4}}, true
+
+	case *js_ast.EString:
+		return pnpIdentAndReference{reference: helpers.UTF16ToString(d.Value), span: logger.Range{Loc: json.Loc}}, true
+
+	case *js_ast.EArray:
+		if len(d.Items) == 2 {
+			if name, ok := getString(d.Items[0]); ok {
+				if reference, ok := getString(d.Items[1]); ok {
+					return pnpIdentAndReference{
+						ident:     name,
+						reference: reference,
+						span:      logger.Range{Loc: json.Loc, Len: d.CloseBracketLoc.Start + 1 - json.Loc.Start},
+					}, true
+				}
+			}
+		}
+	}
+
+	return pnpIdentAndReference{}, false
+}
+
+type pnpDataMode uint8
+
+const (
+	pnpIgnoreErrorsAboutMissingFiles pnpDataMode = iota
+	pnpReportErrorsAboutMissingFiles
+)
+
+func (r resolverQuery) extractYarnPnPDataFromJSON(pnpDataPath string, mode pnpDataMode) (result js_ast.Expr, source logger.Source) {
+	contents, err, originalError := r.caches.FSCache.ReadFile(r.fs, pnpDataPath)
+	if r.debugLogs != nil && originalError != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Failed to read file %q: %s", pnpDataPath, originalError.Error()))
+	}
+	if err != nil {
+		if mode == pnpReportErrorsAboutMissingFiles || err != syscall.ENOENT {
+			r.log.AddError(nil, logger.Range{},
+				fmt.Sprintf("Cannot read file %q: %s",
+					PrettyPath(r.fs, logger.Path{Text: pnpDataPath, Namespace: "file"}), err.Error()))
+		}
+		return
+	}
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("The file %q exists", pnpDataPath))
+	}
+	keyPath := logger.Path{Text: pnpDataPath, Namespace: "file"}
+	source = logger.Source{
+		KeyPath:    keyPath,
+		PrettyPath: PrettyPath(r.fs, keyPath),
+		Contents:   contents,
+	}
+	result, _ = r.caches.JSONCache.Parse(r.log, source, js_parser.JSONOptions{})
+	return
+}
+
+func (r resolverQuery) tryToExtractYarnPnPDataFromJS(pnpDataPath string, mode pnpDataMode) (result js_ast.Expr, source logger.Source) {
+	contents, err, originalError := r.caches.FSCache.ReadFile(r.fs, pnpDataPath)
+	if r.debugLogs != nil && originalError != nil {
+		r.debugLogs.addNote(fmt.Sprintf("Failed to read file %q: %s", pnpDataPath, originalError.Error()))
+	}
+	if err != nil {
+		if mode == pnpReportErrorsAboutMissingFiles || err != syscall.ENOENT {
+			r.log.AddError(nil, logger.Range{},
+				fmt.Sprintf("Cannot read file %q: %s",
+					PrettyPath(r.fs, logger.Path{Text: pnpDataPath, Namespace: "file"}), err.Error()))
+		}
+		return
+	}
+	if r.debugLogs != nil {
+		r.debugLogs.addNote(fmt.Sprintf("The file %q exists", pnpDataPath))
+	}
+
+	keyPath := logger.Path{Text: pnpDataPath, Namespace: "file"}
+	source = logger.Source{
+		KeyPath:    keyPath,
+		PrettyPath: PrettyPath(r.fs, keyPath),
+		Contents:   contents,
+	}
+	ast, _ := r.caches.JSCache.Parse(r.log, source, js_parser.OptionsForYarnPnP())
+
+	if r.debugLogs != nil && ast.ManifestForYarnPnP.Data != nil {
+		r.debugLogs.addNote(fmt.Sprintf("  Extracted JSON data from %q", pnpDataPath))
+	}
+	return ast.ManifestForYarnPnP, source
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/runtime/runtime.go b/source/vendor/github.com/evanw/esbuild/internal/runtime/runtime.go
new file mode 100644
index 0000000..42ccf39
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/runtime/runtime.go
@@ -0,0 +1,604 @@
+package runtime
+
+// This is esbuild's runtime code. It contains helper functions that are
+// automatically injected into output files to implement certain features. For
+// example, the "**" operator is replaced with a call to "__pow" when targeting
+// ES2015. Tree shaking automatically removes unused code from the runtime.
+
+import (
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+// The runtime source is always at a special index. The index is always zero
+// but this constant is always used instead to improve readability and ensure
+// all code that references this index can be discovered easily.
+const SourceIndex = uint32(0)
+
+func Source(unsupportedJSFeatures compat.JSFeature) logger.Source {
+	// Note: These helper functions used to be named similar things to the helper
+	// functions from the TypeScript compiler. However, people sometimes use these
+	// two projects in combination and TypeScript's implementation of these helpers
+	// causes name collisions. Some examples:
+	//
+	// * The "tslib" library will overwrite esbuild's helper functions if the bundled
+	//   code is run in the global scope: https://github.com/evanw/esbuild/issues/1102
+	//
+	// * Running the TypeScript compiler on esbuild's output to convert ES6 to ES5
+	//   will also overwrite esbuild's helper functions because TypeScript doesn't
+	//   change the names of its helper functions to avoid name collisions:
+	//   https://github.com/microsoft/TypeScript/issues/43296
+	//
+	// These can both be considered bugs in TypeScript. However, they are unlikely
+	// to be fixed and it's simplest to just avoid using the same names to avoid
+	// these bugs. Forbidden names (from "tslib"):
+	//
+	//   __assign
+	//   __asyncDelegator
+	//   __asyncGenerator
+	//   __asyncValues
+	//   __await
+	//   __awaiter
+	//   __classPrivateFieldGet
+	//   __classPrivateFieldSet
+	//   __createBinding
+	//   __decorate
+	//   __exportStar
+	//   __extends
+	//   __generator
+	//   __importDefault
+	//   __importStar
+	//   __makeTemplateObject
+	//   __metadata
+	//   __param
+	//   __read
+	//   __rest
+	//   __spread
+	//   __spreadArray
+	//   __spreadArrays
+	//   __values
+	//
+	// Note: The "__objRest" function has a for-of loop which requires ES6, but
+	// transforming destructuring to ES5 isn't even supported so it's ok.
+	text := `
+		var __create = Object.create
+		var __freeze = Object.freeze
+		var __defProp = Object.defineProperty
+		var __defProps = Object.defineProperties
+		var __getOwnPropDesc = Object.getOwnPropertyDescriptor // Note: can return "undefined" due to a Safari bug
+		var __getOwnPropDescs = Object.getOwnPropertyDescriptors
+		var __getOwnPropNames = Object.getOwnPropertyNames
+		var __getOwnPropSymbols = Object.getOwnPropertySymbols
+		var __getProtoOf = Object.getPrototypeOf
+		var __hasOwnProp = Object.prototype.hasOwnProperty
+		var __propIsEnum = Object.prototype.propertyIsEnumerable
+		var __reflectGet = Reflect.get
+		var __reflectSet = Reflect.set
+
+		var __knownSymbol = (name, symbol) => (symbol = Symbol[name]) ? symbol : Symbol.for('Symbol.' + name)
+		var __typeError = msg => { throw TypeError(msg) }
+
+		export var __pow = Math.pow
+
+		var __defNormalProp = (obj, key, value) => key in obj
+			? __defProp(obj, key, {enumerable: true, configurable: true, writable: true, value})
+			: obj[key] = value
+
+		export var __spreadValues = (a, b) => {
+			for (var prop in b ||= {})
+				if (__hasOwnProp.call(b, prop))
+					__defNormalProp(a, prop, b[prop])
+			if (__getOwnPropSymbols)
+		`
+
+	// Avoid "of" when not using ES6
+	if !unsupportedJSFeatures.Has(compat.ForOf) {
+		text += `
+				for (var prop of __getOwnPropSymbols(b)) {
+		`
+	} else {
+		text += `
+				for (var props = __getOwnPropSymbols(b), i = 0, n = props.length, prop; i < n; i++) {
+					prop = props[i]
+		`
+	}
+
+	text += `
+					if (__propIsEnum.call(b, prop))
+						__defNormalProp(a, prop, b[prop])
+				}
+			return a
+		}
+		export var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b))
+
+		// Update the "name" property on the function or class for "--keep-names"
+		export var __name = (target, value) => __defProp(target, 'name', { value, configurable: true })
+
+		// This fallback "require" function exists so that "typeof require" can
+		// naturally be "function" even in non-CommonJS environments since esbuild
+		// emulates a CommonJS environment (issue #1202). However, people want this
+		// shim to fall back to "globalThis.require" even if it's defined later
+		// (including property accesses such as "require.resolve") so we need to
+		// use a proxy (issue #1614).
+		export var __require =
+			/* @__PURE__ */ (x =>
+				typeof require !== 'undefined' ? require :
+				typeof Proxy !== 'undefined' ? new Proxy(x, {
+					get: (a, b) => (typeof require !== 'undefined' ? require : a)[b]
+				}) : x
+			)(function(x) {
+				if (typeof require !== 'undefined') return require.apply(this, arguments)
+				throw Error('Dynamic require of "' + x + '" is not supported')
+			})
+
+		// This is used for glob imports
+		export var __glob = map => path => {
+			var fn = map[path]
+			if (fn) return fn()
+			throw new Error('Module not found in bundle: ' + path)
+		}
+
+		// For object rest patterns
+		export var __restKey = key => typeof key === 'symbol' ? key : key + ''
+		export var __objRest = (source, exclude) => {
+			var target = {}
+			for (var prop in source)
+				if (__hasOwnProp.call(source, prop) && exclude.indexOf(prop) < 0)
+					target[prop] = source[prop]
+			if (source != null && __getOwnPropSymbols)
+	`
+
+	// Avoid "of" when not using ES6
+	if !unsupportedJSFeatures.Has(compat.ForOf) {
+		text += `
+				for (var prop of __getOwnPropSymbols(source)) {
+		`
+	} else {
+		text += `
+				for (var props = __getOwnPropSymbols(source), i = 0, n = props.length, prop; i < n; i++) {
+					prop = props[i]
+		`
+	}
+
+	text += `
+					if (exclude.indexOf(prop) < 0 && __propIsEnum.call(source, prop))
+						target[prop] = source[prop]
+				}
+			return target
+		}
+
+		// This is for lazily-initialized ESM code. This has two implementations, a
+		// compact one for minified code and a verbose one that generates friendly
+		// names in V8's profiler and in stack traces.
+		export var __esm = (fn, res) => function __init() {
+			return fn && (res = (0, fn[__getOwnPropNames(fn)[0]])(fn = 0)), res
+		}
+		export var __esmMin = (fn, res) => () => (fn && (res = fn(fn = 0)), res)
+
+		// Wraps a CommonJS closure and returns a require() function. This has two
+		// implementations, a compact one for minified code and a verbose one that
+		// generates friendly names in V8's profiler and in stack traces.
+		export var __commonJS = (cb, mod) => function __require() {
+			return mod || (0, cb[__getOwnPropNames(cb)[0]])((mod = {exports: {}}).exports, mod), mod.exports
+		}
+		export var __commonJSMin = (cb, mod) => () => (mod || cb((mod = {exports: {}}).exports, mod), mod.exports)
+
+		// Used to implement ESM exports both for "require()" and "import * as"
+		export var __export = (target, all) => {
+			for (var name in all)
+				__defProp(target, name, { get: all[name], enumerable: true })
+		}
+
+		var __copyProps = (to, from, except, desc) => {
+			if (from && typeof from === 'object' || typeof from === 'function')
+	`
+
+	// Avoid "let" when not using ES6
+	if !unsupportedJSFeatures.Has(compat.ForOf) && !unsupportedJSFeatures.Has(compat.ConstAndLet) {
+		text += `
+				for (let key of __getOwnPropNames(from))
+					if (!__hasOwnProp.call(to, key) && key !== except)
+						__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable })
+		`
+	} else {
+		text += `
+				for (var keys = __getOwnPropNames(from), i = 0, n = keys.length, key; i < n; i++) {
+					key = keys[i]
+					if (!__hasOwnProp.call(to, key) && key !== except)
+						__defProp(to, key, { get: (k => from[k]).bind(null, key), enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable })
+				}
+		`
+	}
+
+	text += `
+			return to
+		}
+
+		// This is used to implement "export * from" statements. It copies properties
+		// from the imported module to the current module's ESM export object. If the
+		// current module is an entry point and the target format is CommonJS, we
+		// also copy the properties to "module.exports" in addition to our module's
+		// internal ESM export object.
+		export var __reExport = (target, mod, secondTarget) => (
+			__copyProps(target, mod, 'default'),
+			secondTarget && __copyProps(secondTarget, mod, 'default')
+		)
+
+		// Converts the module from CommonJS to ESM. When in node mode (i.e. in an
+		// ".mjs" file, package.json has "type: module", or the "__esModule" export
+		// in the CommonJS file is falsy or missing), the "default" property is
+		// overridden to point to the original CommonJS exports object instead.
+		export var __toESM = (mod, isNodeMode, target) => (
+			target = mod != null ? __create(__getProtoOf(mod)) : {},
+			__copyProps(
+				// If the importer is in node compatibility mode or this is not an ESM
+				// file that has been converted to a CommonJS file using a Babel-
+				// compatible transform (i.e. "__esModule" has not been set), then set
+				// "default" to the CommonJS "module.exports" for node compatibility.
+				isNodeMode || !mod || !mod.__esModule
+					? __defProp(target, 'default', { value: mod, enumerable: true })
+					: target,
+				mod)
+		)
+
+		// Converts the module from ESM to CommonJS. This clones the input module
+		// object with the addition of a non-enumerable "__esModule" property set
+		// to "true", which overwrites any existing export named "__esModule".
+		export var __toCommonJS = mod => __copyProps(__defProp({}, '__esModule', { value: true }), mod)
+
+		// For TypeScript experimental decorators
+		// - kind === undefined: class
+		// - kind === 1: method, parameter
+		// - kind === 2: field
+		export var __decorateClass = (decorators, target, key, kind) => {
+			var result = kind > 1 ? void 0 : kind ? __getOwnPropDesc(target, key) : target
+			for (var i = decorators.length - 1, decorator; i >= 0; i--)
+				if (decorator = decorators[i])
+					result = (kind ? decorator(target, key, result) : decorator(result)) || result
+			if (kind && result) __defProp(target, key, result)
+			return result
+		}
+		export var __decorateParam = (index, decorator) => (target, key) => decorator(target, key, index)
+
+		// For JavaScript decorators
+		export var __decoratorStart = base => [, , , __create(base?.[__knownSymbol('metadata')] ?? null)]
+		var __decoratorStrings = ['class', 'method', 'getter', 'setter', 'accessor', 'field', 'value', 'get', 'set']
+		var __expectFn = fn => fn !== void 0 && typeof fn !== 'function' ? __typeError('Function expected') : fn
+		var __decoratorContext = (kind, name, done, metadata, fns) => ({ kind: __decoratorStrings[kind], name, metadata, addInitializer: fn =>
+			done._ ? __typeError('Already initialized') : fns.push(__expectFn(fn || null)) })
+		export var __decoratorMetadata = (array, target) => __defNormalProp(target, __knownSymbol('metadata'), array[3])
+		export var __runInitializers = (array, flags, self, value) => {
+			for (var i = 0, fns = array[flags >> 1], n = fns && fns.length; i < n; i++) flags & 1 ? fns[i].call(self) : value = fns[i].call(self, value)
+			return value
+		}
+		export var __decorateElement = (array, flags, name, decorators, target, extra) => {
+			var fn, it, done, ctx, access, k = flags & 7, s = !!(flags & 8), p = !!(flags & 16)
+			var j = k > 3 ? array.length + 1 : k ? s ? 1 : 2 : 0, key = __decoratorStrings[k + 5]
+			var initializers = k > 3 && (array[j - 1] = []), extraInitializers = array[j] || (array[j] = [])
+			var desc = k && (
+				!p && !s && (target = target.prototype),
+				k < 5 && (k > 3 || !p) &&
+			`
+
+	// Avoid object extensions when not using ES6
+	if !unsupportedJSFeatures.Has(compat.ObjectExtensions) && !unsupportedJSFeatures.Has(compat.ObjectAccessors) {
+		text += `__getOwnPropDesc(k < 4 ? target : { get [name]() { return __privateGet(this, extra) }, set [name](x) { return __privateSet(this, extra, x) } }, name)`
+	} else {
+		text += `(k < 4 ? __getOwnPropDesc(target, name) : { get: () => __privateGet(this, extra), set: x => __privateSet(this, extra, x) })`
+	}
+
+	text += `
+			)
+			k ? p && k < 4 && __name(extra, (k > 2 ? 'set ' : k > 1 ? 'get ' : '') + name) : __name(target, name)
+
+			for (var i = decorators.length - 1; i >= 0; i--) {
+				ctx = __decoratorContext(k, name, done = {}, array[3], extraInitializers)
+
+				if (k) {
+					ctx.static = s, ctx.private = p, access = ctx.access = { has: p ? x => __privateIn(target, x) : x => name in x }
+					if (k ^ 3) access.get = p ? x => (k ^ 1 ? __privateGet : __privateMethod)(x, target, k ^ 4 ? extra : desc.get) : x => x[name]
+					if (k > 2) access.set = p ? (x, y) => __privateSet(x, target, y, k ^ 4 ? extra : desc.set) : (x, y) => x[name] = y
+				}
+
+				it = (0, decorators[i])(k ? k < 4 ? p ? extra : desc[key] : k > 4 ? void 0 : { get: desc.get, set: desc.set } : target, ctx), done._ = 1
+
+				if (k ^ 4 || it === void 0) __expectFn(it) && (k > 4 ? initializers.unshift(it) : k ? p ? extra = it : desc[key] = it : target = it)
+				else if (typeof it !== 'object' || it === null) __typeError('Object expected')
+				else __expectFn(fn = it.get) && (desc.get = fn), __expectFn(fn = it.set) && (desc.set = fn), __expectFn(fn = it.init) && initializers.unshift(fn)
+			}
+
+			return k || __decoratorMetadata(array, target),
+				desc && __defProp(target, name, desc),
+				p ? k ^ 4 ? extra : desc : target
+		}
+
+		// For class members
+		export var __publicField = (obj, key, value) => (
+			__defNormalProp(obj, typeof key !== 'symbol' ? key + '' : key, value)
+		)
+		var __accessCheck = (obj, member, msg) => (
+			member.has(obj) || __typeError('Cannot ' + msg)
+		)
+		export var __privateIn = (member, obj) => (
+			Object(obj) !== obj ? __typeError('Cannot use the "in" operator on this value') :
+			member.has(obj)
+		)
+		export var __privateGet = (obj, member, getter) => (
+			__accessCheck(obj, member, 'read from private field'),
+			getter ? getter.call(obj) : member.get(obj)
+		)
+		export var __privateAdd = (obj, member, value) => (
+			member.has(obj) ? __typeError('Cannot add the same private member more than once') :
+			member instanceof WeakSet ? member.add(obj) : member.set(obj, value)
+		)
+		export var __privateSet = (obj, member, value, setter) => (
+			__accessCheck(obj, member, 'write to private field'),
+			setter ? setter.call(obj, value) : member.set(obj, value),
+			value
+		)
+		export var __privateMethod = (obj, member, method) => (
+			__accessCheck(obj, member, 'access private method'),
+			method
+		)
+		export var __earlyAccess = (name) => {
+			throw ReferenceError('Cannot access "' + name + '" before initialization')
+		}
+	`
+
+	if !unsupportedJSFeatures.Has(compat.ObjectAccessors) {
+		text += `
+			export var __privateWrapper = (obj, member, setter, getter) => ({
+				set _(value) { __privateSet(obj, member, value, setter) },
+				get _() { return __privateGet(obj, member, getter) },
+			})
+		`
+	} else {
+		text += `
+		export var __privateWrapper = (obj, member, setter, getter) => __defProp({}, '_', {
+			set: value => __privateSet(obj, member, value, setter),
+			get: () => __privateGet(obj, member, getter),
+		})
+		`
+	}
+
+	text += `
+		// For "super" property accesses
+		export var __superGet = (cls, obj, key) => __reflectGet(__getProtoOf(cls), key, obj)
+		export var __superSet = (cls, obj, key, val) => (__reflectSet(__getProtoOf(cls), key, val, obj), val)
+	`
+
+	if !unsupportedJSFeatures.Has(compat.ObjectAccessors) {
+		text += `
+			export var __superWrapper = (cls, obj, key) => ({
+				get _() { return __superGet(cls, obj, key) },
+				set _(val) { __superSet(cls, obj, key, val) },
+			})
+		`
+	} else {
+		text += `
+			export var __superWrapper = (cls, obj, key) => __defProp({}, '_', {
+				get: () => __superGet(cls, obj, key),
+				set: val => __superSet(cls, obj, key, val),
+			})
+		`
+	}
+
+	text += `
+		// For lowering tagged template literals
+		export var __template = (cooked, raw) => __freeze(__defProp(cooked, 'raw', { value: __freeze(raw || cooked.slice()) }))
+
+		// This helps for lowering async functions
+		export var __async = (__this, __arguments, generator) => {
+			return new Promise((resolve, reject) => {
+				var fulfilled = value => {
+					try {
+						step(generator.next(value))
+					} catch (e) {
+						reject(e)
+					}
+				}
+				var rejected = value => {
+					try {
+						step(generator.throw(value))
+					} catch (e) {
+						reject(e)
+					}
+				}
+				var step = x => x.done ? resolve(x.value) : Promise.resolve(x.value).then(fulfilled, rejected)
+				step((generator = generator.apply(__this, __arguments)).next())
+			})
+		}
+
+		// These help for lowering async generator functions
+		export var __await = function (promise, isYieldStar) {
+			this[0] = promise
+			this[1] = isYieldStar
+		}
+		export var __asyncGenerator = (__this, __arguments, generator) => {
+			var resume = (k, v, yes, no) => {
+				try {
+					var x = generator[k](v), isAwait = (v = x.value) instanceof __await, done = x.done
+					Promise.resolve(isAwait ? v[0] : v)
+						.then(y => isAwait
+							? resume(k === 'return' ? k : 'next', v[1] ? { done: y.done, value: y.value } : y, yes, no)
+							: yes({ value: y, done }))
+						.catch(e => resume('throw', e, yes, no))
+				} catch (e) {
+					no(e)
+				}
+			}, method = k => it[k] = x => new Promise((yes, no) => resume(k, x, yes, no)), it = {}
+			return generator = generator.apply(__this, __arguments),
+				it[__knownSymbol('asyncIterator')] = () => it,
+				method('next'),
+				method('throw'),
+				method('return'),
+				it
+		}
+		export var __yieldStar = value => {
+			var obj = value[__knownSymbol('asyncIterator')], isAwait = false, method, it = {}
+			if (obj == null) {
+				obj = value[__knownSymbol('iterator')]()
+				method = k => it[k] = x => obj[k](x)
+			} else {
+				obj = obj.call(value)
+				method = k => it[k] = v => {
+					if (isAwait) {
+						isAwait = false
+						if (k === 'throw') throw v
+						return v
+					}
+					isAwait = true
+					return {
+						done: false,
+						value: new __await(new Promise(resolve => {
+							var x = obj[k](v)
+							if (!(x instanceof Object)) __typeError('Object expected')
+							resolve(x)
+						}), 1),
+					}
+				}
+			}
+			return it[__knownSymbol('iterator')] = () => it,
+				method('next'),
+				'throw' in obj ? method('throw') : it.throw = x => { throw x },
+				'return' in obj && method('return'),
+				it
+		}
+
+		// This helps for lowering for-await loops
+		export var __forAwait = (obj, it, method) =>
+			(it = obj[__knownSymbol('asyncIterator')])
+				? it.call(obj)
+				: (obj = obj[__knownSymbol('iterator')](),
+					it = {},
+					method = (key, fn) =>
+						(fn = obj[key]) && (it[key] = arg =>
+							new Promise((yes, no, done) => (
+								arg = fn.call(obj, arg),
+								done = arg.done,
+								Promise.resolve(arg.value)
+									.then(value => yes({ value, done }), no)
+							))),
+					method('next'),
+					method('return'),
+					it)
+
+		// This is for the "binary" loader (custom code is ~2x faster than "atob")
+		export var __toBinaryNode = base64 => new Uint8Array(Buffer.from(base64, 'base64'))
+		export var __toBinary = /* @__PURE__ */ (() => {
+			var table = new Uint8Array(128)
+			for (var i = 0; i < 64; i++) table[i < 26 ? i + 65 : i < 52 ? i + 71 : i < 62 ? i - 4 : i * 4 - 205] = i
+			return base64 => {
+				var n = base64.length, bytes = new Uint8Array((n - (base64[n - 1] == '=') - (base64[n - 2] == '=')) * 3 / 4 | 0)
+				for (var i = 0, j = 0; i < n;) {
+					var c0 = table[base64.charCodeAt(i++)], c1 = table[base64.charCodeAt(i++)]
+					var c2 = table[base64.charCodeAt(i++)], c3 = table[base64.charCodeAt(i++)]
+					bytes[j++] = (c0 << 2) | (c1 >> 4)
+					bytes[j++] = (c1 << 4) | (c2 >> 2)
+					bytes[j++] = (c2 << 6) | c3
+				}
+				return bytes
+			}
+		})()
+
+		// These are for the "using" statement in TypeScript 5.2+
+		export var __using = (stack, value, async) => {
+			if (value != null) {
+				if (typeof value !== 'object' && typeof value !== 'function') __typeError('Object expected')
+				var dispose, inner
+				if (async) dispose = value[__knownSymbol('asyncDispose')]
+				if (dispose === void 0) {
+					dispose = value[__knownSymbol('dispose')]
+					if (async) inner = dispose
+				}
+				if (typeof dispose !== 'function') __typeError('Object not disposable')
+				if (inner) dispose = function() { try { inner.call(this) } catch (e) { return Promise.reject(e) } }
+				stack.push([async, dispose, value])
+			} else if (async) {
+				stack.push([async])
+			}
+			return value
+		}
+		export var __callDispose = (stack, error, hasError) => {
+			var E = typeof SuppressedError === 'function' ? SuppressedError :
+				function (e, s, m, _) { return _ = Error(m), _.name = 'SuppressedError', _.error = e, _.suppressed = s, _ }
+			var fail = e => error = hasError ? new E(e, error, 'An error was suppressed during disposal') : (hasError = true, e)
+			var next = (it) => {
+				while (it = stack.pop()) {
+					try {
+						var result = it[1] && it[1].call(it[2])
+						if (it[0]) return Promise.resolve(result).then(next, (e) => (fail(e), next()))
+					} catch (e) {
+						fail(e)
+					}
+				}
+				if (hasError) throw error
+			}
+			return next()
+		}
+	`
+
+	return logger.Source{
+		Index:          SourceIndex,
+		KeyPath:        logger.Path{Text: "<runtime>"},
+		PrettyPath:     "<runtime>",
+		IdentifierName: "runtime",
+		Contents:       text,
+	}
+}
+
+// The TypeScript decorator transform behaves similar to the official
+// TypeScript compiler.
+//
+// One difference is that the "__decorateClass" function doesn't contain a reference
+// to the non-existent "Reflect.decorate" function. This function was never
+// standardized and checking for it is wasted code (as well as a potentially
+// dangerous cause of unintentional behavior changes in the future).
+//
+// Another difference is that the "__decorateClass" function doesn't take in an
+// optional property descriptor like it does in the official TypeScript
+// compiler's support code. This appears to be a dead code path in the official
+// support code that is only there for legacy reasons.
+//
+// Here are some examples of how esbuild's decorator transform works:
+//
+// ============================= Class decorator ==============================
+//
+//   // TypeScript                      // JavaScript
+//   @dec                               let C = class {
+//   class C {                          };
+//   }                                  C = __decorateClass([
+//                                        dec
+//                                      ], C);
+//
+// ============================ Method decorator ==============================
+//
+//   // TypeScript                      // JavaScript
+//   class C {                          class C {
+//     @dec                               foo() {}
+//     foo() {}                         }
+//   }                                  __decorateClass([
+//                                        dec
+//                                      ], C.prototype, 'foo', 1);
+//
+// =========================== Parameter decorator ============================
+//
+//   // TypeScript                      // JavaScript
+//   class C {                          class C {
+//     foo(@dec bar) {}                   foo(bar) {}
+//   }                                  }
+//                                      __decorateClass([
+//                                        __decorateParam(0, dec)
+//                                      ], C.prototype, 'foo', 1);
+//
+// ============================= Field decorator ==============================
+//
+//   // TypeScript                      // JavaScript
+//   class C {                          class C {
+//     @dec                               constructor() {
+//     foo = 123                            this.foo = 123
+//   }                                    }
+//                                      }
+//                                      __decorateClass([
+//                                        dec
+//                                      ], C.prototype, 'foo', 2);
diff --git a/source/vendor/github.com/evanw/esbuild/internal/sourcemap/sourcemap.go b/source/vendor/github.com/evanw/esbuild/internal/sourcemap/sourcemap.go
new file mode 100644
index 0000000..93effc2
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/sourcemap/sourcemap.go
@@ -0,0 +1,834 @@
+package sourcemap
+
+import (
+	"bytes"
+	"unicode/utf8"
+
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type Mapping struct {
+	GeneratedLine   int32 // 0-based
+	GeneratedColumn int32 // 0-based count of UTF-16 code units
+
+	SourceIndex    int32       // 0-based
+	OriginalLine   int32       // 0-based
+	OriginalColumn int32       // 0-based count of UTF-16 code units
+	OriginalName   ast.Index32 // 0-based, optional
+}
+
+type SourceMap struct {
+	Sources        []string
+	SourcesContent []SourceContent
+	Mappings       []Mapping
+	Names          []string
+}
+
+type SourceContent struct {
+	// This stores both the unquoted and the quoted values. We try to use the
+	// already-quoted value if possible so we don't need to re-quote it
+	// unnecessarily for maximum performance.
+	Quoted string
+
+	// But sometimes we need to re-quote the value, such as when it contains
+	// non-ASCII characters and we are in ASCII-only mode. In that case we quote
+	// this parsed UTF-16 value.
+	Value []uint16
+}
+
+func (sm *SourceMap) Find(line int32, column int32) *Mapping {
+	mappings := sm.Mappings
+
+	// Binary search
+	count := len(mappings)
+	index := 0
+	for count > 0 {
+		step := count / 2
+		i := index + step
+		mapping := mappings[i]
+		if mapping.GeneratedLine < line || (mapping.GeneratedLine == line && mapping.GeneratedColumn <= column) {
+			index = i + 1
+			count -= step + 1
+		} else {
+			count = step
+		}
+	}
+
+	// Handle search failure
+	if index > 0 {
+		mapping := &mappings[index-1]
+
+		// Match the behavior of the popular "source-map" library from Mozilla
+		if mapping.GeneratedLine == line {
+			return mapping
+		}
+	}
+	return nil
+}
+
+var base64 = []byte("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/")
+
+// A single base 64 digit can contain 6 bits of data. For the base 64 variable
+// length quantities we use in the source map spec, the first bit is the sign,
+// the next four bits are the actual value, and the 6th bit is the continuation
+// bit. The continuation bit tells us whether there are more digits in this
+// value following this digit.
+//
+//	Continuation
+//	|    Sign
+//	|    |
+//	V    V
+//	101011
+func encodeVLQ(encoded []byte, value int) []byte {
+	var vlq int
+	if value < 0 {
+		vlq = ((-value) << 1) | 1
+	} else {
+		vlq = value << 1
+	}
+
+	// Handle the common case
+	if (vlq >> 5) == 0 {
+		digit := vlq & 31
+		encoded = append(encoded, base64[digit])
+		return encoded
+	}
+
+	for {
+		digit := vlq & 31
+		vlq >>= 5
+
+		// If there are still more digits in this value, we must make sure the
+		// continuation bit is marked
+		if vlq != 0 {
+			digit |= 32
+		}
+
+		encoded = append(encoded, base64[digit])
+
+		if vlq == 0 {
+			break
+		}
+	}
+
+	return encoded
+}
+
+func DecodeVLQ(encoded []byte, start int) (int, int) {
+	shift := 0
+	vlq := 0
+
+	// Scan over the input
+	for {
+		index := bytes.IndexByte(base64, encoded[start])
+		if index < 0 {
+			break
+		}
+
+		// Decode a single byte
+		vlq |= (index & 31) << shift
+		start++
+		shift += 5
+
+		// Stop if there's no continuation bit
+		if (index & 32) == 0 {
+			break
+		}
+	}
+
+	// Recover the value
+	value := vlq >> 1
+	if (vlq & 1) != 0 {
+		value = -value
+	}
+	return value, start
+}
+
+func DecodeVLQUTF16(encoded []uint16) (int32, int, bool) {
+	n := len(encoded)
+	if n == 0 {
+		return 0, 0, false
+	}
+
+	// Scan over the input
+	current := 0
+	shift := 0
+	var vlq int32
+	for {
+		if current >= n {
+			return 0, 0, false
+		}
+		index := int32(bytes.IndexByte(base64, byte(encoded[current])))
+		if index < 0 {
+			return 0, 0, false
+		}
+
+		// Decode a single byte
+		vlq |= (index & 31) << shift
+		current++
+		shift += 5
+
+		// Stop if there's no continuation bit
+		if (index & 32) == 0 {
+			break
+		}
+	}
+
+	// Recover the value
+	var value = vlq >> 1
+	if (vlq & 1) != 0 {
+		value = -value
+	}
+	return value, current, true
+}
+
+type LineColumnOffset struct {
+	Lines   int
+	Columns int
+}
+
+func (a LineColumnOffset) ComesBefore(b LineColumnOffset) bool {
+	return a.Lines < b.Lines || (a.Lines == b.Lines && a.Columns < b.Columns)
+}
+
+func (a *LineColumnOffset) Add(b LineColumnOffset) {
+	if b.Lines == 0 {
+		a.Columns += b.Columns
+	} else {
+		a.Lines += b.Lines
+		a.Columns = b.Columns
+	}
+}
+
+func (offset *LineColumnOffset) AdvanceBytes(bytes []byte) {
+	columns := offset.Columns
+	for len(bytes) > 0 {
+		c, width := utf8.DecodeRune(bytes)
+		bytes = bytes[width:]
+		switch c {
+		case '\r', '\n', '\u2028', '\u2029':
+			// Handle Windows-specific "\r\n" newlines
+			if c == '\r' && len(bytes) > 0 && bytes[0] == '\n' {
+				columns++
+				continue
+			}
+
+			offset.Lines++
+			columns = 0
+
+		default:
+			// Mozilla's "source-map" library counts columns using UTF-16 code units
+			if c <= 0xFFFF {
+				columns++
+			} else {
+				columns += 2
+			}
+		}
+	}
+	offset.Columns = columns
+}
+
+func (offset *LineColumnOffset) AdvanceString(text string) {
+	columns := offset.Columns
+	for i, c := range text {
+		switch c {
+		case '\r', '\n', '\u2028', '\u2029':
+			// Handle Windows-specific "\r\n" newlines
+			if c == '\r' && i+1 < len(text) && text[i+1] == '\n' {
+				columns++
+				continue
+			}
+
+			offset.Lines++
+			columns = 0
+
+		default:
+			// Mozilla's "source-map" library counts columns using UTF-16 code units
+			if c <= 0xFFFF {
+				columns++
+			} else {
+				columns += 2
+			}
+		}
+	}
+	offset.Columns = columns
+}
+
+type SourceMapPieces struct {
+	Prefix   []byte
+	Mappings []byte
+	Suffix   []byte
+}
+
+func (pieces SourceMapPieces) HasContent() bool {
+	return len(pieces.Prefix)+len(pieces.Mappings)+len(pieces.Suffix) > 0
+}
+
+type SourceMapShift struct {
+	Before LineColumnOffset
+	After  LineColumnOffset
+}
+
+func (pieces SourceMapPieces) Finalize(shifts []SourceMapShift) []byte {
+	// An optimized path for when there are no shifts
+	if len(shifts) == 1 {
+		bytes := pieces.Prefix
+		minCap := len(bytes) + len(pieces.Mappings) + len(pieces.Suffix)
+		if cap(bytes) < minCap {
+			bytes = append(make([]byte, 0, minCap), bytes...)
+		}
+		bytes = append(bytes, pieces.Mappings...)
+		bytes = append(bytes, pieces.Suffix...)
+		return bytes
+	}
+
+	startOfRun := 0
+	current := 0
+	generated := LineColumnOffset{}
+	prevShiftColumnDelta := 0
+	j := helpers.Joiner{}
+
+	// Start the source map
+	j.AddBytes(pieces.Prefix)
+
+	// This assumes that a) all mappings are valid and b) all mappings are ordered
+	// by increasing generated position. This should be the case for all mappings
+	// generated by esbuild, which should be the only mappings we process here.
+	for current < len(pieces.Mappings) {
+		// Handle a line break
+		if pieces.Mappings[current] == ';' {
+			generated.Lines++
+			generated.Columns = 0
+			prevShiftColumnDelta = 0
+			current++
+			continue
+		}
+
+		potentialEndOfRun := current
+
+		// Read the generated column
+		generatedColumnDelta, next := DecodeVLQ(pieces.Mappings, current)
+		generated.Columns += generatedColumnDelta
+		current = next
+
+		potentialStartOfRun := current
+
+		// Skip over the original position information
+		_, current = DecodeVLQ(pieces.Mappings, current) // The original source
+		_, current = DecodeVLQ(pieces.Mappings, current) // The original line
+		_, current = DecodeVLQ(pieces.Mappings, current) // The original column
+
+		// Skip over the original name
+		if current < len(pieces.Mappings) {
+			if c := pieces.Mappings[current]; c != ',' && c != ';' {
+				_, current = DecodeVLQ(pieces.Mappings, current)
+			}
+		}
+
+		// Skip a trailing comma
+		if current < len(pieces.Mappings) && pieces.Mappings[current] == ',' {
+			current++
+		}
+
+		// Detect crossing shift boundaries
+		didCrossBoundary := false
+		for len(shifts) > 1 && shifts[1].Before.ComesBefore(generated) {
+			shifts = shifts[1:]
+			didCrossBoundary = true
+		}
+		if !didCrossBoundary {
+			continue
+		}
+
+		// This shift isn't relevant if the next mapping after this shift is on a
+		// following line. In that case, don't split and keep scanning instead.
+		shift := shifts[0]
+		if shift.After.Lines != generated.Lines {
+			continue
+		}
+
+		// Add all previous mappings in a single run for efficiency. Since source
+		// mappings are relative, no data needs to be modified inside this run.
+		j.AddBytes(pieces.Mappings[startOfRun:potentialEndOfRun])
+
+		// Then modify the first mapping across the shift boundary with the updated
+		// generated column value. It's simplest to only support column shifts. This
+		// is reasonable because import paths should not contain newlines.
+		if shift.Before.Lines != shift.After.Lines {
+			panic("Unexpected line change when shifting source maps")
+		}
+		shiftColumnDelta := shift.After.Columns - shift.Before.Columns
+		j.AddBytes(encodeVLQ(nil, generatedColumnDelta+shiftColumnDelta-prevShiftColumnDelta))
+		prevShiftColumnDelta = shiftColumnDelta
+
+		// Finally, start the next run after the end of this generated column offset
+		startOfRun = potentialStartOfRun
+	}
+
+	// Finish the source map
+	j.AddBytes(pieces.Mappings[startOfRun:])
+	j.AddBytes(pieces.Suffix)
+	return j.Done()
+}
+
+// Coordinates in source maps are stored using relative offsets for size
+// reasons. When joining together chunks of a source map that were emitted
+// in parallel for different parts of a file, we need to fix up the first
+// segment of each chunk to be relative to the end of the previous chunk.
+type SourceMapState struct {
+	// This isn't stored in the source map. It's only used by the bundler to join
+	// source map chunks together correctly.
+	GeneratedLine int
+
+	// These are stored in the source map in VLQ format.
+	GeneratedColumn int
+	SourceIndex     int
+	OriginalLine    int
+	OriginalColumn  int
+	OriginalName    int
+	HasOriginalName bool
+}
+
+// Source map chunks are computed in parallel for speed. Each chunk is relative
+// to the zero state instead of being relative to the end state of the previous
+// chunk, since it's impossible to know the end state of the previous chunk in
+// a parallel computation.
+//
+// After all chunks are computed, they are joined together in a second pass.
+// This rewrites the first mapping in each chunk to be relative to the end
+// state of the previous chunk.
+func AppendSourceMapChunk(j *helpers.Joiner, prevEndState SourceMapState, startState SourceMapState, buffer MappingsBuffer) {
+	// Handle line breaks in between this mapping and the previous one
+	if startState.GeneratedLine != 0 {
+		j.AddBytes(bytes.Repeat([]byte{';'}, startState.GeneratedLine))
+		prevEndState.GeneratedColumn = 0
+	}
+
+	// Skip past any leading semicolons, which indicate line breaks
+	semicolons := 0
+	for buffer.Data[semicolons] == ';' {
+		semicolons++
+	}
+	if semicolons > 0 {
+		j.AddBytes(buffer.Data[:semicolons])
+		prevEndState.GeneratedColumn = 0
+		startState.GeneratedColumn = 0
+	}
+
+	// Strip off the first mapping from the buffer. The first mapping should be
+	// for the start of the original file (the printer always generates one for
+	// the start of the file).
+	//
+	// Note that we do not want to strip off the original name, even though it
+	// could be a part of the first mapping. This will be handled using a special
+	// case below instead. Original names are optional and are often omitted, so
+	// we handle it uniformly by saving an index to the first original name,
+	// which may or may not be a part of the first mapping.
+	generatedColumn, i := DecodeVLQ(buffer.Data, semicolons)
+	sourceIndex, i := DecodeVLQ(buffer.Data, i)
+	originalLine, i := DecodeVLQ(buffer.Data, i)
+	originalColumn, i := DecodeVLQ(buffer.Data, i)
+
+	// Rewrite the first mapping to be relative to the end state of the previous
+	// chunk. We now know what the end state is because we're in the second pass
+	// where all chunks have already been generated.
+	startState.SourceIndex += sourceIndex
+	startState.GeneratedColumn += generatedColumn
+	startState.OriginalLine += originalLine
+	startState.OriginalColumn += originalColumn
+	prevEndState.HasOriginalName = false // This is handled separately below
+	rewritten, _ := appendMappingToBuffer(nil, j.LastByte(), prevEndState, startState)
+	j.AddBytes(rewritten)
+
+	// Next, if there's an original name, we need to rewrite that as well to be
+	// relative to that of the previous chunk.
+	if buffer.FirstNameOffset.IsValid() {
+		before := int(buffer.FirstNameOffset.GetIndex())
+		originalName, after := DecodeVLQ(buffer.Data, before)
+		originalName += startState.OriginalName - prevEndState.OriginalName
+		j.AddBytes(buffer.Data[i:before])
+		j.AddBytes(encodeVLQ(nil, originalName))
+		j.AddBytes(buffer.Data[after:])
+		return
+	}
+
+	// Otherwise, just append everything after that without modification
+	j.AddBytes(buffer.Data[i:])
+}
+
+func appendMappingToBuffer(buffer []byte, lastByte byte, prevState SourceMapState, currentState SourceMapState) ([]byte, ast.Index32) {
+	// Put commas in between mappings
+	if lastByte != 0 && lastByte != ';' && lastByte != '"' {
+		buffer = append(buffer, ',')
+	}
+
+	// Record the mapping (note that the generated line is recorded using ';' elsewhere)
+	buffer = encodeVLQ(buffer, currentState.GeneratedColumn-prevState.GeneratedColumn)
+	buffer = encodeVLQ(buffer, currentState.SourceIndex-prevState.SourceIndex)
+	buffer = encodeVLQ(buffer, currentState.OriginalLine-prevState.OriginalLine)
+	buffer = encodeVLQ(buffer, currentState.OriginalColumn-prevState.OriginalColumn)
+
+	// Record the optional original name
+	var nameOffset ast.Index32
+	if currentState.HasOriginalName {
+		nameOffset = ast.MakeIndex32(uint32(len(buffer)))
+		buffer = encodeVLQ(buffer, currentState.OriginalName-prevState.OriginalName)
+	}
+
+	return buffer, nameOffset
+}
+
+type LineOffsetTable struct {
+	// The source map specification is very loose and does not specify what
+	// column numbers actually mean. The popular "source-map" library from Mozilla
+	// appears to interpret them as counts of UTF-16 code units, so we generate
+	// those too for compatibility.
+	//
+	// We keep mapping tables around to accelerate conversion from byte offsets
+	// to UTF-16 code unit counts. However, this mapping takes up a lot of memory
+	// and generates a lot of garbage. Since most JavaScript is ASCII and the
+	// mapping for ASCII is 1:1, we avoid creating a table for ASCII-only lines
+	// as an optimization.
+	columnsForNonASCII        []int32
+	byteOffsetToFirstNonASCII int32
+
+	byteOffsetToStartOfLine int32
+}
+
+func GenerateLineOffsetTables(contents string, approximateLineCount int32) []LineOffsetTable {
+	var ColumnsForNonASCII []int32
+	ByteOffsetToFirstNonASCII := int32(0)
+	lineByteOffset := 0
+	columnByteOffset := 0
+	column := int32(0)
+
+	// Preallocate the top-level table using the approximate line count from the lexer
+	lineOffsetTables := make([]LineOffsetTable, 0, approximateLineCount)
+
+	for i, c := range contents {
+		// Mark the start of the next line
+		if column == 0 {
+			lineByteOffset = i
+		}
+
+		// Start the mapping if this character is non-ASCII
+		if c > 0x7F && ColumnsForNonASCII == nil {
+			columnByteOffset = i - lineByteOffset
+			ByteOffsetToFirstNonASCII = int32(columnByteOffset)
+			ColumnsForNonASCII = []int32{}
+		}
+
+		// Update the per-byte column offsets
+		if ColumnsForNonASCII != nil {
+			for lineBytesSoFar := i - lineByteOffset; columnByteOffset <= lineBytesSoFar; columnByteOffset++ {
+				ColumnsForNonASCII = append(ColumnsForNonASCII, column)
+			}
+		}
+
+		switch c {
+		case '\r', '\n', '\u2028', '\u2029':
+			// Handle Windows-specific "\r\n" newlines
+			if c == '\r' && i+1 < len(contents) && contents[i+1] == '\n' {
+				column++
+				continue
+			}
+
+			lineOffsetTables = append(lineOffsetTables, LineOffsetTable{
+				byteOffsetToStartOfLine:   int32(lineByteOffset),
+				byteOffsetToFirstNonASCII: ByteOffsetToFirstNonASCII,
+				columnsForNonASCII:        ColumnsForNonASCII,
+			})
+			columnByteOffset = 0
+			ByteOffsetToFirstNonASCII = 0
+			ColumnsForNonASCII = nil
+			column = 0
+
+		default:
+			// Mozilla's "source-map" library counts columns using UTF-16 code units
+			if c <= 0xFFFF {
+				column++
+			} else {
+				column += 2
+			}
+		}
+	}
+
+	// Mark the start of the next line
+	if column == 0 {
+		lineByteOffset = len(contents)
+	}
+
+	// Do one last update for the column at the end of the file
+	if ColumnsForNonASCII != nil {
+		for lineBytesSoFar := len(contents) - lineByteOffset; columnByteOffset <= lineBytesSoFar; columnByteOffset++ {
+			ColumnsForNonASCII = append(ColumnsForNonASCII, column)
+		}
+	}
+
+	lineOffsetTables = append(lineOffsetTables, LineOffsetTable{
+		byteOffsetToStartOfLine:   int32(lineByteOffset),
+		byteOffsetToFirstNonASCII: ByteOffsetToFirstNonASCII,
+		columnsForNonASCII:        ColumnsForNonASCII,
+	})
+	return lineOffsetTables
+}
+
+type MappingsBuffer struct {
+	Data            []byte
+	FirstNameOffset ast.Index32
+}
+
+type Chunk struct {
+	Buffer      MappingsBuffer
+	QuotedNames [][]byte
+
+	// This end state will be used to rewrite the start of the following source
+	// map chunk so that the delta-encoded VLQ numbers are preserved.
+	EndState SourceMapState
+
+	// There probably isn't a source mapping at the end of the file (nor should
+	// there be) but if we're appending another source map chunk after this one,
+	// we'll need to know how many characters were in the last line we generated.
+	FinalGeneratedColumn int
+
+	ShouldIgnore bool
+}
+
+type ChunkBuilder struct {
+	inputSourceMap      *SourceMap
+	sourceMap           []byte
+	quotedNames         [][]byte
+	namesMap            map[string]uint32
+	lineOffsetTables    []LineOffsetTable
+	prevOriginalName    string
+	prevState           SourceMapState
+	lastGeneratedUpdate int
+	generatedColumn     int
+	prevGeneratedLen    int
+	prevOriginalLoc     logger.Loc
+	firstNameOffset     ast.Index32
+	hasPrevState        bool
+	asciiOnly           bool
+
+	// This is a workaround for a bug in the popular "source-map" library:
+	// https://github.com/mozilla/source-map/issues/261. The library will
+	// sometimes return null when querying a source map unless every line
+	// starts with a mapping at column zero.
+	//
+	// The workaround is to replicate the previous mapping if a line ends
+	// up not starting with a mapping. This is done lazily because we want
+	// to avoid replicating the previous mapping if we don't need to.
+	lineStartsWithMapping     bool
+	coverLinesWithoutMappings bool
+}
+
+func MakeChunkBuilder(inputSourceMap *SourceMap, lineOffsetTables []LineOffsetTable, asciiOnly bool) ChunkBuilder {
+	return ChunkBuilder{
+		inputSourceMap:   inputSourceMap,
+		prevOriginalLoc:  logger.Loc{Start: -1},
+		lineOffsetTables: lineOffsetTables,
+		asciiOnly:        asciiOnly,
+		namesMap:         make(map[string]uint32),
+
+		// We automatically repeat the previous source mapping if we ever generate
+		// a line that doesn't start with a mapping. This helps give files more
+		// complete mapping coverage without gaps.
+		//
+		// However, we probably shouldn't do this if the input file has a nested
+		// source map that we will be remapping through. We have no idea what state
+		// that source map is in and it could be pretty scrambled.
+		//
+		// I've seen cases where blindly repeating the last mapping for subsequent
+		// lines gives very strange and unhelpful results with source maps from
+		// other tools.
+		coverLinesWithoutMappings: inputSourceMap == nil,
+	}
+}
+
+func (b *ChunkBuilder) AddSourceMapping(originalLoc logger.Loc, originalName string, output []byte) {
+	// Avoid generating duplicate mappings
+	if originalLoc == b.prevOriginalLoc && (b.prevGeneratedLen == len(output) || b.prevOriginalName == originalName) {
+		return
+	}
+
+	b.prevOriginalLoc = originalLoc
+	b.prevGeneratedLen = len(output)
+	b.prevOriginalName = originalName
+
+	// Binary search to find the line
+	lineOffsetTables := b.lineOffsetTables
+	count := len(lineOffsetTables)
+	originalLine := 0
+	for count > 0 {
+		step := count / 2
+		i := originalLine + step
+		if lineOffsetTables[i].byteOffsetToStartOfLine <= originalLoc.Start {
+			originalLine = i + 1
+			count = count - step - 1
+		} else {
+			count = step
+		}
+	}
+	originalLine--
+
+	// Use the line to compute the column
+	line := &lineOffsetTables[originalLine]
+	originalColumn := int(originalLoc.Start - line.byteOffsetToStartOfLine)
+	if line.columnsForNonASCII != nil && originalColumn >= int(line.byteOffsetToFirstNonASCII) {
+		originalColumn = int(line.columnsForNonASCII[originalColumn-int(line.byteOffsetToFirstNonASCII)])
+	}
+
+	b.updateGeneratedLineAndColumn(output)
+
+	// If this line doesn't start with a mapping and we're about to add a mapping
+	// that's not at the start, insert a mapping first so the line starts with one.
+	if b.coverLinesWithoutMappings && !b.lineStartsWithMapping && b.generatedColumn > 0 && b.hasPrevState {
+		b.appendMappingWithoutRemapping(SourceMapState{
+			GeneratedLine:   b.prevState.GeneratedLine,
+			GeneratedColumn: 0,
+			SourceIndex:     b.prevState.SourceIndex,
+			OriginalLine:    b.prevState.OriginalLine,
+			OriginalColumn:  b.prevState.OriginalColumn,
+		})
+	}
+
+	b.appendMapping(originalName, SourceMapState{
+		GeneratedLine:   b.prevState.GeneratedLine,
+		GeneratedColumn: b.generatedColumn,
+		OriginalLine:    originalLine,
+		OriginalColumn:  originalColumn,
+	})
+
+	// This line now has a mapping on it, so don't insert another one
+	b.lineStartsWithMapping = true
+}
+
+func (b *ChunkBuilder) GenerateChunk(output []byte) Chunk {
+	b.updateGeneratedLineAndColumn(output)
+	shouldIgnore := true
+	for _, c := range b.sourceMap {
+		if c != ';' {
+			shouldIgnore = false
+			break
+		}
+	}
+	return Chunk{
+		Buffer: MappingsBuffer{
+			Data:            b.sourceMap,
+			FirstNameOffset: b.firstNameOffset,
+		},
+		QuotedNames:          b.quotedNames,
+		EndState:             b.prevState,
+		FinalGeneratedColumn: b.generatedColumn,
+		ShouldIgnore:         shouldIgnore,
+	}
+}
+
+// Scan over the printed text since the last source mapping and update the
+// generated line and column numbers
+func (b *ChunkBuilder) updateGeneratedLineAndColumn(output []byte) {
+	for i, c := range string(output[b.lastGeneratedUpdate:]) {
+		switch c {
+		case '\r', '\n', '\u2028', '\u2029':
+			// Handle Windows-specific "\r\n" newlines
+			if c == '\r' {
+				newlineCheck := b.lastGeneratedUpdate + i + 1
+				if newlineCheck < len(output) && output[newlineCheck] == '\n' {
+					continue
+				}
+			}
+
+			// If we're about to move to the next line and the previous line didn't have
+			// any mappings, add a mapping at the start of the previous line.
+			if b.coverLinesWithoutMappings && !b.lineStartsWithMapping && b.hasPrevState {
+				b.appendMappingWithoutRemapping(SourceMapState{
+					GeneratedLine:   b.prevState.GeneratedLine,
+					GeneratedColumn: 0,
+					SourceIndex:     b.prevState.SourceIndex,
+					OriginalLine:    b.prevState.OriginalLine,
+					OriginalColumn:  b.prevState.OriginalColumn,
+				})
+			}
+
+			b.prevState.GeneratedLine++
+			b.prevState.GeneratedColumn = 0
+			b.generatedColumn = 0
+			b.sourceMap = append(b.sourceMap, ';')
+
+			// This new line doesn't have a mapping yet
+			b.lineStartsWithMapping = false
+
+		default:
+			// Mozilla's "source-map" library counts columns using UTF-16 code units
+			if c <= 0xFFFF {
+				b.generatedColumn++
+			} else {
+				b.generatedColumn += 2
+			}
+		}
+	}
+
+	b.lastGeneratedUpdate = len(output)
+}
+
+func (b *ChunkBuilder) appendMapping(originalName string, currentState SourceMapState) {
+	// If the input file had a source map, map all the way back to the original
+	if b.inputSourceMap != nil {
+		mapping := b.inputSourceMap.Find(
+			int32(currentState.OriginalLine),
+			int32(currentState.OriginalColumn))
+
+		// Some locations won't have a mapping
+		if mapping == nil {
+			return
+		}
+
+		currentState.SourceIndex = int(mapping.SourceIndex)
+		currentState.OriginalLine = int(mapping.OriginalLine)
+		currentState.OriginalColumn = int(mapping.OriginalColumn)
+
+		// Map all the way back to the original name if present. Otherwise, keep
+		// the original name from esbuild, which corresponds to the name in the
+		// intermediate source code. This is important for tools that only emit
+		// a name mapping when the name is different than the original name.
+		if mapping.OriginalName.IsValid() {
+			originalName = b.inputSourceMap.Names[mapping.OriginalName.GetIndex()]
+		}
+	}
+
+	// Optionally reference the original name
+	if originalName != "" {
+		i, ok := b.namesMap[originalName]
+		if !ok {
+			i = uint32(len(b.quotedNames))
+			b.quotedNames = append(b.quotedNames, helpers.QuoteForJSON(originalName, b.asciiOnly))
+			b.namesMap[originalName] = i
+		}
+		currentState.OriginalName = int(i)
+		currentState.HasOriginalName = true
+	}
+
+	b.appendMappingWithoutRemapping(currentState)
+}
+
+func (b *ChunkBuilder) appendMappingWithoutRemapping(currentState SourceMapState) {
+	var lastByte byte
+	if len(b.sourceMap) != 0 {
+		lastByte = b.sourceMap[len(b.sourceMap)-1]
+	}
+
+	var nameOffset ast.Index32
+	b.sourceMap, nameOffset = appendMappingToBuffer(b.sourceMap, lastByte, b.prevState, currentState)
+	prevOriginalName := b.prevState.OriginalName
+	b.prevState = currentState
+	if !currentState.HasOriginalName {
+		// Revert the original name change if it's invalid
+		b.prevState.OriginalName = prevOriginalName
+	} else if !b.firstNameOffset.IsValid() {
+		// Keep track of the first name offset so we can jump right to it later
+		b.firstNameOffset = nameOffset
+	}
+	b.hasPrevState = true
+}
diff --git a/source/vendor/github.com/evanw/esbuild/internal/xxhash/LICENSE.txt b/source/vendor/github.com/evanw/esbuild/internal/xxhash/LICENSE.txt
new file mode 100644
index 0000000..24b5306
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/xxhash/LICENSE.txt
@@ -0,0 +1,22 @@
+Copyright (c) 2016 Caleb Spare
+
+MIT License
+
+Permission is hereby granted, free of charge, to any person obtaining
+a copy of this software and associated documentation files (the
+"Software"), to deal in the Software without restriction, including
+without limitation the rights to use, copy, modify, merge, publish,
+distribute, sublicense, and/or sell copies of the Software, and to
+permit persons to whom the Software is furnished to do so, subject to
+the following conditions:
+
+The above copyright notice and this permission notice shall be
+included in all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
+EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
diff --git a/source/vendor/github.com/evanw/esbuild/internal/xxhash/README.md b/source/vendor/github.com/evanw/esbuild/internal/xxhash/README.md
new file mode 100644
index 0000000..1c9f0af
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/xxhash/README.md
@@ -0,0 +1 @@
+This Go implementation of xxHash is from https://github.com/cespare/xxhash.
diff --git a/source/vendor/github.com/evanw/esbuild/internal/xxhash/xxhash.go b/source/vendor/github.com/evanw/esbuild/internal/xxhash/xxhash.go
new file mode 100644
index 0000000..15c835d
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/xxhash/xxhash.go
@@ -0,0 +1,235 @@
+// Package xxhash implements the 64-bit variant of xxHash (XXH64) as described
+// at http://cyan4973.github.io/xxHash/.
+package xxhash
+
+import (
+	"encoding/binary"
+	"errors"
+	"math/bits"
+)
+
+const (
+	prime1 uint64 = 11400714785074694791
+	prime2 uint64 = 14029467366897019727
+	prime3 uint64 = 1609587929392839161
+	prime4 uint64 = 9650029242287828579
+	prime5 uint64 = 2870177450012600261
+)
+
+// NOTE(caleb): I'm using both consts and vars of the primes. Using consts where
+// possible in the Go code is worth a small (but measurable) performance boost
+// by avoiding some MOVQs. Vars are needed for the asm and also are useful for
+// convenience in the Go code in a few places where we need to intentionally
+// avoid constant arithmetic (e.g., v1 := prime1 + prime2 fails because the
+// result overflows a uint64).
+var (
+	prime1v = prime1
+	prime2v = prime2
+	prime3v = prime3
+	prime4v = prime4
+	prime5v = prime5
+)
+
+// Digest implements hash.Hash64.
+type Digest struct {
+	v1    uint64
+	v2    uint64
+	v3    uint64
+	v4    uint64
+	total uint64
+	mem   [32]byte
+	n     int // how much of mem is used
+}
+
+// New creates a new Digest that computes the 64-bit xxHash algorithm.
+func New() *Digest {
+	var d Digest
+	d.Reset()
+	return &d
+}
+
+// Reset clears the Digest's state so that it can be reused.
+func (d *Digest) Reset() {
+	d.v1 = prime1v + prime2
+	d.v2 = prime2
+	d.v3 = 0
+	d.v4 = -prime1v
+	d.total = 0
+	d.n = 0
+}
+
+// Size always returns 8 bytes.
+func (d *Digest) Size() int { return 8 }
+
+// BlockSize always returns 32 bytes.
+func (d *Digest) BlockSize() int { return 32 }
+
+// Write adds more data to d. It always returns len(b), nil.
+func (d *Digest) Write(b []byte) (n int, err error) {
+	n = len(b)
+	d.total += uint64(n)
+
+	if d.n+n < 32 {
+		// This new data doesn't even fill the current block.
+		copy(d.mem[d.n:], b)
+		d.n += n
+		return
+	}
+
+	if d.n > 0 {
+		// Finish off the partial block.
+		copy(d.mem[d.n:], b)
+		d.v1 = round(d.v1, u64(d.mem[0:8]))
+		d.v2 = round(d.v2, u64(d.mem[8:16]))
+		d.v3 = round(d.v3, u64(d.mem[16:24]))
+		d.v4 = round(d.v4, u64(d.mem[24:32]))
+		b = b[32-d.n:]
+		d.n = 0
+	}
+
+	if len(b) >= 32 {
+		// One or more full blocks left.
+		nw := writeBlocks(d, b)
+		b = b[nw:]
+	}
+
+	// Store any remaining partial block.
+	copy(d.mem[:], b)
+	d.n = len(b)
+
+	return
+}
+
+// Sum appends the current hash to b and returns the resulting slice.
+func (d *Digest) Sum(b []byte) []byte {
+	s := d.Sum64()
+	return append(
+		b,
+		byte(s>>56),
+		byte(s>>48),
+		byte(s>>40),
+		byte(s>>32),
+		byte(s>>24),
+		byte(s>>16),
+		byte(s>>8),
+		byte(s),
+	)
+}
+
+// Sum64 returns the current hash.
+func (d *Digest) Sum64() uint64 {
+	var h uint64
+
+	if d.total >= 32 {
+		v1, v2, v3, v4 := d.v1, d.v2, d.v3, d.v4
+		h = rol1(v1) + rol7(v2) + rol12(v3) + rol18(v4)
+		h = mergeRound(h, v1)
+		h = mergeRound(h, v2)
+		h = mergeRound(h, v3)
+		h = mergeRound(h, v4)
+	} else {
+		h = d.v3 + prime5
+	}
+
+	h += d.total
+
+	i, end := 0, d.n
+	for ; i+8 <= end; i += 8 {
+		k1 := round(0, u64(d.mem[i:i+8]))
+		h ^= k1
+		h = rol27(h)*prime1 + prime4
+	}
+	if i+4 <= end {
+		h ^= uint64(u32(d.mem[i:i+4])) * prime1
+		h = rol23(h)*prime2 + prime3
+		i += 4
+	}
+	for i < end {
+		h ^= uint64(d.mem[i]) * prime5
+		h = rol11(h) * prime1
+		i++
+	}
+
+	h ^= h >> 33
+	h *= prime2
+	h ^= h >> 29
+	h *= prime3
+	h ^= h >> 32
+
+	return h
+}
+
+const (
+	magic         = "xxh\x06"
+	marshaledSize = len(magic) + 8*5 + 32
+)
+
+// MarshalBinary implements the encoding.BinaryMarshaler interface.
+func (d *Digest) MarshalBinary() ([]byte, error) {
+	b := make([]byte, 0, marshaledSize)
+	b = append(b, magic...)
+	b = appendUint64(b, d.v1)
+	b = appendUint64(b, d.v2)
+	b = appendUint64(b, d.v3)
+	b = appendUint64(b, d.v4)
+	b = appendUint64(b, d.total)
+	b = append(b, d.mem[:d.n]...)
+	b = b[:len(b)+len(d.mem)-d.n]
+	return b, nil
+}
+
+// UnmarshalBinary implements the encoding.BinaryUnmarshaler interface.
+func (d *Digest) UnmarshalBinary(b []byte) error {
+	if len(b) < len(magic) || string(b[:len(magic)]) != magic {
+		return errors.New("xxhash: invalid hash state identifier")
+	}
+	if len(b) != marshaledSize {
+		return errors.New("xxhash: invalid hash state size")
+	}
+	b = b[len(magic):]
+	b, d.v1 = consumeUint64(b)
+	b, d.v2 = consumeUint64(b)
+	b, d.v3 = consumeUint64(b)
+	b, d.v4 = consumeUint64(b)
+	b, d.total = consumeUint64(b)
+	copy(d.mem[:], b)
+	d.n = int(d.total % uint64(len(d.mem)))
+	return nil
+}
+
+func appendUint64(b []byte, x uint64) []byte {
+	var a [8]byte
+	binary.LittleEndian.PutUint64(a[:], x)
+	return append(b, a[:]...)
+}
+
+func consumeUint64(b []byte) ([]byte, uint64) {
+	x := u64(b)
+	return b[8:], x
+}
+
+func u64(b []byte) uint64 { return binary.LittleEndian.Uint64(b) }
+func u32(b []byte) uint32 { return binary.LittleEndian.Uint32(b) }
+
+func round(acc, input uint64) uint64 {
+	acc += input * prime2
+	acc = rol31(acc)
+	acc *= prime1
+	return acc
+}
+
+func mergeRound(acc, val uint64) uint64 {
+	val = round(0, val)
+	acc ^= val
+	acc = acc*prime1 + prime4
+	return acc
+}
+
+func rol1(x uint64) uint64  { return bits.RotateLeft64(x, 1) }
+func rol7(x uint64) uint64  { return bits.RotateLeft64(x, 7) }
+func rol11(x uint64) uint64 { return bits.RotateLeft64(x, 11) }
+func rol12(x uint64) uint64 { return bits.RotateLeft64(x, 12) }
+func rol18(x uint64) uint64 { return bits.RotateLeft64(x, 18) }
+func rol23(x uint64) uint64 { return bits.RotateLeft64(x, 23) }
+func rol27(x uint64) uint64 { return bits.RotateLeft64(x, 27) }
+func rol31(x uint64) uint64 { return bits.RotateLeft64(x, 31) }
diff --git a/source/vendor/github.com/evanw/esbuild/internal/xxhash/xxhash_other.go b/source/vendor/github.com/evanw/esbuild/internal/xxhash/xxhash_other.go
new file mode 100644
index 0000000..ce512f7
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/internal/xxhash/xxhash_other.go
@@ -0,0 +1,74 @@
+package xxhash
+
+// Sum64 computes the 64-bit xxHash digest of b.
+func Sum64(b []byte) uint64 {
+	// A simpler version would be
+	//   d := New()
+	//   d.Write(b)
+	//   return d.Sum64()
+	// but this is faster, particularly for small inputs.
+
+	n := len(b)
+	var h uint64
+
+	if n >= 32 {
+		v1 := prime1v + prime2
+		v2 := prime2
+		v3 := uint64(0)
+		v4 := -prime1v
+		for len(b) >= 32 {
+			v1 = round(v1, u64(b[0:8:len(b)]))
+			v2 = round(v2, u64(b[8:16:len(b)]))
+			v3 = round(v3, u64(b[16:24:len(b)]))
+			v4 = round(v4, u64(b[24:32:len(b)]))
+			b = b[32:len(b):len(b)]
+		}
+		h = rol1(v1) + rol7(v2) + rol12(v3) + rol18(v4)
+		h = mergeRound(h, v1)
+		h = mergeRound(h, v2)
+		h = mergeRound(h, v3)
+		h = mergeRound(h, v4)
+	} else {
+		h = prime5
+	}
+
+	h += uint64(n)
+
+	i, end := 0, len(b)
+	for ; i+8 <= end; i += 8 {
+		k1 := round(0, u64(b[i:i+8:len(b)]))
+		h ^= k1
+		h = rol27(h)*prime1 + prime4
+	}
+	if i+4 <= end {
+		h ^= uint64(u32(b[i:i+4:len(b)])) * prime1
+		h = rol23(h)*prime2 + prime3
+		i += 4
+	}
+	for ; i < end; i++ {
+		h ^= uint64(b[i]) * prime5
+		h = rol11(h) * prime1
+	}
+
+	h ^= h >> 33
+	h *= prime2
+	h ^= h >> 29
+	h *= prime3
+	h ^= h >> 32
+
+	return h
+}
+
+func writeBlocks(d *Digest, b []byte) int {
+	v1, v2, v3, v4 := d.v1, d.v2, d.v3, d.v4
+	n := len(b)
+	for len(b) >= 32 {
+		v1 = round(v1, u64(b[0:8:len(b)]))
+		v2 = round(v2, u64(b[8:16:len(b)]))
+		v3 = round(v3, u64(b[16:24:len(b)]))
+		v4 = round(v4, u64(b[24:32:len(b)]))
+		b = b[32:len(b):len(b)]
+	}
+	d.v1, d.v2, d.v3, d.v4 = v1, v2, v3, v4
+	return n - len(b)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/pkg/api/api.go b/source/vendor/github.com/evanw/esbuild/pkg/api/api.go
new file mode 100644
index 0000000..08a597e
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/pkg/api/api.go
@@ -0,0 +1,718 @@
+// This API exposes esbuild's two main operations: building and transforming.
+// It's intended for integrating esbuild into other tools as a library.
+//
+// If you are just trying to run esbuild from Go without the overhead of
+// creating a child process, there is also an API for the command-line
+// interface itself: https://pkg.go.dev/github.com/evanw/esbuild/pkg/cli.
+//
+// # Build API
+//
+// This function runs an end-to-end build operation. It takes an array of file
+// paths as entry points, parses them and all of their dependencies, and
+// returns the output files to write to the file system. The available options
+// roughly correspond to esbuild's command-line flags.
+//
+// Example usage:
+//
+//	package main
+//
+//	import (
+//	    "os"
+//
+//	    "github.com/evanw/esbuild/pkg/api"
+//	)
+//
+//	func main() {
+//	    result := api.Build(api.BuildOptions{
+//	        EntryPoints: []string{"input.js"},
+//	        Outfile:     "output.js",
+//	        Bundle:      true,
+//	        Write:       true,
+//	        LogLevel:    api.LogLevelInfo,
+//	    })
+//
+//	    if len(result.Errors) > 0 {
+//	        os.Exit(1)
+//	    }
+//	}
+//
+// # Transform API
+//
+// This function transforms a string of source code into JavaScript. It can be
+// used to minify JavaScript, convert TypeScript/JSX to JavaScript, or convert
+// newer JavaScript to older JavaScript. The available options roughly
+// correspond to esbuild's command-line flags.
+//
+// Example usage:
+//
+//	package main
+//
+//	import (
+//	    "fmt"
+//	    "os"
+//
+//	    "github.com/evanw/esbuild/pkg/api"
+//	)
+//
+//	func main() {
+//	    jsx := `
+//	        import * as React from 'react'
+//	        import * as ReactDOM from 'react-dom'
+//
+//	        ReactDOM.render(
+//	            <h1>Hello, world!</h1>,
+//	            document.getElementById('root')
+//	        );
+//	    `
+//
+//	    result := api.Transform(jsx, api.TransformOptions{
+//	        Loader: api.LoaderJSX,
+//	    })
+//
+//	    fmt.Printf("%d errors and %d warnings\n",
+//	        len(result.Errors), len(result.Warnings))
+//
+//	    os.Stdout.Write(result.Code)
+//	}
+package api
+
+import (
+	"time"
+
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+type SourceMap uint8
+
+const (
+	SourceMapNone SourceMap = iota
+	SourceMapInline
+	SourceMapLinked
+	SourceMapExternal
+	SourceMapInlineAndExternal
+)
+
+type SourcesContent uint8
+
+const (
+	SourcesContentInclude SourcesContent = iota
+	SourcesContentExclude
+)
+
+type LegalComments uint8
+
+const (
+	LegalCommentsDefault LegalComments = iota
+	LegalCommentsNone
+	LegalCommentsInline
+	LegalCommentsEndOfFile
+	LegalCommentsLinked
+	LegalCommentsExternal
+)
+
+type JSX uint8
+
+const (
+	JSXTransform JSX = iota
+	JSXPreserve
+	JSXAutomatic
+)
+
+type Target uint8
+
+const (
+	DefaultTarget Target = iota
+	ESNext
+	ES5
+	ES2015
+	ES2016
+	ES2017
+	ES2018
+	ES2019
+	ES2020
+	ES2021
+	ES2022
+	ES2023
+	ES2024
+)
+
+type Loader uint16
+
+const (
+	LoaderNone Loader = iota
+	LoaderBase64
+	LoaderBinary
+	LoaderCopy
+	LoaderCSS
+	LoaderDataURL
+	LoaderDefault
+	LoaderEmpty
+	LoaderFile
+	LoaderGlobalCSS
+	LoaderJS
+	LoaderJSON
+	LoaderJSX
+	LoaderLocalCSS
+	LoaderText
+	LoaderTS
+	LoaderTSX
+)
+
+type Platform uint8
+
+const (
+	PlatformDefault Platform = iota
+	PlatformBrowser
+	PlatformNode
+	PlatformNeutral
+)
+
+type Format uint8
+
+const (
+	FormatDefault Format = iota
+	FormatIIFE
+	FormatCommonJS
+	FormatESModule
+)
+
+type Packages uint8
+
+const (
+	PackagesDefault Packages = iota
+	PackagesBundle
+	PackagesExternal
+)
+
+type Engine struct {
+	Name    EngineName
+	Version string
+}
+
+type Location struct {
+	File       string
+	Namespace  string
+	Line       int // 1-based
+	Column     int // 0-based, in bytes
+	Length     int // in bytes
+	LineText   string
+	Suggestion string
+}
+
+type Message struct {
+	ID         string
+	PluginName string
+	Text       string
+	Location   *Location
+	Notes      []Note
+
+	// Optional user-specified data that is passed through unmodified. You can
+	// use this to stash the original error, for example.
+	Detail interface{}
+}
+
+type Note struct {
+	Text     string
+	Location *Location
+}
+
+type StderrColor uint8
+
+const (
+	ColorIfTerminal StderrColor = iota
+	ColorNever
+	ColorAlways
+)
+
+type LogLevel uint8
+
+const (
+	LogLevelSilent LogLevel = iota
+	LogLevelVerbose
+	LogLevelDebug
+	LogLevelInfo
+	LogLevelWarning
+	LogLevelError
+)
+
+type Charset uint8
+
+const (
+	CharsetDefault Charset = iota
+	CharsetASCII
+	CharsetUTF8
+)
+
+type TreeShaking uint8
+
+const (
+	TreeShakingDefault TreeShaking = iota
+	TreeShakingFalse
+	TreeShakingTrue
+)
+
+type Drop uint8
+
+const (
+	DropConsole Drop = 1 << iota
+	DropDebugger
+)
+
+type MangleQuoted uint8
+
+const (
+	MangleQuotedFalse MangleQuoted = iota
+	MangleQuotedTrue
+)
+
+////////////////////////////////////////////////////////////////////////////////
+// Build API
+
+type BuildOptions struct {
+	Color       StderrColor         // Documentation: https://esbuild.github.io/api/#color
+	LogLevel    LogLevel            // Documentation: https://esbuild.github.io/api/#log-level
+	LogLimit    int                 // Documentation: https://esbuild.github.io/api/#log-limit
+	LogOverride map[string]LogLevel // Documentation: https://esbuild.github.io/api/#log-override
+
+	Sourcemap      SourceMap      // Documentation: https://esbuild.github.io/api/#sourcemap
+	SourceRoot     string         // Documentation: https://esbuild.github.io/api/#source-root
+	SourcesContent SourcesContent // Documentation: https://esbuild.github.io/api/#sources-content
+
+	Target    Target          // Documentation: https://esbuild.github.io/api/#target
+	Engines   []Engine        // Documentation: https://esbuild.github.io/api/#target
+	Supported map[string]bool // Documentation: https://esbuild.github.io/api/#supported
+
+	MangleProps       string                 // Documentation: https://esbuild.github.io/api/#mangle-props
+	ReserveProps      string                 // Documentation: https://esbuild.github.io/api/#mangle-props
+	MangleQuoted      MangleQuoted           // Documentation: https://esbuild.github.io/api/#mangle-props
+	MangleCache       map[string]interface{} // Documentation: https://esbuild.github.io/api/#mangle-props
+	Drop              Drop                   // Documentation: https://esbuild.github.io/api/#drop
+	DropLabels        []string               // Documentation: https://esbuild.github.io/api/#drop-labels
+	MinifyWhitespace  bool                   // Documentation: https://esbuild.github.io/api/#minify
+	MinifyIdentifiers bool                   // Documentation: https://esbuild.github.io/api/#minify
+	MinifySyntax      bool                   // Documentation: https://esbuild.github.io/api/#minify
+	LineLimit         int                    // Documentation: https://esbuild.github.io/api/#line-limit
+	Charset           Charset                // Documentation: https://esbuild.github.io/api/#charset
+	TreeShaking       TreeShaking            // Documentation: https://esbuild.github.io/api/#tree-shaking
+	IgnoreAnnotations bool                   // Documentation: https://esbuild.github.io/api/#ignore-annotations
+	LegalComments     LegalComments          // Documentation: https://esbuild.github.io/api/#legal-comments
+
+	JSX             JSX    // Documentation: https://esbuild.github.io/api/#jsx-mode
+	JSXFactory      string // Documentation: https://esbuild.github.io/api/#jsx-factory
+	JSXFragment     string // Documentation: https://esbuild.github.io/api/#jsx-fragment
+	JSXImportSource string // Documentation: https://esbuild.github.io/api/#jsx-import-source
+	JSXDev          bool   // Documentation: https://esbuild.github.io/api/#jsx-dev
+	JSXSideEffects  bool   // Documentation: https://esbuild.github.io/api/#jsx-side-effects
+
+	Define    map[string]string // Documentation: https://esbuild.github.io/api/#define
+	Pure      []string          // Documentation: https://esbuild.github.io/api/#pure
+	KeepNames bool              // Documentation: https://esbuild.github.io/api/#keep-names
+
+	GlobalName        string            // Documentation: https://esbuild.github.io/api/#global-name
+	Bundle            bool              // Documentation: https://esbuild.github.io/api/#bundle
+	PreserveSymlinks  bool              // Documentation: https://esbuild.github.io/api/#preserve-symlinks
+	Splitting         bool              // Documentation: https://esbuild.github.io/api/#splitting
+	Outfile           string            // Documentation: https://esbuild.github.io/api/#outfile
+	Metafile          bool              // Documentation: https://esbuild.github.io/api/#metafile
+	Outdir            string            // Documentation: https://esbuild.github.io/api/#outdir
+	Outbase           string            // Documentation: https://esbuild.github.io/api/#outbase
+	AbsWorkingDir     string            // Documentation: https://esbuild.github.io/api/#working-directory
+	Platform          Platform          // Documentation: https://esbuild.github.io/api/#platform
+	Format            Format            // Documentation: https://esbuild.github.io/api/#format
+	External          []string          // Documentation: https://esbuild.github.io/api/#external
+	Packages          Packages          // Documentation: https://esbuild.github.io/api/#packages
+	Alias             map[string]string // Documentation: https://esbuild.github.io/api/#alias
+	MainFields        []string          // Documentation: https://esbuild.github.io/api/#main-fields
+	Conditions        []string          // Documentation: https://esbuild.github.io/api/#conditions
+	Loader            map[string]Loader // Documentation: https://esbuild.github.io/api/#loader
+	ResolveExtensions []string          // Documentation: https://esbuild.github.io/api/#resolve-extensions
+	Tsconfig          string            // Documentation: https://esbuild.github.io/api/#tsconfig
+	TsconfigRaw       string            // Documentation: https://esbuild.github.io/api/#tsconfig-raw
+	OutExtension      map[string]string // Documentation: https://esbuild.github.io/api/#out-extension
+	PublicPath        string            // Documentation: https://esbuild.github.io/api/#public-path
+	Inject            []string          // Documentation: https://esbuild.github.io/api/#inject
+	Banner            map[string]string // Documentation: https://esbuild.github.io/api/#banner
+	Footer            map[string]string // Documentation: https://esbuild.github.io/api/#footer
+	NodePaths         []string          // Documentation: https://esbuild.github.io/api/#node-paths
+
+	EntryNames string // Documentation: https://esbuild.github.io/api/#entry-names
+	ChunkNames string // Documentation: https://esbuild.github.io/api/#chunk-names
+	AssetNames string // Documentation: https://esbuild.github.io/api/#asset-names
+
+	EntryPoints         []string     // Documentation: https://esbuild.github.io/api/#entry-points
+	EntryPointsAdvanced []EntryPoint // Documentation: https://esbuild.github.io/api/#entry-points
+
+	Stdin          *StdinOptions // Documentation: https://esbuild.github.io/api/#stdin
+	Write          bool          // Documentation: https://esbuild.github.io/api/#write
+	AllowOverwrite bool          // Documentation: https://esbuild.github.io/api/#allow-overwrite
+	Plugins        []Plugin      // Documentation: https://esbuild.github.io/plugins/
+}
+
+type EntryPoint struct {
+	InputPath  string
+	OutputPath string
+}
+
+type StdinOptions struct {
+	Contents   string
+	ResolveDir string
+	Sourcefile string
+	Loader     Loader
+}
+
+type BuildResult struct {
+	Errors   []Message
+	Warnings []Message
+
+	OutputFiles []OutputFile
+	Metafile    string
+	MangleCache map[string]interface{}
+}
+
+type OutputFile struct {
+	Path     string
+	Contents []byte
+	Hash     string
+}
+
+// Documentation: https://esbuild.github.io/api/#build
+func Build(options BuildOptions) BuildResult {
+	start := time.Now()
+
+	ctx, errors := contextImpl(options)
+	if ctx == nil {
+		return BuildResult{Errors: errors}
+	}
+
+	result := ctx.Rebuild()
+
+	// Print a summary of the generated files to stderr. Except don't do
+	// this if the terminal is already being used for something else.
+	if ctx.args.logOptions.LogLevel <= logger.LevelInfo && !ctx.args.options.WriteToStdout {
+		printSummary(ctx.args.logOptions.Color, result.OutputFiles, start)
+	}
+
+	ctx.Dispose()
+	return result
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// Transform API
+
+type TransformOptions struct {
+	Color       StderrColor         // Documentation: https://esbuild.github.io/api/#color
+	LogLevel    LogLevel            // Documentation: https://esbuild.github.io/api/#log-level
+	LogLimit    int                 // Documentation: https://esbuild.github.io/api/#log-limit
+	LogOverride map[string]LogLevel // Documentation: https://esbuild.github.io/api/#log-override
+
+	Sourcemap      SourceMap      // Documentation: https://esbuild.github.io/api/#sourcemap
+	SourceRoot     string         // Documentation: https://esbuild.github.io/api/#source-root
+	SourcesContent SourcesContent // Documentation: https://esbuild.github.io/api/#sources-content
+
+	Target    Target          // Documentation: https://esbuild.github.io/api/#target
+	Engines   []Engine        // Documentation: https://esbuild.github.io/api/#target
+	Supported map[string]bool // Documentation: https://esbuild.github.io/api/#supported
+
+	Platform   Platform // Documentation: https://esbuild.github.io/api/#platform
+	Format     Format   // Documentation: https://esbuild.github.io/api/#format
+	GlobalName string   // Documentation: https://esbuild.github.io/api/#global-name
+
+	MangleProps       string                 // Documentation: https://esbuild.github.io/api/#mangle-props
+	ReserveProps      string                 // Documentation: https://esbuild.github.io/api/#mangle-props
+	MangleQuoted      MangleQuoted           // Documentation: https://esbuild.github.io/api/#mangle-props
+	MangleCache       map[string]interface{} // Documentation: https://esbuild.github.io/api/#mangle-props
+	Drop              Drop                   // Documentation: https://esbuild.github.io/api/#drop
+	DropLabels        []string               // Documentation: https://esbuild.github.io/api/#drop-labels
+	MinifyWhitespace  bool                   // Documentation: https://esbuild.github.io/api/#minify
+	MinifyIdentifiers bool                   // Documentation: https://esbuild.github.io/api/#minify
+	MinifySyntax      bool                   // Documentation: https://esbuild.github.io/api/#minify
+	LineLimit         int                    // Documentation: https://esbuild.github.io/api/#line-limit
+	Charset           Charset                // Documentation: https://esbuild.github.io/api/#charset
+	TreeShaking       TreeShaking            // Documentation: https://esbuild.github.io/api/#tree-shaking
+	IgnoreAnnotations bool                   // Documentation: https://esbuild.github.io/api/#ignore-annotations
+	LegalComments     LegalComments          // Documentation: https://esbuild.github.io/api/#legal-comments
+
+	JSX             JSX    // Documentation: https://esbuild.github.io/api/#jsx
+	JSXFactory      string // Documentation: https://esbuild.github.io/api/#jsx-factory
+	JSXFragment     string // Documentation: https://esbuild.github.io/api/#jsx-fragment
+	JSXImportSource string // Documentation: https://esbuild.github.io/api/#jsx-import-source
+	JSXDev          bool   // Documentation: https://esbuild.github.io/api/#jsx-dev
+	JSXSideEffects  bool   // Documentation: https://esbuild.github.io/api/#jsx-side-effects
+
+	TsconfigRaw string // Documentation: https://esbuild.github.io/api/#tsconfig-raw
+	Banner      string // Documentation: https://esbuild.github.io/api/#banner
+	Footer      string // Documentation: https://esbuild.github.io/api/#footer
+
+	Define    map[string]string // Documentation: https://esbuild.github.io/api/#define
+	Pure      []string          // Documentation: https://esbuild.github.io/api/#pure
+	KeepNames bool              // Documentation: https://esbuild.github.io/api/#keep-names
+
+	Sourcefile string // Documentation: https://esbuild.github.io/api/#sourcefile
+	Loader     Loader // Documentation: https://esbuild.github.io/api/#loader
+}
+
+type TransformResult struct {
+	Errors   []Message
+	Warnings []Message
+
+	Code          []byte
+	Map           []byte
+	LegalComments []byte
+
+	MangleCache map[string]interface{}
+}
+
+// Documentation: https://esbuild.github.io/api/#transform
+func Transform(input string, options TransformOptions) TransformResult {
+	return transformImpl(input, options)
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// Context API
+
+// Documentation: https://esbuild.github.io/api/#serve-arguments
+type ServeOptions struct {
+	Port      uint16
+	Host      string
+	Servedir  string
+	Keyfile   string
+	Certfile  string
+	Fallback  string
+	OnRequest func(ServeOnRequestArgs)
+}
+
+type ServeOnRequestArgs struct {
+	RemoteAddress string
+	Method        string
+	Path          string
+	Status        int
+	TimeInMS      int // The time to generate the response, not to send it
+}
+
+// Documentation: https://esbuild.github.io/api/#serve-return-values
+type ServeResult struct {
+	Port uint16
+	Host string
+}
+
+type WatchOptions struct {
+}
+
+type BuildContext interface {
+	// Documentation: https://esbuild.github.io/api/#rebuild
+	Rebuild() BuildResult
+
+	// Documentation: https://esbuild.github.io/api/#watch
+	Watch(options WatchOptions) error
+
+	// Documentation: https://esbuild.github.io/api/#serve
+	Serve(options ServeOptions) (ServeResult, error)
+
+	Cancel()
+	Dispose()
+}
+
+type ContextError struct {
+	Errors []Message // Option validation errors are returned here
+}
+
+func (err *ContextError) Error() string {
+	if len(err.Errors) > 0 {
+		return err.Errors[0].Text
+	}
+	return "Context creation failed"
+}
+
+// Documentation: https://esbuild.github.io/api/#build
+func Context(buildOptions BuildOptions) (BuildContext, *ContextError) {
+	ctx, errors := contextImpl(buildOptions)
+	if ctx == nil {
+		return nil, &ContextError{Errors: errors}
+	}
+	return ctx, nil
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// Plugin API
+
+type SideEffects uint8
+
+const (
+	SideEffectsTrue SideEffects = iota
+	SideEffectsFalse
+)
+
+type Plugin struct {
+	Name  string
+	Setup func(PluginBuild)
+}
+
+type PluginBuild struct {
+	// Documentation: https://esbuild.github.io/plugins/#build-options
+	InitialOptions *BuildOptions
+
+	// Documentation: https://esbuild.github.io/plugins/#resolve
+	Resolve func(path string, options ResolveOptions) ResolveResult
+
+	// Documentation: https://esbuild.github.io/plugins/#on-start
+	OnStart func(callback func() (OnStartResult, error))
+
+	// Documentation: https://esbuild.github.io/plugins/#on-end
+	OnEnd func(callback func(result *BuildResult) (OnEndResult, error))
+
+	// Documentation: https://esbuild.github.io/plugins/#on-resolve
+	OnResolve func(options OnResolveOptions, callback func(OnResolveArgs) (OnResolveResult, error))
+
+	// Documentation: https://esbuild.github.io/plugins/#on-load
+	OnLoad func(options OnLoadOptions, callback func(OnLoadArgs) (OnLoadResult, error))
+
+	// Documentation: https://esbuild.github.io/plugins/#on-dispose
+	OnDispose func(callback func())
+}
+
+// Documentation: https://esbuild.github.io/plugins/#resolve-options
+type ResolveOptions struct {
+	PluginName string
+	Importer   string
+	Namespace  string
+	ResolveDir string
+	Kind       ResolveKind
+	PluginData interface{}
+	With       map[string]string
+}
+
+// Documentation: https://esbuild.github.io/plugins/#resolve-results
+type ResolveResult struct {
+	Errors   []Message
+	Warnings []Message
+
+	Path        string
+	External    bool
+	SideEffects bool
+	Namespace   string
+	Suffix      string
+	PluginData  interface{}
+}
+
+type OnStartResult struct {
+	Errors   []Message
+	Warnings []Message
+}
+
+type OnEndResult struct {
+	Errors   []Message
+	Warnings []Message
+}
+
+// Documentation: https://esbuild.github.io/plugins/#on-resolve-options
+type OnResolveOptions struct {
+	Filter    string
+	Namespace string
+}
+
+// Documentation: https://esbuild.github.io/plugins/#on-resolve-arguments
+type OnResolveArgs struct {
+	Path       string
+	Importer   string
+	Namespace  string
+	ResolveDir string
+	Kind       ResolveKind
+	PluginData interface{}
+	With       map[string]string
+}
+
+// Documentation: https://esbuild.github.io/plugins/#on-resolve-results
+type OnResolveResult struct {
+	PluginName string
+
+	Errors   []Message
+	Warnings []Message
+
+	Path        string
+	External    bool
+	SideEffects SideEffects
+	Namespace   string
+	Suffix      string
+	PluginData  interface{}
+
+	WatchFiles []string
+	WatchDirs  []string
+}
+
+// Documentation: https://esbuild.github.io/plugins/#on-load-options
+type OnLoadOptions struct {
+	Filter    string
+	Namespace string
+}
+
+// Documentation: https://esbuild.github.io/plugins/#on-load-arguments
+type OnLoadArgs struct {
+	Path       string
+	Namespace  string
+	Suffix     string
+	PluginData interface{}
+	With       map[string]string
+}
+
+// Documentation: https://esbuild.github.io/plugins/#on-load-results
+type OnLoadResult struct {
+	PluginName string
+
+	Errors   []Message
+	Warnings []Message
+
+	Contents   *string
+	ResolveDir string
+	Loader     Loader
+	PluginData interface{}
+
+	WatchFiles []string
+	WatchDirs  []string
+}
+
+type ResolveKind uint8
+
+const (
+	ResolveNone ResolveKind = iota
+	ResolveEntryPoint
+	ResolveJSImportStatement
+	ResolveJSRequireCall
+	ResolveJSDynamicImport
+	ResolveJSRequireResolve
+	ResolveCSSImportRule
+	ResolveCSSComposesFrom
+	ResolveCSSURLToken
+)
+
+////////////////////////////////////////////////////////////////////////////////
+// FormatMessages API
+
+type MessageKind uint8
+
+const (
+	ErrorMessage MessageKind = iota
+	WarningMessage
+)
+
+type FormatMessagesOptions struct {
+	TerminalWidth int
+	Kind          MessageKind
+	Color         bool
+}
+
+func FormatMessages(msgs []Message, opts FormatMessagesOptions) []string {
+	return formatMsgsImpl(msgs, opts)
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// AnalyzeMetafile API
+
+type AnalyzeMetafileOptions struct {
+	Color   bool
+	Verbose bool
+}
+
+// Documentation: https://esbuild.github.io/api/#analyze
+func AnalyzeMetafile(metafile string, opts AnalyzeMetafileOptions) string {
+	return analyzeMetafileImpl(metafile, opts)
+}
diff --git a/source/vendor/github.com/evanw/esbuild/pkg/api/api_impl.go b/source/vendor/github.com/evanw/esbuild/pkg/api/api_impl.go
new file mode 100644
index 0000000..7c73777
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/pkg/api/api_impl.go
@@ -0,0 +1,2530 @@
+package api
+
+// This file implements most of the API. This includes the "Build", "Transform",
+// "FormatMessages", and "AnalyzeMetafile" functions.
+
+import (
+	"bytes"
+	"encoding/base64"
+	"encoding/binary"
+	"errors"
+	"fmt"
+	"io/ioutil"
+	"math"
+	"os"
+	"path"
+	"regexp"
+	"sort"
+	"strconv"
+	"strings"
+	"sync"
+	"time"
+	"unicode/utf8"
+
+	"github.com/evanw/esbuild/internal/api_helpers"
+	"github.com/evanw/esbuild/internal/ast"
+	"github.com/evanw/esbuild/internal/bundler"
+	"github.com/evanw/esbuild/internal/cache"
+	"github.com/evanw/esbuild/internal/compat"
+	"github.com/evanw/esbuild/internal/config"
+	"github.com/evanw/esbuild/internal/css_ast"
+	"github.com/evanw/esbuild/internal/fs"
+	"github.com/evanw/esbuild/internal/graph"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/js_ast"
+	"github.com/evanw/esbuild/internal/js_parser"
+	"github.com/evanw/esbuild/internal/linker"
+	"github.com/evanw/esbuild/internal/logger"
+	"github.com/evanw/esbuild/internal/resolver"
+	"github.com/evanw/esbuild/internal/xxhash"
+)
+
+func validatePathTemplate(template string) []config.PathTemplate {
+	if template == "" {
+		return nil
+	}
+	template = "./" + strings.ReplaceAll(template, "\\", "/")
+
+	parts := make([]config.PathTemplate, 0, 4)
+	search := 0
+
+	// Split by placeholders
+	for search < len(template) {
+		// Jump to the next "["
+		if found := strings.IndexByte(template[search:], '['); found == -1 {
+			break
+		} else {
+			search += found
+		}
+		head, tail := template[:search], template[search:]
+		placeholder := config.NoPlaceholder
+
+		// Check for a placeholder
+		switch {
+		case strings.HasPrefix(tail, "[dir]"):
+			placeholder = config.DirPlaceholder
+			search += len("[dir]")
+
+		case strings.HasPrefix(tail, "[name]"):
+			placeholder = config.NamePlaceholder
+			search += len("[name]")
+
+		case strings.HasPrefix(tail, "[hash]"):
+			placeholder = config.HashPlaceholder
+			search += len("[hash]")
+
+		case strings.HasPrefix(tail, "[ext]"):
+			placeholder = config.ExtPlaceholder
+			search += len("[ext]")
+
+		default:
+			// Skip past the "[" so we don't find it again
+			search++
+			continue
+		}
+
+		// Add a part for everything up to and including this placeholder
+		parts = append(parts, config.PathTemplate{
+			Data:        head,
+			Placeholder: placeholder,
+		})
+
+		// Reset the search after this placeholder
+		template = template[search:]
+		search = 0
+	}
+
+	// Append any remaining data as a part without a placeholder
+	if search < len(template) {
+		parts = append(parts, config.PathTemplate{
+			Data:        template,
+			Placeholder: config.NoPlaceholder,
+		})
+	}
+
+	return parts
+}
+
+func validatePlatform(value Platform) config.Platform {
+	switch value {
+	case PlatformDefault, PlatformBrowser:
+		return config.PlatformBrowser
+	case PlatformNode:
+		return config.PlatformNode
+	case PlatformNeutral:
+		return config.PlatformNeutral
+	default:
+		panic("Invalid platform")
+	}
+}
+
+func validateFormat(value Format) config.Format {
+	switch value {
+	case FormatDefault:
+		return config.FormatPreserve
+	case FormatIIFE:
+		return config.FormatIIFE
+	case FormatCommonJS:
+		return config.FormatCommonJS
+	case FormatESModule:
+		return config.FormatESModule
+	default:
+		panic("Invalid format")
+	}
+}
+
+func validateSourceMap(value SourceMap) config.SourceMap {
+	switch value {
+	case SourceMapNone:
+		return config.SourceMapNone
+	case SourceMapLinked:
+		return config.SourceMapLinkedWithComment
+	case SourceMapInline:
+		return config.SourceMapInline
+	case SourceMapExternal:
+		return config.SourceMapExternalWithoutComment
+	case SourceMapInlineAndExternal:
+		return config.SourceMapInlineAndExternal
+	default:
+		panic("Invalid source map")
+	}
+}
+
+func validateLegalComments(value LegalComments, bundle bool) config.LegalComments {
+	switch value {
+	case LegalCommentsDefault:
+		if bundle {
+			return config.LegalCommentsEndOfFile
+		} else {
+			return config.LegalCommentsInline
+		}
+	case LegalCommentsNone:
+		return config.LegalCommentsNone
+	case LegalCommentsInline:
+		return config.LegalCommentsInline
+	case LegalCommentsEndOfFile:
+		return config.LegalCommentsEndOfFile
+	case LegalCommentsLinked:
+		return config.LegalCommentsLinkedWithComment
+	case LegalCommentsExternal:
+		return config.LegalCommentsExternalWithoutComment
+	default:
+		panic("Invalid source map")
+	}
+}
+
+func validateColor(value StderrColor) logger.UseColor {
+	switch value {
+	case ColorIfTerminal:
+		return logger.ColorIfTerminal
+	case ColorNever:
+		return logger.ColorNever
+	case ColorAlways:
+		return logger.ColorAlways
+	default:
+		panic("Invalid color")
+	}
+}
+
+func validateLogLevel(value LogLevel) logger.LogLevel {
+	switch value {
+	case LogLevelVerbose:
+		return logger.LevelVerbose
+	case LogLevelDebug:
+		return logger.LevelDebug
+	case LogLevelInfo:
+		return logger.LevelInfo
+	case LogLevelWarning:
+		return logger.LevelWarning
+	case LogLevelError:
+		return logger.LevelError
+	case LogLevelSilent:
+		return logger.LevelSilent
+	default:
+		panic("Invalid log level")
+	}
+}
+
+func validateASCIIOnly(value Charset) bool {
+	switch value {
+	case CharsetDefault, CharsetASCII:
+		return true
+	case CharsetUTF8:
+		return false
+	default:
+		panic("Invalid charset")
+	}
+}
+
+func validateExternalPackages(value Packages) bool {
+	switch value {
+	case PackagesDefault, PackagesBundle:
+		return false
+	case PackagesExternal:
+		return true
+	default:
+		panic("Invalid packages")
+	}
+}
+
+func validateTreeShaking(value TreeShaking, bundle bool, format Format) bool {
+	switch value {
+	case TreeShakingDefault:
+		// If we're in an IIFE then there's no way to concatenate additional code
+		// to the end of our output so we assume tree shaking is safe. And when
+		// bundling we assume that tree shaking is safe because if you want to add
+		// code to the bundle, you should be doing that by including it in the
+		// bundle instead of concatenating it afterward, so we also assume tree
+		// shaking is safe then. Otherwise we assume tree shaking is not safe.
+		return bundle || format == FormatIIFE
+	case TreeShakingFalse:
+		return false
+	case TreeShakingTrue:
+		return true
+	default:
+		panic("Invalid tree shaking")
+	}
+}
+
+func validateLoader(value Loader) config.Loader {
+	switch value {
+	case LoaderBase64:
+		return config.LoaderBase64
+	case LoaderBinary:
+		return config.LoaderBinary
+	case LoaderCopy:
+		return config.LoaderCopy
+	case LoaderCSS:
+		return config.LoaderCSS
+	case LoaderDataURL:
+		return config.LoaderDataURL
+	case LoaderDefault:
+		return config.LoaderDefault
+	case LoaderEmpty:
+		return config.LoaderEmpty
+	case LoaderFile:
+		return config.LoaderFile
+	case LoaderGlobalCSS:
+		return config.LoaderGlobalCSS
+	case LoaderJS:
+		return config.LoaderJS
+	case LoaderJSON:
+		return config.LoaderJSON
+	case LoaderJSX:
+		return config.LoaderJSX
+	case LoaderLocalCSS:
+		return config.LoaderLocalCSS
+	case LoaderNone:
+		return config.LoaderNone
+	case LoaderText:
+		return config.LoaderText
+	case LoaderTS:
+		return config.LoaderTS
+	case LoaderTSX:
+		return config.LoaderTSX
+	default:
+		panic("Invalid loader")
+	}
+}
+
+var versionRegex = regexp.MustCompile(`^([0-9]+)(?:\.([0-9]+))?(?:\.([0-9]+))?(-[A-Za-z0-9]+(?:\.[A-Za-z0-9]+)*)?$`)
+
+func validateFeatures(log logger.Log, target Target, engines []Engine) (compat.JSFeature, compat.CSSFeature, map[css_ast.D]compat.CSSPrefix, string) {
+	if target == DefaultTarget && len(engines) == 0 {
+		return 0, 0, nil, ""
+	}
+
+	constraints := make(map[compat.Engine]compat.Semver)
+	targets := make([]string, 0, 1+len(engines))
+
+	switch target {
+	case ES5:
+		constraints[compat.ES] = compat.Semver{Parts: []int{5}}
+	case ES2015:
+		constraints[compat.ES] = compat.Semver{Parts: []int{2015}}
+	case ES2016:
+		constraints[compat.ES] = compat.Semver{Parts: []int{2016}}
+	case ES2017:
+		constraints[compat.ES] = compat.Semver{Parts: []int{2017}}
+	case ES2018:
+		constraints[compat.ES] = compat.Semver{Parts: []int{2018}}
+	case ES2019:
+		constraints[compat.ES] = compat.Semver{Parts: []int{2019}}
+	case ES2020:
+		constraints[compat.ES] = compat.Semver{Parts: []int{2020}}
+	case ES2021:
+		constraints[compat.ES] = compat.Semver{Parts: []int{2021}}
+	case ES2022:
+		constraints[compat.ES] = compat.Semver{Parts: []int{2022}}
+	case ES2023:
+		constraints[compat.ES] = compat.Semver{Parts: []int{2023}}
+	case ES2024:
+		constraints[compat.ES] = compat.Semver{Parts: []int{2024}}
+	case ESNext, DefaultTarget:
+	default:
+		panic("Invalid target")
+	}
+
+	for _, engine := range engines {
+		if match := versionRegex.FindStringSubmatch(engine.Version); match != nil {
+			if major, err := strconv.Atoi(match[1]); err == nil {
+				parts := []int{major}
+				if minor, err := strconv.Atoi(match[2]); err == nil {
+					parts = append(parts, minor)
+					if patch, err := strconv.Atoi(match[3]); err == nil {
+						parts = append(parts, patch)
+					}
+				}
+				constraints[convertEngineName(engine.Name)] = compat.Semver{
+					Parts:      parts,
+					PreRelease: match[4],
+				}
+				continue
+			}
+		}
+
+		text := "All version numbers passed to esbuild must be in the format \"X\", \"X.Y\", or \"X.Y.Z\" where X, Y, and Z are non-negative integers."
+
+		log.AddErrorWithNotes(nil, logger.Range{}, fmt.Sprintf("Invalid version: %q", engine.Version),
+			[]logger.MsgData{{Text: text}})
+	}
+
+	for engine, version := range constraints {
+		targets = append(targets, engine.String()+version.String())
+	}
+	if target == ESNext {
+		targets = append(targets, "esnext")
+	}
+
+	sort.Strings(targets)
+	targetEnv := helpers.StringArrayToQuotedCommaSeparatedString(targets)
+
+	return compat.UnsupportedJSFeatures(constraints), compat.UnsupportedCSSFeatures(constraints), compat.CSSPrefixData(constraints), targetEnv
+}
+
+func validateSupported(log logger.Log, supported map[string]bool) (
+	jsFeature compat.JSFeature,
+	jsMask compat.JSFeature,
+	cssFeature compat.CSSFeature,
+	cssMask compat.CSSFeature,
+) {
+	for k, v := range supported {
+		if js, ok := compat.StringToJSFeature[k]; ok {
+			jsMask |= js
+			if !v {
+				jsFeature |= js
+			}
+		} else if css, ok := compat.StringToCSSFeature[k]; ok {
+			cssMask |= css
+			if !v {
+				cssFeature |= css
+			}
+		} else {
+			log.AddError(nil, logger.Range{}, fmt.Sprintf("%q is not a valid feature name for the \"supported\" setting", k))
+		}
+	}
+	return
+}
+
+func validateGlobalName(log logger.Log, text string) []string {
+	if text != "" {
+		source := logger.Source{
+			KeyPath:    logger.Path{Text: "(global path)"},
+			PrettyPath: "(global name)",
+			Contents:   text,
+		}
+
+		if result, ok := js_parser.ParseGlobalName(log, source); ok {
+			return result
+		}
+	}
+
+	return nil
+}
+
+func validateRegex(log logger.Log, what string, value string) *regexp.Regexp {
+	if value == "" {
+		return nil
+	}
+	regex, err := regexp.Compile(value)
+	if err != nil {
+		log.AddError(nil, logger.Range{},
+			fmt.Sprintf("The %q setting is not a valid Go regular expression: %s", what, value))
+		return nil
+	}
+	return regex
+}
+
+func validateExternals(log logger.Log, fs fs.FS, paths []string) config.ExternalSettings {
+	result := config.ExternalSettings{
+		PreResolve:  config.ExternalMatchers{Exact: make(map[string]bool)},
+		PostResolve: config.ExternalMatchers{Exact: make(map[string]bool)},
+	}
+
+	for _, path := range paths {
+		if index := strings.IndexByte(path, '*'); index != -1 {
+			// Wildcard behavior
+			if strings.ContainsRune(path[index+1:], '*') {
+				log.AddError(nil, logger.Range{}, fmt.Sprintf("External path %q cannot have more than one \"*\" wildcard", path))
+			} else {
+				result.PreResolve.Patterns = append(result.PreResolve.Patterns, config.WildcardPattern{Prefix: path[:index], Suffix: path[index+1:]})
+				if !resolver.IsPackagePath(path) {
+					if absPath := validatePath(log, fs, path, "external path"); absPath != "" {
+						if absIndex := strings.IndexByte(absPath, '*'); absIndex != -1 && !strings.ContainsRune(absPath[absIndex+1:], '*') {
+							result.PostResolve.Patterns = append(result.PostResolve.Patterns, config.WildcardPattern{Prefix: absPath[:absIndex], Suffix: absPath[absIndex+1:]})
+						}
+					}
+				}
+			}
+		} else {
+			// Non-wildcard behavior
+			result.PreResolve.Exact[path] = true
+			if resolver.IsPackagePath(path) {
+				result.PreResolve.Patterns = append(result.PreResolve.Patterns, config.WildcardPattern{Prefix: path + "/"})
+			} else if absPath := validatePath(log, fs, path, "external path"); absPath != "" {
+				result.PostResolve.Exact[absPath] = true
+			}
+		}
+	}
+
+	return result
+}
+
+func validateAlias(log logger.Log, fs fs.FS, alias map[string]string) map[string]string {
+	valid := make(map[string]string, len(alias))
+
+	for old, new := range alias {
+		if new == "" {
+			log.AddError(nil, logger.Range{}, fmt.Sprintf("Invalid alias substitution: %q", new))
+			continue
+		}
+
+		// Valid alias names:
+		//   "foo"
+		//   "foo/bar"
+		//   "@foo"
+		//   "@foo/bar"
+		//   "@foo/bar/baz"
+		//
+		// Invalid alias names:
+		//   "./foo"
+		//   "../foo"
+		//   "/foo"
+		//   "C:\\foo"
+		//   ".foo"
+		//   "foo/"
+		//   "@foo/"
+		//   "foo/../bar"
+		//
+		if !strings.HasPrefix(old, ".") && !strings.HasPrefix(old, "/") && !fs.IsAbs(old) && path.Clean(strings.ReplaceAll(old, "\\", "/")) == old {
+			valid[old] = new
+			continue
+		}
+
+		log.AddError(nil, logger.Range{}, fmt.Sprintf("Invalid alias name: %q", old))
+	}
+
+	return valid
+}
+
+func isValidExtension(ext string) bool {
+	return len(ext) >= 2 && ext[0] == '.' && ext[len(ext)-1] != '.'
+}
+
+func validateResolveExtensions(log logger.Log, order []string) []string {
+	if order == nil {
+		return []string{".tsx", ".ts", ".jsx", ".js", ".css", ".json"}
+	}
+	for _, ext := range order {
+		if !isValidExtension(ext) {
+			log.AddError(nil, logger.Range{}, fmt.Sprintf("Invalid file extension: %q", ext))
+		}
+	}
+	return order
+}
+
+func validateLoaders(log logger.Log, loaders map[string]Loader) map[string]config.Loader {
+	result := bundler.DefaultExtensionToLoaderMap()
+	for ext, loader := range loaders {
+		if ext != "" && !isValidExtension(ext) {
+			log.AddError(nil, logger.Range{}, fmt.Sprintf("Invalid file extension: %q", ext))
+		}
+		result[ext] = validateLoader(loader)
+	}
+	return result
+}
+
+func validateJSXExpr(log logger.Log, text string, name string) config.DefineExpr {
+	if text != "" {
+		if expr, _ := js_parser.ParseDefineExprOrJSON(text); len(expr.Parts) > 0 || (name == "fragment" && expr.Constant != nil) {
+			return expr
+		}
+		log.AddError(nil, logger.Range{}, fmt.Sprintf("Invalid JSX %s: %q", name, text))
+	}
+	return config.DefineExpr{}
+}
+
+func validateDefines(
+	log logger.Log,
+	defines map[string]string,
+	pureFns []string,
+	platform config.Platform,
+	isBuildAPI bool,
+	minify bool,
+	drop Drop,
+) (*config.ProcessedDefines, []config.InjectedDefine) {
+	rawDefines := make(map[string]config.DefineData)
+	var valueToInject map[string]config.InjectedDefine
+	var definesToInject []string
+
+	for key, value := range defines {
+		// The key must be a dot-separated identifier list
+		for _, part := range strings.Split(key, ".") {
+			if !js_ast.IsIdentifier(part) {
+				if part == key {
+					log.AddError(nil, logger.Range{}, fmt.Sprintf("The define key %q must be a valid identifier", key))
+				} else {
+					log.AddError(nil, logger.Range{}, fmt.Sprintf("The define key %q contains invalid identifier %q", key, part))
+				}
+				continue
+			}
+		}
+
+		// Parse the value
+		defineExpr, injectExpr := js_parser.ParseDefineExprOrJSON(value)
+
+		// Define simple expressions
+		if defineExpr.Constant != nil || len(defineExpr.Parts) > 0 {
+			rawDefines[key] = config.DefineData{DefineExpr: &defineExpr}
+
+			// Try to be helpful for common mistakes
+			if len(defineExpr.Parts) == 1 && key == "process.env.NODE_ENV" {
+				data := logger.MsgData{
+					Text: fmt.Sprintf("%q is defined as an identifier instead of a string (surround %q with quotes to get a string)", key, value),
+				}
+				part := defineExpr.Parts[0]
+
+				switch logger.API {
+				case logger.CLIAPI:
+					data.Location = &logger.MsgLocation{
+						File:       "<cli>",
+						Line:       1,
+						Column:     30,
+						Length:     len(part),
+						LineText:   fmt.Sprintf("--define:process.env.NODE_ENV=%s", part),
+						Suggestion: fmt.Sprintf("\\\"%s\\\"", part),
+					}
+
+				case logger.JSAPI:
+					data.Location = &logger.MsgLocation{
+						File:       "<js>",
+						Line:       1,
+						Column:     34,
+						Length:     len(part) + 2,
+						LineText:   fmt.Sprintf("define: { 'process.env.NODE_ENV': '%s' }", part),
+						Suggestion: fmt.Sprintf("'\"%s\"'", part),
+					}
+
+				case logger.GoAPI:
+					data.Location = &logger.MsgLocation{
+						File:       "<go>",
+						Line:       1,
+						Column:     50,
+						Length:     len(part) + 2,
+						LineText:   fmt.Sprintf("Define: map[string]string{\"process.env.NODE_ENV\": \"%s\"}", part),
+						Suggestion: fmt.Sprintf("\"\\\"%s\\\"\"", part),
+					}
+				}
+
+				log.AddMsgID(logger.MsgID_JS_SuspiciousDefine, logger.Msg{
+					Kind: logger.Warning,
+					Data: data,
+				})
+			}
+			continue
+		}
+
+		// Inject complex expressions
+		if injectExpr != nil {
+			definesToInject = append(definesToInject, key)
+			if valueToInject == nil {
+				valueToInject = make(map[string]config.InjectedDefine)
+			}
+			valueToInject[key] = config.InjectedDefine{
+				Source: logger.Source{Contents: value},
+				Data:   injectExpr,
+				Name:   key,
+			}
+			continue
+		}
+
+		// Anything else is unsupported
+		log.AddError(nil, logger.Range{}, fmt.Sprintf("Invalid define value (must be an entity name or valid JSON syntax): %s", value))
+	}
+
+	// Sort injected defines for determinism, since the imports will be injected
+	// into every file in the order that we return them from this function
+	var injectedDefines []config.InjectedDefine
+	if len(definesToInject) > 0 {
+		injectedDefines = make([]config.InjectedDefine, len(definesToInject))
+		sort.Strings(definesToInject)
+		for i, key := range definesToInject {
+			injectedDefines[i] = valueToInject[key]
+			rawDefines[key] = config.DefineData{DefineExpr: &config.DefineExpr{InjectedDefineIndex: ast.MakeIndex32(uint32(i))}}
+		}
+	}
+
+	// If we're bundling for the browser, add a special-cased define for
+	// "process.env.NODE_ENV" that is "development" when not minifying and
+	// "production" when minifying. This is a convention from the React world
+	// that must be handled to avoid all React code crashing instantly. This
+	// is only done if it's not already defined so that you can override it if
+	// necessary.
+	if isBuildAPI && platform == config.PlatformBrowser {
+		if _, process := rawDefines["process"]; !process {
+			if _, processEnv := rawDefines["process.env"]; !processEnv {
+				if _, processEnvNodeEnv := rawDefines["process.env.NODE_ENV"]; !processEnvNodeEnv {
+					var value []uint16
+					if minify {
+						value = helpers.StringToUTF16("production")
+					} else {
+						value = helpers.StringToUTF16("development")
+					}
+					rawDefines["process.env.NODE_ENV"] = config.DefineData{DefineExpr: &config.DefineExpr{Constant: &js_ast.EString{Value: value}}}
+				}
+			}
+		}
+	}
+
+	// If we're dropping all console API calls, replace each one with undefined
+	if (drop & DropConsole) != 0 {
+		define := rawDefines["console"]
+		define.Flags |= config.MethodCallsMustBeReplacedWithUndefined
+		rawDefines["console"] = define
+	}
+
+	for _, key := range pureFns {
+		// The key must be a dot-separated identifier list
+		for _, part := range strings.Split(key, ".") {
+			if !js_ast.IsIdentifier(part) {
+				log.AddError(nil, logger.Range{}, fmt.Sprintf("Invalid pure function: %q", key))
+				continue
+			}
+		}
+
+		// Merge with any previously-specified defines
+		define := rawDefines[key]
+		define.Flags |= config.CallCanBeUnwrappedIfUnused
+		rawDefines[key] = define
+	}
+
+	// Processing defines is expensive. Process them once here so the same object
+	// can be shared between all parsers we create using these arguments.
+	processed := config.ProcessDefines(rawDefines)
+	return &processed, injectedDefines
+}
+
+func validateLogOverrides(input map[string]LogLevel) (output map[logger.MsgID]logger.LogLevel) {
+	output = make(map[uint8]logger.LogLevel)
+	for k, v := range input {
+		logger.StringToMsgIDs(k, validateLogLevel(v), output)
+	}
+	return
+}
+
+func validatePath(log logger.Log, fs fs.FS, relPath string, pathKind string) string {
+	if relPath == "" {
+		return ""
+	}
+	absPath, ok := fs.Abs(relPath)
+	if !ok {
+		log.AddError(nil, logger.Range{}, fmt.Sprintf("Invalid %s: %s", pathKind, relPath))
+	}
+	return absPath
+}
+
+func validateOutputExtensions(log logger.Log, outExtensions map[string]string) (js string, css string) {
+	for key, value := range outExtensions {
+		if !isValidExtension(value) {
+			log.AddError(nil, logger.Range{}, fmt.Sprintf("Invalid output extension: %q", value))
+		}
+		switch key {
+		case ".js":
+			js = value
+		case ".css":
+			css = value
+		default:
+			log.AddError(nil, logger.Range{}, fmt.Sprintf("Invalid output extension: %q (valid: .css, .js)", key))
+		}
+	}
+	return
+}
+
+func validateBannerOrFooter(log logger.Log, name string, values map[string]string) (js string, css string) {
+	for key, value := range values {
+		switch key {
+		case "js":
+			js = value
+		case "css":
+			css = value
+		default:
+			log.AddError(nil, logger.Range{}, fmt.Sprintf("Invalid %s file type: %q (valid: css, js)", name, key))
+		}
+	}
+	return
+}
+
+func validateKeepNames(log logger.Log, options *config.Options) {
+	if options.KeepNames && options.UnsupportedJSFeatures.Has(compat.FunctionNameConfigurable) {
+		where := config.PrettyPrintTargetEnvironment(options.OriginalTargetEnv, options.UnsupportedJSFeatureOverridesMask)
+		log.AddErrorWithNotes(nil, logger.Range{}, fmt.Sprintf("The \"keep names\" setting cannot be used with %s", where), []logger.MsgData{{
+			Text: "In this environment, the \"Function.prototype.name\" property is not configurable and assigning to it will throw an error. " +
+				"Either use a newer target environment or disable the \"keep names\" setting."}})
+	}
+}
+
+func convertLocationToPublic(loc *logger.MsgLocation) *Location {
+	if loc != nil {
+		return &Location{
+			File:       loc.File,
+			Namespace:  loc.Namespace,
+			Line:       loc.Line,
+			Column:     loc.Column,
+			Length:     loc.Length,
+			LineText:   loc.LineText,
+			Suggestion: loc.Suggestion,
+		}
+	}
+	return nil
+}
+
+func convertMessagesToPublic(kind logger.MsgKind, msgs []logger.Msg) []Message {
+	var filtered []Message
+	for _, msg := range msgs {
+		if msg.Kind == kind {
+			var notes []Note
+			for _, note := range msg.Notes {
+				notes = append(notes, Note{
+					Text:     note.Text,
+					Location: convertLocationToPublic(note.Location),
+				})
+			}
+			filtered = append(filtered, Message{
+				ID:         logger.MsgIDToString(msg.ID),
+				PluginName: msg.PluginName,
+				Text:       msg.Data.Text,
+				Location:   convertLocationToPublic(msg.Data.Location),
+				Notes:      notes,
+				Detail:     msg.Data.UserDetail,
+			})
+		}
+	}
+	return filtered
+}
+
+func convertLocationToInternal(loc *Location) *logger.MsgLocation {
+	if loc != nil {
+		namespace := loc.Namespace
+		if namespace == "" {
+			namespace = "file"
+		}
+		return &logger.MsgLocation{
+			File:       loc.File,
+			Namespace:  namespace,
+			Line:       loc.Line,
+			Column:     loc.Column,
+			Length:     loc.Length,
+			LineText:   loc.LineText,
+			Suggestion: loc.Suggestion,
+		}
+	}
+	return nil
+}
+
+func convertMessagesToInternal(msgs []logger.Msg, kind logger.MsgKind, messages []Message) []logger.Msg {
+	for _, message := range messages {
+		var notes []logger.MsgData
+		for _, note := range message.Notes {
+			notes = append(notes, logger.MsgData{
+				Text:     note.Text,
+				Location: convertLocationToInternal(note.Location),
+			})
+		}
+		msgs = append(msgs, logger.Msg{
+			ID:         logger.StringToMaximumMsgID(message.ID),
+			PluginName: message.PluginName,
+			Kind:       kind,
+			Data: logger.MsgData{
+				Text:       message.Text,
+				Location:   convertLocationToInternal(message.Location),
+				UserDetail: message.Detail,
+			},
+			Notes: notes,
+		})
+	}
+	return msgs
+}
+
+func convertErrorsAndWarningsToInternal(errors []Message, warnings []Message) []logger.Msg {
+	if len(errors)+len(warnings) > 0 {
+		msgs := make(logger.SortableMsgs, 0, len(errors)+len(warnings))
+		msgs = convertMessagesToInternal(msgs, logger.Error, errors)
+		msgs = convertMessagesToInternal(msgs, logger.Warning, warnings)
+		sort.Stable(msgs)
+		return msgs
+	}
+	return nil
+}
+
+func cloneMangleCache(log logger.Log, mangleCache map[string]interface{}) map[string]interface{} {
+	if mangleCache == nil {
+		return nil
+	}
+	clone := make(map[string]interface{}, len(mangleCache))
+	for k, v := range mangleCache {
+		if v == "__proto__" {
+			// This could cause problems for our binary serialization protocol. It's
+			// also unnecessary because we already avoid mangling this property name.
+			log.AddError(nil, logger.Range{},
+				fmt.Sprintf("Invalid identifier name %q in mangle cache", k))
+		} else if _, ok := v.(string); ok || v == false {
+			clone[k] = v
+		} else {
+			log.AddError(nil, logger.Range{},
+				fmt.Sprintf("Expected %q in mangle cache to map to either a string or false", k))
+		}
+	}
+	return clone
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// Build API
+
+func contextImpl(buildOpts BuildOptions) (*internalContext, []Message) {
+	logOptions := logger.OutputOptions{
+		IncludeSource: true,
+		MessageLimit:  buildOpts.LogLimit,
+		Color:         validateColor(buildOpts.Color),
+		LogLevel:      validateLogLevel(buildOpts.LogLevel),
+		Overrides:     validateLogOverrides(buildOpts.LogOverride),
+	}
+
+	// Validate that the current working directory is an absolute path
+	absWorkingDir := buildOpts.AbsWorkingDir
+	realFS, err := fs.RealFS(fs.RealFSOptions{
+		AbsWorkingDir: absWorkingDir,
+
+		// This is a long-lived file system object so do not cache calls to
+		// ReadDirectory() (they are normally cached for the duration of a build
+		// for performance).
+		DoNotCache: true,
+	})
+	if err != nil {
+		log := logger.NewStderrLog(logOptions)
+		log.AddError(nil, logger.Range{}, err.Error())
+		return nil, convertMessagesToPublic(logger.Error, log.Done())
+	}
+
+	// Do not re-evaluate plugins when rebuilding. Also make sure the working
+	// directory doesn't change, since breaking that invariant would break the
+	// validation that we just did above.
+	caches := cache.MakeCacheSet()
+	log := logger.NewDeferLog(logger.DeferLogNoVerboseOrDebug, logOptions.Overrides)
+	onEndCallbacks, onDisposeCallbacks, finalizeBuildOptions := loadPlugins(&buildOpts, realFS, log, caches)
+	options, entryPoints := validateBuildOptions(buildOpts, log, realFS)
+	finalizeBuildOptions(&options)
+	if buildOpts.AbsWorkingDir != absWorkingDir {
+		panic("Mutating \"AbsWorkingDir\" is not allowed")
+	}
+
+	// If we have errors already, then refuse to build any further. This only
+	// happens when the build options themselves contain validation errors.
+	msgs := log.Done()
+	if log.HasErrors() {
+		if logOptions.LogLevel < logger.LevelSilent {
+			// Print all deferred validation log messages to stderr. We defer all log
+			// messages that are generated above because warnings are re-printed for
+			// every rebuild and we don't want to double-print these warnings for the
+			// first build.
+			stderr := logger.NewStderrLog(logOptions)
+			for _, msg := range msgs {
+				stderr.AddMsg(msg)
+			}
+			stderr.Done()
+		}
+		return nil, convertMessagesToPublic(logger.Error, msgs)
+	}
+
+	args := rebuildArgs{
+		caches:             caches,
+		onEndCallbacks:     onEndCallbacks,
+		onDisposeCallbacks: onDisposeCallbacks,
+		logOptions:         logOptions,
+		logWarnings:        msgs,
+		entryPoints:        entryPoints,
+		options:            options,
+		mangleCache:        buildOpts.MangleCache,
+		absWorkingDir:      absWorkingDir,
+		write:              buildOpts.Write,
+	}
+
+	return &internalContext{
+		args:          args,
+		realFS:        realFS,
+		absWorkingDir: absWorkingDir,
+	}, nil
+}
+
+type buildInProgress struct {
+	state     rebuildState
+	waitGroup sync.WaitGroup
+	cancel    config.CancelFlag
+}
+
+type internalContext struct {
+	mutex         sync.Mutex
+	args          rebuildArgs
+	activeBuild   *buildInProgress
+	recentBuild   *BuildResult
+	realFS        fs.FS
+	absWorkingDir string
+	watcher       *watcher
+	handler       *apiHandler
+	didDispose    bool
+
+	// This saves just enough information to be able to compute a useful diff
+	// between two sets of output files. That way we don't need to hold both
+	// sets of output files in memory at once to compute a diff.
+	latestHashes map[string]string
+}
+
+func (ctx *internalContext) rebuild() rebuildState {
+	ctx.mutex.Lock()
+
+	// Ignore disposed contexts
+	if ctx.didDispose {
+		ctx.mutex.Unlock()
+		return rebuildState{}
+	}
+
+	// If there's already an active build, just return that build's result
+	if build := ctx.activeBuild; build != nil {
+		ctx.mutex.Unlock()
+		build.waitGroup.Wait()
+		return build.state
+	}
+
+	// Otherwise, start a new build
+	build := &buildInProgress{}
+	build.waitGroup.Add(1)
+	ctx.activeBuild = build
+	args := ctx.args
+	watcher := ctx.watcher
+	handler := ctx.handler
+	oldHashes := ctx.latestHashes
+	args.options.CancelFlag = &build.cancel
+	ctx.mutex.Unlock()
+
+	// Do the build without holding the mutex
+	var newHashes map[string]string
+	build.state, newHashes = rebuildImpl(args, oldHashes)
+	if handler != nil {
+		handler.broadcastBuildResult(build.state.result, newHashes)
+	}
+	if watcher != nil {
+		watcher.setWatchData(build.state.watchData)
+	}
+
+	// Store the recent build for the dev server
+	recentBuild := &build.state.result
+	ctx.mutex.Lock()
+	ctx.activeBuild = nil
+	ctx.recentBuild = recentBuild
+	ctx.latestHashes = newHashes
+	ctx.mutex.Unlock()
+
+	// Clear the recent build after it goes stale
+	go func() {
+		time.Sleep(250 * time.Millisecond)
+		ctx.mutex.Lock()
+		if ctx.recentBuild == recentBuild {
+			ctx.recentBuild = nil
+		}
+		ctx.mutex.Unlock()
+	}()
+
+	build.waitGroup.Done()
+	return build.state
+}
+
+// This is used by the dev server. The dev server does a rebuild on each
+// incoming request since a) we want incoming requests to always be up to
+// date and b) we don't necessarily know what output paths to even serve
+// without running another build (e.g. the hashes may have changed).
+//
+// However, there is a small period of time where we reuse old build results
+// instead of generating new ones. This is because page loads likely involve
+// multiple requests, and don't want to rebuild separately for each of those
+// requests.
+func (ctx *internalContext) activeBuildOrRecentBuildOrRebuild() BuildResult {
+	ctx.mutex.Lock()
+
+	// If there's already an active build, wait for it and return that
+	if build := ctx.activeBuild; build != nil {
+		ctx.mutex.Unlock()
+		build.waitGroup.Wait()
+		return build.state.result
+	}
+
+	// Then try to return a recentl already-completed build
+	if build := ctx.recentBuild; build != nil {
+		ctx.mutex.Unlock()
+		return *build
+	}
+
+	// Otherwise, fall back to rebuilding
+	ctx.mutex.Unlock()
+	return ctx.Rebuild()
+}
+
+func (ctx *internalContext) Rebuild() BuildResult {
+	return ctx.rebuild().result
+}
+
+func (ctx *internalContext) Watch(options WatchOptions) error {
+	ctx.mutex.Lock()
+	defer ctx.mutex.Unlock()
+
+	// Ignore disposed contexts
+	if ctx.didDispose {
+		return errors.New("Cannot watch a disposed context")
+	}
+
+	// Don't allow starting watch mode multiple times
+	if ctx.watcher != nil {
+		return errors.New("Watch mode has already been enabled")
+	}
+
+	logLevel := ctx.args.logOptions.LogLevel
+	ctx.watcher = &watcher{
+		fs:        ctx.realFS,
+		shouldLog: logLevel == logger.LevelInfo || logLevel == logger.LevelDebug || logLevel == logger.LevelVerbose,
+		useColor:  ctx.args.logOptions.Color,
+		rebuild: func() fs.WatchData {
+			return ctx.rebuild().watchData
+		},
+	}
+
+	// All subsequent builds will be watch mode builds
+	ctx.args.options.WatchMode = true
+
+	// Start the file watcher goroutine
+	ctx.watcher.start()
+
+	// Do the first watch mode build on another goroutine
+	go func() {
+		ctx.mutex.Lock()
+		build := ctx.activeBuild
+		ctx.mutex.Unlock()
+
+		// If there's an active build, then it's not a watch build. Wait for it to
+		// finish first so we don't just get this build when we call "Rebuild()".
+		if build != nil {
+			build.waitGroup.Wait()
+		}
+
+		// Trigger a rebuild now that we know all future builds will pick up on
+		// our watcher. This build will populate the initial watch data, which is
+		// necessary to be able to know what file system changes are relevant.
+		ctx.Rebuild()
+	}()
+	return nil
+}
+
+func (ctx *internalContext) Cancel() {
+	ctx.mutex.Lock()
+
+	// Ignore disposed contexts
+	if ctx.didDispose {
+		ctx.mutex.Unlock()
+		return
+	}
+
+	build := ctx.activeBuild
+	ctx.mutex.Unlock()
+
+	if build != nil {
+		// Tell observers to cut this build short
+		build.cancel.Cancel()
+
+		// Wait for the build to finish before returning
+		build.waitGroup.Wait()
+	}
+}
+
+func (ctx *internalContext) Dispose() {
+	// Only dispose once
+	ctx.mutex.Lock()
+	if ctx.didDispose {
+		ctx.mutex.Unlock()
+		return
+	}
+	ctx.didDispose = true
+	ctx.recentBuild = nil
+	build := ctx.activeBuild
+	ctx.mutex.Unlock()
+
+	if ctx.watcher != nil {
+		ctx.watcher.stop()
+	}
+	if ctx.handler != nil {
+		ctx.handler.stop()
+	}
+
+	// It's important to wait for the build to finish before returning. The JS
+	// API will unregister its callbacks when it returns. If that happens while
+	// the build is still in progress, that might cause the JS API to generate
+	// errors when we send it events (e.g. when it runs "onEnd" callbacks) that
+	// we then print to the terminal, which would be confusing.
+	if build != nil {
+		build.waitGroup.Wait()
+	}
+
+	// Run each "OnDispose" callback on its own goroutine
+	for _, fn := range ctx.args.onDisposeCallbacks {
+		go fn()
+	}
+}
+
+func prettyPrintByteCount(n int) string {
+	var size string
+	if n < 1024 {
+		size = fmt.Sprintf("%db ", n)
+	} else if n < 1024*1024 {
+		size = fmt.Sprintf("%.1fkb", float64(n)/(1024))
+	} else if n < 1024*1024*1024 {
+		size = fmt.Sprintf("%.1fmb", float64(n)/(1024*1024))
+	} else {
+		size = fmt.Sprintf("%.1fgb", float64(n)/(1024*1024*1024))
+	}
+	return size
+}
+
+func printSummary(color logger.UseColor, outputFiles []OutputFile, start time.Time) {
+	if len(outputFiles) == 0 {
+		return
+	}
+
+	var table logger.SummaryTable = make([]logger.SummaryTableEntry, len(outputFiles))
+
+	if cwd, err := os.Getwd(); err == nil {
+		if realFS, err := fs.RealFS(fs.RealFSOptions{AbsWorkingDir: cwd}); err == nil {
+			for i, file := range outputFiles {
+				path, ok := realFS.Rel(realFS.Cwd(), file.Path)
+				if !ok {
+					path = file.Path
+				}
+				base := realFS.Base(path)
+				n := len(file.Contents)
+				table[i] = logger.SummaryTableEntry{
+					Dir:         path[:len(path)-len(base)],
+					Base:        base,
+					Size:        prettyPrintByteCount(n),
+					Bytes:       n,
+					IsSourceMap: strings.HasSuffix(base, ".map"),
+				}
+			}
+		}
+	}
+
+	// Don't print the time taken by the build if we're running under Yarn 1
+	// since Yarn 1 always prints its own copy of the time taken by each command
+	if userAgent, ok := os.LookupEnv("npm_config_user_agent"); ok {
+		if strings.Contains(userAgent, "yarn/1.") {
+			logger.PrintSummary(color, table, nil)
+			return
+		}
+	}
+
+	logger.PrintSummary(color, table, &start)
+}
+
+func validateBuildOptions(
+	buildOpts BuildOptions,
+	log logger.Log,
+	realFS fs.FS,
+) (
+	options config.Options,
+	entryPoints []bundler.EntryPoint,
+) {
+	jsFeatures, cssFeatures, cssPrefixData, targetEnv := validateFeatures(log, buildOpts.Target, buildOpts.Engines)
+	jsOverrides, jsMask, cssOverrides, cssMask := validateSupported(log, buildOpts.Supported)
+	outJS, outCSS := validateOutputExtensions(log, buildOpts.OutExtension)
+	bannerJS, bannerCSS := validateBannerOrFooter(log, "banner", buildOpts.Banner)
+	footerJS, footerCSS := validateBannerOrFooter(log, "footer", buildOpts.Footer)
+	minify := buildOpts.MinifyWhitespace && buildOpts.MinifyIdentifiers && buildOpts.MinifySyntax
+	platform := validatePlatform(buildOpts.Platform)
+	defines, injectedDefines := validateDefines(log, buildOpts.Define, buildOpts.Pure, platform, true /* isBuildAPI */, minify, buildOpts.Drop)
+	options = config.Options{
+		CSSPrefixData:                      cssPrefixData,
+		UnsupportedJSFeatures:              jsFeatures.ApplyOverrides(jsOverrides, jsMask),
+		UnsupportedCSSFeatures:             cssFeatures.ApplyOverrides(cssOverrides, cssMask),
+		UnsupportedJSFeatureOverrides:      jsOverrides,
+		UnsupportedJSFeatureOverridesMask:  jsMask,
+		UnsupportedCSSFeatureOverrides:     cssOverrides,
+		UnsupportedCSSFeatureOverridesMask: cssMask,
+		OriginalTargetEnv:                  targetEnv,
+		JSX: config.JSXOptions{
+			Preserve:         buildOpts.JSX == JSXPreserve,
+			AutomaticRuntime: buildOpts.JSX == JSXAutomatic,
+			Factory:          validateJSXExpr(log, buildOpts.JSXFactory, "factory"),
+			Fragment:         validateJSXExpr(log, buildOpts.JSXFragment, "fragment"),
+			Development:      buildOpts.JSXDev,
+			ImportSource:     buildOpts.JSXImportSource,
+			SideEffects:      buildOpts.JSXSideEffects,
+		},
+		Defines:               defines,
+		InjectedDefines:       injectedDefines,
+		Platform:              platform,
+		SourceMap:             validateSourceMap(buildOpts.Sourcemap),
+		LegalComments:         validateLegalComments(buildOpts.LegalComments, buildOpts.Bundle),
+		SourceRoot:            buildOpts.SourceRoot,
+		ExcludeSourcesContent: buildOpts.SourcesContent == SourcesContentExclude,
+		MinifySyntax:          buildOpts.MinifySyntax,
+		MinifyWhitespace:      buildOpts.MinifyWhitespace,
+		MinifyIdentifiers:     buildOpts.MinifyIdentifiers,
+		LineLimit:             buildOpts.LineLimit,
+		MangleProps:           validateRegex(log, "mangle props", buildOpts.MangleProps),
+		ReserveProps:          validateRegex(log, "reserve props", buildOpts.ReserveProps),
+		MangleQuoted:          buildOpts.MangleQuoted == MangleQuotedTrue,
+		DropLabels:            append([]string{}, buildOpts.DropLabels...),
+		DropDebugger:          (buildOpts.Drop & DropDebugger) != 0,
+		AllowOverwrite:        buildOpts.AllowOverwrite,
+		ASCIIOnly:             validateASCIIOnly(buildOpts.Charset),
+		IgnoreDCEAnnotations:  buildOpts.IgnoreAnnotations,
+		TreeShaking:           validateTreeShaking(buildOpts.TreeShaking, buildOpts.Bundle, buildOpts.Format),
+		GlobalName:            validateGlobalName(log, buildOpts.GlobalName),
+		CodeSplitting:         buildOpts.Splitting,
+		OutputFormat:          validateFormat(buildOpts.Format),
+		AbsOutputFile:         validatePath(log, realFS, buildOpts.Outfile, "outfile path"),
+		AbsOutputDir:          validatePath(log, realFS, buildOpts.Outdir, "outdir path"),
+		AbsOutputBase:         validatePath(log, realFS, buildOpts.Outbase, "outbase path"),
+		NeedsMetafile:         buildOpts.Metafile,
+		EntryPathTemplate:     validatePathTemplate(buildOpts.EntryNames),
+		ChunkPathTemplate:     validatePathTemplate(buildOpts.ChunkNames),
+		AssetPathTemplate:     validatePathTemplate(buildOpts.AssetNames),
+		OutputExtensionJS:     outJS,
+		OutputExtensionCSS:    outCSS,
+		ExtensionToLoader:     validateLoaders(log, buildOpts.Loader),
+		ExtensionOrder:        validateResolveExtensions(log, buildOpts.ResolveExtensions),
+		ExternalSettings:      validateExternals(log, realFS, buildOpts.External),
+		ExternalPackages:      validateExternalPackages(buildOpts.Packages),
+		PackageAliases:        validateAlias(log, realFS, buildOpts.Alias),
+		TSConfigPath:          validatePath(log, realFS, buildOpts.Tsconfig, "tsconfig path"),
+		TSConfigRaw:           buildOpts.TsconfigRaw,
+		MainFields:            buildOpts.MainFields,
+		PublicPath:            buildOpts.PublicPath,
+		KeepNames:             buildOpts.KeepNames,
+		InjectPaths:           append([]string{}, buildOpts.Inject...),
+		AbsNodePaths:          make([]string, len(buildOpts.NodePaths)),
+		JSBanner:              bannerJS,
+		JSFooter:              footerJS,
+		CSSBanner:             bannerCSS,
+		CSSFooter:             footerCSS,
+		PreserveSymlinks:      buildOpts.PreserveSymlinks,
+	}
+	validateKeepNames(log, &options)
+	if buildOpts.Conditions != nil {
+		options.Conditions = append([]string{}, buildOpts.Conditions...)
+	}
+	if options.MainFields != nil {
+		options.MainFields = append([]string{}, options.MainFields...)
+	}
+	for i, path := range buildOpts.NodePaths {
+		options.AbsNodePaths[i] = validatePath(log, realFS, path, "node path")
+	}
+	entryPoints = make([]bundler.EntryPoint, 0, len(buildOpts.EntryPoints)+len(buildOpts.EntryPointsAdvanced))
+	hasEntryPointWithWildcard := false
+	for _, ep := range buildOpts.EntryPoints {
+		entryPoints = append(entryPoints, bundler.EntryPoint{InputPath: ep})
+		if strings.ContainsRune(ep, '*') {
+			hasEntryPointWithWildcard = true
+		}
+	}
+	for _, ep := range buildOpts.EntryPointsAdvanced {
+		entryPoints = append(entryPoints, bundler.EntryPoint{InputPath: ep.InputPath, OutputPath: ep.OutputPath})
+		if strings.ContainsRune(ep.InputPath, '*') {
+			hasEntryPointWithWildcard = true
+		}
+	}
+	entryPointCount := len(entryPoints)
+	if buildOpts.Stdin != nil {
+		entryPointCount++
+		options.Stdin = &config.StdinInfo{
+			Loader:        validateLoader(buildOpts.Stdin.Loader),
+			Contents:      buildOpts.Stdin.Contents,
+			SourceFile:    buildOpts.Stdin.Sourcefile,
+			AbsResolveDir: validatePath(log, realFS, buildOpts.Stdin.ResolveDir, "resolve directory path"),
+		}
+	}
+
+	if options.AbsOutputDir == "" && (entryPointCount > 1 || hasEntryPointWithWildcard) {
+		log.AddError(nil, logger.Range{},
+			"Must use \"outdir\" when there are multiple input files")
+	} else if options.AbsOutputDir == "" && options.CodeSplitting {
+		log.AddError(nil, logger.Range{},
+			"Must use \"outdir\" when code splitting is enabled")
+	} else if options.AbsOutputFile != "" && options.AbsOutputDir != "" {
+		log.AddError(nil, logger.Range{}, "Cannot use both \"outfile\" and \"outdir\"")
+	} else if options.AbsOutputFile != "" {
+		// If the output file is specified, use it to derive the output directory
+		options.AbsOutputDir = realFS.Dir(options.AbsOutputFile)
+	} else if options.AbsOutputDir == "" {
+		options.WriteToStdout = true
+
+		// Forbid certain features when writing to stdout
+		if options.SourceMap != config.SourceMapNone && options.SourceMap != config.SourceMapInline {
+			log.AddError(nil, logger.Range{}, "Cannot use an external source map without an output path")
+		}
+		if options.LegalComments.HasExternalFile() {
+			log.AddError(nil, logger.Range{}, "Cannot use linked or external legal comments without an output path")
+		}
+		for _, loader := range options.ExtensionToLoader {
+			if loader == config.LoaderFile {
+				log.AddError(nil, logger.Range{}, "Cannot use the \"file\" loader without an output path")
+				break
+			}
+			if loader == config.LoaderCopy {
+				log.AddError(nil, logger.Range{}, "Cannot use the \"copy\" loader without an output path")
+				break
+			}
+		}
+
+		// Use the current directory as the output directory instead of an empty
+		// string because external modules with relative paths need a base directory.
+		options.AbsOutputDir = realFS.Cwd()
+	}
+
+	if !buildOpts.Bundle {
+		// Disallow bundle-only options when not bundling
+		if options.ExternalSettings.PreResolve.HasMatchers() || options.ExternalSettings.PostResolve.HasMatchers() {
+			log.AddError(nil, logger.Range{}, "Cannot use \"external\" without \"bundle\"")
+		}
+		if len(options.PackageAliases) > 0 {
+			log.AddError(nil, logger.Range{}, "Cannot use \"alias\" without \"bundle\"")
+		}
+	} else if options.OutputFormat == config.FormatPreserve {
+		// If the format isn't specified, set the default format using the platform
+		switch options.Platform {
+		case config.PlatformBrowser:
+			options.OutputFormat = config.FormatIIFE
+		case config.PlatformNode:
+			options.OutputFormat = config.FormatCommonJS
+		case config.PlatformNeutral:
+			options.OutputFormat = config.FormatESModule
+		}
+	}
+
+	// Set the output mode using other settings
+	if buildOpts.Bundle {
+		options.Mode = config.ModeBundle
+	} else if options.OutputFormat != config.FormatPreserve {
+		options.Mode = config.ModeConvertFormat
+	}
+
+	// Automatically enable the "module" condition for better tree shaking
+	if options.Conditions == nil && options.Platform != config.PlatformNeutral {
+		options.Conditions = []string{"module"}
+	}
+
+	// Code splitting is experimental and currently only enabled for ES6 modules
+	if options.CodeSplitting && options.OutputFormat != config.FormatESModule {
+		log.AddError(nil, logger.Range{}, "Splitting currently only works with the \"esm\" format")
+	}
+
+	// Code splitting is experimental and currently only enabled for ES6 modules
+	if options.TSConfigPath != "" && options.TSConfigRaw != "" {
+		log.AddError(nil, logger.Range{}, "Cannot provide \"tsconfig\" as both a raw string and a path")
+	}
+
+	// If we aren't writing the output to the file system, then we can allow the
+	// output paths to be the same as the input paths. This helps when serving.
+	if !buildOpts.Write {
+		options.AllowOverwrite = true
+	}
+
+	return
+}
+
+type onEndCallback struct {
+	pluginName string
+	fn         func(*BuildResult) (OnEndResult, error)
+}
+
+type rebuildArgs struct {
+	caches             *cache.CacheSet
+	onEndCallbacks     []onEndCallback
+	onDisposeCallbacks []func()
+	logOptions         logger.OutputOptions
+	logWarnings        []logger.Msg
+	entryPoints        []bundler.EntryPoint
+	options            config.Options
+	mangleCache        map[string]interface{}
+	absWorkingDir      string
+	write              bool
+}
+
+type rebuildState struct {
+	result    BuildResult
+	watchData fs.WatchData
+	options   config.Options
+}
+
+func rebuildImpl(args rebuildArgs, oldHashes map[string]string) (rebuildState, map[string]string) {
+	log := logger.NewStderrLog(args.logOptions)
+
+	// All validation warnings are repeated for every rebuild
+	for _, msg := range args.logWarnings {
+		log.AddMsg(msg)
+	}
+
+	// Convert and validate the buildOpts
+	realFS, err := fs.RealFS(fs.RealFSOptions{
+		AbsWorkingDir: args.absWorkingDir,
+		WantWatchData: args.options.WatchMode,
+	})
+	if err != nil {
+		// This should already have been checked by the caller
+		panic(err.Error())
+	}
+
+	var result BuildResult
+	var watchData fs.WatchData
+	var toWriteToStdout []byte
+
+	var timer *helpers.Timer
+	if api_helpers.UseTimer {
+		timer = &helpers.Timer{}
+	}
+
+	// Scan over the bundle
+	bundle := bundler.ScanBundle(config.BuildCall, log, realFS, args.caches, args.entryPoints, args.options, timer)
+	watchData = realFS.WatchData()
+
+	// The new build summary remains the same as the old one when there are
+	// errors. A failed build shouldn't erase the previous successful build.
+	newHashes := oldHashes
+
+	// Stop now if there were errors
+	if !log.HasErrors() {
+		// Compile the bundle
+		result.MangleCache = cloneMangleCache(log, args.mangleCache)
+		results, metafile := bundle.Compile(log, timer, result.MangleCache, linker.Link)
+
+		// Canceling a build generates a single error at the end of the build
+		if args.options.CancelFlag.DidCancel() {
+			log.AddError(nil, logger.Range{}, "The build was canceled")
+		}
+
+		// Stop now if there were errors
+		if !log.HasErrors() {
+			result.Metafile = metafile
+
+			// Populate the results to return
+			var hashBytes [8]byte
+			result.OutputFiles = make([]OutputFile, len(results))
+			newHashes = make(map[string]string)
+			for i, item := range results {
+				if args.options.WriteToStdout {
+					item.AbsPath = "<stdout>"
+				}
+				hasher := xxhash.New()
+				hasher.Write(item.Contents)
+				binary.LittleEndian.PutUint64(hashBytes[:], hasher.Sum64())
+				hash := base64.RawStdEncoding.EncodeToString(hashBytes[:])
+				result.OutputFiles[i] = OutputFile{
+					Path:     item.AbsPath,
+					Contents: item.Contents,
+					Hash:     hash,
+				}
+				newHashes[item.AbsPath] = hash
+			}
+
+			// Write output files before "OnEnd" callbacks run so they can expect
+			// output files to exist on the file system. "OnEnd" callbacks can be
+			// used to move output files to a different location after the build.
+			if args.write {
+				timer.Begin("Write output files")
+				if args.options.WriteToStdout {
+					// Special-case writing to stdout
+					if len(results) != 1 {
+						log.AddError(nil, logger.Range{}, fmt.Sprintf(
+							"Internal error: did not expect to generate %d files when writing to stdout", len(results)))
+					} else {
+						// Print this later on, at the end of the current function
+						toWriteToStdout = results[0].Contents
+					}
+				} else {
+					// Delete old files that are no longer relevant
+					var toDelete []string
+					for absPath := range oldHashes {
+						if _, ok := newHashes[absPath]; !ok {
+							toDelete = append(toDelete, absPath)
+						}
+					}
+
+					// Process all file operations in parallel
+					waitGroup := sync.WaitGroup{}
+					waitGroup.Add(len(results) + len(toDelete))
+					for _, result := range results {
+						go func(result graph.OutputFile) {
+							defer waitGroup.Done()
+							fs.BeforeFileOpen()
+							defer fs.AfterFileClose()
+							if oldHash, ok := oldHashes[result.AbsPath]; ok && oldHash == newHashes[result.AbsPath] {
+								if contents, err := ioutil.ReadFile(result.AbsPath); err == nil && bytes.Equal(contents, result.Contents) {
+									// Skip writing out files that haven't changed since last time
+									return
+								}
+							}
+							if err := fs.MkdirAll(realFS, realFS.Dir(result.AbsPath), 0755); err != nil {
+								log.AddError(nil, logger.Range{}, fmt.Sprintf(
+									"Failed to create output directory: %s", err.Error()))
+							} else {
+								var mode os.FileMode = 0666
+								if result.IsExecutable {
+									mode = 0777
+								}
+								if err := ioutil.WriteFile(result.AbsPath, result.Contents, mode); err != nil {
+									log.AddError(nil, logger.Range{}, fmt.Sprintf(
+										"Failed to write to output file: %s", err.Error()))
+								}
+							}
+						}(result)
+					}
+					for _, absPath := range toDelete {
+						go func(absPath string) {
+							defer waitGroup.Done()
+							fs.BeforeFileOpen()
+							defer fs.AfterFileClose()
+							os.Remove(absPath)
+						}(absPath)
+					}
+					waitGroup.Wait()
+				}
+				timer.End("Write output files")
+			}
+		}
+	}
+
+	// Only return the mangle cache for a successful build
+	if log.HasErrors() {
+		result.MangleCache = nil
+	}
+
+	// Populate the result object with the messages so far
+	msgs := log.Peek()
+	result.Errors = convertMessagesToPublic(logger.Error, msgs)
+	result.Warnings = convertMessagesToPublic(logger.Warning, msgs)
+
+	// Run any registered "OnEnd" callbacks now. These always run regardless of
+	// whether the current build has bee canceled or not. They can check for
+	// errors by checking the error array in the build result, and canceled
+	// builds should always have at least one error.
+	timer.Begin("On-end callbacks")
+	for _, onEnd := range args.onEndCallbacks {
+		fromPlugin, thrown := onEnd.fn(&result)
+
+		// Report errors and warnings generated by the plugin
+		for i := range fromPlugin.Errors {
+			if fromPlugin.Errors[i].PluginName == "" {
+				fromPlugin.Errors[i].PluginName = onEnd.pluginName
+			}
+		}
+		for i := range fromPlugin.Warnings {
+			if fromPlugin.Warnings[i].PluginName == "" {
+				fromPlugin.Warnings[i].PluginName = onEnd.pluginName
+			}
+		}
+
+		// Report errors thrown by the plugin itself
+		if thrown != nil {
+			fromPlugin.Errors = append(fromPlugin.Errors, Message{
+				PluginName: onEnd.pluginName,
+				Text:       thrown.Error(),
+			})
+		}
+
+		// Log any errors and warnings generated above
+		for _, msg := range convertErrorsAndWarningsToInternal(fromPlugin.Errors, fromPlugin.Warnings) {
+			log.AddMsg(msg)
+		}
+
+		// Add the errors and warnings to the result object
+		result.Errors = append(result.Errors, fromPlugin.Errors...)
+		result.Warnings = append(result.Warnings, fromPlugin.Warnings...)
+
+		// Stop if an "onEnd" callback failed. This counts as a build failure.
+		if len(fromPlugin.Errors) > 0 {
+			break
+		}
+	}
+	timer.End("On-end callbacks")
+
+	// Log timing information now that we're all done
+	timer.Log(log)
+
+	// End the log after "OnEnd" callbacks have added any additional errors and/or
+	// warnings. This may may print any warnings that were deferred up until this
+	// point, as well as a message with the number of errors and/or warnings
+	// omitted due to the configured log limit.
+	log.Done()
+
+	// Only write to stdout after the log has been finalized. We want this output
+	// to show up in the terminal after the message that was printed above.
+	if toWriteToStdout != nil {
+		os.Stdout.Write(toWriteToStdout)
+	}
+
+	return rebuildState{
+		result:    result,
+		options:   args.options,
+		watchData: watchData,
+	}, newHashes
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// Transform API
+
+func transformImpl(input string, transformOpts TransformOptions) TransformResult {
+	log := logger.NewStderrLog(logger.OutputOptions{
+		IncludeSource: true,
+		MessageLimit:  transformOpts.LogLimit,
+		Color:         validateColor(transformOpts.Color),
+		LogLevel:      validateLogLevel(transformOpts.LogLevel),
+		Overrides:     validateLogOverrides(transformOpts.LogOverride),
+	})
+	caches := cache.MakeCacheSet()
+
+	// Apply default values
+	if transformOpts.Sourcefile == "" {
+		transformOpts.Sourcefile = "<stdin>"
+	}
+	if transformOpts.Loader == LoaderNone {
+		transformOpts.Loader = LoaderJS
+	}
+
+	// Convert and validate the transformOpts
+	jsFeatures, cssFeatures, cssPrefixData, targetEnv := validateFeatures(log, transformOpts.Target, transformOpts.Engines)
+	jsOverrides, jsMask, cssOverrides, cssMask := validateSupported(log, transformOpts.Supported)
+	platform := validatePlatform(transformOpts.Platform)
+	defines, injectedDefines := validateDefines(log, transformOpts.Define, transformOpts.Pure, platform, false /* isBuildAPI */, false /* minify */, transformOpts.Drop)
+	mangleCache := cloneMangleCache(log, transformOpts.MangleCache)
+	options := config.Options{
+		CSSPrefixData:                      cssPrefixData,
+		UnsupportedJSFeatures:              jsFeatures.ApplyOverrides(jsOverrides, jsMask),
+		UnsupportedCSSFeatures:             cssFeatures.ApplyOverrides(cssOverrides, cssMask),
+		UnsupportedJSFeatureOverrides:      jsOverrides,
+		UnsupportedJSFeatureOverridesMask:  jsMask,
+		UnsupportedCSSFeatureOverrides:     cssOverrides,
+		UnsupportedCSSFeatureOverridesMask: cssMask,
+		OriginalTargetEnv:                  targetEnv,
+		TSConfigRaw:                        transformOpts.TsconfigRaw,
+		JSX: config.JSXOptions{
+			Preserve:         transformOpts.JSX == JSXPreserve,
+			AutomaticRuntime: transformOpts.JSX == JSXAutomatic,
+			Factory:          validateJSXExpr(log, transformOpts.JSXFactory, "factory"),
+			Fragment:         validateJSXExpr(log, transformOpts.JSXFragment, "fragment"),
+			Development:      transformOpts.JSXDev,
+			ImportSource:     transformOpts.JSXImportSource,
+			SideEffects:      transformOpts.JSXSideEffects,
+		},
+		Defines:               defines,
+		InjectedDefines:       injectedDefines,
+		Platform:              platform,
+		SourceMap:             validateSourceMap(transformOpts.Sourcemap),
+		LegalComments:         validateLegalComments(transformOpts.LegalComments, false /* bundle */),
+		SourceRoot:            transformOpts.SourceRoot,
+		ExcludeSourcesContent: transformOpts.SourcesContent == SourcesContentExclude,
+		OutputFormat:          validateFormat(transformOpts.Format),
+		GlobalName:            validateGlobalName(log, transformOpts.GlobalName),
+		MinifySyntax:          transformOpts.MinifySyntax,
+		MinifyWhitespace:      transformOpts.MinifyWhitespace,
+		MinifyIdentifiers:     transformOpts.MinifyIdentifiers,
+		LineLimit:             transformOpts.LineLimit,
+		MangleProps:           validateRegex(log, "mangle props", transformOpts.MangleProps),
+		ReserveProps:          validateRegex(log, "reserve props", transformOpts.ReserveProps),
+		MangleQuoted:          transformOpts.MangleQuoted == MangleQuotedTrue,
+		DropLabels:            append([]string{}, transformOpts.DropLabels...),
+		DropDebugger:          (transformOpts.Drop & DropDebugger) != 0,
+		ASCIIOnly:             validateASCIIOnly(transformOpts.Charset),
+		IgnoreDCEAnnotations:  transformOpts.IgnoreAnnotations,
+		TreeShaking:           validateTreeShaking(transformOpts.TreeShaking, false /* bundle */, transformOpts.Format),
+		AbsOutputFile:         transformOpts.Sourcefile + "-out",
+		KeepNames:             transformOpts.KeepNames,
+		Stdin: &config.StdinInfo{
+			Loader:     validateLoader(transformOpts.Loader),
+			Contents:   input,
+			SourceFile: transformOpts.Sourcefile,
+		},
+	}
+	validateKeepNames(log, &options)
+	if options.Stdin.Loader.IsCSS() {
+		options.CSSBanner = transformOpts.Banner
+		options.CSSFooter = transformOpts.Footer
+	} else {
+		options.JSBanner = transformOpts.Banner
+		options.JSFooter = transformOpts.Footer
+	}
+	if options.SourceMap == config.SourceMapLinkedWithComment {
+		// Linked source maps don't make sense because there's no output file name
+		log.AddError(nil, logger.Range{}, "Cannot transform with linked source maps")
+	}
+	if options.SourceMap != config.SourceMapNone && options.Stdin.SourceFile == "" {
+		log.AddError(nil, logger.Range{},
+			"Must use \"sourcefile\" with \"sourcemap\" to set the original file name")
+	}
+	if logger.API == logger.CLIAPI {
+		if options.LegalComments.HasExternalFile() {
+			log.AddError(nil, logger.Range{}, "Cannot transform with linked or external legal comments")
+		}
+	} else if options.LegalComments == config.LegalCommentsLinkedWithComment {
+		log.AddError(nil, logger.Range{}, "Cannot transform with linked legal comments")
+	}
+
+	// Set the output mode using other settings
+	if options.OutputFormat != config.FormatPreserve {
+		options.Mode = config.ModeConvertFormat
+	}
+
+	var results []graph.OutputFile
+
+	// Stop now if there were errors
+	if !log.HasErrors() {
+		var timer *helpers.Timer
+		if api_helpers.UseTimer {
+			timer = &helpers.Timer{}
+		}
+
+		// Scan over the bundle
+		mockFS := fs.MockFS(make(map[string]string), fs.MockUnix, "/")
+		bundle := bundler.ScanBundle(config.TransformCall, log, mockFS, caches, nil, options, timer)
+
+		// Stop now if there were errors
+		if !log.HasErrors() {
+			// Compile the bundle
+			results, _ = bundle.Compile(log, timer, mangleCache, linker.Link)
+		}
+
+		timer.Log(log)
+	}
+
+	// Return the results
+	var code []byte
+	var sourceMap []byte
+	var legalComments []byte
+
+	var shortestAbsPath string
+	for _, result := range results {
+		if shortestAbsPath == "" || len(result.AbsPath) < len(shortestAbsPath) {
+			shortestAbsPath = result.AbsPath
+		}
+	}
+
+	// Unpack the JavaScript file, the source map file, and the legal comments file
+	for _, result := range results {
+		switch result.AbsPath {
+		case shortestAbsPath:
+			code = result.Contents
+		case shortestAbsPath + ".map":
+			sourceMap = result.Contents
+		case shortestAbsPath + ".LEGAL.txt":
+			legalComments = result.Contents
+		}
+	}
+
+	// Only return the mangle cache for a successful build
+	if log.HasErrors() {
+		mangleCache = nil
+	}
+
+	msgs := log.Done()
+	return TransformResult{
+		Errors:        convertMessagesToPublic(logger.Error, msgs),
+		Warnings:      convertMessagesToPublic(logger.Warning, msgs),
+		Code:          code,
+		Map:           sourceMap,
+		LegalComments: legalComments,
+		MangleCache:   mangleCache,
+	}
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// Plugin API
+
+type pluginImpl struct {
+	log    logger.Log
+	fs     fs.FS
+	plugin config.Plugin
+}
+
+func (impl *pluginImpl) onStart(callback func() (OnStartResult, error)) {
+	impl.plugin.OnStart = append(impl.plugin.OnStart, config.OnStart{
+		Name: impl.plugin.Name,
+		Callback: func() (result config.OnStartResult) {
+			response, err := callback()
+
+			if err != nil {
+				result.ThrownError = err
+				return
+			}
+
+			// Convert log messages
+			result.Msgs = convertErrorsAndWarningsToInternal(response.Errors, response.Warnings)
+			return
+		},
+	})
+}
+
+func importKindToResolveKind(kind ast.ImportKind) ResolveKind {
+	switch kind {
+	case ast.ImportEntryPoint:
+		return ResolveEntryPoint
+	case ast.ImportStmt:
+		return ResolveJSImportStatement
+	case ast.ImportRequire:
+		return ResolveJSRequireCall
+	case ast.ImportDynamic:
+		return ResolveJSDynamicImport
+	case ast.ImportRequireResolve:
+		return ResolveJSRequireResolve
+	case ast.ImportAt:
+		return ResolveCSSImportRule
+	case ast.ImportComposesFrom:
+		return ResolveCSSComposesFrom
+	case ast.ImportURL:
+		return ResolveCSSURLToken
+	default:
+		panic("Internal error")
+	}
+}
+
+func resolveKindToImportKind(kind ResolveKind) ast.ImportKind {
+	switch kind {
+	case ResolveEntryPoint:
+		return ast.ImportEntryPoint
+	case ResolveJSImportStatement:
+		return ast.ImportStmt
+	case ResolveJSRequireCall:
+		return ast.ImportRequire
+	case ResolveJSDynamicImport:
+		return ast.ImportDynamic
+	case ResolveJSRequireResolve:
+		return ast.ImportRequireResolve
+	case ResolveCSSImportRule:
+		return ast.ImportAt
+	case ResolveCSSComposesFrom:
+		return ast.ImportComposesFrom
+	case ResolveCSSURLToken:
+		return ast.ImportURL
+	default:
+		panic("Internal error")
+	}
+}
+
+func (impl *pluginImpl) onResolve(options OnResolveOptions, callback func(OnResolveArgs) (OnResolveResult, error)) {
+	filter, err := config.CompileFilterForPlugin(impl.plugin.Name, "OnResolve", options.Filter)
+	if filter == nil {
+		impl.log.AddError(nil, logger.Range{}, err.Error())
+		return
+	}
+
+	impl.plugin.OnResolve = append(impl.plugin.OnResolve, config.OnResolve{
+		Name:      impl.plugin.Name,
+		Filter:    filter,
+		Namespace: options.Namespace,
+		Callback: func(args config.OnResolveArgs) (result config.OnResolveResult) {
+			response, err := callback(OnResolveArgs{
+				Path:       args.Path,
+				Importer:   args.Importer.Text,
+				Namespace:  args.Importer.Namespace,
+				ResolveDir: args.ResolveDir,
+				Kind:       importKindToResolveKind(args.Kind),
+				PluginData: args.PluginData,
+				With:       args.With.DecodeIntoMap(),
+			})
+			result.PluginName = response.PluginName
+			result.AbsWatchFiles = impl.validatePathsArray(response.WatchFiles, "watch file")
+			result.AbsWatchDirs = impl.validatePathsArray(response.WatchDirs, "watch directory")
+
+			// Restrict the suffix to start with "?" or "#" for now to match esbuild's behavior
+			if err == nil && response.Suffix != "" && response.Suffix[0] != '?' && response.Suffix[0] != '#' {
+				err = fmt.Errorf("Invalid path suffix %q returned from plugin (must start with \"?\" or \"#\")", response.Suffix)
+			}
+
+			if err != nil {
+				result.ThrownError = err
+				return
+			}
+
+			result.Path = logger.Path{
+				Text:          response.Path,
+				Namespace:     response.Namespace,
+				IgnoredSuffix: response.Suffix,
+			}
+			result.External = response.External
+			result.IsSideEffectFree = response.SideEffects == SideEffectsFalse
+			result.PluginData = response.PluginData
+
+			// Convert log messages
+			result.Msgs = convertErrorsAndWarningsToInternal(response.Errors, response.Warnings)
+
+			// Warn if the plugin returned things without resolving the path
+			if response.Path == "" && !response.External {
+				var what string
+				if response.Namespace != "" {
+					what = "namespace"
+				} else if response.Suffix != "" {
+					what = "suffix"
+				} else if response.PluginData != nil {
+					what = "pluginData"
+				} else if response.WatchFiles != nil {
+					what = "watchFiles"
+				} else if response.WatchDirs != nil {
+					what = "watchDirs"
+				}
+				if what != "" {
+					path := "path"
+					if logger.API == logger.GoAPI {
+						what = strings.Title(what)
+						path = strings.Title(path)
+					}
+					result.Msgs = append(result.Msgs, logger.Msg{
+						Kind: logger.Warning,
+						Data: logger.MsgData{Text: fmt.Sprintf("Returning %q doesn't do anything when %q is empty", what, path)},
+					})
+				}
+			}
+			return
+		},
+	})
+}
+
+func (impl *pluginImpl) onLoad(options OnLoadOptions, callback func(OnLoadArgs) (OnLoadResult, error)) {
+	filter, err := config.CompileFilterForPlugin(impl.plugin.Name, "OnLoad", options.Filter)
+	if filter == nil {
+		impl.log.AddError(nil, logger.Range{}, err.Error())
+		return
+	}
+
+	impl.plugin.OnLoad = append(impl.plugin.OnLoad, config.OnLoad{
+		Filter:    filter,
+		Namespace: options.Namespace,
+		Callback: func(args config.OnLoadArgs) (result config.OnLoadResult) {
+			response, err := callback(OnLoadArgs{
+				Path:       args.Path.Text,
+				Namespace:  args.Path.Namespace,
+				PluginData: args.PluginData,
+				Suffix:     args.Path.IgnoredSuffix,
+				With:       args.Path.ImportAttributes.DecodeIntoMap(),
+			})
+			result.PluginName = response.PluginName
+			result.AbsWatchFiles = impl.validatePathsArray(response.WatchFiles, "watch file")
+			result.AbsWatchDirs = impl.validatePathsArray(response.WatchDirs, "watch directory")
+
+			if err != nil {
+				result.ThrownError = err
+				return
+			}
+
+			result.Contents = response.Contents
+			result.Loader = validateLoader(response.Loader)
+			result.PluginData = response.PluginData
+			pathKind := fmt.Sprintf("resolve directory path for plugin %q", impl.plugin.Name)
+			if absPath := validatePath(impl.log, impl.fs, response.ResolveDir, pathKind); absPath != "" {
+				result.AbsResolveDir = absPath
+			}
+
+			// Convert log messages
+			result.Msgs = convertErrorsAndWarningsToInternal(response.Errors, response.Warnings)
+			return
+		},
+	})
+}
+
+func (impl *pluginImpl) validatePathsArray(pathsIn []string, name string) (pathsOut []string) {
+	if len(pathsIn) > 0 {
+		pathKind := fmt.Sprintf("%s path for plugin %q", name, impl.plugin.Name)
+		for _, relPath := range pathsIn {
+			if absPath := validatePath(impl.log, impl.fs, relPath, pathKind); absPath != "" {
+				pathsOut = append(pathsOut, absPath)
+			}
+		}
+	}
+	return
+}
+
+func loadPlugins(initialOptions *BuildOptions, fs fs.FS, log logger.Log, caches *cache.CacheSet) (
+	onEndCallbacks []onEndCallback,
+	onDisposeCallbacks []func(),
+	finalizeBuildOptions func(*config.Options),
+) {
+	// Clone the plugin array to guard against mutation during iteration
+	clone := append(make([]Plugin, 0, len(initialOptions.Plugins)), initialOptions.Plugins...)
+
+	var optionsForResolve *config.Options
+	var plugins []config.Plugin
+
+	// This is called after the build options have been validated
+	finalizeBuildOptions = func(options *config.Options) {
+		options.Plugins = plugins
+		optionsForResolve = options
+	}
+
+	for i, item := range clone {
+		if item.Name == "" {
+			log.AddError(nil, logger.Range{}, fmt.Sprintf("Plugin at index %d is missing a name", i))
+			continue
+		}
+
+		impl := &pluginImpl{
+			fs:     fs,
+			log:    log,
+			plugin: config.Plugin{Name: item.Name},
+		}
+
+		resolve := func(path string, options ResolveOptions) (result ResolveResult) {
+			// If options are missing, then this is being called before plugin setup
+			// has finished. That isn't allowed because plugin setup is allowed to
+			// change the initial options object, which can affect path resolution.
+			if optionsForResolve == nil {
+				return ResolveResult{Errors: []Message{{Text: "Cannot call \"resolve\" before plugin setup has completed"}}}
+			}
+
+			if options.Kind == ResolveNone {
+				return ResolveResult{Errors: []Message{{Text: "Must specify \"kind\" when calling \"resolve\""}}}
+			}
+
+			// Make a new resolver so it has its own log
+			log := logger.NewDeferLog(logger.DeferLogNoVerboseOrDebug, validateLogOverrides(initialOptions.LogOverride))
+			optionsClone := *optionsForResolve
+			resolver := resolver.NewResolver(config.BuildCall, fs, log, caches, &optionsClone)
+
+			// Make sure the resolve directory is an absolute path, which can fail
+			absResolveDir := validatePath(log, fs, options.ResolveDir, "resolve directory")
+			if log.HasErrors() {
+				msgs := log.Done()
+				result.Errors = convertMessagesToPublic(logger.Error, msgs)
+				result.Warnings = convertMessagesToPublic(logger.Warning, msgs)
+				return
+			}
+
+			// Run path resolution
+			kind := resolveKindToImportKind(options.Kind)
+			resolveResult, _, _ := bundler.RunOnResolvePlugins(
+				plugins,
+				resolver,
+				log,
+				fs,
+				&caches.FSCache,
+				nil,            // importSource
+				logger.Range{}, // importPathRange
+				logger.Path{Text: options.Importer, Namespace: options.Namespace},
+				path,
+				logger.EncodeImportAttributes(options.With),
+				kind,
+				absResolveDir,
+				options.PluginData,
+			)
+			msgs := log.Done()
+
+			// Populate the result
+			result.Errors = convertMessagesToPublic(logger.Error, msgs)
+			result.Warnings = convertMessagesToPublic(logger.Warning, msgs)
+			if resolveResult != nil {
+				result.Path = resolveResult.PathPair.Primary.Text
+				result.External = resolveResult.PathPair.IsExternal
+				result.SideEffects = resolveResult.PrimarySideEffectsData == nil
+				result.Namespace = resolveResult.PathPair.Primary.Namespace
+				result.Suffix = resolveResult.PathPair.Primary.IgnoredSuffix
+				result.PluginData = resolveResult.PluginData
+			} else if len(result.Errors) == 0 {
+				// Always fail with at least one error
+				pluginName := item.Name
+				if options.PluginName != "" {
+					pluginName = options.PluginName
+				}
+				text, _, notes := bundler.ResolveFailureErrorTextSuggestionNotes(resolver, path, kind, pluginName, fs, absResolveDir, optionsForResolve.Platform, "", "")
+				result.Errors = append(result.Errors, convertMessagesToPublic(logger.Error, []logger.Msg{{
+					Data:  logger.MsgData{Text: text},
+					Notes: notes,
+				}})...)
+			}
+			return
+		}
+
+		onEnd := func(fn func(*BuildResult) (OnEndResult, error)) {
+			onEndCallbacks = append(onEndCallbacks, onEndCallback{
+				pluginName: item.Name,
+				fn:         fn,
+			})
+		}
+
+		onDispose := func(fn func()) {
+			onDisposeCallbacks = append(onDisposeCallbacks, fn)
+		}
+
+		item.Setup(PluginBuild{
+			InitialOptions: initialOptions,
+			Resolve:        resolve,
+			OnStart:        impl.onStart,
+			OnEnd:          onEnd,
+			OnDispose:      onDispose,
+			OnResolve:      impl.onResolve,
+			OnLoad:         impl.onLoad,
+		})
+
+		plugins = append(plugins, impl.plugin)
+	}
+
+	return
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// FormatMessages API
+
+func formatMsgsImpl(msgs []Message, opts FormatMessagesOptions) []string {
+	kind := logger.Error
+	if opts.Kind == WarningMessage {
+		kind = logger.Warning
+	}
+	logMsgs := convertMessagesToInternal(nil, kind, msgs)
+	strings := make([]string, len(logMsgs))
+	for i, msg := range logMsgs {
+		strings[i] = msg.String(
+			logger.OutputOptions{
+				IncludeSource: true,
+			},
+			logger.TerminalInfo{
+				UseColorEscapes: opts.Color,
+				Width:           opts.TerminalWidth,
+			},
+		)
+	}
+	return strings
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// AnalyzeMetafile API
+
+type metafileEntry struct {
+	name       string
+	entryPoint string
+	entries    []metafileEntry
+	size       int
+}
+
+// This type is just so we can use Go's native sort function
+type metafileArray []metafileEntry
+
+func (a metafileArray) Len() int          { return len(a) }
+func (a metafileArray) Swap(i int, j int) { a[i], a[j] = a[j], a[i] }
+
+func (a metafileArray) Less(i int, j int) bool {
+	ai := a[i]
+	aj := a[j]
+	return ai.size > aj.size || (ai.size == aj.size && ai.name < aj.name)
+}
+
+func getObjectProperty(expr js_ast.Expr, key string) js_ast.Expr {
+	if obj, ok := expr.Data.(*js_ast.EObject); ok {
+		for _, prop := range obj.Properties {
+			if helpers.UTF16EqualsString(prop.Key.Data.(*js_ast.EString).Value, key) {
+				return prop.ValueOrNil
+			}
+		}
+	}
+	return js_ast.Expr{}
+}
+
+func getObjectPropertyNumber(expr js_ast.Expr, key string) *js_ast.ENumber {
+	value, _ := getObjectProperty(expr, key).Data.(*js_ast.ENumber)
+	return value
+}
+
+func getObjectPropertyString(expr js_ast.Expr, key string) *js_ast.EString {
+	value, _ := getObjectProperty(expr, key).Data.(*js_ast.EString)
+	return value
+}
+
+func getObjectPropertyObject(expr js_ast.Expr, key string) *js_ast.EObject {
+	value, _ := getObjectProperty(expr, key).Data.(*js_ast.EObject)
+	return value
+}
+
+func getObjectPropertyArray(expr js_ast.Expr, key string) *js_ast.EArray {
+	value, _ := getObjectProperty(expr, key).Data.(*js_ast.EArray)
+	return value
+}
+
+func analyzeMetafileImpl(metafile string, opts AnalyzeMetafileOptions) string {
+	log := logger.NewDeferLog(logger.DeferLogNoVerboseOrDebug, nil)
+	source := logger.Source{Contents: metafile}
+
+	if result, ok := js_parser.ParseJSON(log, source, js_parser.JSONOptions{}); ok {
+		if outputs := getObjectPropertyObject(result, "outputs"); outputs != nil {
+			var entries metafileArray
+			var entryPoints []string
+
+			// Scan over the "outputs" object
+			for _, output := range outputs.Properties {
+				if key := helpers.UTF16ToString(output.Key.Data.(*js_ast.EString).Value); !strings.HasSuffix(key, ".map") {
+					entryPointPath := ""
+					if entryPoint := getObjectPropertyString(output.ValueOrNil, "entryPoint"); entryPoint != nil {
+						entryPointPath = helpers.UTF16ToString(entryPoint.Value)
+						entryPoints = append(entryPoints, entryPointPath)
+					}
+
+					if bytes := getObjectPropertyNumber(output.ValueOrNil, "bytes"); bytes != nil {
+						if inputs := getObjectPropertyObject(output.ValueOrNil, "inputs"); inputs != nil {
+							var children metafileArray
+
+							for _, input := range inputs.Properties {
+								if bytesInOutput := getObjectPropertyNumber(input.ValueOrNil, "bytesInOutput"); bytesInOutput != nil && bytesInOutput.Value > 0 {
+									children = append(children, metafileEntry{
+										name: helpers.UTF16ToString(input.Key.Data.(*js_ast.EString).Value),
+										size: int(bytesInOutput.Value),
+									})
+								}
+							}
+
+							sort.Sort(children)
+
+							entries = append(entries, metafileEntry{
+								name:       key,
+								size:       int(bytes.Value),
+								entries:    children,
+								entryPoint: entryPointPath,
+							})
+						}
+					}
+				}
+			}
+
+			sort.Sort(entries)
+
+			type importData struct {
+				imports []string
+			}
+
+			type graphData struct {
+				parent string
+				depth  uint32
+			}
+
+			importsForPath := make(map[string]importData)
+
+			// Scan over the "inputs" object
+			if inputs := getObjectPropertyObject(result, "inputs"); inputs != nil {
+				for _, prop := range inputs.Properties {
+					if imports := getObjectPropertyArray(prop.ValueOrNil, "imports"); imports != nil {
+						var data importData
+
+						for _, item := range imports.Items {
+							if path := getObjectPropertyString(item, "path"); path != nil {
+								data.imports = append(data.imports, helpers.UTF16ToString(path.Value))
+							}
+						}
+
+						importsForPath[helpers.UTF16ToString(prop.Key.Data.(*js_ast.EString).Value)] = data
+					}
+				}
+			}
+
+			// Returns a graph with links pointing from imports to importers
+			graphForEntryPoints := func(worklist []string) map[string]graphData {
+				if !opts.Verbose {
+					return nil
+				}
+
+				graph := make(map[string]graphData)
+
+				for _, entryPoint := range worklist {
+					graph[entryPoint] = graphData{}
+				}
+
+				for len(worklist) > 0 {
+					top := worklist[len(worklist)-1]
+					worklist = worklist[:len(worklist)-1]
+					childDepth := graph[top].depth + 1
+
+					for _, importPath := range importsForPath[top].imports {
+						imported, ok := graph[importPath]
+						if !ok {
+							imported.depth = math.MaxUint32
+						}
+
+						if imported.depth > childDepth {
+							imported.depth = childDepth
+							imported.parent = top
+							graph[importPath] = imported
+							worklist = append(worklist, importPath)
+						}
+					}
+				}
+
+				return graph
+			}
+
+			graphForAllEntryPoints := graphForEntryPoints(entryPoints)
+
+			type tableEntry struct {
+				first      string
+				second     string
+				third      string
+				firstLen   int
+				secondLen  int
+				thirdLen   int
+				isTopLevel bool
+			}
+
+			var table []tableEntry
+			var colors logger.Colors
+
+			if opts.Color {
+				colors = logger.TerminalColors
+			}
+
+			// Build up the table with an entry for each output file (other than ".map" files)
+			for _, entry := range entries {
+				second := prettyPrintByteCount(entry.size)
+				third := "100.0%"
+
+				table = append(table, tableEntry{
+					first:      entry.name,
+					firstLen:   utf8.RuneCountInString(entry.name),
+					second:     second,
+					secondLen:  len(second),
+					third:      third,
+					thirdLen:   len(third),
+					isTopLevel: true,
+				})
+
+				graph := graphForAllEntryPoints
+				if entry.entryPoint != "" {
+					// If there are multiple entry points and this output file is from an
+					// entry point, prefer import paths for this entry point. This is less
+					// confusing than showing import paths for another entry point.
+					graph = graphForEntryPoints([]string{entry.entryPoint})
+				}
+
+				// Add a sub-entry for each input file in this output file
+				for j, child := range entry.entries {
+					indent := " ├ "
+					if j+1 == len(entry.entries) {
+						indent = " └ "
+					}
+					percent := 100.0 * float64(child.size) / float64(entry.size)
+
+					first := indent + child.name
+					second := prettyPrintByteCount(child.size)
+					third := fmt.Sprintf("%.1f%%", percent)
+
+					table = append(table, tableEntry{
+						first:     first,
+						firstLen:  utf8.RuneCountInString(first),
+						second:    second,
+						secondLen: len(second),
+						third:     third,
+						thirdLen:  len(third),
+					})
+
+					// If we're in verbose mode, also print the import chain from this file
+					// up toward an entry point to show why this file is in the bundle
+					if opts.Verbose {
+						indent = " │ "
+						if j+1 == len(entry.entries) {
+							indent = "   "
+						}
+						data := graph[child.name]
+						depth := 0
+
+						for data.depth != 0 {
+							table = append(table, tableEntry{
+								first: fmt.Sprintf("%s%s%s └ %s%s", indent, colors.Dim, strings.Repeat(" ", depth), data.parent, colors.Reset),
+							})
+							data = graph[data.parent]
+							depth += 3
+						}
+					}
+				}
+			}
+
+			maxFirstLen := 0
+			maxSecondLen := 0
+			maxThirdLen := 0
+
+			// Calculate column widths
+			for _, entry := range table {
+				if maxFirstLen < entry.firstLen {
+					maxFirstLen = entry.firstLen
+				}
+				if maxSecondLen < entry.secondLen {
+					maxSecondLen = entry.secondLen
+				}
+				if maxThirdLen < entry.thirdLen {
+					maxThirdLen = entry.thirdLen
+				}
+			}
+
+			sb := strings.Builder{}
+
+			// Render the columns now that we know the widths
+			for _, entry := range table {
+				prefix := "\n"
+				color := colors.Bold
+				if !entry.isTopLevel {
+					prefix = ""
+					color = ""
+				}
+
+				// Import paths don't have second and third columns
+				if entry.second == "" && entry.third == "" {
+					sb.WriteString(fmt.Sprintf("%s  %s\n",
+						prefix,
+						entry.first,
+					))
+					continue
+				}
+
+				second := entry.second
+				secondTrimmed := strings.TrimRight(second, " ")
+				lineChar := " "
+				extraSpace := 0
+
+				if opts.Verbose {
+					lineChar = "─"
+					extraSpace = 1
+				}
+
+				sb.WriteString(fmt.Sprintf("%s  %s%s%s %s%s%s %s%s%s %s%s%s %s%s%s\n",
+					prefix,
+					color,
+					entry.first,
+					colors.Reset,
+					colors.Dim,
+					strings.Repeat(lineChar, extraSpace+maxFirstLen-entry.firstLen+maxSecondLen-entry.secondLen),
+					colors.Reset,
+					color,
+					secondTrimmed,
+					colors.Reset,
+					colors.Dim,
+					strings.Repeat(lineChar, extraSpace+maxThirdLen-entry.thirdLen+len(second)-len(secondTrimmed)),
+					colors.Reset,
+					color,
+					entry.third,
+					colors.Reset,
+				))
+			}
+
+			return sb.String()
+		}
+	}
+
+	return ""
+}
+
+func stripDirPrefix(path string, prefix string, allowedSlashes string) (string, bool) {
+	if strings.HasPrefix(path, prefix) {
+		pathLen := len(path)
+		prefixLen := len(prefix)
+
+		// Just return the path if there is no prefix
+		if prefixLen == 0 {
+			return path, true
+		}
+
+		// Return the empty string if the path equals the prefix
+		if pathLen == prefixLen {
+			return "", true
+		}
+
+		if strings.IndexByte(allowedSlashes, prefix[prefixLen-1]) >= 0 {
+			// Return the suffix if the prefix ends in a slash. Examples:
+			//
+			//   stripDirPrefix(`/foo`, `/`, `/`) => `foo`
+			//   stripDirPrefix(`C:\foo`, `C:\`, `\/`) => `foo`
+			//
+			return path[prefixLen:], true
+		} else if strings.IndexByte(allowedSlashes, path[prefixLen]) >= 0 {
+			// Return the suffix if there's a slash after the prefix. Examples:
+			//
+			//   stripDirPrefix(`/foo/bar`, `/foo`, `/`) => `bar`
+			//   stripDirPrefix(`C:\foo\bar`, `C:\foo`, `\/`) => `bar`
+			//
+			return path[prefixLen+1:], true
+		}
+	}
+
+	return "", false
+}
diff --git a/source/vendor/github.com/evanw/esbuild/pkg/api/api_js_table.go b/source/vendor/github.com/evanw/esbuild/pkg/api/api_js_table.go
new file mode 100644
index 0000000..72177f1
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/pkg/api/api_js_table.go
@@ -0,0 +1,50 @@
+// This file was automatically generated by "js_table.ts"
+
+package api
+
+import "github.com/evanw/esbuild/internal/compat"
+
+type EngineName uint8
+
+const (
+	EngineChrome EngineName = iota
+	EngineDeno
+	EngineEdge
+	EngineFirefox
+	EngineHermes
+	EngineIE
+	EngineIOS
+	EngineNode
+	EngineOpera
+	EngineRhino
+	EngineSafari
+)
+
+func convertEngineName(engine EngineName) compat.Engine {
+	switch engine {
+	case EngineChrome:
+		return compat.Chrome
+	case EngineDeno:
+		return compat.Deno
+	case EngineEdge:
+		return compat.Edge
+	case EngineFirefox:
+		return compat.Firefox
+	case EngineHermes:
+		return compat.Hermes
+	case EngineIE:
+		return compat.IE
+	case EngineIOS:
+		return compat.IOS
+	case EngineNode:
+		return compat.Node
+	case EngineOpera:
+		return compat.Opera
+	case EngineRhino:
+		return compat.Rhino
+	case EngineSafari:
+		return compat.Safari
+	default:
+		panic("Invalid engine name")
+	}
+}
diff --git a/source/vendor/github.com/evanw/esbuild/pkg/api/favicon.go b/source/vendor/github.com/evanw/esbuild/pkg/api/favicon.go
new file mode 100644
index 0000000..c3ec13a
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/pkg/api/favicon.go
@@ -0,0 +1,31 @@
+package api
+
+// This is the "favicon.ico" file used by esbuild's built-in development server
+var favicon_ico_gz = []byte{
+	0x1F, 0x8B, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0x03, 0x63, 0x60, 0x60, 0x64, 0x60, 0x62,
+	0x10, 0x10, 0x60, 0x00, 0xD2, 0x0A, 0x0C, 0x19, 0x2C, 0x0C, 0x0C, 0x6A, 0x0C, 0x0C, 0x0C, 0x0A,
+	0x0A, 0x10, 0xBE, 0x86, 0x20, 0x03, 0x43, 0x1F, 0x50, 0x4C, 0x03, 0x28, 0x26, 0x00, 0x12, 0x67,
+	0x80, 0x88, 0x13, 0x04, 0xE7, 0xFF, 0xFF, 0x27, 0x09, 0xD3, 0x4A, 0xFF, 0xC9, 0xEF, 0xFF, 0x59,
+	0x97, 0x9F, 0x81, 0xAB, 0x63, 0x5B, 0x72, 0xF2, 0x3F, 0xC3, 0x99, 0xDF, 0x44, 0xEB, 0xE7, 0x29,
+	0xE9, 0xF9, 0x2F, 0xAE, 0xA8, 0x02, 0xD6, 0xC7, 0x74, 0xE0, 0xCD, 0x7F, 0x09, 0x45, 0xE5, 0xFF,
+	0x02, 0xA1, 0xA9, 0x98, 0x66, 0xE0, 0xB1, 0x5F, 0xC8, 0x27, 0x12, 0x6E, 0x06, 0xFB, 0xEC, 0x7D,
+	0xFF, 0x25, 0xE4, 0x14, 0x30, 0xCD, 0xC0, 0xE7, 0x7F, 0xA0, 0x19, 0xC2, 0x0E, 0xDE, 0x60, 0x33,
+	0x58, 0x36, 0x5C, 0xFF, 0xCF, 0x31, 0x79, 0xF3, 0x7F, 0x49, 0x49, 0xC9, 0xFF, 0xFC, 0xB1, 0xF9,
+	0x44, 0xE9, 0x47, 0xB6, 0x93, 0xF1, 0xD8, 0x17, 0x14, 0xF7, 0x10, 0xD2, 0x4F, 0x94, 0x5E, 0x02,
+	0xFA, 0x05, 0x42, 0x53, 0xC0, 0x7E, 0x85, 0xE9, 0xC7, 0xD0, 0x4B, 0xCB, 0xF8, 0xA7, 0x85, 0xFE,
+	0x9A, 0x99, 0x68, 0x78, 0x56, 0x3D, 0xF9, 0xFA, 0xB1, 0xE8, 0x25, 0x5A, 0x3F, 0x0E, 0xBD, 0x44,
+	0xE9, 0xC7, 0xA3, 0x97, 0xA0, 0x7E, 0x02, 0x7A, 0x29, 0x00, 0x1A, 0xD0, 0x32, 0xC6, 0x81, 0xD8,
+	0x72, 0x86, 0xDC, 0xF8, 0xA6, 0x34, 0x7D, 0x8C, 0xDA, 0x3F, 0x6A, 0x3F, 0x01, 0xFB, 0x99, 0x77,
+	0x3C, 0xFA, 0x2F, 0xEC, 0x1E, 0x0C, 0xA6, 0xD1, 0xE5, 0x58, 0xD7, 0x5C, 0x06, 0xCB, 0x31, 0x1D,
+	0xF9, 0x48, 0x13, 0xFB, 0x41, 0x76, 0x8A, 0x19, 0x98, 0x81, 0xEB, 0x09, 0x11, 0x1B, 0x57, 0x14,
+	0x7B, 0x40, 0x76, 0x4B, 0xA8, 0xA8, 0x63, 0x95, 0xA3, 0x85, 0xFD, 0xE8, 0xF6, 0xE0, 0x93, 0xA3,
+	0x76, 0xF8, 0x53, 0xCD, 0x0D, 0x64, 0xA6, 0x3F, 0xAA, 0xB9, 0x81, 0x82, 0xF4, 0x4F, 0x15, 0x37,
+	0x50, 0x98, 0xFF, 0xD8, 0x16, 0x1E, 0x85, 0xDB, 0x01, 0xC2, 0x02, 0x71, 0x05, 0x70, 0x39, 0x8E,
+	0x69, 0xDB, 0x71, 0xCA, 0x0D, 0x75, 0xFF, 0x0F, 0x64, 0xFC, 0x0F, 0x64, 0xFA, 0x1F, 0xE8, 0xFC,
+	0x3F, 0xD0, 0xE5, 0xDF, 0x40, 0x97, 0xFF, 0xA3, 0xF5, 0xEF, 0xA8, 0xFD, 0x44, 0x61, 0x8C, 0xBE,
+	0x0C, 0x36, 0x3C, 0xA7, 0x7E, 0xE0, 0xEC, 0x9F, 0x53, 0x4F, 0xD3, 0xF6, 0x3F, 0x35, 0xEC, 0xA6,
+	0x89, 0xFD, 0x73, 0x48, 0xEB, 0x63, 0x51, 0xD5, 0xFE, 0x39, 0xA4, 0xF7, 0xEF, 0xA8, 0x66, 0xFF,
+	0x1C, 0xF2, 0xFA, 0x96, 0x54, 0xB1, 0x7F, 0x0E, 0xF9, 0xFD, 0x5A, 0x8A, 0xED, 0x9F, 0x43, 0x59,
+	0x9F, 0x9A, 0x22, 0xFB, 0xE7, 0x50, 0xDE, 0x9F, 0x27, 0xDB, 0xFE, 0x39, 0x34, 0x1B, 0x4B, 0x18,
+	0xCE, 0x00, 0x00, 0xDA, 0xEB, 0x61, 0xFD, 0xB6, 0x15, 0x00, 0x00,
+}
diff --git a/source/vendor/github.com/evanw/esbuild/pkg/api/serve_other.go b/source/vendor/github.com/evanw/esbuild/pkg/api/serve_other.go
new file mode 100644
index 0000000..0f5f3aa
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/pkg/api/serve_other.go
@@ -0,0 +1,990 @@
+//go:build !js || !wasm
+// +build !js !wasm
+
+package api
+
+// This file implements the "Serve()" function in esbuild's public API. It
+// provides a basic web server that can serve a directory tree over HTTP. When
+// a directory is visited the "index.html" will be served if present, otherwise
+// esbuild will automatically generate a directory listing page with links for
+// each file in the directory. If there is a build configured that generates
+// output files, those output files are not written to disk but are instead
+// "overlayed" virtually on top of the real file system. The server responds to
+// HTTP requests for output files from the build with the latest in-memory
+// build results.
+
+import (
+	"errors"
+	"fmt"
+	"net"
+	"net/http"
+	"os"
+	"path"
+	"sort"
+	"strconv"
+	"strings"
+	"sync"
+	"sync/atomic"
+	"syscall"
+	"time"
+
+	"github.com/evanw/esbuild/internal/fs"
+	"github.com/evanw/esbuild/internal/helpers"
+	"github.com/evanw/esbuild/internal/logger"
+)
+
+////////////////////////////////////////////////////////////////////////////////
+// Serve API
+
+type apiHandler struct {
+	onRequest        func(ServeOnRequestArgs)
+	rebuild          func() BuildResult
+	stop             func()
+	fs               fs.FS
+	absOutputDir     string
+	outdirPathPrefix string
+	publicPath       string
+	servedir         string
+	keyfileToLower   string
+	certfileToLower  string
+	fallback         string
+	serveWaitGroup   sync.WaitGroup
+	activeStreams    []chan serverSentEvent
+	currentHashes    map[string]string
+	mutex            sync.Mutex
+}
+
+type serverSentEvent struct {
+	event string
+	data  string
+}
+
+func escapeForHTML(text string) string {
+	text = strings.ReplaceAll(text, "&", "&amp;")
+	text = strings.ReplaceAll(text, "<", "&lt;")
+	text = strings.ReplaceAll(text, ">", "&gt;")
+	return text
+}
+
+func escapeForAttribute(text string) string {
+	text = escapeForHTML(text)
+	text = strings.ReplaceAll(text, "\"", "&quot;")
+	text = strings.ReplaceAll(text, "'", "&apos;")
+	return text
+}
+
+func (h *apiHandler) notifyRequest(duration time.Duration, req *http.Request, status int) {
+	if h.onRequest != nil {
+		h.onRequest(ServeOnRequestArgs{
+			RemoteAddress: req.RemoteAddr,
+			Method:        req.Method,
+			Path:          req.URL.Path,
+			Status:        status,
+			TimeInMS:      int(duration.Milliseconds()),
+		})
+	}
+}
+
+func errorsToString(errors []Message) string {
+	stderrOptions := logger.OutputOptions{IncludeSource: true}
+	terminalOptions := logger.TerminalInfo{}
+	sb := strings.Builder{}
+	limit := 5
+	for i, msg := range convertMessagesToInternal(nil, logger.Error, errors) {
+		if i == limit {
+			sb.WriteString(fmt.Sprintf("%d out of %d errors shown\n", limit, len(errors)))
+			break
+		}
+		sb.WriteString(msg.String(stderrOptions, terminalOptions))
+	}
+	return sb.String()
+}
+
+func (h *apiHandler) ServeHTTP(res http.ResponseWriter, req *http.Request) {
+	start := time.Now()
+
+	// Special-case the esbuild event stream
+	if req.Method == "GET" && req.URL.Path == "/esbuild" && req.Header.Get("Accept") == "text/event-stream" {
+		h.serveEventStream(start, req, res)
+		return
+	}
+
+	// HEAD requests omit the body
+	maybeWriteResponseBody := func(bytes []byte) { res.Write(bytes) }
+	isHEAD := req.Method == "HEAD"
+	if isHEAD {
+		maybeWriteResponseBody = func([]byte) { res.Write(nil) }
+	}
+
+	// Handle GET and HEAD requests
+	if (isHEAD || req.Method == "GET") && strings.HasPrefix(req.URL.Path, "/") {
+		res.Header().Set("Access-Control-Allow-Origin", "*")
+		queryPath := path.Clean(req.URL.Path)[1:]
+		result := h.rebuild()
+
+		// Requests fail if the build had errors
+		if len(result.Errors) > 0 {
+			res.Header().Set("Content-Type", "text/plain; charset=utf-8")
+			go h.notifyRequest(time.Since(start), req, http.StatusServiceUnavailable)
+			res.WriteHeader(http.StatusServiceUnavailable)
+			maybeWriteResponseBody([]byte(errorsToString(result.Errors)))
+			return
+		}
+
+		type fileToServe struct {
+			absPath  string
+			contents fs.OpenedFile
+		}
+
+		var kind fs.EntryKind
+		var file fileToServe
+		dirEntries := make(map[string]bool)
+		fileEntries := make(map[string]bool)
+
+		// Check for a match with the results if we're within the output directory
+		if outdirQueryPath, ok := stripDirPrefix(queryPath, h.outdirPathPrefix, "/"); ok {
+			resultKind, inMemoryBytes, absPath, isImplicitIndexHTML := h.matchQueryPathToResult(outdirQueryPath, &result, dirEntries, fileEntries)
+			kind = resultKind
+			file = fileToServe{
+				absPath:  absPath,
+				contents: &fs.InMemoryOpenedFile{Contents: inMemoryBytes},
+			}
+			if isImplicitIndexHTML {
+				queryPath = path.Join(queryPath, "index.html")
+			}
+		} else {
+			// Create a fake directory entry for the output path so that it appears to be a real directory
+			p := h.outdirPathPrefix
+			for p != "" {
+				var dir string
+				var base string
+				if slash := strings.IndexByte(p, '/'); slash == -1 {
+					base = p
+				} else {
+					dir = p[:slash]
+					base = p[slash+1:]
+				}
+				if dir == queryPath {
+					kind = fs.DirEntry
+					dirEntries[base] = true
+					break
+				}
+				p = dir
+			}
+		}
+
+		// Check for a file in the "servedir" directory
+		if h.servedir != "" && kind != fs.FileEntry {
+			absPath := h.fs.Join(h.servedir, queryPath)
+			if absDir := h.fs.Dir(absPath); absDir != absPath {
+				if entries, err, _ := h.fs.ReadDirectory(absDir); err == nil {
+					if entry, _ := entries.Get(h.fs.Base(absPath)); entry != nil && entry.Kind(h.fs) == fs.FileEntry {
+						if h.keyfileToLower != "" || h.certfileToLower != "" {
+							if toLower := strings.ToLower(absPath); toLower == h.keyfileToLower || toLower == h.certfileToLower {
+								// Don't serve the HTTPS key or certificate. This uses a case-
+								// insensitive check because some file systems are case-sensitive.
+								go h.notifyRequest(time.Since(start), req, http.StatusForbidden)
+								res.WriteHeader(http.StatusForbidden)
+								maybeWriteResponseBody([]byte("403 - Forbidden"))
+								return
+							}
+						}
+						if contents, err, _ := h.fs.OpenFile(absPath); err == nil {
+							defer contents.Close()
+							file = fileToServe{absPath: absPath, contents: contents}
+							kind = fs.FileEntry
+						} else if err != syscall.ENOENT {
+							go h.notifyRequest(time.Since(start), req, http.StatusInternalServerError)
+							res.WriteHeader(http.StatusInternalServerError)
+							maybeWriteResponseBody([]byte(fmt.Sprintf("500 - Internal server error: %s", err.Error())))
+							return
+						}
+					}
+				}
+			}
+		}
+
+		// Check for a directory in the "servedir" directory
+		var servedirIndexName string
+		if h.servedir != "" && kind != fs.FileEntry {
+			if entries, err, _ := h.fs.ReadDirectory(h.fs.Join(h.servedir, queryPath)); err == nil {
+				kind = fs.DirEntry
+				for _, name := range entries.SortedKeys() {
+					entry, _ := entries.Get(name)
+					switch entry.Kind(h.fs) {
+					case fs.DirEntry:
+						dirEntries[name] = true
+					case fs.FileEntry:
+						fileEntries[name] = true
+						if name == "index.html" {
+							servedirIndexName = name
+						}
+					}
+				}
+			} else if err != syscall.ENOENT {
+				go h.notifyRequest(time.Since(start), req, http.StatusInternalServerError)
+				res.WriteHeader(http.StatusInternalServerError)
+				maybeWriteResponseBody([]byte(fmt.Sprintf("500 - Internal server error: %s", err.Error())))
+				return
+			}
+		}
+
+		// Redirect to a trailing slash for directories
+		if kind == fs.DirEntry && !strings.HasSuffix(req.URL.Path, "/") {
+			res.Header().Set("Location", path.Clean(req.URL.Path)+"/")
+			go h.notifyRequest(time.Since(start), req, http.StatusFound)
+			res.WriteHeader(http.StatusFound)
+			maybeWriteResponseBody(nil)
+			return
+		}
+
+		// Serve an "index.html" file if present
+		if kind == fs.DirEntry && servedirIndexName != "" {
+			queryPath += "/" + servedirIndexName
+			absPath := h.fs.Join(h.servedir, queryPath)
+			if contents, err, _ := h.fs.OpenFile(absPath); err == nil {
+				defer contents.Close()
+				file = fileToServe{absPath: absPath, contents: contents}
+				kind = fs.FileEntry
+			} else if err != syscall.ENOENT {
+				go h.notifyRequest(time.Since(start), req, http.StatusInternalServerError)
+				res.WriteHeader(http.StatusInternalServerError)
+				maybeWriteResponseBody([]byte(fmt.Sprintf("500 - Internal server error: %s", err.Error())))
+				return
+			}
+		}
+
+		// Serve the fallback HTML page if one was provided
+		if kind != fs.FileEntry && h.fallback != "" {
+			if contents, err, _ := h.fs.OpenFile(h.fallback); err == nil {
+				defer contents.Close()
+				file = fileToServe{absPath: h.fallback, contents: contents}
+				kind = fs.FileEntry
+			} else if err != syscall.ENOENT {
+				go h.notifyRequest(time.Since(start), req, http.StatusInternalServerError)
+				res.WriteHeader(http.StatusInternalServerError)
+				maybeWriteResponseBody([]byte(fmt.Sprintf("500 - Internal server error: %s", err.Error())))
+				return
+			}
+		}
+
+		// Serve a file
+		if kind == fs.FileEntry {
+			// Default to serving the whole file
+			status := http.StatusOK
+			fileContentsLen := file.contents.Len()
+			begin := 0
+			end := fileContentsLen
+			isRange := false
+
+			// Handle range requests so that video playback works in Safari
+			if rangeBegin, rangeEnd, ok := parseRangeHeader(req.Header.Get("Range"), fileContentsLen); ok && rangeBegin < rangeEnd {
+				// Note: The content range is inclusive so subtract 1 from the end
+				isRange = true
+				begin = rangeBegin
+				end = rangeEnd
+				status = http.StatusPartialContent
+			}
+
+			// Try to read the range from the file, which may fail
+			fileBytes, err := file.contents.Read(begin, end)
+			if err != nil {
+				go h.notifyRequest(time.Since(start), req, http.StatusInternalServerError)
+				res.WriteHeader(http.StatusInternalServerError)
+				maybeWriteResponseBody([]byte(fmt.Sprintf("500 - Internal server error: %s", err.Error())))
+				return
+			}
+
+			// If we get here, the request was successful
+			if contentType := helpers.MimeTypeByExtension(h.fs.Ext(file.absPath)); contentType != "" {
+				res.Header().Set("Content-Type", contentType)
+			} else {
+				res.Header().Set("Content-Type", "application/octet-stream")
+			}
+			if isRange {
+				res.Header().Set("Content-Range", fmt.Sprintf("bytes %d-%d/%d", begin, end-1, fileContentsLen))
+			}
+			res.Header().Set("Content-Length", fmt.Sprintf("%d", len(fileBytes)))
+			go h.notifyRequest(time.Since(start), req, status)
+			res.WriteHeader(status)
+			maybeWriteResponseBody(fileBytes)
+			return
+		}
+
+		// Serve a directory listing
+		if kind == fs.DirEntry {
+			html := respondWithDirList(queryPath, dirEntries, fileEntries)
+			res.Header().Set("Content-Type", "text/html; charset=utf-8")
+			res.Header().Set("Content-Length", fmt.Sprintf("%d", len(html)))
+			go h.notifyRequest(time.Since(start), req, http.StatusOK)
+			maybeWriteResponseBody(html)
+			return
+		}
+	}
+
+	// Satisfy requests for "favicon.ico" to avoid errors in Firefox developer tools
+	if req.Method == "GET" && req.URL.Path == "/favicon.ico" {
+		for _, encoding := range strings.Split(req.Header.Get("Accept-Encoding"), ",") {
+			if semi := strings.IndexByte(encoding, ';'); semi >= 0 {
+				encoding = encoding[:semi]
+			}
+			if strings.TrimSpace(encoding) == "gzip" {
+				res.Header().Set("Content-Encoding", "gzip")
+				res.Header().Set("Content-Type", "image/vnd.microsoft.icon")
+				go h.notifyRequest(time.Since(start), req, http.StatusOK)
+				maybeWriteResponseBody(favicon_ico_gz)
+				return
+			}
+		}
+	}
+
+	// Default to a 404
+	res.Header().Set("Content-Type", "text/plain; charset=utf-8")
+	go h.notifyRequest(time.Since(start), req, http.StatusNotFound)
+	res.WriteHeader(http.StatusNotFound)
+	maybeWriteResponseBody([]byte("404 - Not Found"))
+}
+
+// This exposes an event stream to clients using server-sent events:
+// https://developer.mozilla.org/en-US/docs/Web/API/Server-sent_events
+func (h *apiHandler) serveEventStream(start time.Time, req *http.Request, res http.ResponseWriter) {
+	if flusher, ok := res.(http.Flusher); ok {
+		if closer, ok := res.(http.CloseNotifier); ok {
+			// Add a new stream to the array of active streams
+			stream := make(chan serverSentEvent)
+			h.mutex.Lock()
+			h.activeStreams = append(h.activeStreams, stream)
+			h.mutex.Unlock()
+
+			// Start the event stream
+			res.Header().Set("Content-Type", "text/event-stream")
+			res.Header().Set("Connection", "keep-alive")
+			res.Header().Set("Cache-Control", "no-cache")
+			res.Header().Set("Access-Control-Allow-Origin", "*")
+			go h.notifyRequest(time.Since(start), req, http.StatusOK)
+			res.WriteHeader(http.StatusOK)
+			res.Write([]byte("retry: 500\n"))
+			flusher.Flush()
+
+			// Send incoming messages over the stream
+			streamWasClosed := make(chan struct{}, 1)
+			go func() {
+				for {
+					var msg []byte
+					select {
+					case next, ok := <-stream:
+						if !ok {
+							streamWasClosed <- struct{}{}
+							return
+						}
+						msg = []byte(fmt.Sprintf("event: %s\ndata: %s\n\n", next.event, next.data))
+					case <-time.After(30 * time.Second):
+						// Send an occasional keep-alive
+						msg = []byte(":\n\n")
+					}
+					if _, err := res.Write(msg); err != nil {
+						return
+					}
+					flusher.Flush()
+				}
+			}()
+
+			// When the stream is closed (either by them or by us), remove it
+			// from the array and end the response body to clean up resources
+			select {
+			case <-closer.CloseNotify():
+			case <-streamWasClosed:
+			}
+			h.mutex.Lock()
+			for i := range h.activeStreams {
+				if h.activeStreams[i] == stream {
+					end := len(h.activeStreams) - 1
+					h.activeStreams[i] = h.activeStreams[end]
+					h.activeStreams = h.activeStreams[:end]
+
+					// Only close the stream if it's present in the list of active
+					// streams. Stopping the server can also call close on this
+					// stream and Go only lets you close a channel once before
+					// panicking, so we don't want to close it twice.
+					close(stream)
+					break
+				}
+			}
+			h.mutex.Unlock()
+			return
+		}
+	}
+
+	// If we get here, then event streaming isn't possible
+	go h.notifyRequest(time.Since(start), req, http.StatusInternalServerError)
+	res.WriteHeader(http.StatusInternalServerError)
+	res.Write([]byte("500 - Event stream error"))
+}
+
+func (h *apiHandler) broadcastBuildResult(result BuildResult, newHashes map[string]string) {
+	h.mutex.Lock()
+
+	var added []string
+	var removed []string
+	var updated []string
+
+	urlForPath := func(absPath string) (string, bool) {
+		if relPath, ok := stripDirPrefix(absPath, h.absOutputDir, "\\/"); ok {
+			relPath = strings.ReplaceAll(relPath, "\\", "/")
+			relPath = path.Join(h.outdirPathPrefix, relPath)
+			publicPath := h.publicPath
+			slash := "/"
+			if publicPath != "" && strings.HasSuffix(h.publicPath, "/") {
+				slash = ""
+			}
+			return fmt.Sprintf("%s%s%s", publicPath, slash, relPath), true
+		}
+		return "", false
+	}
+
+	// Diff the old and new states, but only if the build succeeded. We shouldn't
+	// make it appear as if all files were removed when there is a build error.
+	if len(result.Errors) == 0 {
+		oldHashes := h.currentHashes
+		h.currentHashes = newHashes
+
+		for absPath, newHash := range newHashes {
+			if oldHash, ok := oldHashes[absPath]; !ok {
+				if url, ok := urlForPath(absPath); ok {
+					added = append(added, url)
+				}
+			} else if newHash != oldHash {
+				if url, ok := urlForPath(absPath); ok {
+					updated = append(updated, url)
+				}
+			}
+		}
+
+		for absPath := range oldHashes {
+			if _, ok := newHashes[absPath]; !ok {
+				if url, ok := urlForPath(absPath); ok {
+					removed = append(removed, url)
+				}
+			}
+		}
+	}
+
+	// Only notify listeners if there's a change that's worth sending. That way
+	// you can implement a simple "reload on any change" script without having
+	// to do this check in the script.
+	if len(added) > 0 || len(removed) > 0 || len(updated) > 0 {
+		sort.Strings(added)
+		sort.Strings(removed)
+		sort.Strings(updated)
+
+		// Assemble the diff
+		var sb strings.Builder
+		sb.WriteString("{\"added\":[")
+		for i, path := range added {
+			if i > 0 {
+				sb.WriteRune(',')
+			}
+			sb.Write(helpers.QuoteForJSON(path, false))
+		}
+		sb.WriteString("],\"removed\":[")
+		for i, path := range removed {
+			if i > 0 {
+				sb.WriteRune(',')
+			}
+			sb.Write(helpers.QuoteForJSON(path, false))
+		}
+		sb.WriteString("],\"updated\":[")
+		for i, path := range updated {
+			if i > 0 {
+				sb.WriteRune(',')
+			}
+			sb.Write(helpers.QuoteForJSON(path, false))
+		}
+		sb.WriteString("]}")
+		json := sb.String()
+
+		// Broadcast the diff to all streams
+		for _, stream := range h.activeStreams {
+			stream <- serverSentEvent{event: "change", data: json}
+		}
+	}
+
+	h.mutex.Unlock()
+}
+
+// Handle enough of the range specification so that video playback works in Safari
+func parseRangeHeader(r string, contentLength int) (int, int, bool) {
+	if strings.HasPrefix(r, "bytes=") {
+		r = r[len("bytes="):]
+		if dash := strings.IndexByte(r, '-'); dash != -1 {
+			// Note: The range is inclusive so the limit is deliberately "length - 1"
+			if begin, ok := parseRangeInt(r[:dash], contentLength-1); ok {
+				if end, ok := parseRangeInt(r[dash+1:], contentLength-1); ok {
+					// Note: The range is inclusive so a range of "0-1" is two bytes long
+					return begin, end + 1, true
+				}
+			}
+		}
+	}
+	return 0, 0, false
+}
+
+func parseRangeInt(text string, maxValue int) (int, bool) {
+	if text == "" {
+		return 0, false
+	}
+	value := 0
+	for _, c := range text {
+		if c < '0' || c > '9' {
+			return 0, false
+		}
+		value = value*10 + int(c-'0')
+		if value > maxValue {
+			return 0, false
+		}
+	}
+	return value, true
+}
+
+func (h *apiHandler) matchQueryPathToResult(
+	queryPath string,
+	result *BuildResult,
+	dirEntries map[string]bool,
+	fileEntries map[string]bool,
+) (fs.EntryKind, []byte, string, bool) {
+	queryIsDir := false
+	queryDir := queryPath
+	if queryDir != "" {
+		queryDir += "/"
+	}
+
+	// Check the output files for a match
+	for _, file := range result.OutputFiles {
+		if relPath, ok := h.fs.Rel(h.absOutputDir, file.Path); ok {
+			relPath = strings.ReplaceAll(relPath, "\\", "/")
+
+			// An exact match
+			if relPath == queryPath {
+				return fs.FileEntry, file.Contents, file.Path, false
+			}
+
+			// Serve an "index.html" file if present
+			if dir, base := path.Split(relPath); base == "index.html" && queryDir == dir {
+				return fs.FileEntry, file.Contents, file.Path, true
+			}
+
+			// A match inside this directory
+			if strings.HasPrefix(relPath, queryDir) {
+				entry := relPath[len(queryDir):]
+				queryIsDir = true
+				if slash := strings.IndexByte(entry, '/'); slash == -1 {
+					fileEntries[entry] = true
+				} else if dir := entry[:slash]; !dirEntries[dir] {
+					dirEntries[dir] = true
+				}
+			}
+		}
+	}
+
+	// Treat this as a directory if it's non-empty
+	if queryIsDir {
+		return fs.DirEntry, nil, "", false
+	}
+
+	return 0, nil, "", false
+}
+
+func respondWithDirList(queryPath string, dirEntries map[string]bool, fileEntries map[string]bool) []byte {
+	queryPath = "/" + queryPath
+	queryDir := queryPath
+	if queryDir != "/" {
+		queryDir += "/"
+	}
+	html := strings.Builder{}
+	html.WriteString("<!doctype html>\n")
+	html.WriteString("<meta charset=\"utf8\">\n")
+	html.WriteString("<style>\n")
+	html.WriteString("body { margin: 30px; color: #222; background: #fff; font: 16px/22px sans-serif; }\n")
+	html.WriteString("a { color: inherit; text-decoration: none; }\n")
+	html.WriteString("a:hover { text-decoration: underline; }\n")
+	html.WriteString("a:visited { color: #777; }\n")
+	html.WriteString("@media (prefers-color-scheme: dark) {\n")
+	html.WriteString("  body { color: #fff; background: #222; }\n")
+	html.WriteString("  a:visited { color: #aaa; }\n")
+	html.WriteString("}\n")
+	html.WriteString("</style>\n")
+	html.WriteString("<title>Directory: ")
+	html.WriteString(escapeForHTML(queryDir))
+	html.WriteString("</title>\n")
+	html.WriteString("<h1>Directory: ")
+	var parts []string
+	if queryPath == "/" {
+		parts = []string{""}
+	} else {
+		parts = strings.Split(queryPath, "/")
+	}
+	for i, part := range parts {
+		if i+1 < len(parts) {
+			html.WriteString("<a href=\"")
+			html.WriteString(escapeForHTML(strings.Join(parts[:i+1], "/")))
+			html.WriteString("/\">")
+		}
+		html.WriteString(escapeForHTML(part))
+		html.WriteString("/")
+		if i+1 < len(parts) {
+			html.WriteString("</a>")
+		}
+	}
+	html.WriteString("</h1>\n")
+
+	// Link to the parent directory
+	if queryPath != "/" {
+		parentDir := path.Dir(queryPath)
+		if parentDir != "/" {
+			parentDir += "/"
+		}
+		html.WriteString(fmt.Sprintf("<div>📁 <a href=\"%s\">../</a></div>\n", escapeForAttribute(parentDir)))
+	}
+
+	// Link to child directories
+	strings := make([]string, 0, len(dirEntries)+len(fileEntries))
+	for entry := range dirEntries {
+		strings = append(strings, entry)
+	}
+	sort.Strings(strings)
+	for _, entry := range strings {
+		html.WriteString(fmt.Sprintf("<div>📁 <a href=\"%s/\">%s/</a></div>\n", escapeForAttribute(path.Join(queryPath, entry)), escapeForHTML(entry)))
+	}
+
+	// Link to files in the directory
+	strings = strings[:0]
+	for entry := range fileEntries {
+		strings = append(strings, entry)
+	}
+	sort.Strings(strings)
+	for _, entry := range strings {
+		html.WriteString(fmt.Sprintf("<div>📄 <a href=\"%s\">%s</a></div>\n", escapeForAttribute(path.Join(queryPath, entry)), escapeForHTML(entry)))
+	}
+
+	return []byte(html.String())
+}
+
+// This is used to make error messages platform-independent
+func prettyPrintPath(fs fs.FS, path string) string {
+	if relPath, ok := fs.Rel(fs.Cwd(), path); ok {
+		return strings.ReplaceAll(relPath, "\\", "/")
+	}
+	return path
+}
+
+func (ctx *internalContext) Serve(serveOptions ServeOptions) (ServeResult, error) {
+	ctx.mutex.Lock()
+	defer ctx.mutex.Unlock()
+
+	// Ignore disposed contexts
+	if ctx.didDispose {
+		return ServeResult{}, errors.New("Cannot serve a disposed context")
+	}
+
+	// Don't allow starting serve mode multiple times
+	if ctx.handler != nil {
+		return ServeResult{}, errors.New("Serve mode has already been enabled")
+	}
+
+	// Don't allow starting serve mode multiple times
+	if (serveOptions.Keyfile != "") != (serveOptions.Certfile != "") {
+		return ServeResult{}, errors.New("Must specify both key and certificate for HTTPS")
+	}
+
+	// Validate the "servedir" path
+	if serveOptions.Servedir != "" {
+		if absPath, ok := ctx.realFS.Abs(serveOptions.Servedir); ok {
+			serveOptions.Servedir = absPath
+		} else {
+			return ServeResult{}, fmt.Errorf("Invalid serve path: %s", serveOptions.Servedir)
+		}
+	}
+
+	// Validate the "fallback" path
+	if serveOptions.Fallback != "" {
+		if absPath, ok := ctx.realFS.Abs(serveOptions.Fallback); ok {
+			serveOptions.Fallback = absPath
+		} else {
+			return ServeResult{}, fmt.Errorf("Invalid fallback path: %s", serveOptions.Fallback)
+		}
+	}
+
+	// Stuff related to the output directory only matters if there are entry points
+	outdirPathPrefix := ""
+	if len(ctx.args.entryPoints) > 0 {
+		// Don't allow serving when builds are written to stdout
+		if ctx.args.options.WriteToStdout {
+			what := "entry points"
+			if len(ctx.args.entryPoints) == 1 {
+				what = "an entry point"
+			}
+			return ServeResult{}, fmt.Errorf("Cannot serve %s without an output path", what)
+		}
+
+		// Compute the output path prefix
+		if serveOptions.Servedir != "" && ctx.args.options.AbsOutputDir != "" {
+			// Make sure the output directory is contained in the "servedir" directory
+			relPath, ok := ctx.realFS.Rel(serveOptions.Servedir, ctx.args.options.AbsOutputDir)
+			if !ok {
+				return ServeResult{}, fmt.Errorf(
+					"Cannot compute relative path from %q to %q\n", serveOptions.Servedir, ctx.args.options.AbsOutputDir)
+			}
+			relPath = strings.ReplaceAll(relPath, "\\", "/") // Fix paths on Windows
+			if relPath == ".." || strings.HasPrefix(relPath, "../") {
+				return ServeResult{}, fmt.Errorf(
+					"Output directory %q must be contained in serve directory %q",
+					prettyPrintPath(ctx.realFS, ctx.args.options.AbsOutputDir),
+					prettyPrintPath(ctx.realFS, serveOptions.Servedir),
+				)
+			}
+			if relPath != "." {
+				outdirPathPrefix = relPath
+			}
+		}
+	}
+
+	// Determine the host
+	var listener net.Listener
+	network := "tcp4"
+	host := "0.0.0.0"
+	if serveOptions.Host != "" {
+		host = serveOptions.Host
+
+		// Only use "tcp4" if this is an IPv4 address, otherwise use "tcp"
+		if ip := net.ParseIP(host); ip == nil || ip.To4() == nil {
+			network = "tcp"
+		}
+	}
+
+	// Pick the port
+	if serveOptions.Port == 0 {
+		// Default to picking a "800X" port
+		for port := 8000; port <= 8009; port++ {
+			if result, err := net.Listen(network, net.JoinHostPort(host, fmt.Sprintf("%d", port))); err == nil {
+				listener = result
+				break
+			}
+		}
+	}
+	if listener == nil {
+		// Otherwise pick the provided port
+		if result, err := net.Listen(network, net.JoinHostPort(host, fmt.Sprintf("%d", serveOptions.Port))); err != nil {
+			return ServeResult{}, err
+		} else {
+			listener = result
+		}
+	}
+
+	// Try listening on the provided port
+	addr := listener.Addr().String()
+
+	// Extract the real port in case we passed a port of "0"
+	var result ServeResult
+	if host, text, err := net.SplitHostPort(addr); err == nil {
+		if port, err := strconv.ParseInt(text, 10, 32); err == nil {
+			result.Port = uint16(port)
+			result.Host = host
+		}
+	}
+
+	// HTTPS-related files should be absolute paths
+	isHTTPS := serveOptions.Keyfile != "" && serveOptions.Certfile != ""
+	if isHTTPS {
+		serveOptions.Keyfile, _ = ctx.realFS.Abs(serveOptions.Keyfile)
+		serveOptions.Certfile, _ = ctx.realFS.Abs(serveOptions.Certfile)
+	}
+
+	var shouldStop int32
+
+	// The first build will just build normally
+	handler := &apiHandler{
+		onRequest:        serveOptions.OnRequest,
+		outdirPathPrefix: outdirPathPrefix,
+		absOutputDir:     ctx.args.options.AbsOutputDir,
+		publicPath:       ctx.args.options.PublicPath,
+		servedir:         serveOptions.Servedir,
+		keyfileToLower:   strings.ToLower(serveOptions.Keyfile),
+		certfileToLower:  strings.ToLower(serveOptions.Certfile),
+		fallback:         serveOptions.Fallback,
+		rebuild: func() BuildResult {
+			if atomic.LoadInt32(&shouldStop) != 0 {
+				// Don't start more rebuilds if we were told to stop
+				return BuildResult{}
+			} else {
+				return ctx.activeBuildOrRecentBuildOrRebuild()
+			}
+		},
+		fs: ctx.realFS,
+	}
+
+	// Create the server
+	server := &http.Server{Addr: addr, Handler: handler}
+
+	// When stop is called, block further rebuilds and then close the server
+	handler.stop = func() {
+		atomic.StoreInt32(&shouldStop, 1)
+
+		// Close the server and wait for it to close
+		server.Close()
+
+		// Close all open event streams
+		handler.mutex.Lock()
+		for _, stream := range handler.activeStreams {
+			close(stream)
+		}
+		handler.activeStreams = nil
+		handler.mutex.Unlock()
+
+		handler.serveWaitGroup.Wait()
+	}
+
+	// HACK: Go's HTTP API doesn't appear to provide a way to separate argument
+	// validation errors from eventual network errors. Specifically "ServeTLS"
+	// blocks for an arbitrarily long time before returning an error. So we
+	// intercept the first call to "Accept" on the listener and say that the
+	// serve call succeeded without an error if we get to that point.
+	hack := &hackListener{Listener: listener}
+	hack.waitGroup.Add(1)
+
+	// Start the server and signal on "serveWaitGroup" when it stops
+	handler.serveWaitGroup.Add(1)
+	go func() {
+		var err error
+		if isHTTPS {
+			err = server.ServeTLS(hack, serveOptions.Certfile, serveOptions.Keyfile)
+		} else {
+			err = server.Serve(hack)
+		}
+		if err != http.ErrServerClosed {
+			hack.mutex.Lock()
+			if !hack.done {
+				hack.done = true
+				hack.err = err
+				hack.waitGroup.Done()
+			}
+			hack.mutex.Unlock()
+		}
+		handler.serveWaitGroup.Done()
+	}()
+
+	// Return an error if the server failed to start accepting connections
+	hack.waitGroup.Wait()
+	if hack.err != nil {
+		return ServeResult{}, hack.err
+	}
+
+	// There appears to be some issue with Linux (but not with macOS) where
+	// destroying and recreating a server with the same port as the previous
+	// server had sometimes causes subsequent connections to fail with
+	// ECONNRESET (shows up in node as "Error: socket hang up").
+	//
+	// I think the problem is sort of that Go sets SO_REUSEADDR to 1 for listener
+	// sockets (specifically in "setDefaultListenerSockopts"). In some ways this
+	// is good, because it's more convenient for the user if the port is the
+	// same. However, I believe this sends a TCP RST packet to kill any previous
+	// connections. That can then be received by clients attempting to connect
+	// to the new server.
+	//
+	// As a hack to work around this problem, we wait for an additional short
+	// amount of time before returning. I observed this problem even with a 5ms
+	// timeout but I did not observe this problem with a 10ms timeout. So I'm
+	// setting this timeout to 50ms to be extra safe.
+	time.Sleep(50 * time.Millisecond)
+
+	// Only set the context handler if the server started successfully
+	ctx.handler = handler
+
+	// Print the URL(s) that the server can be reached at
+	if ctx.args.logOptions.LogLevel <= logger.LevelInfo {
+		printURLs(result.Host, result.Port, isHTTPS, ctx.args.logOptions.Color)
+	}
+
+	// Start the first build shortly after this function returns (but not
+	// immediately so that stuff we print right after this will come first).
+	//
+	// This also helps the CLI not do two builds when serve and watch mode
+	// are enabled together. Watch mode is enabled after serve mode because
+	// we want the stderr output for watch to come after the stderr output for
+	// serve, but watch mode will do another build if the current build is
+	// not a watch mode build.
+	go func() {
+		time.Sleep(10 * time.Millisecond)
+		handler.rebuild()
+	}()
+	return result, nil
+}
+
+type hackListener struct {
+	net.Listener
+	mutex     sync.Mutex
+	waitGroup sync.WaitGroup
+	err       error
+	done      bool
+}
+
+func (hack *hackListener) Accept() (net.Conn, error) {
+	hack.mutex.Lock()
+	if !hack.done {
+		hack.done = true
+		hack.waitGroup.Done()
+	}
+	hack.mutex.Unlock()
+	return hack.Listener.Accept()
+}
+
+func printURLs(host string, port uint16, https bool, useColor logger.UseColor) {
+	logger.PrintTextWithColor(os.Stderr, useColor, func(colors logger.Colors) string {
+		var hosts []string
+		sb := strings.Builder{}
+		sb.WriteString(colors.Reset)
+
+		// If this is "0.0.0.0" or "::", list all relevant IP addresses
+		if ip := net.ParseIP(host); ip != nil && ip.IsUnspecified() {
+			if addrs, err := net.InterfaceAddrs(); err == nil {
+				for _, addr := range addrs {
+					if addr, ok := addr.(*net.IPNet); ok && (addr.IP.To4() != nil) == (ip.To4() != nil) && !addr.IP.IsLinkLocalUnicast() {
+						hosts = append(hosts, addr.IP.String())
+					}
+				}
+			}
+		}
+
+		// Otherwise, just list the one IP address
+		if len(hosts) == 0 {
+			hosts = append(hosts, host)
+		}
+
+		// Determine the host kinds
+		kinds := make([]string, len(hosts))
+		maxLen := 0
+		for i, host := range hosts {
+			kind := "Network"
+			if ip := net.ParseIP(host); ip != nil && ip.IsLoopback() {
+				kind = "Local"
+			}
+			kinds[i] = kind
+			if len(kind) > maxLen {
+				maxLen = len(kind)
+			}
+		}
+
+		// Pretty-print the host list
+		protocol := "http"
+		if https {
+			protocol = "https"
+		}
+		for i, kind := range kinds {
+			sb.WriteString(fmt.Sprintf("\n > %s:%s %s%s://%s/%s",
+				kind, strings.Repeat(" ", maxLen-len(kind)), colors.Underline, protocol,
+				net.JoinHostPort(hosts[i], fmt.Sprintf("%d", port)), colors.Reset))
+		}
+
+		sb.WriteString("\n\n")
+		return sb.String()
+	})
+}
diff --git a/source/vendor/github.com/evanw/esbuild/pkg/api/serve_wasm.go b/source/vendor/github.com/evanw/esbuild/pkg/api/serve_wasm.go
new file mode 100644
index 0000000..bdae038
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/pkg/api/serve_wasm.go
@@ -0,0 +1,21 @@
+//go:build js && wasm
+// +build js,wasm
+
+package api
+
+import "fmt"
+
+// Remove the serve API in the WebAssembly build. This removes 2.7mb of stuff.
+
+func (*internalContext) Serve(ServeOptions) (ServeResult, error) {
+	return ServeResult{}, fmt.Errorf("The \"serve\" API is not supported when using WebAssembly")
+}
+
+type apiHandler struct {
+}
+
+func (*apiHandler) broadcastBuildResult(BuildResult, map[string]string) {
+}
+
+func (*apiHandler) stop() {
+}
diff --git a/source/vendor/github.com/evanw/esbuild/pkg/api/watcher.go b/source/vendor/github.com/evanw/esbuild/pkg/api/watcher.go
new file mode 100644
index 0000000..efe6277
--- /dev/null
+++ b/source/vendor/github.com/evanw/esbuild/pkg/api/watcher.go
@@ -0,0 +1,187 @@
+package api
+
+// This file implements a polling file watcher for esbuild (i.e. it detects
+// when files are changed by repeatedly checking their contents). Polling is
+// used instead of more efficient platform-specific file system APIs because:
+//
+//   * Go's standard library doesn't have built-in APIs for file watching
+//   * Using platform-specific APIs means using cgo, which I want to avoid
+//   * Polling is cross-platform and esbuild needs to work on 20+ platforms
+//   * Platform-specific APIs might be unreliable and could introduce bugs
+//
+// That said, this polling system is designed to use relatively little CPU vs.
+// a more traditional polling system that scans the whole directory tree at
+// once. The file system is still scanned regularly but each scan only checks
+// a random subset of your files, which means a change to a file will be picked
+// up soon after the change is made but not necessarily instantly.
+//
+// With the current heuristics, large projects should be completely scanned
+// around every 2 seconds so in the worst case it could take up to 2 seconds
+// for a change to be noticed. However, after a change has been noticed the
+// change's path goes on a short list of recently changed paths which are
+// checked on every scan, so further changes to recently changed files should
+// be noticed almost instantly.
+
+import (
+	"fmt"
+	"math/rand"
+	"os"
+	"sync"
+	"sync/atomic"
+	"time"
+
+	"github.com/evanw/esbuild/internal/fs"
+	"github.com/evanw/esbuild/internal/logger"
+	"github.com/evanw/esbuild/internal/resolver"
+)
+
+// The time to wait between watch intervals
+const watchIntervalSleep = 100 * time.Millisecond
+
+// The maximum number of recently-edited items to check every interval
+const maxRecentItemCount = 16
+
+// The minimum number of non-recent items to check every interval
+const minItemCountPerIter = 64
+
+// The maximum number of intervals before a change is detected
+const maxIntervalsBeforeUpdate = 20
+
+type watcher struct {
+	data              fs.WatchData
+	fs                fs.FS
+	rebuild           func() fs.WatchData
+	recentItems       []string
+	itemsToScan       []string
+	mutex             sync.Mutex
+	itemsPerIteration int
+	shouldStop        int32
+	shouldLog         bool
+	useColor          logger.UseColor
+	stopWaitGroup     sync.WaitGroup
+}
+
+func (w *watcher) setWatchData(data fs.WatchData) {
+	defer w.mutex.Unlock()
+	w.mutex.Lock()
+
+	// Print something for the end of the first build
+	if w.shouldLog && w.data.Paths == nil {
+		logger.PrintTextWithColor(os.Stderr, w.useColor, func(colors logger.Colors) string {
+			return fmt.Sprintf("%s[watch] build finished, watching for changes...%s\n", colors.Dim, colors.Reset)
+		})
+	}
+
+	w.data = data
+	w.itemsToScan = w.itemsToScan[:0] // Reuse memory
+
+	// Remove any recent items that weren't a part of the latest build
+	end := 0
+	for _, path := range w.recentItems {
+		if data.Paths[path] != nil {
+			w.recentItems[end] = path
+			end++
+		}
+	}
+	w.recentItems = w.recentItems[:end]
+}
+
+func (w *watcher) start() {
+	w.stopWaitGroup.Add(1)
+
+	go func() {
+		// Note: Do not change these log messages without a breaking version change.
+		// People want to run regexes over esbuild's stderr stream to look for these
+		// messages instead of using esbuild's API.
+
+		for atomic.LoadInt32(&w.shouldStop) == 0 {
+			// Sleep for the watch interval
+			time.Sleep(watchIntervalSleep)
+
+			// Rebuild if we're dirty
+			if absPath := w.tryToFindDirtyPath(); absPath != "" {
+				if w.shouldLog {
+					logger.PrintTextWithColor(os.Stderr, w.useColor, func(colors logger.Colors) string {
+						prettyPath := resolver.PrettyPath(w.fs, logger.Path{Text: absPath, Namespace: "file"})
+						return fmt.Sprintf("%s[watch] build started (change: %q)%s\n", colors.Dim, prettyPath, colors.Reset)
+					})
+				}
+
+				// Run the build
+				w.setWatchData(w.rebuild())
+
+				if w.shouldLog {
+					logger.PrintTextWithColor(os.Stderr, w.useColor, func(colors logger.Colors) string {
+						return fmt.Sprintf("%s[watch] build finished%s\n", colors.Dim, colors.Reset)
+					})
+				}
+			}
+		}
+
+		w.stopWaitGroup.Done()
+	}()
+}
+
+func (w *watcher) stop() {
+	atomic.StoreInt32(&w.shouldStop, 1)
+	w.stopWaitGroup.Wait()
+}
+
+func (w *watcher) tryToFindDirtyPath() string {
+	defer w.mutex.Unlock()
+	w.mutex.Lock()
+
+	// If we ran out of items to scan, fill the items back up in a random order
+	if len(w.itemsToScan) == 0 {
+		items := w.itemsToScan[:0] // Reuse memory
+		for path := range w.data.Paths {
+			items = append(items, path)
+		}
+		rand.Seed(time.Now().UnixNano())
+		for i := int32(len(items) - 1); i > 0; i-- { // Fisher-Yates shuffle
+			j := rand.Int31n(i + 1)
+			items[i], items[j] = items[j], items[i]
+		}
+		w.itemsToScan = items
+
+		// Determine how many items to check every iteration, rounded up
+		perIter := (len(items) + maxIntervalsBeforeUpdate - 1) / maxIntervalsBeforeUpdate
+		if perIter < minItemCountPerIter {
+			perIter = minItemCountPerIter
+		}
+		w.itemsPerIteration = perIter
+	}
+
+	// Always check all recent items every iteration
+	for i, path := range w.recentItems {
+		if dirtyPath := w.data.Paths[path](); dirtyPath != "" {
+			// Move this path to the back of the list (i.e. the "most recent" position)
+			copy(w.recentItems[i:], w.recentItems[i+1:])
+			w.recentItems[len(w.recentItems)-1] = path
+			return dirtyPath
+		}
+	}
+
+	// Check a constant number of items every iteration
+	remainingCount := len(w.itemsToScan) - w.itemsPerIteration
+	if remainingCount < 0 {
+		remainingCount = 0
+	}
+	toCheck, remaining := w.itemsToScan[remainingCount:], w.itemsToScan[:remainingCount]
+	w.itemsToScan = remaining
+
+	// Check if any of the entries in this iteration have been modified
+	for _, path := range toCheck {
+		if dirtyPath := w.data.Paths[path](); dirtyPath != "" {
+			// Mark this item as recent by adding it to the back of the list
+			w.recentItems = append(w.recentItems, path)
+			if len(w.recentItems) > maxRecentItemCount {
+				// Remove items from the front of the list when we hit the limit
+				copy(w.recentItems, w.recentItems[1:])
+				w.recentItems = w.recentItems[:maxRecentItemCount]
+			}
+			return dirtyPath
+		}
+	}
+	return ""
+}
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/.envrc b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/.envrc
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/.envrc
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/.envrc
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/.gitignore b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/.gitignore
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/.gitignore
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/.gitignore
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/.gitlab-ci.yml b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/.gitlab-ci.yml
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/.gitlab-ci.yml
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/.gitlab-ci.yml
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/CHANGELOG.md b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/CHANGELOG.md
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/CHANGELOG.md
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/CHANGELOG.md
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/CONTRIBUTING.md b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/CONTRIBUTING.md
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/CONTRIBUTING.md
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/CONTRIBUTING.md
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/LICENSE b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/LICENSE
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/LICENSE
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/LICENSE
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/README.md b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/README.md
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/README.md
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/README.md
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/api.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/api.go
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/api.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/api.go
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/command.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/command.go
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/command.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/command.go
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/devenv.lock b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/devenv.lock
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/devenv.lock
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/devenv.lock
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/devenv.nix b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/devenv.nix
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/devenv.nix
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/devenv.nix
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/devenv.yaml b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/devenv.yaml
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/devenv.yaml
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/devenv.yaml
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/doc.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/doc.go
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/doc.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/doc.go
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/error.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/error.go
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/error.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/error.go
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/execute.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/execute.go
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/execute.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/execute.go
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/flake.lock b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/flake.lock
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/flake.lock
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/flake.lock
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/flake.nix b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/flake.nix
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/flake.nix
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/flake.nix
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/help-util.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/help-util.go
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/help-util.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/help-util.go
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/help.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/help.go
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/help.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/help.go
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/hint.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/hint.go
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/hint.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/hint.go
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/mapping.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/mapping.go
similarity index 97%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/mapping.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/mapping.go
index f4f0438..b1efa64 100644
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/mapping.go
+++ b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/mapping.go
@@ -5,7 +5,7 @@ package xflags
 
 import (
 	"flag"
-	"gitlab.schukai.com/oss/libraries/go/utilities/pathfinder"
+	"gitlab.schukai.com/oss/libraries/go/utilities/pathfinder.git"
 	"reflect"
 	"strconv"
 	"strings"
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/parse.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/parse.go
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/parse.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/parse.go
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/release.json b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/release.json
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/release.json
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/release.json
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/setting.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/setting.go
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/setting.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/setting.go
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/tags.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/tags.go
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/tags.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/tags.go
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/type.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/type.go
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags/type.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/application/xflags.git/type.go
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/markup/html/LICENSE b/source/vendor/gitlab.schukai.com/oss/libraries/go/markup/html.git/LICENSE
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/markup/html/LICENSE
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/markup/html.git/LICENSE
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/markup/html/engine/engine.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/markup/html.git/engine/engine.go
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/markup/html/engine/engine.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/markup/html.git/engine/engine.go
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/markup/html/engine/error.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/markup/html.git/engine/error.go
similarity index 100%
rename from source/vendor/gitlab.schukai.com/oss/libraries/go/markup/html/engine/error.go
rename to source/vendor/gitlab.schukai.com/oss/libraries/go/markup/html.git/engine/error.go
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/.envrc b/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/.envrc
deleted file mode 100644
index 0da5bcc..0000000
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/.envrc
+++ /dev/null
@@ -1,2 +0,0 @@
-watch_file ./flake.nix ./nix/scripts/*.nix ./nix/config/*.nix ./nix/packages/*.nix 
-use flake
\ No newline at end of file
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/.gitignore b/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/.gitignore
deleted file mode 100644
index 13bb7b5..0000000
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/.gitignore
+++ /dev/null
@@ -1,155 +0,0 @@
-# Created by https://www.toptal.com/developers/gitignore/api/intellij,go
-# Edit at https://www.toptal.com/developers/gitignore?templates=intellij,go
-
-### Go ###
-# If you prefer the allow list template instead of the deny list, see community template:
-# https://github.com/github/gitignore/blob/main/community/Golang/Go.AllowList.gitignore
-#
-# Binaries for programs and plugins
-*.exe
-*.exe~
-*.dll
-*.so
-*.dylib
-
-# Test binary, built with `go test -c`
-*.test
-
-# Output of the go coverage tool, specifically when used with LiteIDE
-*.out
-
-# Dependency directories (remove the comment below to include it)
-# vendor/
-
-# Go workspace file
-go.work
-
-### Go Patch ###
-/vendor/
-/Godeps/
-
-### Intellij ###
-# Covers JetBrains IDEs: IntelliJ, RubyMine, PhpStorm, AppCode, PyCharm, CLion, Android Studio, WebStorm and Rider
-# Reference: https://intellij-support.jetbrains.com/hc/en-us/articles/206544839
-
-# User-specific stuff
-.idea/**/workspace.xml
-.idea/**/tasks.xml
-.idea/**/usage.statistics.xml
-.idea/**/dictionaries
-.idea/**/shelf
-
-# AWS User-specific
-.idea/**/aws.xml
-
-# Generated files
-.idea/**/contentModel.xml
-
-# Sensitive or high-churn files
-.idea/**/dataSources/
-.idea/**/dataSources.ids
-.idea/**/dataSources.local.xml
-.idea/**/sqlDataSources.xml
-.idea/**/dynamic.xml
-.idea/**/uiDesigner.xml
-.idea/**/dbnavigator.xml
-
-# Gradle
-.idea/**/gradle.xml
-.idea/**/libraries
-
-# Gradle and Maven with auto-import
-# When using Gradle or Maven with auto-import, you should exclude module files,
-# since they will be recreated, and may cause churn.  Uncomment if using
-# auto-import.
-# .idea/artifacts
-# .idea/compiler.xml
-# .idea/jarRepositories.xml
-# .idea/modules.xml
-# .idea/*.iml
-# .idea/modules
-# *.iml
-# *.ipr
-
-# CMake
-cmake-build-*/
-
-# Mongo Explorer plugin
-.idea/**/mongoSettings.xml
-
-# File-based project format
-*.iws
-
-# IntelliJ
-out/
-
-# mpeltonen/sbt-idea plugin
-.idea_modules/
-
-# JIRA plugin
-atlassian-ide-plugin.xml
-
-# Cursive Clojure plugin
-.idea/replstate.xml
-
-# SonarLint plugin
-.idea/sonarlint/
-
-# Crashlytics plugin (for Android Studio and IntelliJ)
-com_crashlytics_export_strings.xml
-crashlytics.properties
-crashlytics-build.properties
-fabric.properties
-
-# Editor-based Rest Client
-.idea/httpRequests
-
-# Android studio 3.1+ serialized cache file
-.idea/caches/build_file_checksums.ser
-
-### Intellij Patch ###
-# Comment Reason: https://github.com/joeblau/gitignore.io/issues/186#issuecomment-215987721
-
-# *.iml
-# modules.xml
-# .idea/misc.xml
-# *.ipr
-
-# Sonarlint plugin
-# https://plugins.jetbrains.com/plugin/7973-sonarlint
-.idea/**/sonarlint/
-
-# SonarQube Plugin
-# https://plugins.jetbrains.com/plugin/7238-sonarqube-community-plugin
-.idea/**/sonarIssues.xml
-
-# Markdown Navigator plugin
-# https://plugins.jetbrains.com/plugin/7896-markdown-navigator-enhanced
-.idea/**/markdown-navigator.xml
-.idea/**/markdown-navigator-enh.xml
-.idea/**/markdown-navigator/
-
-# Cache file creation bug
-# See https://youtrack.jetbrains.com/issue/JBR-2257
-.idea/$CACHE_FILE$
-
-# CodeStream plugin
-# https://plugins.jetbrains.com/plugin/12206-codestream
-.idea/codestream.xml
-
-# Azure Toolkit for IntelliJ plugin
-# https://plugins.jetbrains.com/plugin/8053-azure-toolkit-for-intellij
-.idea/**/azureSettings.xml
-
-# End of https://www.toptal.com/developers/gitignore/api/intellij,go
-# Devenv
-.devenv*
-devenv.local.nix
-
-# direnv
-.direnv
-
-# pre-commit
-.pre-commit-config.yaml
-
-/Session.vim
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/.gitlab-ci.yml b/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/.gitlab-ci.yml
deleted file mode 100644
index 78b9b3f..0000000
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/.gitlab-ci.yml
+++ /dev/null
@@ -1,43 +0,0 @@
-variables:
-  NIXPKGS_ALLOW_UNFREE: "1"
-  NIXPKGS_ALLOW_INSECURE: "1"
-  GIT_DEPTH: 10
-  GIT_SUBMODULE_STRATEGY: normal
-  GIT_SUBMODULE_DEPTH: 1
-
-
-stages:
-  - test
-  - release
-
-before_script:
-  - git config --global user.email "${GITLAB_USER_EMAIL}"
-  - git config --global user.name "${GITLAB_USER_NAME:-"Gitlab CI"}"
-  - git config --global credential.helper '!f() { echo "username=gitlab-ci-token"; echo "password=${CI_JOB_TOKEN}"; }; f'
-  - git config --global pull.rebase true
-  - git config --global http.sslVerify "false"
-
-
-after_script:
-  - nix develop .#gitlab --command clean-up
-
-tests:
-  stage: test
-  tags:
-    - nixos-gen3
-  script:
-    - nix develop .#gitlab --command run-ci-tests
-  artifacts:
-    paths:
-      - last-phpunit-result.xml
-
-release:
-  stage: release
-  tags:
-    - nixos-gen3
-  script:
-    - nix develop .#gitlab --command release
-  when: on_success
-  rules:
-    - if: $CI_COMMIT_BRANCH == "master"
-  
\ No newline at end of file
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/CONTRIBUTING.md b/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/CONTRIBUTING.md
deleted file mode 100644
index 2713a85..0000000
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/CONTRIBUTING.md
+++ /dev/null
@@ -1,58 +0,0 @@
-# Contributing to schukai GmbH Projects
-
-## Code of Conduct
-
-Be a human, not an asshole. Common sense and basic human decency apply.
-
-## Getting Started
-
-### Setting up the Project
-
-1. Fork the project on GitLab.
-2. Clone your fork locally. Replace `[your-username]` with your GitLab username and `[project-name]` with the actual project name:
-    ```bash
-    git clone $(git config --get remote.origin.url)
-    ```
-3. Add the upstream repository. Replace `[original-username]` and `[project-name]` with the original repository's username and project name:
-    ```bash
-    git remote add upstream https://gitlab.schukai.com/[original-username]/[project-name].git
-    ```
-
-### Making Changes
-
-1. Create a new branch:
-    ```bash
-    git checkout -b new-feature-branch
-    ```
-2. Make your changes.
-3. Commit your changes:
-    ```bash
-    git commit -m "Description of change"
-    ```
-
-### Submitting a Merge Request
-
-1. Push your changes to your fork:
-    ```bash
-    git push origin new-feature-branch
-    ```
-2. Navigate to the original project repository on `gitlab.schukai.com`.
-3. Open a Merge Request and provide a clear description of the changes.
-
-## Coding Guidelines
-
-- Follow the coding style used in the project.
-- Write unit tests for new features.
-- Ensure that all tests pass before submitting a Merge Request.
-
-## Reporting Issues
-
-If you find an issue, please create a new issue on `gitlab.schukai.com`.
-
-## Additional Resources
-
-- [GitLab Flow](https://docs.gitlab.com/ee/topics/gitlab_flow.html)
-- [GitLab Merge Request Guidelines](https://docs.gitlab.com/ee/user/project/merge_requests/)
-
-Thank you for your contribution! 
-    
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/LICENSE b/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/LICENSE
deleted file mode 100644
index 5694d30..0000000
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/LICENSE
+++ /dev/null
@@ -1,14 +0,0 @@
-Copyright (C) 2023 schukai GmbH 
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published
-by the Free Software Foundation, either version 3 of the License.
-
-This program is distributed in the hope that it will be useful,
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program.  If not, see <https://www.gnu.org/licenses/>.  
-    
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/README.md b/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/README.md
deleted file mode 100644
index 4d1522c..0000000
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/README.md
+++ /dev/null
@@ -1,69 +0,0 @@
-## Pathfinder
-
-## What does this library?
-
-This library provides a simple way to get and set values in a nested structure.
-
-It supports:
-
-* [X]  Set values in a nested structure
-* [X]  Get values from a nested structure
-
-## Installation
-
-```shell
-go get gitlab.schukai.com/oss/libraries/go/utilities/pathfinder
-```
-
-**Note:** This library uses [Go Modules](https://github.com/golang/go/wiki/Modules) to manage dependencies.
-
-## Usage
-
-### Set values
-
-```go
-s := &StructA{}
-err := GetValue[*StructA](s, "my.key")
-```
-
-### Get values
-
-```go
-s := &StructA{}
-err := SetValue[*StructA](s, "my.key", "value")
-```
-
-## Contributing
-
-Merge requests are welcome. For major changes, please open an issue first to discuss what
-you would like to change. **Please make sure to update tests as appropriate.**
-
-Versioning is done with [SemVer](https://semver.org/).
-Changelog is generated with [git-chglog](https://github.com/git-chglog/git-chglog#git-chglog)
-
-Commit messages should follow the [Conventional Commits](https://www.conventionalcommits.org/en/v1.0.0/) specification.
-Messages are started with a type, which is one of the following:
-
-- **feat**: A new feature
-- **fix**: A bug fix
-- **doc**: Documentation only changes
-- **refactor**: A code change that neither fixes a bug nor adds a feature
-- **perf**: A code change that improves performance
-- **test**: Adding missing or correcting existing tests
-- **chore**: Other changes that don't modify src or test files
-
-The footer would be used for a reference to an issue or a breaking change.
-
-A commit that has a footer `BREAKING CHANGE:`, or appends a ! after the type/scope,
-introduces a breaking API change (correlating with MAJOR in semantic versioning).
-A BREAKING CHANGE can be part of commits of any type.
-
-the following is an example of a commit message:
-
-```text
-feat: add 'extras' field
-```
-
-## License
-
-[AGPL-3.0](https://choosealicense.com/licenses/agpl-3.0/)
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/error.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/error.go
deleted file mode 100644
index 614b13e..0000000
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/error.go
+++ /dev/null
@@ -1,39 +0,0 @@
-// Copyright 2022 schukai GmbH
-// SPDX-License-Identifier: AGPL-3.0
-
-package pathfinder
-
-import (
-	"errors"
-	"reflect"
-)
-
-type InvalidPathError error
-
-func newInvalidPathError(path string) InvalidPathError {
-	return InvalidPathError(errors.New("invalid path " + path))
-}
-
-type UnsupportedTypeAtTopOfPathError error
-
-func newUnsupportedTypeAtTopOfPathError(path string, t reflect.Type) UnsupportedTypeAtTopOfPathError {
-	return UnsupportedTypeAtTopOfPathError(errors.New("unsupported type " + t.String() + " at top of path " + path))
-}
-
-type UnsupportedTypePathError error
-
-func newUnsupportedTypePathError(path string, t reflect.Type) UnsupportedTypePathError {
-	return UnsupportedTypePathError(errors.New("unsupported type " + t.String() + " at path " + path))
-}
-
-type CannotSetError error
-
-func newCannotSetError(name string) CannotSetError {
-	return CannotSetError(errors.New("cannot set " + name))
-}
-
-type InvalidTypeForPathError error
-
-func newInvalidTypeForPathError(path string, pt string, nt string) InvalidTypeForPathError {
-	return InvalidTypeForPathError(errors.New("invalid type for path " + path + ": expected " + pt + ", got " + nt))
-}
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/find.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/find.go
deleted file mode 100644
index 0495ecf..0000000
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/find.go
+++ /dev/null
@@ -1,51 +0,0 @@
-// Copyright 2024 schukai GmbH
-// SPDX-License-Identifier: proprietary
-
-package pathfinder
-
-import (
-	"fmt"
-	"reflect"
-	"strings"
-)
-
-func FindPaths(v reflect.Value, targetType reflect.Type, path []string, paths *[]string) {
-
-	if v.Kind() == reflect.Invalid {
-		return
-	}
-
-	vType := v.Type()
-
-	switch v.Kind() {
-	case reflect.Ptr:
-		FindPaths(v.Elem(), targetType, path, paths)
-	case reflect.Struct:
-		for i := 0; i < v.NumField(); i++ {
-			newPath := append(path, vType.Field(i).Name)
-			FindPaths(v.Field(i), targetType, newPath, paths)
-		}
-	case reflect.Map:
-		for _, key := range v.MapKeys() {
-			newPath := append(path, fmt.Sprint(key))
-			FindPaths(v.MapIndex(key), targetType, newPath, paths)
-		}
-	case reflect.Slice, reflect.Array:
-		for i := 0; i < v.Len(); i++ {
-			newPath := append(path, fmt.Sprint(i))
-			FindPaths(v.Index(i), targetType, newPath, paths)
-		}
-
-	case reflect.String, reflect.Bool, reflect.Int64, reflect.Int32, reflect.Int16, reflect.Int8, reflect.Int, reflect.Uint64, reflect.Uint32, reflect.Uint16, reflect.Uint8, reflect.Uint:
-		if vType != targetType {
-			return
-		}
-
-	default:
-		return
-	}
-
-	if vType == targetType {
-		*paths = append(*paths, strings.Join(path, "."))
-	}
-}
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/flake.lock b/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/flake.lock
deleted file mode 100644
index fd83cb2..0000000
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/flake.lock
+++ /dev/null
@@ -1,181 +0,0 @@
-{
-  "nodes": {
-    "commonFlake": {
-      "inputs": {
-        "nixpkgs": "nixpkgs"
-      },
-      "locked": {
-        "dir": "common",
-        "lastModified": 1718788884,
-        "narHash": "sha256-PefMbkGNMK9TN1qcNL9OkFVTNdv6wo6XoaS8eTdsY04=",
-        "ref": "refs/heads/master",
-        "rev": "abda2dc723e13dfc835535593321c514666e679e",
-        "revCount": 39,
-        "type": "git",
-        "url": "https://gitlab.schukai.com/schukai/entwicklung/nix-flakes.git?dir=common"
-      },
-      "original": {
-        "dir": "common",
-        "type": "git",
-        "url": "https://gitlab.schukai.com/schukai/entwicklung/nix-flakes.git?dir=common"
-      }
-    },
-    "flake-utils": {
-      "inputs": {
-        "systems": "systems"
-      },
-      "locked": {
-        "lastModified": 1726560853,
-        "narHash": "sha256-X6rJYSESBVr3hBoH0WbKE5KvhPU5bloyZ2L4K60/fPQ=",
-        "owner": "numtide",
-        "repo": "flake-utils",
-        "rev": "c1dfcf08411b08f6b8615f7d8971a2bfa81d5e8a",
-        "type": "github"
-      },
-      "original": {
-        "id": "flake-utils",
-        "type": "indirect"
-      }
-    },
-    "flakeUtils": {
-      "inputs": {
-        "systems": "systems_2"
-      },
-      "locked": {
-        "lastModified": 1726560853,
-        "narHash": "sha256-X6rJYSESBVr3hBoH0WbKE5KvhPU5bloyZ2L4K60/fPQ=",
-        "owner": "numtide",
-        "repo": "flake-utils",
-        "rev": "c1dfcf08411b08f6b8615f7d8971a2bfa81d5e8a",
-        "type": "github"
-      },
-      "original": {
-        "owner": "numtide",
-        "repo": "flake-utils",
-        "type": "github"
-      }
-    },
-    "nixpkgs": {
-      "locked": {
-        "lastModified": 1714971268,
-        "narHash": "sha256-IKwMSwHj9+ec660l+I4tki/1NRoeGpyA2GdtdYpAgEw=",
-        "owner": "NixOS",
-        "repo": "nixpkgs",
-        "rev": "27c13997bf450a01219899f5a83bd6ffbfc70d3c",
-        "type": "github"
-      },
-      "original": {
-        "id": "nixpkgs",
-        "ref": "nixos-23.11",
-        "type": "indirect"
-      }
-    },
-    "nixpkgsUnstable": {
-      "locked": {
-        "lastModified": 1727348695,
-        "narHash": "sha256-J+PeFKSDV+pHL7ukkfpVzCOO7mBSrrpJ3svwBFABbhI=",
-        "owner": "nixos",
-        "repo": "nixpkgs",
-        "rev": "1925c603f17fc89f4c8f6bf6f631a802ad85d784",
-        "type": "github"
-      },
-      "original": {
-        "owner": "nixos",
-        "ref": "nixos-unstable",
-        "repo": "nixpkgs",
-        "type": "github"
-      }
-    },
-    "nixpkgs_2": {
-      "locked": {
-        "lastModified": 1727397532,
-        "narHash": "sha256-pojbL/qteElw/nIXlN8kmHn/w6PQbEHr7Iz+WOXs0EM=",
-        "owner": "nixos",
-        "repo": "nixpkgs",
-        "rev": "f65141456289e81ea0d5a05af8898333cab5c53d",
-        "type": "github"
-      },
-      "original": {
-        "owner": "nixos",
-        "ref": "nixos-24.05",
-        "repo": "nixpkgs",
-        "type": "github"
-      }
-    },
-    "nixpkgs_3": {
-      "locked": {
-        "lastModified": 1704145853,
-        "narHash": "sha256-G/1AMt9ibpeMlcxvD1vNaC8imGaK+g7zZ99e29BLgWw=",
-        "owner": "NixOS",
-        "repo": "nixpkgs",
-        "rev": "2d2ea8eab9e400618748ab1a6a108255233b602c",
-        "type": "github"
-      },
-      "original": {
-        "id": "nixpkgs",
-        "ref": "nixos-23.11",
-        "type": "indirect"
-      }
-    },
-    "root": {
-      "inputs": {
-        "commonFlake": "commonFlake",
-        "flake-utils": "flake-utils",
-        "flakeUtils": "flakeUtils",
-        "nixpkgs": "nixpkgs_2",
-        "nixpkgsUnstable": "nixpkgsUnstable",
-        "versionFlake": "versionFlake"
-      }
-    },
-    "systems": {
-      "locked": {
-        "lastModified": 1681028828,
-        "narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=",
-        "owner": "nix-systems",
-        "repo": "default",
-        "rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e",
-        "type": "github"
-      },
-      "original": {
-        "owner": "nix-systems",
-        "repo": "default",
-        "type": "github"
-      }
-    },
-    "systems_2": {
-      "locked": {
-        "lastModified": 1681028828,
-        "narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=",
-        "owner": "nix-systems",
-        "repo": "default",
-        "rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e",
-        "type": "github"
-      },
-      "original": {
-        "owner": "nix-systems",
-        "repo": "default",
-        "type": "github"
-      }
-    },
-    "versionFlake": {
-      "inputs": {
-        "nixpkgs": "nixpkgs_3"
-      },
-      "locked": {
-        "lastModified": 1716914109,
-        "narHash": "sha256-JY0PLGWzYRDQ9daKLGOBWHHuYun9nSpH9J3aSk8iDmQ=",
-        "ref": "refs/heads/master",
-        "rev": "fe8dd932d6c414a93b4a69c470792b2db038e0fb",
-        "revCount": 129,
-        "type": "git",
-        "url": "https://gitlab.schukai.com/oss/utilities/version.git"
-      },
-      "original": {
-        "type": "git",
-        "url": "https://gitlab.schukai.com/oss/utilities/version.git"
-      }
-    }
-  },
-  "root": "root",
-  "version": 7
-}
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/flake.nix b/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/flake.nix
deleted file mode 100644
index 64f1699..0000000
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/flake.nix
+++ /dev/null
@@ -1,158 +0,0 @@
-{
-  description = "Configuration is a library";
-
-  inputs = {
-    nixpkgs = {url = "github:nixos/nixpkgs/nixos-24.05";};
-    nixpkgsUnstable = {url = "github:nixos/nixpkgs/nixos-unstable";};
-    flakeUtils = {url = "github:numtide/flake-utils";};
-
-    commonFlake = {
-      url = "git+https://gitlab.schukai.com/schukai/entwicklung/nix-flakes.git?dir=common";
-      flake = true;
-    };
-
-    versionFlake = {
-      url = "git+https://gitlab.schukai.com/oss/utilities/version.git";
-      flake = true;
-    };
-  };
-
-  outputs = {
-    self,
-    nixpkgs,
-    nixpkgsUnstable,
-    flake-utils,
-    versionFlake,
-    commonFlake,
-    ...
-  } @ inputs:
-    flake-utils.lib.eachDefaultSystem (system: let
-      inherit (nixpkgs.lib) optional;
-
-      commonPck = commonFlake.packages.${system}.common;
-      versionPck = versionFlake.packages.${system}.version;
-
-      pkgs' = import nixpkgs {
-        inherit system;
-        overlays = [
-          (final: prev: {
-            common = commonPck;
-          })
-
-          (final: prev: {
-            version = versionPck;
-          })
-
-          (final: prev: {
-            dolt =
-              (import nixpkgsUnstable {
-                inherit system;
-              })
-              .dolt;
-          })
-        ];
-      };
-    in {
-      packages = rec {
-        manual = pkgs'.callPackage ./nix/packages/manual.nix {inherit self pkgs';};
-      };
-
-      devShells = {
-        default = let
-          commonPck = commonFlake.packages.${system}.common;
-          commonScript = commonPck + "/bin/common";
-
-          versionPck = versionFlake.packages.${system}.version;
-          versionBin = versionPck + "/bin/version";
-
-          scriptGoTask = import ./nix/scripts/go-task.nix {inherit self pkgs' system;};
-
-          commonPackages = import ./nix/config/common-packages.nix {inherit pkgs';};
-
-          extendedPackages = [
-            scriptGoTask
-          ];
-
-          scriptPackages = [
-            versionPck
-          ];
-
-          shellPackages =
-            commonPackages
-            ++ extendedPackages
-            ++ scriptPackages;
-        in
-          pkgs'.mkShell {
-            nativeBuildInputs = shellPackages;
-
-            shellHook = ''
-              source ${commonScript}
-
-              if [ -n "$CI_JOB_TOKEN" ]; then
-                  echo_fail "You are in a CI environment, this shell is not intended for CI, but for local development"
-                  exit 1
-              fi
-
-              echo_header "Configuration Lib development shell"
-              readonly worktree=$(get_working_dir)
-              echo_hint "Working directory: ''${worktree}"
-              currentVersion=$(${versionBin} print -g)
-              if [ -z "''${currentVersion}" ]; then
-                  echo_fail "No version found, check your git tags"
-              else
-                  echo_hint "Current version: ''${currentVersion}"
-              fi
-
-              currentGitBranch=$(git rev-parse --abbrev-ref HEAD)
-              echo_hint "Current branch: ''${currentGitBranch}"
-              echo_hint "You can run the task command to see the available tasks"
-
-              echo_section "Happy hacking!"
-            '';
-          };
-
-        gitlab = let
-          commonPck = commonFlake.packages.${system}.common;
-          commonScript = commonPck + "/bin/common";
-
-          versionPck = versionFlake.packages.${system}.version;
-
-          scriptCleanUp = pkgs'.callPackage ./nix/scripts/clean-up.nix {inherit pkgs';};
-          scriptRunCITests = pkgs'.callPackage ./nix/scripts/run-ci-tests.nix {inherit pkgs';};
-          scriptRelease = pkgs'.callPackage ./nix/scripts/release.nix {inherit pkgs';};
-
-          commonPackages = import ./nix/config/common-packages.nix {inherit pkgs';};
-
-          extendedPackages = [
-            scriptCleanUp
-            scriptRunCITests
-            scriptRelease
-          ];
-
-          scriptPackages = [
-            versionPck
-          ];
-
-          shellPackages =
-            commonPackages
-            ++ extendedPackages
-            ++ scriptPackages;
-        in
-          pkgs'.mkShell {
-            nativeBuildInputs = shellPackages;
-
-            shellHook = ''
-              source ${commonScript}
-
-              if [ -z "$CI_JOB_TOKEN" ]; then
-                 echo_fail "You are not in a CI environment, this shell is intended for CI, but for local development"
-                 exit 1
-              fi
-
-              cd ''${CI_PROJECT_DIR} || exit 1
-
-            '';
-          };
-      };
-    });
-}
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/get.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/get.go
deleted file mode 100644
index 3d53650..0000000
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/get.go
+++ /dev/null
@@ -1,74 +0,0 @@
-// Copyright 2022 schukai GmbH
-// SPDX-License-Identifier: AGPL-3.0
-
-package pathfinder
-
-import (
-	"reflect"
-	"strconv"
-	"strings"
-)
-
-// GetValue returns the value of a field in a struct, given a path to the field.
-// The path can contain dots to access nested fields.
-// The object must be a pointer to a struct, a struct, a map, a slice or an array,
-// otherwise an error is returned.
-func GetValue[D any](obj D, keyWithDots string) (any, error) {
-	keySlice := strings.Split(keyWithDots, ".")
-	v := reflect.ValueOf(obj)
-
-	for _, key := range keySlice[0:] {
-
-		if !v.IsValid() {
-			return nil, newInvalidPathError(keyWithDots)
-		}
-
-		switch v.Kind() {
-		case reflect.Ptr, reflect.Interface:
-			v = v.Elem()
-		}
-
-		switch v.Kind() {
-		case reflect.Map:
-			v = v.MapIndex(reflect.ValueOf(key))
-			if !v.IsValid() {
-				return nil, newInvalidPathError(keyWithDots)
-			}
-
-		case reflect.Slice, reflect.Array:
-			index, err := strconv.Atoi(key)
-			if err != nil {
-				return nil, newInvalidPathError(keyWithDots)
-			}
-			// check if index is in range
-			if index >= v.Len() {
-				return nil, newInvalidPathError(keyWithDots)
-			}
-			v = v.Index(index)
-		case reflect.Struct:
-			v = v.FieldByName(key)
-			if !v.IsValid() {
-				return nil, newInvalidPathError(keyWithDots)
-			}
-		default:
-			return nil, newInvalidPathError(keyWithDots)
-		}
-
-	}
-
-	if v.Kind() == reflect.Invalid {
-		return nil, newInvalidPathError(keyWithDots)
-	}
-
-	//for v.Kind() == reflect.Ptr {
-	//	v = v.Elem()
-	//}
-
-	// check if v can interface
-	if !v.CanInterface() {
-		return nil, newInvalidPathError(keyWithDots)
-	}
-
-	return v.Interface(), nil
-
-}
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/pathfinder.iml b/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/pathfinder.iml
deleted file mode 100644
index 789c0e8..0000000
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/pathfinder.iml
+++ /dev/null
@@ -1,12 +0,0 @@
-<?xml version="1.0" encoding="UTF-8"?>
-<module type="JAVA_MODULE" version="4">
-  <component name="NewModuleRootManager" inherit-compiler-output="true">
-    <exclude-output />
-    <content url="file://$MODULE_DIR$">
-      <sourceFolder url="file://$MODULE_DIR$/.devenv/state/go/pkg/mod/github.com/google/addlicense@v1.1.1/testdata/expected" isTestSource="false" />
-      <sourceFolder url="file://$MODULE_DIR$/.devenv/state/go/pkg/mod/github.com/google/addlicense@v1.1.1/testdata/initial" isTestSource="false" />
-    </content>
-    <orderEntry type="inheritedJdk" />
-    <orderEntry type="sourceFolder" forTests="false" />
-  </component>
-</module>
\ No newline at end of file
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/release.json b/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/release.json
deleted file mode 100644
index ccd00c2..0000000
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/release.json
+++ /dev/null
@@ -1 +0,0 @@
-{"version":"0.5.2"}
diff --git a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/set.go b/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/set.go
deleted file mode 100644
index 7c37da1..0000000
--- a/source/vendor/gitlab.schukai.com/oss/libraries/go/utilities/pathfinder/set.go
+++ /dev/null
@@ -1,293 +0,0 @@
-// Copyright 2022 schukai GmbH
-// SPDX-License-Identifier: AGPL-3.0
-
-package pathfinder
-
-import (
-	"bytes"
-	"encoding/gob"
-	"fmt"
-	"reflect"
-	"strconv"
-	"strings"
-)
-
-func deepCopy(src, dst interface{}) error {
-	var buf bytes.Buffer
-	enc := gob.NewEncoder(&buf)
-	dec := gob.NewDecoder(&buf)
-
-	if err := enc.Encode(src); err != nil {
-		return err
-	}
-
-	return dec.Decode(dst)
-}
-
-// SetValue sets the value of a field in a struct, given a path to the field.
-// The object must be a pointer to a struct, otherwise an error is returned.
-func SetValue[D any](obj D, keyWithDots string, newValue any) error {
-
-	keySlice := strings.Split(keyWithDots, ".")
-	reflectionOfObject := reflect.ValueOf(obj)
-
-	for keyIndex, key := range keySlice[0 : len(keySlice)-1] {
-
-		if reflectionOfObject.Kind() == reflect.Map {
-
-			if reflectionOfObject.IsNil() {
-				return newInvalidPathError(keyWithDots)
-			}
-
-			currentValue := reflectionOfObject.MapIndex(reflect.ValueOf(key)).Interface()
-			newValueCopy := reflect.New(reflect.TypeOf(currentValue)).Interface()
-			if err := deepCopy(currentValue, newValueCopy); err != nil {
-				return err
-			}
-
-			newValueCopyPtr := &newValueCopy
-			newValueCopyReflect := reflect.ValueOf(newValueCopyPtr).Elem()
-			if !newValueCopyReflect.CanAddr() {
-				return newCannotSetError("Wert ist nicht adressierbar")
-			}
-			newKey := strings.Join(keySlice[keyIndex+1:], ".")
-
-			err := SetValue(newValueCopyPtr, newKey, newValue)
-			if err != nil {
-				return err
-			}
-
-			reflectionOfObject.SetMapIndex(reflect.ValueOf(key), reflect.ValueOf(newValueCopy).Elem())
-			return nil
-
-		}
-
-		if reflectionOfObject.Kind() == reflect.Ptr && reflectionOfObject.Elem().Kind() == reflect.Interface {
-			reflectionOfObject = reflectionOfObject.Elem().Elem()
-		}
-
-		for reflectionOfObject.Kind() != reflect.Ptr {
-			if reflectionOfObject.Kind() == reflect.Invalid {
-				return newInvalidPathError(keyWithDots)
-			}
-
-			if reflectionOfObject.CanAddr() {
-				reflectionOfObject = reflectionOfObject.Addr()
-			} else {
-				return newCannotSetError(keyWithDots)
-			}
-
-		}
-
-		if reflectionOfObject.Kind() != reflect.Ptr {
-			return newUnsupportedTypePathError(keyWithDots, reflectionOfObject.Type())
-		}
-
-		switch reflectionOfObject.Elem().Kind() {
-		case reflect.Struct:
-			reflectionOfObject = reflectionOfObject.Elem().FieldByName(key)
-
-		case reflect.Slice:
-			// index is a number and get reflectionOfObject from slice with index
-			index, err := strconv.Atoi(key)
-			if err != nil {
-				return newInvalidPathError(keyWithDots)
-			}
-
-			if index >= reflectionOfObject.Elem().Len() {
-				return newInvalidPathError(keyWithDots)
-			}
-
-			reflectionOfObject = reflectionOfObject.Elem().Index(index)
-		default:
-			return newUnsupportedTypePathError(keyWithDots, reflectionOfObject.Type())
-		}
-
-	}
-
-	if reflectionOfObject.Kind() == reflect.Invalid {
-		return newInvalidPathError(keyWithDots)
-	}
-
-	for reflectionOfObject.Kind() == reflect.Ptr {
-		reflectionOfObject = reflectionOfObject.Elem()
-	}
-
-	// non-supporter type at the top of the path
-	switch reflectionOfObject.Kind() {
-	case reflect.Struct:
-
-		reflectionOfObject = reflectionOfObject.FieldByName(keySlice[len(keySlice)-1])
-		if !reflectionOfObject.IsValid() {
-			return newInvalidPathError(keyWithDots)
-		}
-
-		if !reflectionOfObject.CanSet() {
-			return newCannotSetError(keyWithDots)
-		}
-
-	case reflect.Map:
-
-		key := keySlice[len(keySlice)-1]
-		m := reflectionOfObject
-
-		keyVal := reflect.ValueOf(key)
-		newVal := reflect.ValueOf(newValue)
-
-		if !keyVal.Type().ConvertibleTo(m.Type().Key()) {
-			return fmt.Errorf("key type mismatch")
-		}
-
-		if !newVal.Type().ConvertibleTo(m.Type().Elem()) {
-			return fmt.Errorf("value type mismatch")
-		}
-
-		keyValConverted := keyVal.Convert(m.Type().Key())
-		newValConverted := newVal.Convert(m.Type().Elem())
-		m.SetMapIndex(keyValConverted, newValConverted)
-		return nil
-
-	case reflect.Slice:
-
-		index, err := strconv.Atoi(keySlice[len(keySlice)-1])
-		if err != nil {
-			return newInvalidPathError(keyWithDots)
-		}
-
-		if index >= reflectionOfObject.Len() {
-			return newInvalidPathError(keyWithDots)
-		}
-
-		reflectionOfObject = reflectionOfObject.Index(index)
-
-	case reflect.Array:
-		return newUnsupportedTypeAtTopOfPathError(keyWithDots, reflectionOfObject.Type())
-	case reflect.Ptr:
-		if newValue == nil {
-			reflectionOfObject.Set(reflect.Zero(reflectionOfObject.Type()))
-		} else {
-			reflectionOfObject.Set(reflect.ValueOf(&newValue))
-		}
-		return nil
-	case reflect.Interface:
-
-		// check if reflectionOfObject is an interface to an struct pointer
-		if reflectionOfObject.Elem().Kind() == reflect.Ptr && reflectionOfObject.Elem().Elem().Kind() == reflect.Struct {
-			return SetValue(reflectionOfObject.Elem().Interface(), keySlice[len(keySlice)-1], newValue)
-		}
-
-	case reflect.Chan:
-		return newUnsupportedTypeAtTopOfPathError(keyWithDots, reflectionOfObject.Type())
-	case reflect.Func:
-		return newUnsupportedTypeAtTopOfPathError(keyWithDots, reflectionOfObject.Type())
-	case reflect.UnsafePointer:
-		return newUnsupportedTypeAtTopOfPathError(keyWithDots, reflectionOfObject.Type())
-	case reflect.Uintptr:
-		return newUnsupportedTypeAtTopOfPathError(keyWithDots, reflectionOfObject.Type())
-	case reflect.Complex64:
-		return newUnsupportedTypeAtTopOfPathError(keyWithDots, reflectionOfObject.Type())
-	case reflect.Complex128:
-		return newUnsupportedTypeAtTopOfPathError(keyWithDots, reflectionOfObject.Type())
-	case reflect.Invalid:
-		return newUnsupportedTypeAtTopOfPathError(keyWithDots, reflectionOfObject.Type())
-	default:
-		return newUnsupportedTypeAtTopOfPathError(keyWithDots, reflectionOfObject.Type())
-	}
-
-	newValueType := reflect.TypeOf(newValue)
-	if newValueType == nil {
-		return newUnsupportedTypePathError(keyWithDots, reflectionOfObject.Type())
-	}
-
-	newValueKind := reflect.TypeOf(newValue).Kind()
-
-	switch reflectionOfObject.Kind() {
-	case reflect.String:
-		if reflectionOfObject.Kind() == reflect.Ptr || reflectionOfObject.Kind() == reflect.Interface {
-			if reflectionOfObject.Elem().CanSet() && reflectionOfObject.Elem().Kind() == reflect.String {
-				if newValueKind == reflect.String {
-					reflectionOfObject.Elem().SetString(newValue.(string))
-				} else {
-					reflectionOfObject.Elem().SetString(fmt.Sprintf("%v", newValue))
-				}
-			}
-		} else if newValueKind == reflect.String {
-
-			if reflect.TypeOf(newValue).ConvertibleTo(reflect.TypeOf("")) {
-				newValueString := reflect.ValueOf(newValue).Convert(reflect.TypeOf("")).Interface().(string)
-				reflectionOfObject.SetString(newValueString)
-			} else {
-				return newUnsupportedTypePathError(keyWithDots, reflectionOfObject.Type())
-			}
-		} else {
-			reflectionOfObject.SetString(fmt.Sprintf("%v", newValue))
-		}
-
-	case reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64, reflect.Int:
-
-		if newValueKind == reflect.Int {
-			reflectionOfObject.SetInt(int64(newValue.(int)))
-		} else {
-			s, err := strconv.ParseInt(fmt.Sprintf("%v", newValue), 10, 64)
-			if err != nil {
-				return err
-			}
-			reflectionOfObject.SetInt(s)
-		}
-
-	case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
-
-		if newValueKind == reflect.Int {
-			reflectionOfObject.SetUint(uint64(newValue.(int)))
-		} else {
-			s, err := strconv.ParseInt(fmt.Sprintf("%v", newValue), 10, 64)
-			if err != nil {
-				return err
-			}
-			reflectionOfObject.SetUint(uint64(s))
-		}
-
-	case reflect.Bool:
-
-		if newValueKind == reflect.Bool {
-			reflectionOfObject.SetBool(newValue.(bool))
-		} else {
-			b, err := strconv.ParseBool(fmt.Sprintf("%v", newValue))
-			if err != nil {
-				return err
-			}
-
-			reflectionOfObject.SetBool(b)
-		}
-
-	case reflect.Float64, reflect.Float32:
-
-		if newValueKind == reflect.Float64 {
-			reflectionOfObject.SetFloat(newValue.(float64))
-		} else {
-			s, err := strconv.ParseFloat(fmt.Sprintf("%v", newValue), 64)
-			if err != nil {
-				return err
-			}
-
-			reflectionOfObject.SetFloat(s)
-		}
-
-	case reflect.Slice, reflect.Array:
-
-		if newValueKind == reflect.Ptr {
-			newValue = reflect.ValueOf(newValue).Elem().Interface()
-			reflectionOfObject.Set(reflect.ValueOf(newValue))
-		} else if newValueKind == reflect.Slice {
-			reflectionOfObject.Set(reflect.ValueOf(newValue))
-		} else {
-			return newUnsupportedTypePathError(keyWithDots, reflectionOfObject.Type())
-		}
-
-	default:
-		return newInvalidTypeForPathError(keyWithDots, reflectionOfObject.Type().String(), newValueKind.String())
-	}
-
-	return nil
-
-}
diff --git a/source/vendor/golang.org/x/net/html/doc.go b/source/vendor/golang.org/x/net/html/doc.go
index 3a7e5ab..885c4c5 100644
--- a/source/vendor/golang.org/x/net/html/doc.go
+++ b/source/vendor/golang.org/x/net/html/doc.go
@@ -78,16 +78,11 @@ example, to process each anchor node in depth-first order:
 	if err != nil {
 		// ...
 	}
-	var f func(*html.Node)
-	f = func(n *html.Node) {
+	for n := range doc.Descendants() {
 		if n.Type == html.ElementNode && n.Data == "a" {
 			// Do something with n...
 		}
-		for c := n.FirstChild; c != nil; c = c.NextSibling {
-			f(c)
-		}
 	}
-	f(doc)
 
 The relevant specifications include:
 https://html.spec.whatwg.org/multipage/syntax.html and
diff --git a/source/vendor/golang.org/x/net/html/iter.go b/source/vendor/golang.org/x/net/html/iter.go
new file mode 100644
index 0000000..54be8fd
--- /dev/null
+++ b/source/vendor/golang.org/x/net/html/iter.go
@@ -0,0 +1,56 @@
+// Copyright 2024 The Go Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style
+// license that can be found in the LICENSE file.
+
+//go:build go1.23
+
+package html
+
+import "iter"
+
+// Ancestors returns an iterator over the ancestors of n, starting with n.Parent.
+//
+// Mutating a Node or its parents while iterating may have unexpected results.
+func (n *Node) Ancestors() iter.Seq[*Node] {
+	_ = n.Parent // eager nil check
+
+	return func(yield func(*Node) bool) {
+		for p := n.Parent; p != nil && yield(p); p = p.Parent {
+		}
+	}
+}
+
+// ChildNodes returns an iterator over the immediate children of n,
+// starting with n.FirstChild.
+//
+// Mutating a Node or its children while iterating may have unexpected results.
+func (n *Node) ChildNodes() iter.Seq[*Node] {
+	_ = n.FirstChild // eager nil check
+
+	return func(yield func(*Node) bool) {
+		for c := n.FirstChild; c != nil && yield(c); c = c.NextSibling {
+		}
+	}
+
+}
+
+// Descendants returns an iterator over all nodes recursively beneath
+// n, excluding n itself. Nodes are visited in depth-first preorder.
+//
+// Mutating a Node or its descendants while iterating may have unexpected results.
+func (n *Node) Descendants() iter.Seq[*Node] {
+	_ = n.FirstChild // eager nil check
+
+	return func(yield func(*Node) bool) {
+		n.descendants(yield)
+	}
+}
+
+func (n *Node) descendants(yield func(*Node) bool) bool {
+	for c := range n.ChildNodes() {
+		if !yield(c) || !c.descendants(yield) {
+			return false
+		}
+	}
+	return true
+}
diff --git a/source/vendor/golang.org/x/net/html/node.go b/source/vendor/golang.org/x/net/html/node.go
index 1350eef..77741a1 100644
--- a/source/vendor/golang.org/x/net/html/node.go
+++ b/source/vendor/golang.org/x/net/html/node.go
@@ -38,6 +38,10 @@ var scopeMarker = Node{Type: scopeMarkerNode}
 // that it looks like "a<b" rather than "a&lt;b". For element nodes, DataAtom
 // is the atom for Data, or zero if Data is not a known tag name.
 //
+// Node trees may be navigated using the link fields (Parent,
+// FirstChild, and so on) or a range loop over iterators such as
+// [Node.Descendants].
+//
 // An empty Namespace implies a "http://www.w3.org/1999/xhtml" namespace.
 // Similarly, "math" is short for "http://www.w3.org/1998/Math/MathML", and
 // "svg" is short for "http://www.w3.org/2000/svg".
diff --git a/source/vendor/golang.org/x/sys/unix/ioctl_linux.go b/source/vendor/golang.org/x/sys/unix/ioctl_linux.go
index dbe680e..7ca4fa1 100644
--- a/source/vendor/golang.org/x/sys/unix/ioctl_linux.go
+++ b/source/vendor/golang.org/x/sys/unix/ioctl_linux.go
@@ -58,6 +58,102 @@ func IoctlGetEthtoolDrvinfo(fd int, ifname string) (*EthtoolDrvinfo, error) {
 	return &value, err
 }
 
+// IoctlGetEthtoolTsInfo fetches ethtool timestamping and PHC
+// association for the network device specified by ifname.
+func IoctlGetEthtoolTsInfo(fd int, ifname string) (*EthtoolTsInfo, error) {
+	ifr, err := NewIfreq(ifname)
+	if err != nil {
+		return nil, err
+	}
+
+	value := EthtoolTsInfo{Cmd: ETHTOOL_GET_TS_INFO}
+	ifrd := ifr.withData(unsafe.Pointer(&value))
+
+	err = ioctlIfreqData(fd, SIOCETHTOOL, &ifrd)
+	return &value, err
+}
+
+// IoctlGetHwTstamp retrieves the hardware timestamping configuration
+// for the network device specified by ifname.
+func IoctlGetHwTstamp(fd int, ifname string) (*HwTstampConfig, error) {
+	ifr, err := NewIfreq(ifname)
+	if err != nil {
+		return nil, err
+	}
+
+	value := HwTstampConfig{}
+	ifrd := ifr.withData(unsafe.Pointer(&value))
+
+	err = ioctlIfreqData(fd, SIOCGHWTSTAMP, &ifrd)
+	return &value, err
+}
+
+// IoctlSetHwTstamp updates the hardware timestamping configuration for
+// the network device specified by ifname.
+func IoctlSetHwTstamp(fd int, ifname string, cfg *HwTstampConfig) error {
+	ifr, err := NewIfreq(ifname)
+	if err != nil {
+		return err
+	}
+	ifrd := ifr.withData(unsafe.Pointer(cfg))
+	return ioctlIfreqData(fd, SIOCSHWTSTAMP, &ifrd)
+}
+
+// FdToClockID derives the clock ID from the file descriptor number
+// - see clock_gettime(3), FD_TO_CLOCKID macros. The resulting ID is
+// suitable for system calls like ClockGettime.
+func FdToClockID(fd int) int32 { return int32((int(^fd) << 3) | 3) }
+
+// IoctlPtpClockGetcaps returns the description of a given PTP device.
+func IoctlPtpClockGetcaps(fd int) (*PtpClockCaps, error) {
+	var value PtpClockCaps
+	err := ioctlPtr(fd, PTP_CLOCK_GETCAPS2, unsafe.Pointer(&value))
+	return &value, err
+}
+
+// IoctlPtpSysOffsetPrecise returns a description of the clock
+// offset compared to the system clock.
+func IoctlPtpSysOffsetPrecise(fd int) (*PtpSysOffsetPrecise, error) {
+	var value PtpSysOffsetPrecise
+	err := ioctlPtr(fd, PTP_SYS_OFFSET_PRECISE2, unsafe.Pointer(&value))
+	return &value, err
+}
+
+// IoctlPtpSysOffsetExtended returns an extended description of the
+// clock offset compared to the system clock. The samples parameter
+// specifies the desired number of measurements.
+func IoctlPtpSysOffsetExtended(fd int, samples uint) (*PtpSysOffsetExtended, error) {
+	value := PtpSysOffsetExtended{Samples: uint32(samples)}
+	err := ioctlPtr(fd, PTP_SYS_OFFSET_EXTENDED2, unsafe.Pointer(&value))
+	return &value, err
+}
+
+// IoctlPtpPinGetfunc returns the configuration of the specified
+// I/O pin on given PTP device.
+func IoctlPtpPinGetfunc(fd int, index uint) (*PtpPinDesc, error) {
+	value := PtpPinDesc{Index: uint32(index)}
+	err := ioctlPtr(fd, PTP_PIN_GETFUNC2, unsafe.Pointer(&value))
+	return &value, err
+}
+
+// IoctlPtpPinSetfunc updates configuration of the specified PTP
+// I/O pin.
+func IoctlPtpPinSetfunc(fd int, pd *PtpPinDesc) error {
+	return ioctlPtr(fd, PTP_PIN_SETFUNC2, unsafe.Pointer(pd))
+}
+
+// IoctlPtpPeroutRequest configures the periodic output mode of the
+// PTP I/O pins.
+func IoctlPtpPeroutRequest(fd int, r *PtpPeroutRequest) error {
+	return ioctlPtr(fd, PTP_PEROUT_REQUEST2, unsafe.Pointer(r))
+}
+
+// IoctlPtpExttsRequest configures the external timestamping mode
+// of the PTP I/O pins.
+func IoctlPtpExttsRequest(fd int, r *PtpExttsRequest) error {
+	return ioctlPtr(fd, PTP_EXTTS_REQUEST2, unsafe.Pointer(r))
+}
+
 // IoctlGetWatchdogInfo fetches information about a watchdog device from the
 // Linux watchdog API. For more information, see:
 // https://www.kernel.org/doc/html/latest/watchdog/watchdog-api.html.
diff --git a/source/vendor/golang.org/x/sys/unix/mkerrors.sh b/source/vendor/golang.org/x/sys/unix/mkerrors.sh
index ac54eca..6ab02b6 100644
--- a/source/vendor/golang.org/x/sys/unix/mkerrors.sh
+++ b/source/vendor/golang.org/x/sys/unix/mkerrors.sh
@@ -158,6 +158,16 @@ includes_Linux='
 #endif
 #define _GNU_SOURCE
 
+// See the description in unix/linux/types.go
+#if defined(__ARM_EABI__) || \
+	(defined(__mips__) && (_MIPS_SIM == _ABIO32)) || \
+	(defined(__powerpc__) && (!defined(__powerpc64__)))
+# ifdef   _TIME_BITS
+#  undef  _TIME_BITS
+# endif
+# define  _TIME_BITS 32
+#endif
+
 // <sys/ioctl.h> is broken on powerpc64, as it fails to include definitions of
 // these structures. We just include them copied from <bits/termios.h>.
 #if defined(__powerpc__)
@@ -256,6 +266,7 @@ struct ltchars {
 #include <linux/nsfs.h>
 #include <linux/perf_event.h>
 #include <linux/pps.h>
+#include <linux/ptp_clock.h>
 #include <linux/ptrace.h>
 #include <linux/random.h>
 #include <linux/reboot.h>
@@ -527,6 +538,7 @@ ccflags="$@"
 		$2 ~ /^(AF|SOCK|SO|SOL|IPPROTO|IP|IPV6|TCP|MCAST|EVFILT|NOTE|SHUT|PROT|MAP|MREMAP|MFD|T?PACKET|MSG|SCM|MCL|DT|MADV|PR|LOCAL|TCPOPT|UDP)_/ ||
 		$2 ~ /^NFC_(GENL|PROTO|COMM|RF|SE|DIRECTION|LLCP|SOCKPROTO)_/ ||
 		$2 ~ /^NFC_.*_(MAX)?SIZE$/ ||
+		$2 ~ /^PTP_/ ||
 		$2 ~ /^RAW_PAYLOAD_/ ||
 		$2 ~ /^[US]F_/ ||
 		$2 ~ /^TP_STATUS_/ ||
diff --git a/source/vendor/golang.org/x/sys/unix/syscall_linux.go b/source/vendor/golang.org/x/sys/unix/syscall_linux.go
index f08abd4..230a945 100644
--- a/source/vendor/golang.org/x/sys/unix/syscall_linux.go
+++ b/source/vendor/golang.org/x/sys/unix/syscall_linux.go
@@ -1860,6 +1860,7 @@ func Sendfile(outfd int, infd int, offset *int64, count int) (written int, err e
 //sys	ClockAdjtime(clockid int32, buf *Timex) (state int, err error)
 //sys	ClockGetres(clockid int32, res *Timespec) (err error)
 //sys	ClockGettime(clockid int32, time *Timespec) (err error)
+//sys	ClockSettime(clockid int32, time *Timespec) (err error)
 //sys	ClockNanosleep(clockid int32, flags int, request *Timespec, remain *Timespec) (err error)
 //sys	Close(fd int) (err error)
 //sys	CloseRange(first uint, last uint, flags uint) (err error)
diff --git a/source/vendor/golang.org/x/sys/unix/syscall_zos_s390x.go b/source/vendor/golang.org/x/sys/unix/syscall_zos_s390x.go
index 312ae6a..7bf5c04 100644
--- a/source/vendor/golang.org/x/sys/unix/syscall_zos_s390x.go
+++ b/source/vendor/golang.org/x/sys/unix/syscall_zos_s390x.go
@@ -768,6 +768,15 @@ func Munmap(b []byte) (err error) {
 	return mapper.Munmap(b)
 }
 
+func MmapPtr(fd int, offset int64, addr unsafe.Pointer, length uintptr, prot int, flags int) (ret unsafe.Pointer, err error) {
+	xaddr, err := mapper.mmap(uintptr(addr), length, prot, flags, fd, offset)
+	return unsafe.Pointer(xaddr), err
+}
+
+func MunmapPtr(addr unsafe.Pointer, length uintptr) (err error) {
+	return mapper.munmap(uintptr(addr), length)
+}
+
 //sys   Gethostname(buf []byte) (err error) = SYS___GETHOSTNAME_A
 //sysnb	Getgid() (gid int)
 //sysnb	Getpid() (pid int)
@@ -816,10 +825,10 @@ func Lstat(path string, stat *Stat_t) (err error) {
 // for checking symlinks begins with $VERSION/ $SYSNAME/ $SYSSYMR/ $SYSSYMA/
 func isSpecialPath(path []byte) (v bool) {
 	var special = [4][8]byte{
-		[8]byte{'V', 'E', 'R', 'S', 'I', 'O', 'N', '/'},
-		[8]byte{'S', 'Y', 'S', 'N', 'A', 'M', 'E', '/'},
-		[8]byte{'S', 'Y', 'S', 'S', 'Y', 'M', 'R', '/'},
-		[8]byte{'S', 'Y', 'S', 'S', 'Y', 'M', 'A', '/'}}
+		{'V', 'E', 'R', 'S', 'I', 'O', 'N', '/'},
+		{'S', 'Y', 'S', 'N', 'A', 'M', 'E', '/'},
+		{'S', 'Y', 'S', 'S', 'Y', 'M', 'R', '/'},
+		{'S', 'Y', 'S', 'S', 'Y', 'M', 'A', '/'}}
 
 	var i, j int
 	for i = 0; i < len(special); i++ {
@@ -3115,3 +3124,90 @@ func legacy_Mkfifoat(dirfd int, path string, mode uint32) (err error) {
 //sys	Posix_openpt(oflag int) (fd int, err error) = SYS_POSIX_OPENPT
 //sys	Grantpt(fildes int) (rc int, err error) = SYS_GRANTPT
 //sys	Unlockpt(fildes int) (rc int, err error) = SYS_UNLOCKPT
+
+func fcntlAsIs(fd uintptr, cmd int, arg uintptr) (val int, err error) {
+	runtime.EnterSyscall()
+	r0, e2, e1 := CallLeFuncWithErr(GetZosLibVec()+SYS_FCNTL<<4, uintptr(fd), uintptr(cmd), arg)
+	runtime.ExitSyscall()
+	val = int(r0)
+	if int64(r0) == -1 {
+		err = errnoErr2(e1, e2)
+	}
+	return
+}
+
+func Fcntl(fd uintptr, cmd int, op interface{}) (ret int, err error) {
+	switch op.(type) {
+	case *Flock_t:
+		err = FcntlFlock(fd, cmd, op.(*Flock_t))
+		if err != nil {
+			ret = -1
+		}
+		return
+	case int:
+		return FcntlInt(fd, cmd, op.(int))
+	case *F_cnvrt:
+		return fcntlAsIs(fd, cmd, uintptr(unsafe.Pointer(op.(*F_cnvrt))))
+	case unsafe.Pointer:
+		return fcntlAsIs(fd, cmd, uintptr(op.(unsafe.Pointer)))
+	default:
+		return -1, EINVAL
+	}
+	return
+}
+
+func Sendfile(outfd int, infd int, offset *int64, count int) (written int, err error) {
+	if raceenabled {
+		raceReleaseMerge(unsafe.Pointer(&ioSync))
+	}
+	return sendfile(outfd, infd, offset, count)
+}
+
+func sendfile(outfd int, infd int, offset *int64, count int) (written int, err error) {
+	// TODO: use LE call instead if the call is implemented
+	originalOffset, err := Seek(infd, 0, SEEK_CUR)
+	if err != nil {
+		return -1, err
+	}
+	//start reading data from in_fd
+	if offset != nil {
+		_, err := Seek(infd, *offset, SEEK_SET)
+		if err != nil {
+			return -1, err
+		}
+	}
+
+	buf := make([]byte, count)
+	readBuf := make([]byte, 0)
+	var n int = 0
+	for i := 0; i < count; i += n {
+		n, err := Read(infd, buf)
+		if n == 0 {
+			if err != nil {
+				return -1, err
+			} else { // EOF
+				break
+			}
+		}
+		readBuf = append(readBuf, buf...)
+		buf = buf[0:0]
+	}
+
+	n2, err := Write(outfd, readBuf)
+	if err != nil {
+		return -1, err
+	}
+
+	//When sendfile() returns, this variable will be set to the
+	// offset of the byte following the last byte that was read.
+	if offset != nil {
+		*offset = *offset + int64(n)
+		// If offset is not NULL, then sendfile() does not modify the file
+		// offset of in_fd
+		_, err := Seek(infd, originalOffset, SEEK_SET)
+		if err != nil {
+			return -1, err
+		}
+	}
+	return n2, nil
+}
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux.go
index de3b462..ccba391 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux.go
@@ -2625,6 +2625,28 @@ const (
 	PR_UNALIGN_NOPRINT                          = 0x1
 	PR_UNALIGN_SIGBUS                           = 0x2
 	PSTOREFS_MAGIC                              = 0x6165676c
+	PTP_CLK_MAGIC                               = '='
+	PTP_ENABLE_FEATURE                          = 0x1
+	PTP_EXTTS_EDGES                             = 0x6
+	PTP_EXTTS_EVENT_VALID                       = 0x1
+	PTP_EXTTS_V1_VALID_FLAGS                    = 0x7
+	PTP_EXTTS_VALID_FLAGS                       = 0x1f
+	PTP_EXT_OFFSET                              = 0x10
+	PTP_FALLING_EDGE                            = 0x4
+	PTP_MAX_SAMPLES                             = 0x19
+	PTP_PEROUT_DUTY_CYCLE                       = 0x2
+	PTP_PEROUT_ONE_SHOT                         = 0x1
+	PTP_PEROUT_PHASE                            = 0x4
+	PTP_PEROUT_V1_VALID_FLAGS                   = 0x0
+	PTP_PEROUT_VALID_FLAGS                      = 0x7
+	PTP_PIN_GETFUNC                             = 0xc0603d06
+	PTP_PIN_GETFUNC2                            = 0xc0603d0f
+	PTP_RISING_EDGE                             = 0x2
+	PTP_STRICT_FLAGS                            = 0x8
+	PTP_SYS_OFFSET_EXTENDED                     = 0xc4c03d09
+	PTP_SYS_OFFSET_EXTENDED2                    = 0xc4c03d12
+	PTP_SYS_OFFSET_PRECISE                      = 0xc0403d08
+	PTP_SYS_OFFSET_PRECISE2                     = 0xc0403d11
 	PTRACE_ATTACH                               = 0x10
 	PTRACE_CONT                                 = 0x7
 	PTRACE_DETACH                               = 0x11
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_386.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_386.go
index 8aa6d77..0c00cb3 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_386.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_386.go
@@ -237,6 +237,20 @@ const (
 	PPPIOCUNBRIDGECHAN               = 0x7434
 	PPPIOCXFERUNIT                   = 0x744e
 	PR_SET_PTRACER_ANY               = 0xffffffff
+	PTP_CLOCK_GETCAPS                = 0x80503d01
+	PTP_CLOCK_GETCAPS2               = 0x80503d0a
+	PTP_ENABLE_PPS                   = 0x40043d04
+	PTP_ENABLE_PPS2                  = 0x40043d0d
+	PTP_EXTTS_REQUEST                = 0x40103d02
+	PTP_EXTTS_REQUEST2               = 0x40103d0b
+	PTP_MASK_CLEAR_ALL               = 0x3d13
+	PTP_MASK_EN_SINGLE               = 0x40043d14
+	PTP_PEROUT_REQUEST               = 0x40383d03
+	PTP_PEROUT_REQUEST2              = 0x40383d0c
+	PTP_PIN_SETFUNC                  = 0x40603d07
+	PTP_PIN_SETFUNC2                 = 0x40603d10
+	PTP_SYS_OFFSET                   = 0x43403d05
+	PTP_SYS_OFFSET2                  = 0x43403d0e
 	PTRACE_GETFPREGS                 = 0xe
 	PTRACE_GETFPXREGS                = 0x12
 	PTRACE_GET_THREAD_AREA           = 0x19
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_amd64.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_amd64.go
index da428f4..dfb3645 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_amd64.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_amd64.go
@@ -237,6 +237,20 @@ const (
 	PPPIOCUNBRIDGECHAN               = 0x7434
 	PPPIOCXFERUNIT                   = 0x744e
 	PR_SET_PTRACER_ANY               = 0xffffffffffffffff
+	PTP_CLOCK_GETCAPS                = 0x80503d01
+	PTP_CLOCK_GETCAPS2               = 0x80503d0a
+	PTP_ENABLE_PPS                   = 0x40043d04
+	PTP_ENABLE_PPS2                  = 0x40043d0d
+	PTP_EXTTS_REQUEST                = 0x40103d02
+	PTP_EXTTS_REQUEST2               = 0x40103d0b
+	PTP_MASK_CLEAR_ALL               = 0x3d13
+	PTP_MASK_EN_SINGLE               = 0x40043d14
+	PTP_PEROUT_REQUEST               = 0x40383d03
+	PTP_PEROUT_REQUEST2              = 0x40383d0c
+	PTP_PIN_SETFUNC                  = 0x40603d07
+	PTP_PIN_SETFUNC2                 = 0x40603d10
+	PTP_SYS_OFFSET                   = 0x43403d05
+	PTP_SYS_OFFSET2                  = 0x43403d0e
 	PTRACE_ARCH_PRCTL                = 0x1e
 	PTRACE_GETFPREGS                 = 0xe
 	PTRACE_GETFPXREGS                = 0x12
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_arm.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_arm.go
index bf45bfe..d46dcf7 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_arm.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_arm.go
@@ -234,6 +234,20 @@ const (
 	PPPIOCUNBRIDGECHAN               = 0x7434
 	PPPIOCXFERUNIT                   = 0x744e
 	PR_SET_PTRACER_ANY               = 0xffffffff
+	PTP_CLOCK_GETCAPS                = 0x80503d01
+	PTP_CLOCK_GETCAPS2               = 0x80503d0a
+	PTP_ENABLE_PPS                   = 0x40043d04
+	PTP_ENABLE_PPS2                  = 0x40043d0d
+	PTP_EXTTS_REQUEST                = 0x40103d02
+	PTP_EXTTS_REQUEST2               = 0x40103d0b
+	PTP_MASK_CLEAR_ALL               = 0x3d13
+	PTP_MASK_EN_SINGLE               = 0x40043d14
+	PTP_PEROUT_REQUEST               = 0x40383d03
+	PTP_PEROUT_REQUEST2              = 0x40383d0c
+	PTP_PIN_SETFUNC                  = 0x40603d07
+	PTP_PIN_SETFUNC2                 = 0x40603d10
+	PTP_SYS_OFFSET                   = 0x43403d05
+	PTP_SYS_OFFSET2                  = 0x43403d0e
 	PTRACE_GETCRUNCHREGS             = 0x19
 	PTRACE_GETFDPIC                  = 0x1f
 	PTRACE_GETFDPIC_EXEC             = 0x0
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_arm64.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_arm64.go
index 71c6716..3af3248 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_arm64.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_arm64.go
@@ -240,6 +240,20 @@ const (
 	PROT_BTI                         = 0x10
 	PROT_MTE                         = 0x20
 	PR_SET_PTRACER_ANY               = 0xffffffffffffffff
+	PTP_CLOCK_GETCAPS                = 0x80503d01
+	PTP_CLOCK_GETCAPS2               = 0x80503d0a
+	PTP_ENABLE_PPS                   = 0x40043d04
+	PTP_ENABLE_PPS2                  = 0x40043d0d
+	PTP_EXTTS_REQUEST                = 0x40103d02
+	PTP_EXTTS_REQUEST2               = 0x40103d0b
+	PTP_MASK_CLEAR_ALL               = 0x3d13
+	PTP_MASK_EN_SINGLE               = 0x40043d14
+	PTP_PEROUT_REQUEST               = 0x40383d03
+	PTP_PEROUT_REQUEST2              = 0x40383d0c
+	PTP_PIN_SETFUNC                  = 0x40603d07
+	PTP_PIN_SETFUNC2                 = 0x40603d10
+	PTP_SYS_OFFSET                   = 0x43403d05
+	PTP_SYS_OFFSET2                  = 0x43403d0e
 	PTRACE_PEEKMTETAGS               = 0x21
 	PTRACE_POKEMTETAGS               = 0x22
 	PTRACE_SYSEMU                    = 0x1f
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_loong64.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_loong64.go
index 9476628..292bcf0 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_loong64.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_loong64.go
@@ -238,6 +238,20 @@ const (
 	PPPIOCUNBRIDGECHAN               = 0x7434
 	PPPIOCXFERUNIT                   = 0x744e
 	PR_SET_PTRACER_ANY               = 0xffffffffffffffff
+	PTP_CLOCK_GETCAPS                = 0x80503d01
+	PTP_CLOCK_GETCAPS2               = 0x80503d0a
+	PTP_ENABLE_PPS                   = 0x40043d04
+	PTP_ENABLE_PPS2                  = 0x40043d0d
+	PTP_EXTTS_REQUEST                = 0x40103d02
+	PTP_EXTTS_REQUEST2               = 0x40103d0b
+	PTP_MASK_CLEAR_ALL               = 0x3d13
+	PTP_MASK_EN_SINGLE               = 0x40043d14
+	PTP_PEROUT_REQUEST               = 0x40383d03
+	PTP_PEROUT_REQUEST2              = 0x40383d0c
+	PTP_PIN_SETFUNC                  = 0x40603d07
+	PTP_PIN_SETFUNC2                 = 0x40603d10
+	PTP_SYS_OFFSET                   = 0x43403d05
+	PTP_SYS_OFFSET2                  = 0x43403d0e
 	PTRACE_SYSEMU                    = 0x1f
 	PTRACE_SYSEMU_SINGLESTEP         = 0x20
 	RLIMIT_AS                        = 0x9
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_mips.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_mips.go
index b9e85f3..782b711 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_mips.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_mips.go
@@ -234,6 +234,20 @@ const (
 	PPPIOCUNBRIDGECHAN               = 0x20007434
 	PPPIOCXFERUNIT                   = 0x2000744e
 	PR_SET_PTRACER_ANY               = 0xffffffff
+	PTP_CLOCK_GETCAPS                = 0x40503d01
+	PTP_CLOCK_GETCAPS2               = 0x40503d0a
+	PTP_ENABLE_PPS                   = 0x80043d04
+	PTP_ENABLE_PPS2                  = 0x80043d0d
+	PTP_EXTTS_REQUEST                = 0x80103d02
+	PTP_EXTTS_REQUEST2               = 0x80103d0b
+	PTP_MASK_CLEAR_ALL               = 0x20003d13
+	PTP_MASK_EN_SINGLE               = 0x80043d14
+	PTP_PEROUT_REQUEST               = 0x80383d03
+	PTP_PEROUT_REQUEST2              = 0x80383d0c
+	PTP_PIN_SETFUNC                  = 0x80603d07
+	PTP_PIN_SETFUNC2                 = 0x80603d10
+	PTP_SYS_OFFSET                   = 0x83403d05
+	PTP_SYS_OFFSET2                  = 0x83403d0e
 	PTRACE_GETFPREGS                 = 0xe
 	PTRACE_GET_THREAD_AREA           = 0x19
 	PTRACE_GET_THREAD_AREA_3264      = 0xc4
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_mips64.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_mips64.go
index a48b68a..84973fd 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_mips64.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_mips64.go
@@ -234,6 +234,20 @@ const (
 	PPPIOCUNBRIDGECHAN               = 0x20007434
 	PPPIOCXFERUNIT                   = 0x2000744e
 	PR_SET_PTRACER_ANY               = 0xffffffffffffffff
+	PTP_CLOCK_GETCAPS                = 0x40503d01
+	PTP_CLOCK_GETCAPS2               = 0x40503d0a
+	PTP_ENABLE_PPS                   = 0x80043d04
+	PTP_ENABLE_PPS2                  = 0x80043d0d
+	PTP_EXTTS_REQUEST                = 0x80103d02
+	PTP_EXTTS_REQUEST2               = 0x80103d0b
+	PTP_MASK_CLEAR_ALL               = 0x20003d13
+	PTP_MASK_EN_SINGLE               = 0x80043d14
+	PTP_PEROUT_REQUEST               = 0x80383d03
+	PTP_PEROUT_REQUEST2              = 0x80383d0c
+	PTP_PIN_SETFUNC                  = 0x80603d07
+	PTP_PIN_SETFUNC2                 = 0x80603d10
+	PTP_SYS_OFFSET                   = 0x83403d05
+	PTP_SYS_OFFSET2                  = 0x83403d0e
 	PTRACE_GETFPREGS                 = 0xe
 	PTRACE_GET_THREAD_AREA           = 0x19
 	PTRACE_GET_THREAD_AREA_3264      = 0xc4
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_mips64le.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_mips64le.go
index ea00e85..6d9cbc3 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_mips64le.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_mips64le.go
@@ -234,6 +234,20 @@ const (
 	PPPIOCUNBRIDGECHAN               = 0x20007434
 	PPPIOCXFERUNIT                   = 0x2000744e
 	PR_SET_PTRACER_ANY               = 0xffffffffffffffff
+	PTP_CLOCK_GETCAPS                = 0x40503d01
+	PTP_CLOCK_GETCAPS2               = 0x40503d0a
+	PTP_ENABLE_PPS                   = 0x80043d04
+	PTP_ENABLE_PPS2                  = 0x80043d0d
+	PTP_EXTTS_REQUEST                = 0x80103d02
+	PTP_EXTTS_REQUEST2               = 0x80103d0b
+	PTP_MASK_CLEAR_ALL               = 0x20003d13
+	PTP_MASK_EN_SINGLE               = 0x80043d14
+	PTP_PEROUT_REQUEST               = 0x80383d03
+	PTP_PEROUT_REQUEST2              = 0x80383d0c
+	PTP_PIN_SETFUNC                  = 0x80603d07
+	PTP_PIN_SETFUNC2                 = 0x80603d10
+	PTP_SYS_OFFSET                   = 0x83403d05
+	PTP_SYS_OFFSET2                  = 0x83403d0e
 	PTRACE_GETFPREGS                 = 0xe
 	PTRACE_GET_THREAD_AREA           = 0x19
 	PTRACE_GET_THREAD_AREA_3264      = 0xc4
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_mipsle.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_mipsle.go
index 91c6468..5f9fedb 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_mipsle.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_mipsle.go
@@ -234,6 +234,20 @@ const (
 	PPPIOCUNBRIDGECHAN               = 0x20007434
 	PPPIOCXFERUNIT                   = 0x2000744e
 	PR_SET_PTRACER_ANY               = 0xffffffff
+	PTP_CLOCK_GETCAPS                = 0x40503d01
+	PTP_CLOCK_GETCAPS2               = 0x40503d0a
+	PTP_ENABLE_PPS                   = 0x80043d04
+	PTP_ENABLE_PPS2                  = 0x80043d0d
+	PTP_EXTTS_REQUEST                = 0x80103d02
+	PTP_EXTTS_REQUEST2               = 0x80103d0b
+	PTP_MASK_CLEAR_ALL               = 0x20003d13
+	PTP_MASK_EN_SINGLE               = 0x80043d14
+	PTP_PEROUT_REQUEST               = 0x80383d03
+	PTP_PEROUT_REQUEST2              = 0x80383d0c
+	PTP_PIN_SETFUNC                  = 0x80603d07
+	PTP_PIN_SETFUNC2                 = 0x80603d10
+	PTP_SYS_OFFSET                   = 0x83403d05
+	PTP_SYS_OFFSET2                  = 0x83403d0e
 	PTRACE_GETFPREGS                 = 0xe
 	PTRACE_GET_THREAD_AREA           = 0x19
 	PTRACE_GET_THREAD_AREA_3264      = 0xc4
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_ppc.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_ppc.go
index 8cbf38d..bb0026e 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_ppc.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_ppc.go
@@ -237,6 +237,20 @@ const (
 	PPPIOCXFERUNIT                   = 0x2000744e
 	PROT_SAO                         = 0x10
 	PR_SET_PTRACER_ANY               = 0xffffffff
+	PTP_CLOCK_GETCAPS                = 0x40503d01
+	PTP_CLOCK_GETCAPS2               = 0x40503d0a
+	PTP_ENABLE_PPS                   = 0x80043d04
+	PTP_ENABLE_PPS2                  = 0x80043d0d
+	PTP_EXTTS_REQUEST                = 0x80103d02
+	PTP_EXTTS_REQUEST2               = 0x80103d0b
+	PTP_MASK_CLEAR_ALL               = 0x20003d13
+	PTP_MASK_EN_SINGLE               = 0x80043d14
+	PTP_PEROUT_REQUEST               = 0x80383d03
+	PTP_PEROUT_REQUEST2              = 0x80383d0c
+	PTP_PIN_SETFUNC                  = 0x80603d07
+	PTP_PIN_SETFUNC2                 = 0x80603d10
+	PTP_SYS_OFFSET                   = 0x83403d05
+	PTP_SYS_OFFSET2                  = 0x83403d0e
 	PTRACE_GETEVRREGS                = 0x14
 	PTRACE_GETFPREGS                 = 0xe
 	PTRACE_GETREGS64                 = 0x16
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_ppc64.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_ppc64.go
index a2df734..46120db 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_ppc64.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_ppc64.go
@@ -237,6 +237,20 @@ const (
 	PPPIOCXFERUNIT                   = 0x2000744e
 	PROT_SAO                         = 0x10
 	PR_SET_PTRACER_ANY               = 0xffffffffffffffff
+	PTP_CLOCK_GETCAPS                = 0x40503d01
+	PTP_CLOCK_GETCAPS2               = 0x40503d0a
+	PTP_ENABLE_PPS                   = 0x80043d04
+	PTP_ENABLE_PPS2                  = 0x80043d0d
+	PTP_EXTTS_REQUEST                = 0x80103d02
+	PTP_EXTTS_REQUEST2               = 0x80103d0b
+	PTP_MASK_CLEAR_ALL               = 0x20003d13
+	PTP_MASK_EN_SINGLE               = 0x80043d14
+	PTP_PEROUT_REQUEST               = 0x80383d03
+	PTP_PEROUT_REQUEST2              = 0x80383d0c
+	PTP_PIN_SETFUNC                  = 0x80603d07
+	PTP_PIN_SETFUNC2                 = 0x80603d10
+	PTP_SYS_OFFSET                   = 0x83403d05
+	PTP_SYS_OFFSET2                  = 0x83403d0e
 	PTRACE_GETEVRREGS                = 0x14
 	PTRACE_GETFPREGS                 = 0xe
 	PTRACE_GETREGS64                 = 0x16
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_ppc64le.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_ppc64le.go
index 2479137..5c95163 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_ppc64le.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_ppc64le.go
@@ -237,6 +237,20 @@ const (
 	PPPIOCXFERUNIT                   = 0x2000744e
 	PROT_SAO                         = 0x10
 	PR_SET_PTRACER_ANY               = 0xffffffffffffffff
+	PTP_CLOCK_GETCAPS                = 0x40503d01
+	PTP_CLOCK_GETCAPS2               = 0x40503d0a
+	PTP_ENABLE_PPS                   = 0x80043d04
+	PTP_ENABLE_PPS2                  = 0x80043d0d
+	PTP_EXTTS_REQUEST                = 0x80103d02
+	PTP_EXTTS_REQUEST2               = 0x80103d0b
+	PTP_MASK_CLEAR_ALL               = 0x20003d13
+	PTP_MASK_EN_SINGLE               = 0x80043d14
+	PTP_PEROUT_REQUEST               = 0x80383d03
+	PTP_PEROUT_REQUEST2              = 0x80383d0c
+	PTP_PIN_SETFUNC                  = 0x80603d07
+	PTP_PIN_SETFUNC2                 = 0x80603d10
+	PTP_SYS_OFFSET                   = 0x83403d05
+	PTP_SYS_OFFSET2                  = 0x83403d0e
 	PTRACE_GETEVRREGS                = 0x14
 	PTRACE_GETFPREGS                 = 0xe
 	PTRACE_GETREGS64                 = 0x16
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_riscv64.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_riscv64.go
index d265f14..11a84d5 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_riscv64.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_riscv64.go
@@ -234,6 +234,20 @@ const (
 	PPPIOCUNBRIDGECHAN               = 0x7434
 	PPPIOCXFERUNIT                   = 0x744e
 	PR_SET_PTRACER_ANY               = 0xffffffffffffffff
+	PTP_CLOCK_GETCAPS                = 0x80503d01
+	PTP_CLOCK_GETCAPS2               = 0x80503d0a
+	PTP_ENABLE_PPS                   = 0x40043d04
+	PTP_ENABLE_PPS2                  = 0x40043d0d
+	PTP_EXTTS_REQUEST                = 0x40103d02
+	PTP_EXTTS_REQUEST2               = 0x40103d0b
+	PTP_MASK_CLEAR_ALL               = 0x3d13
+	PTP_MASK_EN_SINGLE               = 0x40043d14
+	PTP_PEROUT_REQUEST               = 0x40383d03
+	PTP_PEROUT_REQUEST2              = 0x40383d0c
+	PTP_PIN_SETFUNC                  = 0x40603d07
+	PTP_PIN_SETFUNC2                 = 0x40603d10
+	PTP_SYS_OFFSET                   = 0x43403d05
+	PTP_SYS_OFFSET2                  = 0x43403d0e
 	PTRACE_GETFDPIC                  = 0x21
 	PTRACE_GETFDPIC_EXEC             = 0x0
 	PTRACE_GETFDPIC_INTERP           = 0x1
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_s390x.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_s390x.go
index 3f2d644..f78c461 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_s390x.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_s390x.go
@@ -234,6 +234,20 @@ const (
 	PPPIOCUNBRIDGECHAN               = 0x7434
 	PPPIOCXFERUNIT                   = 0x744e
 	PR_SET_PTRACER_ANY               = 0xffffffffffffffff
+	PTP_CLOCK_GETCAPS                = 0x80503d01
+	PTP_CLOCK_GETCAPS2               = 0x80503d0a
+	PTP_ENABLE_PPS                   = 0x40043d04
+	PTP_ENABLE_PPS2                  = 0x40043d0d
+	PTP_EXTTS_REQUEST                = 0x40103d02
+	PTP_EXTTS_REQUEST2               = 0x40103d0b
+	PTP_MASK_CLEAR_ALL               = 0x3d13
+	PTP_MASK_EN_SINGLE               = 0x40043d14
+	PTP_PEROUT_REQUEST               = 0x40383d03
+	PTP_PEROUT_REQUEST2              = 0x40383d0c
+	PTP_PIN_SETFUNC                  = 0x40603d07
+	PTP_PIN_SETFUNC2                 = 0x40603d10
+	PTP_SYS_OFFSET                   = 0x43403d05
+	PTP_SYS_OFFSET2                  = 0x43403d0e
 	PTRACE_DISABLE_TE                = 0x5010
 	PTRACE_ENABLE_TE                 = 0x5009
 	PTRACE_GET_LAST_BREAK            = 0x5006
diff --git a/source/vendor/golang.org/x/sys/unix/zerrors_linux_sparc64.go b/source/vendor/golang.org/x/sys/unix/zerrors_linux_sparc64.go
index 5d8b727..aeb777c 100644
--- a/source/vendor/golang.org/x/sys/unix/zerrors_linux_sparc64.go
+++ b/source/vendor/golang.org/x/sys/unix/zerrors_linux_sparc64.go
@@ -239,6 +239,20 @@ const (
 	PPPIOCUNBRIDGECHAN               = 0x20007434
 	PPPIOCXFERUNIT                   = 0x2000744e
 	PR_SET_PTRACER_ANY               = 0xffffffffffffffff
+	PTP_CLOCK_GETCAPS                = 0x40503d01
+	PTP_CLOCK_GETCAPS2               = 0x40503d0a
+	PTP_ENABLE_PPS                   = 0x80043d04
+	PTP_ENABLE_PPS2                  = 0x80043d0d
+	PTP_EXTTS_REQUEST                = 0x80103d02
+	PTP_EXTTS_REQUEST2               = 0x80103d0b
+	PTP_MASK_CLEAR_ALL               = 0x20003d13
+	PTP_MASK_EN_SINGLE               = 0x80043d14
+	PTP_PEROUT_REQUEST               = 0x80383d03
+	PTP_PEROUT_REQUEST2              = 0x80383d0c
+	PTP_PIN_SETFUNC                  = 0x80603d07
+	PTP_PIN_SETFUNC2                 = 0x80603d10
+	PTP_SYS_OFFSET                   = 0x83403d05
+	PTP_SYS_OFFSET2                  = 0x83403d0e
 	PTRACE_GETFPAREGS                = 0x14
 	PTRACE_GETFPREGS                 = 0xe
 	PTRACE_GETFPREGS64               = 0x19
diff --git a/source/vendor/golang.org/x/sys/unix/zsyscall_linux.go b/source/vendor/golang.org/x/sys/unix/zsyscall_linux.go
index af30da5..5cc1e8e 100644
--- a/source/vendor/golang.org/x/sys/unix/zsyscall_linux.go
+++ b/source/vendor/golang.org/x/sys/unix/zsyscall_linux.go
@@ -592,6 +592,16 @@ func ClockGettime(clockid int32, time *Timespec) (err error) {
 
 // THIS FILE IS GENERATED BY THE COMMAND AT THE TOP; DO NOT EDIT
 
+func ClockSettime(clockid int32, time *Timespec) (err error) {
+	_, _, e1 := Syscall(SYS_CLOCK_SETTIME, uintptr(clockid), uintptr(unsafe.Pointer(time)), 0)
+	if e1 != 0 {
+		err = errnoErr(e1)
+	}
+	return
+}
+
+// THIS FILE IS GENERATED BY THE COMMAND AT THE TOP; DO NOT EDIT
+
 func ClockNanosleep(clockid int32, flags int, request *Timespec, remain *Timespec) (err error) {
 	_, _, e1 := Syscall6(SYS_CLOCK_NANOSLEEP, uintptr(clockid), uintptr(flags), uintptr(unsafe.Pointer(request)), uintptr(unsafe.Pointer(remain)), 0, 0)
 	if e1 != 0 {
diff --git a/source/vendor/golang.org/x/sys/unix/ztypes_linux.go b/source/vendor/golang.org/x/sys/unix/ztypes_linux.go
index 3a69e45..8daaf3f 100644
--- a/source/vendor/golang.org/x/sys/unix/ztypes_linux.go
+++ b/source/vendor/golang.org/x/sys/unix/ztypes_linux.go
@@ -1752,12 +1752,6 @@ const (
 	IFLA_IPVLAN_UNSPEC                         = 0x0
 	IFLA_IPVLAN_MODE                           = 0x1
 	IFLA_IPVLAN_FLAGS                          = 0x2
-	NETKIT_NEXT                                = -0x1
-	NETKIT_PASS                                = 0x0
-	NETKIT_DROP                                = 0x2
-	NETKIT_REDIRECT                            = 0x7
-	NETKIT_L2                                  = 0x0
-	NETKIT_L3                                  = 0x1
 	IFLA_NETKIT_UNSPEC                         = 0x0
 	IFLA_NETKIT_PEER_INFO                      = 0x1
 	IFLA_NETKIT_PRIMARY                        = 0x2
@@ -1796,6 +1790,7 @@ const (
 	IFLA_VXLAN_DF                              = 0x1d
 	IFLA_VXLAN_VNIFILTER                       = 0x1e
 	IFLA_VXLAN_LOCALBYPASS                     = 0x1f
+	IFLA_VXLAN_LABEL_POLICY                    = 0x20
 	IFLA_GENEVE_UNSPEC                         = 0x0
 	IFLA_GENEVE_ID                             = 0x1
 	IFLA_GENEVE_REMOTE                         = 0x2
@@ -1825,6 +1820,8 @@ const (
 	IFLA_GTP_ROLE                              = 0x4
 	IFLA_GTP_CREATE_SOCKETS                    = 0x5
 	IFLA_GTP_RESTART_COUNT                     = 0x6
+	IFLA_GTP_LOCAL                             = 0x7
+	IFLA_GTP_LOCAL6                            = 0x8
 	IFLA_BOND_UNSPEC                           = 0x0
 	IFLA_BOND_MODE                             = 0x1
 	IFLA_BOND_ACTIVE_SLAVE                     = 0x2
@@ -1857,6 +1854,7 @@ const (
 	IFLA_BOND_AD_LACP_ACTIVE                   = 0x1d
 	IFLA_BOND_MISSED_MAX                       = 0x1e
 	IFLA_BOND_NS_IP6_TARGET                    = 0x1f
+	IFLA_BOND_COUPLED_CONTROL                  = 0x20
 	IFLA_BOND_AD_INFO_UNSPEC                   = 0x0
 	IFLA_BOND_AD_INFO_AGGREGATOR               = 0x1
 	IFLA_BOND_AD_INFO_NUM_PORTS                = 0x2
@@ -1925,6 +1923,7 @@ const (
 	IFLA_HSR_SEQ_NR                            = 0x5
 	IFLA_HSR_VERSION                           = 0x6
 	IFLA_HSR_PROTOCOL                          = 0x7
+	IFLA_HSR_INTERLINK                         = 0x8
 	IFLA_STATS_UNSPEC                          = 0x0
 	IFLA_STATS_LINK_64                         = 0x1
 	IFLA_STATS_LINK_XSTATS                     = 0x2
@@ -1977,6 +1976,15 @@ const (
 	IFLA_DSA_MASTER                            = 0x1
 )
 
+const (
+	NETKIT_NEXT     = -0x1
+	NETKIT_PASS     = 0x0
+	NETKIT_DROP     = 0x2
+	NETKIT_REDIRECT = 0x7
+	NETKIT_L2       = 0x0
+	NETKIT_L3       = 0x1
+)
+
 const (
 	NF_INET_PRE_ROUTING  = 0x0
 	NF_INET_LOCAL_IN     = 0x1
@@ -4110,6 +4118,106 @@ type EthtoolDrvinfo struct {
 	Regdump_len  uint32
 }
 
+type EthtoolTsInfo struct {
+	Cmd             uint32
+	So_timestamping uint32
+	Phc_index       int32
+	Tx_types        uint32
+	Tx_reserved     [3]uint32
+	Rx_filters      uint32
+	Rx_reserved     [3]uint32
+}
+
+type HwTstampConfig struct {
+	Flags     int32
+	Tx_type   int32
+	Rx_filter int32
+}
+
+const (
+	HWTSTAMP_FILTER_NONE            = 0x0
+	HWTSTAMP_FILTER_ALL             = 0x1
+	HWTSTAMP_FILTER_SOME            = 0x2
+	HWTSTAMP_FILTER_PTP_V1_L4_EVENT = 0x3
+	HWTSTAMP_FILTER_PTP_V2_L4_EVENT = 0x6
+	HWTSTAMP_FILTER_PTP_V2_L2_EVENT = 0x9
+	HWTSTAMP_FILTER_PTP_V2_EVENT    = 0xc
+)
+
+const (
+	HWTSTAMP_TX_OFF          = 0x0
+	HWTSTAMP_TX_ON           = 0x1
+	HWTSTAMP_TX_ONESTEP_SYNC = 0x2
+)
+
+type (
+	PtpClockCaps struct {
+		Max_adj            int32
+		N_alarm            int32
+		N_ext_ts           int32
+		N_per_out          int32
+		Pps                int32
+		N_pins             int32
+		Cross_timestamping int32
+		Adjust_phase       int32
+		Max_phase_adj      int32
+		Rsv                [11]int32
+	}
+	PtpClockTime struct {
+		Sec      int64
+		Nsec     uint32
+		Reserved uint32
+	}
+	PtpExttsEvent struct {
+		T     PtpClockTime
+		Index uint32
+		Flags uint32
+		Rsv   [2]uint32
+	}
+	PtpExttsRequest struct {
+		Index uint32
+		Flags uint32
+		Rsv   [2]uint32
+	}
+	PtpPeroutRequest struct {
+		StartOrPhase PtpClockTime
+		Period       PtpClockTime
+		Index        uint32
+		Flags        uint32
+		On           PtpClockTime
+	}
+	PtpPinDesc struct {
+		Name  [64]byte
+		Index uint32
+		Func  uint32
+		Chan  uint32
+		Rsv   [5]uint32
+	}
+	PtpSysOffset struct {
+		Samples uint32
+		Rsv     [3]uint32
+		Ts      [51]PtpClockTime
+	}
+	PtpSysOffsetExtended struct {
+		Samples uint32
+		Rsv     [3]uint32
+		Ts      [25][3]PtpClockTime
+	}
+	PtpSysOffsetPrecise struct {
+		Device   PtpClockTime
+		Realtime PtpClockTime
+		Monoraw  PtpClockTime
+		Rsv      [4]uint32
+	}
+)
+
+const (
+	PTP_PF_NONE    = 0x0
+	PTP_PF_EXTTS   = 0x1
+	PTP_PF_PEROUT  = 0x2
+	PTP_PF_PHYSYNC = 0x3
+)
+
 type (
 	HIDRawReportDescriptor struct {
 		Size  uint32
diff --git a/source/vendor/golang.org/x/sys/unix/ztypes_zos_s390x.go b/source/vendor/golang.org/x/sys/unix/ztypes_zos_s390x.go
index d9a13af..2e5d5a4 100644
--- a/source/vendor/golang.org/x/sys/unix/ztypes_zos_s390x.go
+++ b/source/vendor/golang.org/x/sys/unix/ztypes_zos_s390x.go
@@ -377,6 +377,12 @@ type Flock_t struct {
 	Pid    int32
 }
 
+type F_cnvrt struct {
+	Cvtcmd int32
+	Pccsid int16
+	Fccsid int16
+}
+
 type Termios struct {
 	Cflag uint32
 	Iflag uint32
diff --git a/source/vendor/golang.org/x/sys/windows/syscall_windows.go b/source/vendor/golang.org/x/sys/windows/syscall_windows.go
index 5cee9a3..4510bfc 100644
--- a/source/vendor/golang.org/x/sys/windows/syscall_windows.go
+++ b/source/vendor/golang.org/x/sys/windows/syscall_windows.go
@@ -725,20 +725,12 @@ func DurationSinceBoot() time.Duration {
 }
 
 func Ftruncate(fd Handle, length int64) (err error) {
-	curoffset, e := Seek(fd, 0, 1)
-	if e != nil {
-		return e
-	}
-	defer Seek(fd, curoffset, 0)
-	_, e = Seek(fd, length, 0)
-	if e != nil {
-		return e
+	type _FILE_END_OF_FILE_INFO struct {
+		EndOfFile int64
 	}
-	e = SetEndOfFile(fd)
-	if e != nil {
-		return e
-	}
-	return nil
+	var info _FILE_END_OF_FILE_INFO
+	info.EndOfFile = length
+	return SetFileInformationByHandle(fd, FileEndOfFileInfo, (*byte)(unsafe.Pointer(&info)), uint32(unsafe.Sizeof(info)))
 }
 
 func Gettimeofday(tv *Timeval) (err error) {
@@ -894,6 +886,11 @@ const socket_error = uintptr(^uint32(0))
 //sys	GetACP() (acp uint32) = kernel32.GetACP
 //sys	MultiByteToWideChar(codePage uint32, dwFlags uint32, str *byte, nstr int32, wchar *uint16, nwchar int32) (nwrite int32, err error) = kernel32.MultiByteToWideChar
 //sys	getBestInterfaceEx(sockaddr unsafe.Pointer, pdwBestIfIndex *uint32) (errcode error) = iphlpapi.GetBestInterfaceEx
+//sys   GetIfEntry2Ex(level uint32, row *MibIfRow2) (errcode error) = iphlpapi.GetIfEntry2Ex
+//sys   GetUnicastIpAddressEntry(row *MibUnicastIpAddressRow) (errcode error) = iphlpapi.GetUnicastIpAddressEntry
+//sys   NotifyIpInterfaceChange(family uint16, callback uintptr, callerContext unsafe.Pointer, initialNotification bool, notificationHandle *Handle) (errcode error) = iphlpapi.NotifyIpInterfaceChange
+//sys   NotifyUnicastIpAddressChange(family uint16, callback uintptr, callerContext unsafe.Pointer, initialNotification bool, notificationHandle *Handle) (errcode error) = iphlpapi.NotifyUnicastIpAddressChange
+//sys   CancelMibChangeNotify2(notificationHandle Handle) (errcode error) = iphlpapi.CancelMibChangeNotify2
 
 // For testing: clients can set this flag to force
 // creation of IPv6 sockets to return EAFNOSUPPORT.
@@ -1685,13 +1682,16 @@ func (s NTStatus) Error() string {
 // do not use NTUnicodeString, and instead UTF16PtrFromString should be used for
 // the more common *uint16 string type.
 func NewNTUnicodeString(s string) (*NTUnicodeString, error) {
-	var u NTUnicodeString
-	s16, err := UTF16PtrFromString(s)
+	s16, err := UTF16FromString(s)
 	if err != nil {
 		return nil, err
 	}
-	RtlInitUnicodeString(&u, s16)
-	return &u, nil
+	n := uint16(len(s16) * 2)
+	return &NTUnicodeString{
+		Length:        n - 2, // subtract 2 bytes for the NULL terminator
+		MaximumLength: n,
+		Buffer:        &s16[0],
+	}, nil
 }
 
 // Slice returns a uint16 slice that aliases the data in the NTUnicodeString.
diff --git a/source/vendor/golang.org/x/sys/windows/types_windows.go b/source/vendor/golang.org/x/sys/windows/types_windows.go
index 7b97a15..51311e2 100644
--- a/source/vendor/golang.org/x/sys/windows/types_windows.go
+++ b/source/vendor/golang.org/x/sys/windows/types_windows.go
@@ -2203,6 +2203,132 @@ const (
 	IfOperStatusLowerLayerDown = 7
 )
 
+const (
+	IF_MAX_PHYS_ADDRESS_LENGTH = 32
+	IF_MAX_STRING_SIZE         = 256
+)
+
+// MIB_IF_ENTRY_LEVEL enumeration from netioapi.h or
+// https://learn.microsoft.com/en-us/windows/win32/api/netioapi/nf-netioapi-getifentry2ex.
+const (
+	MibIfEntryNormal                  = 0
+	MibIfEntryNormalWithoutStatistics = 2
+)
+
+// MIB_NOTIFICATION_TYPE enumeration from netioapi.h or
+// https://learn.microsoft.com/en-us/windows/win32/api/netioapi/ne-netioapi-mib_notification_type.
+const (
+	MibParameterNotification = 0
+	MibAddInstance           = 1
+	MibDeleteInstance        = 2
+	MibInitialNotification   = 3
+)
+
+// MibIfRow2 stores information about a particular interface. See
+// https://learn.microsoft.com/en-us/windows/win32/api/netioapi/ns-netioapi-mib_if_row2.
+type MibIfRow2 struct {
+	InterfaceLuid               uint64
+	InterfaceIndex              uint32
+	InterfaceGuid               GUID
+	Alias                       [IF_MAX_STRING_SIZE + 1]uint16
+	Description                 [IF_MAX_STRING_SIZE + 1]uint16
+	PhysicalAddressLength       uint32
+	PhysicalAddress             [IF_MAX_PHYS_ADDRESS_LENGTH]uint8
+	PermanentPhysicalAddress    [IF_MAX_PHYS_ADDRESS_LENGTH]uint8
+	Mtu                         uint32
+	Type                        uint32
+	TunnelType                  uint32
+	MediaType                   uint32
+	PhysicalMediumType          uint32
+	AccessType                  uint32
+	DirectionType               uint32
+	InterfaceAndOperStatusFlags uint8
+	OperStatus                  uint32
+	AdminStatus                 uint32
+	MediaConnectState           uint32
+	NetworkGuid                 GUID
+	ConnectionType              uint32
+	TransmitLinkSpeed           uint64
+	ReceiveLinkSpeed            uint64
+	InOctets                    uint64
+	InUcastPkts                 uint64
+	InNUcastPkts                uint64
+	InDiscards                  uint64
+	InErrors                    uint64
+	InUnknownProtos             uint64
+	InUcastOctets               uint64
+	InMulticastOctets           uint64
+	InBroadcastOctets           uint64
+	OutOctets                   uint64
+	OutUcastPkts                uint64
+	OutNUcastPkts               uint64
+	OutDiscards                 uint64
+	OutErrors                   uint64
+	OutUcastOctets              uint64
+	OutMulticastOctets          uint64
+	OutBroadcastOctets          uint64
+	OutQLen                     uint64
+}
+
+// MIB_UNICASTIPADDRESS_ROW stores information about a unicast IP address. See
+// https://learn.microsoft.com/en-us/windows/win32/api/netioapi/ns-netioapi-mib_unicastipaddress_row.
+type MibUnicastIpAddressRow struct {
+	Address            RawSockaddrInet6 // SOCKADDR_INET union
+	InterfaceLuid      uint64
+	InterfaceIndex     uint32
+	PrefixOrigin       uint32
+	SuffixOrigin       uint32
+	ValidLifetime      uint32
+	PreferredLifetime  uint32
+	OnLinkPrefixLength uint8
+	SkipAsSource       uint8
+	DadState           uint32
+	ScopeId            uint32
+	CreationTimeStamp  Filetime
+}
+
+const ScopeLevelCount = 16
+
+// MIB_IPINTERFACE_ROW stores interface management information for a particular IP address family on a network interface.
+// See https://learn.microsoft.com/en-us/windows/win32/api/netioapi/ns-netioapi-mib_ipinterface_row.
+type MibIpInterfaceRow struct {
+	Family                               uint16
+	InterfaceLuid                        uint64
+	InterfaceIndex                       uint32
+	MaxReassemblySize                    uint32
+	InterfaceIdentifier                  uint64
+	MinRouterAdvertisementInterval       uint32
+	MaxRouterAdvertisementInterval       uint32
+	AdvertisingEnabled                   uint8
+	ForwardingEnabled                    uint8
+	WeakHostSend                         uint8
+	WeakHostReceive                      uint8
+	UseAutomaticMetric                   uint8
+	UseNeighborUnreachabilityDetection   uint8
+	ManagedAddressConfigurationSupported uint8
+	OtherStatefulConfigurationSupported  uint8
+	AdvertiseDefaultRoute                uint8
+	RouterDiscoveryBehavior              uint32
+	DadTransmits                         uint32
+	BaseReachableTime                    uint32
+	RetransmitTime                       uint32
+	PathMtuDiscoveryTimeout              uint32
+	LinkLocalAddressBehavior             uint32
+	LinkLocalAddressTimeout              uint32
+	ZoneIndices                          [ScopeLevelCount]uint32
+	SitePrefixLength                     uint32
+	Metric                               uint32
+	NlMtu                                uint32
+	Connected                            uint8
+	SupportsWakeUpPatterns               uint8
+	SupportsNeighborDiscovery            uint8
+	SupportsRouterDiscovery              uint8
+	ReachableTime                        uint32
+	TransmitOffload                      uint32
+	ReceiveOffload                       uint32
+	DisableDefaultRoutes                 uint8
+}
+
 // Console related constants used for the mode parameter to SetConsoleMode. See
 // https://docs.microsoft.com/en-us/windows/console/setconsolemode for details.
 
diff --git a/source/vendor/golang.org/x/sys/windows/zsyscall_windows.go b/source/vendor/golang.org/x/sys/windows/zsyscall_windows.go
index 4c2e1bd..6f52528 100644
--- a/source/vendor/golang.org/x/sys/windows/zsyscall_windows.go
+++ b/source/vendor/golang.org/x/sys/windows/zsyscall_windows.go
@@ -181,10 +181,15 @@ var (
 	procDnsRecordListFree                                    = moddnsapi.NewProc("DnsRecordListFree")
 	procDwmGetWindowAttribute                                = moddwmapi.NewProc("DwmGetWindowAttribute")
 	procDwmSetWindowAttribute                                = moddwmapi.NewProc("DwmSetWindowAttribute")
+	procCancelMibChangeNotify2                               = modiphlpapi.NewProc("CancelMibChangeNotify2")
 	procGetAdaptersAddresses                                 = modiphlpapi.NewProc("GetAdaptersAddresses")
 	procGetAdaptersInfo                                      = modiphlpapi.NewProc("GetAdaptersInfo")
 	procGetBestInterfaceEx                                   = modiphlpapi.NewProc("GetBestInterfaceEx")
 	procGetIfEntry                                           = modiphlpapi.NewProc("GetIfEntry")
+	procGetIfEntry2Ex                                        = modiphlpapi.NewProc("GetIfEntry2Ex")
+	procGetUnicastIpAddressEntry                             = modiphlpapi.NewProc("GetUnicastIpAddressEntry")
+	procNotifyIpInterfaceChange                              = modiphlpapi.NewProc("NotifyIpInterfaceChange")
+	procNotifyUnicastIpAddressChange                         = modiphlpapi.NewProc("NotifyUnicastIpAddressChange")
 	procAddDllDirectory                                      = modkernel32.NewProc("AddDllDirectory")
 	procAssignProcessToJobObject                             = modkernel32.NewProc("AssignProcessToJobObject")
 	procCancelIo                                             = modkernel32.NewProc("CancelIo")
@@ -1606,6 +1611,14 @@ func DwmSetWindowAttribute(hwnd HWND, attribute uint32, value unsafe.Pointer, si
 	return
 }
 
+func CancelMibChangeNotify2(notificationHandle Handle) (errcode error) {
+	r0, _, _ := syscall.SyscallN(procCancelMibChangeNotify2.Addr(), uintptr(notificationHandle))
+	if r0 != 0 {
+		errcode = syscall.Errno(r0)
+	}
+	return
+}
+
 func GetAdaptersAddresses(family uint32, flags uint32, reserved uintptr, adapterAddresses *IpAdapterAddresses, sizePointer *uint32) (errcode error) {
 	r0, _, _ := syscall.Syscall6(procGetAdaptersAddresses.Addr(), 5, uintptr(family), uintptr(flags), uintptr(reserved), uintptr(unsafe.Pointer(adapterAddresses)), uintptr(unsafe.Pointer(sizePointer)), 0)
 	if r0 != 0 {
@@ -1638,6 +1651,46 @@ func GetIfEntry(pIfRow *MibIfRow) (errcode error) {
 	return
 }
 
+func GetIfEntry2Ex(level uint32, row *MibIfRow2) (errcode error) {
+	r0, _, _ := syscall.SyscallN(procGetIfEntry2Ex.Addr(), uintptr(level), uintptr(unsafe.Pointer(row)))
+	if r0 != 0 {
+		errcode = syscall.Errno(r0)
+	}
+	return
+}
+
+func GetUnicastIpAddressEntry(row *MibUnicastIpAddressRow) (errcode error) {
+	r0, _, _ := syscall.SyscallN(procGetUnicastIpAddressEntry.Addr(), uintptr(unsafe.Pointer(row)))
+	if r0 != 0 {
+		errcode = syscall.Errno(r0)
+	}
+	return
+}
+
+func NotifyIpInterfaceChange(family uint16, callback uintptr, callerContext unsafe.Pointer, initialNotification bool, notificationHandle *Handle) (errcode error) {
+	var _p0 uint32
+	if initialNotification {
+		_p0 = 1
+	}
+	r0, _, _ := syscall.SyscallN(procNotifyIpInterfaceChange.Addr(), uintptr(family), uintptr(callback), uintptr(callerContext), uintptr(_p0), uintptr(unsafe.Pointer(notificationHandle)))
+	if r0 != 0 {
+		errcode = syscall.Errno(r0)
+	}
+	return
+}
+
+func NotifyUnicastIpAddressChange(family uint16, callback uintptr, callerContext unsafe.Pointer, initialNotification bool, notificationHandle *Handle) (errcode error) {
+	var _p0 uint32
+	if initialNotification {
+		_p0 = 1
+	}
+	r0, _, _ := syscall.SyscallN(procNotifyUnicastIpAddressChange.Addr(), uintptr(family), uintptr(callback), uintptr(callerContext), uintptr(_p0), uintptr(unsafe.Pointer(notificationHandle)))
+	if r0 != 0 {
+		errcode = syscall.Errno(r0)
+	}
+	return
+}
+
 func AddDllDirectory(path *uint16) (cookie uintptr, err error) {
 	r0, _, e1 := syscall.Syscall(procAddDllDirectory.Addr(), 1, uintptr(unsafe.Pointer(path)), 0, 0)
 	cookie = uintptr(r0)
diff --git a/source/vendor/modules.txt b/source/vendor/modules.txt
index f3fa8cb..219eadc 100644
--- a/source/vendor/modules.txt
+++ b/source/vendor/modules.txt
@@ -14,6 +14,33 @@ github.com/charmbracelet/log
 ## explicit; go 1.18
 github.com/charmbracelet/x/ansi
 github.com/charmbracelet/x/ansi/parser
+# github.com/evanw/esbuild v0.24.0
+## explicit; go 1.13
+github.com/evanw/esbuild/internal/api_helpers
+github.com/evanw/esbuild/internal/ast
+github.com/evanw/esbuild/internal/bundler
+github.com/evanw/esbuild/internal/cache
+github.com/evanw/esbuild/internal/compat
+github.com/evanw/esbuild/internal/config
+github.com/evanw/esbuild/internal/css_ast
+github.com/evanw/esbuild/internal/css_lexer
+github.com/evanw/esbuild/internal/css_parser
+github.com/evanw/esbuild/internal/css_printer
+github.com/evanw/esbuild/internal/fs
+github.com/evanw/esbuild/internal/graph
+github.com/evanw/esbuild/internal/helpers
+github.com/evanw/esbuild/internal/js_ast
+github.com/evanw/esbuild/internal/js_lexer
+github.com/evanw/esbuild/internal/js_parser
+github.com/evanw/esbuild/internal/js_printer
+github.com/evanw/esbuild/internal/linker
+github.com/evanw/esbuild/internal/logger
+github.com/evanw/esbuild/internal/renamer
+github.com/evanw/esbuild/internal/resolver
+github.com/evanw/esbuild/internal/runtime
+github.com/evanw/esbuild/internal/sourcemap
+github.com/evanw/esbuild/internal/xxhash
+github.com/evanw/esbuild/pkg/api
 # github.com/go-logfmt/logfmt v0.6.0
 ## explicit; go 1.17
 github.com/go-logfmt/logfmt
@@ -42,37 +69,46 @@ github.com/tdewolff/parse/v2/css
 # github.com/volker-schukai/tokenizer v1.0.0
 ## explicit; go 1.13
 github.com/volker-schukai/tokenizer
+# gitlab.schukai.com/oss/libraries/go/application/configuration.git v1.22.9
+## explicit; go 1.22.0
 # gitlab.schukai.com/oss/libraries/go/application/xflags v1.16.3
 ## explicit; go 1.22
-gitlab.schukai.com/oss/libraries/go/application/xflags
+# gitlab.schukai.com/oss/libraries/go/application/xflags.git v1.16.5
+## explicit; go 1.22
+gitlab.schukai.com/oss/libraries/go/application/xflags.git
 # gitlab.schukai.com/oss/libraries/go/markup/html v0.4.6
 ## explicit; go 1.22
-gitlab.schukai.com/oss/libraries/go/markup/html/engine
+# gitlab.schukai.com/oss/libraries/go/markup/html.git v0.4.7
+## explicit; go 1.22
+gitlab.schukai.com/oss/libraries/go/markup/html.git/engine
+# gitlab.schukai.com/oss/libraries/go/services/job-queues.git v1.20.2
+## explicit; go 1.22
 # gitlab.schukai.com/oss/libraries/go/utilities/data.git v0.2.2
 ## explicit; go 1.22
 gitlab.schukai.com/oss/libraries/go/utilities/data.git
 # gitlab.schukai.com/oss/libraries/go/utilities/pathfinder v0.9.4
 ## explicit; go 1.21
-gitlab.schukai.com/oss/libraries/go/utilities/pathfinder
 # gitlab.schukai.com/oss/libraries/go/utilities/pathfinder.git v0.9.5
 ## explicit; go 1.21
 gitlab.schukai.com/oss/libraries/go/utilities/pathfinder.git
-# golang.org/x/crypto v0.28.0
+# gitlab.schukai.com/oss/libraries/go/utilities/watch.git v0.4.2
+## explicit; go 1.22
+# golang.org/x/crypto v0.29.0
 ## explicit; go 1.20
 golang.org/x/crypto/bcrypt
 golang.org/x/crypto/blowfish
-# golang.org/x/exp v0.0.0-20241009180824-f66d83c29e7c
+# golang.org/x/exp v0.0.0-20241108190413-2d47ceb2692f
 ## explicit; go 1.22.0
 golang.org/x/exp/constraints
 golang.org/x/exp/slices
 golang.org/x/exp/slog
 golang.org/x/exp/slog/internal
 golang.org/x/exp/slog/internal/buffer
-# golang.org/x/net v0.30.0
+# golang.org/x/net v0.31.0
 ## explicit; go 1.18
 golang.org/x/net/html
 golang.org/x/net/html/atom
-# golang.org/x/sys v0.26.0
+# golang.org/x/sys v0.27.0
 ## explicit; go 1.18
 golang.org/x/sys/unix
 golang.org/x/sys/windows
-- 
GitLab