@nartallax/imploder
v1.1.29
Published
A tool to work with Typescript projects.
Downloads
34
Readme
Imploder
This tool allows you to pack Typescript projects into launchable single-js-file bundles.
Disclaimer: this tool does not support, and is not aiming to support all possible ways of using Typescript. It does not support triple-slash directives, it does not support non-module code files and so on.
Usage
Install
First, you need to install peer dependencies for the tool:
npm install --save-dev typescript
npm install --save-dev tslib
Then you install the tool itself:
npm install --save-dev @nartallax/imploder
npm of version 7 and higher will install peer dependencies automatically.
Note that Typescript, as said earlier, is peer dependency for the tool. That means tool will use any Typescript version installed.
Examples of tool usage are available in tests directory; however, keep in mind that some of the options shown there is required just for testing and won't work well with your project.
Basic usage: Bundling
To get started, you need to modify your tsconfig.json.
Add block imploderConfig to your tsconfig.json like this:
{
"imploderConfig": {
"entryModule": "my_main_file.ts",
"entryFunction": "myEntryPoint",
"outFile": "js/bundle.js",
"target": "ES2018"
},
"compilerOptions": { ... }
}
In this block you can see that tool is pointed to .ts file (my_main_file.ts), which is exporting function myEntryPoint. This is how entry point is defined. When bundle produced by the tool runs, this function is invoked. The function should not expect any arguments (they won't be passed) and should not return any value (it will be ignored).
There is also outFile defined. This file will contain the bundle. A bundle contains code of modules of project you bundling as well as loader. A bundle could be launched as any javascript file, and will execute entrypoint of project it contains.
Defining target is not required, but strongly recommended. Default target is ES5. ES3 is not supported. Target constants are the same as in compilerOptions.
Some of compilerOptions won't be compatible with the tool. In this case, you will see error messages during tool launch.
After configuration block is added, it's time to build. Launch build:
./node_modules/.bin/imploder --tsconfig ./tsconfig.json
After build is finished, you should end up with either bundle in desired location, or with bunch of errors in tool output.
Note that entryFunction is not required; it's just a nice way to export actual entrypoint instead of writing launch code all over the module.
Everything that entryModule exports will be available as module product. That is, you can later do the following in some other file (assuming NodeJS):
const {myEntryPoint} = require("elsewhere/my_bundle.js");
myEntryPoint();
Watch mode
Watch mode is special mode supported by Typescript compiler. It allows to watch .ts files for changes and recompile them after changes immediately.
This mode is supported by the tool. To activate it, add following option to imploderConfig block:
"watchMode": true
Next time tool is launched, it will not stop after first build; instead, it will run indefinitely.
Note that not each build results in bundle file update. To actually update the bundle, you will need to trigger the bundling; it is done through HTTP service. To make tool create HTTP service, you can pass following options:
"httpPort": 7570,
"showErrorsOverHttp": true
Port number is arbitrary.
Now, after tool is restarted, when HTTP request is sent to http://localhost:7570/assemble_bundle , a bundle is assembled and put into designated file, as well as sent over HTTP as response. Build errors are also sent over HTTP, if any. In case of errors, bundle is not produced.
Alternatively you may call /assemble_bundle_errors_only endpoint, which will only output errors, but not bundle code.
Profiles
Now you may want to separate production and development options. The tool offers means to do this: profiles.
You may define profile like this (within imploderConfig block):
"profiles": {
"development": {
"watchMode": true,
"target": "ES2018"
},
"production": {
"target": "ES5"
}
}
After these changes are made, you may pass profile name on tool launch:
--profile development
Values from profile definition will override values from "base" profile, which is options in imploderConfig block. Note that if no profile name passed, just the values of the imploderConfig block will be used.
Minification
The tool is able to minify modules before putting them into bundle. To do this, add following option to config:
"minify": true
Terser minifier is used.
If you need, you may pass overrides to terser:
"minificationOverrides": {
"collapse_vars": false
}
Some overrides could break the tool.
Transformers
Example of transformer usage
Example of transformer definition, another example of transformer definition
The tool is able to apply user-defined source code transformations at compilation. Transformer could alter resulting code in arbitrary way, as well as generate new files.
Transformers are referenced in plugins property. This property could appear within compilerOptions, imploderConfig and profiles:
{
"imploderConfig": {
...
"plugins": [{
"transform": "../some_great_logging_transformer/tsconfig.json",
"loggingFunctionName": "logText2",
"imploderProject": true,
"type": "imploder"
}],
"profiles": {
"test": {
"plugins": [{
"transform": "@nartallax/clamsensor",
"type": "imploder"
}]
}
}
},
"compilerOptions": {
...
"plugins": [
{
"transform": "../list_all_classes/main.js",
"type": "imploder"
}
]
...
}
}
Default order of transformers is: compilerOptions, then base profile, then target profile (if --profile is passed). It could be overriden, see below.
The interface of plugin object mostly resembles ttypescript's transformer definitions and should be compatible with them. However, Imploder offers some more options:
- imploderProject - if this property is true, transform property is treated as path to tsconfig.json of some other Imploder project. This project will be built, and then build result will be used as transformer.
- type: "imploder" - this is special type of transformer added by Imploder. Transformers with this type will receive Imploder.Context on creation. Also with this type you could return not just transformer factory, but promise of transformer factory.
- transformerExecutionOrder - a number that determines order of execution of transformers. Transformers with lower value of the property will get executed first. Transformers without such property get executed last (i.e. default property value is Number.MAX_SAFE_INTEGER). Transformers with equal order values are executed in appearance order.
Other options
TSLib embedding
By default, the tool embeds tslib as one of the modules if any other module requires it. You can prevent it with following option:
"embedTslib": false
Note that path to tslib.js is resolved relative to tsconfig.json directory path by Node default package search rules.
Module blacklisting/whitelisting
You could blacklist/whitelist some modules with following tsconfig.json options:
"moduleBlacklistRegexp": ["^/bad_modules/.*?$"],
"moduleWhitelistRegexp": ["^/good_modules/.*?$"]
If bundler detects that bundle includes module which name is matches any blacklist entry, bundling is failed.
If whitelist regexps are defined and non-empty, then names of all modules included in bundle must match at least one of the regexps, otherwise bundling is failed.
Use-case of this option is sanity checks (to prevent accidently bundling a ton of code you won't actually use), or to keep some secret server-side logic away from client-side bundle.
Prevent module tree pruning
By default, the tool won't include in bundle modules that are not referenced (directly or indirectly) from the entrypoint module.
To actually include all of the compiled modules (not really, see below), you can set the following option:
"preventModuleTreePruning": true
Note that enabling this option won't include any blacklisted or not whitelisted modules. The only way how whitelist/blacklist check can be triggered is for module to be referenced from entrypoint.
Output directory deletion
By default, output directory is deleted on tool start. This is done to ensure build consistency (there are bugs related to changing "target" value without purging old js files, resulting in inconsistency of ES version within single bundle).
You can disable such behavior with following option (which is not recommended):
"preserveOutDir": true
Lazy start
By default, in watchmode compiler will start as soon as the tool is launched. You can prevent this passing following option:
"lazyStart": true
With this option, compiler will be started after first request to HTTP server.
This option is mainly intended for case when you have a multitude of (frontend) projects, but do not want to load them all at once.
Better circular links resolution
The tool provides its own module loader. One of reasons behind this is need for better circular dependencies resolution between modules.
By default (that is, in requirejs loader) if there is circular dependency between modules, one of the modules receives empty module object (which is filled with values later), which sometimes could lead to runtime errors if said module tries to use any value from that module object at definition time (for example, during creation of subclass).
The loader of the tool have knowledge about names of values each module exports, and this allows the loader to actually run definition of module only when module value is actually used.
Note that there is still possible to generate circular dependency link that will not possible to be resolved, and this will fail at runtime; the loader could only resolve circular dependencies that are actually resolvable.
Compatibility with other loaders
Bundles could be used as CommonJS modules, as well as RequireJS (AMD) modules.
Note that if RequireJS is present and entryFunction is passed to the tool, bundle won't be launched on its own, as RequireJS provides no means of doing so. entryFunction will be called when the module is required.
Compiling the tool
As a user of the tool you will probably never need this, but I'll write this just in case:
npm install
npm run compile
npm run test
Running tests is optional, but you definitely should do this.
To run compile script, you will need bash shell and bunch of utilities. It runs successfully on Debian 9, probably will run fine on most Linux distributions, and maybe will run on bash on Windows (like gitbash) - not tested and not explicitly supported.
Naming
This tools grabs a lot of redundant source files and squeezes them into single compact file. This is comparable to implosion process; thus the name.
TODO
These features probably will be implemented at some point; just not yet.
Decide about asynchronous module loading and separation of project into several bundles
Support for modules in C - asmjs/wasm
Test for not enough file watchers to properly watch all the files (it could lead to interesting results in projects with file-generating transformers, even in single-build launch)