Compiling and bundling JavaScript, the painless way
Michele Riva
Michele Riva
Senior Software Architect @NearForm
Google Developer Expert
Microsoft MVP
data:image/s3,"s3://crabby-images/eb9a7/eb9a7dfb70b1dbfd0eb1fcb625599f71411bd71e" alt=""
MicheleRivaCode
data:image/s3,"s3://crabby-images/32b1e/32b1e6dfe31b83c7819364bbd3f0e8619a15410c" alt=""
data:image/s3,"s3://crabby-images/8125d/8125d0495c3b5d1fbd2d2667ed3e4f58f82fe7a7" alt=""
data:image/s3,"s3://crabby-images/c79f7/c79f75469280d9dede5cc9013a57f837d6c3c9f8" alt=""
data:image/s3,"s3://crabby-images/5c0f6/5c0f6474c686755a326fffafeca48bbb901875f0" alt=""
Real-World Next.js
Build scalable, high performances and modern web applications using Next.js, the React framework for production
MicheleRivaCode
Compiling and bundling JavaScript
is (often) a pain*
*but it shouldn't
MicheleRivaCode
A bit of terminology: compiling
to change a computer program into a machine language
Cambridge Dictionary
MicheleRivaCode
A bit of terminology: compiling transpiling
to translate a source code into a different language source code
Me, myself
MicheleRivaCode
A bit of terminology: bundling
to encapsulate code and resources into a single executable file
Me, myself
MicheleRivaCode
Why do we want to transpile our code?
- To make it compatible with different platforms
- To write our scripts in different languages
- To adopt new language features
- To run large-scale refactors
MicheleRivaCode
data:image/s3,"s3://crabby-images/da6ac/da6ac4f808adfec49afd4839fa6cbfb13d41d548" alt=""
data:image/s3,"s3://crabby-images/ddfc5/ddfc503260dda59ef23c44183eddf7e370c01704" alt=""
MicheleRivaCode
data:image/s3,"s3://crabby-images/4db79/4db7999ab53ca118ef8b7b1ced8de5a4502ecce4" alt=""
Scala.js
https://www.scala-js.org
MicheleRivaCode
ReasonML
ReScript
data:image/s3,"s3://crabby-images/dbda6/dbda6606aafc93dcb0e98454e49950edde0eb6a8" alt=""
F# (via Fable)
Gleam
data:image/s3,"s3://crabby-images/da895/da89567359c06e51cead3991563167e35f8055aa" alt=""
Elm
data:image/s3,"s3://crabby-images/6cc7c/6cc7c0fed9f0187b12d4ac2dc128c8ac4177c58e" alt=""
Kotlin
Nim, Haxe, C/C++ (via Emscripten), ClojureScript, Dart, PureScript, Haskell (via GHCJS)
MicheleRivaCode
MicheleRivaCode
data:image/s3,"s3://crabby-images/0e93d/0e93d68e66f61c1280bb7af3346295c20920c5d4" alt=""
LLVM
MicheleRivaCode
function toUpper(x) {
return x.toUpperCase();
}
function addExclamationMarks(x) {
return x + '!!!'
}
function scream(input) {
return input
|> toUpper
|> addExclamationMarks
|> console.log
}
scream('Hello, Serbia');
// HELLO, SERBIA!!!
"use strict";
function toUpper(x) {
return x.toUpperCase();
}
function addExclamationMarks(x) {
return x + '!!!';
}
function scream(input) {
var _ref, _ref2, _input;
return _ref = (
_ref2 = (_input = input, toUpper(_input)),
addExclamationMarks(_ref2)
), console.log(_ref);
}
scream('Hello, Serbia');
// HELLO, SERBIA!!!
MicheleRivaCode
import { VFC } from 'react';
enum UserType {
ADMIN = "admin",
EDITOR = "editor",
USER = "user",
ANONYMOUS = "guest"
}
type MyProps = {
userType: UserType;
}
const MyComponent: VFC<MyProps> = ({ userType }) => {
return (
<div>
User is of type: {props.userType}
</div>
)
}
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
var UserType;
(function (UserType) {
UserType["ADMIN"] = "admin";
UserType["EDITOR"] = "editor";
UserType["USER"] = "user";
UserType["ANONYMOUS"] = "guest";
})(UserType || (UserType = {}));
const MyComponent = ({ userType }) => {
return /*#__PURE__*/ React.createElement(
"div",
null,
"User is of type: ",
props.userType
);
};
MicheleRivaCode
Why do we want to bundle our code?
- To create a single executable file
- To serve a single JS file over the net
MicheleRivaCode
data:image/s3,"s3://crabby-images/da6ac/da6ac4f808adfec49afd4839fa6cbfb13d41d548" alt=""
MicheleRivaCode
data:image/s3,"s3://crabby-images/54e6d/54e6d9093359bf8da299b3a98c8b0b414151008b" alt=""
ReadArticle.jsx
data:image/s3,"s3://crabby-images/54e6d/54e6d9093359bf8da299b3a98c8b0b414151008b" alt=""
NewArticle.jsx
data:image/s3,"s3://crabby-images/54e6d/54e6d9093359bf8da299b3a98c8b0b414151008b" alt=""
AuthorProfile.jsx
data:image/s3,"s3://crabby-images/54e6d/54e6d9093359bf8da299b3a98c8b0b414151008b" alt=""
BundledPage.js
MicheleRivaCode
in depths
transpiling
MicheleRivaCode
Language-to-language
ClojureScript
(defn simple-component []
[:div
[:p "I am a component!"]
[:p.someclass
"I have " [:strong "bold"]
[:span {:style {:color "red"}} " and red "] "text."]])
cljs.user.simple_component = (function cljs$user$simple_component(){
return new cljs.core.PersistentVector(null, 3, 5, cljs.core.PersistentVector.EMPTY_NODE,
[new cljs.core.Keyword(null,"div","div",(1057191632)),
new cljs.core.PersistentVector(null, 2, 5, cljs.core.PersistentVector.EMPTY_NODE,
[new cljs.core.Keyword(null,"p","p",(151049309)),"I am a component!"], null),
new cljs.core.PersistentVector(null, 5, 5, cljs.core.PersistentVector.EMPTY_NODE,
[new cljs.core.Keyword(null,"p.someclass","p.someclass",(-1904646929)),
"I have ",new cljs.core.PersistentVector(null, 2, 5, cljs.core.PersistentVector.EMPTY_NODE,
[new cljs.core.Keyword(null,"strong","strong",(269529000)),"bold"], null),
new cljs.core.PersistentVector(null, 3, 5, cljs.core.PersistentVector.EMPTY_NODE,
[new cljs.core.Keyword(null,"span","span",(1394872991)),
new cljs.core.PersistentArrayMap(null, 1, [new cljs.core.Keyword(null,"style","style",(-496642736)),
new cljs.core.PersistentArrayMap(null, 1, [
new cljs.core.Keyword(null,"color","color",(1011675173)),"red"]
, null)], null)," and red "], null),"text."], null)], null);
});
https://reagent-project.github.io
MicheleRivaCode
[@bs.config {jsx: 3}];
module Greeting = {
[@react.component]
let make = () => {
<button> {React.string("Hello!")} </button>
};
};
ReactDOMRe.renderToElementWithId(<Greeting />, "preview");
// Generated by BUCKLESCRIPT, PLEASE EDIT WITH CARE
'use strict';
var React = require("react");
var ReactDOMRe = require("./stdlib/reactDOMRe.js");
function _none_$Greeting(Props) {
return React.createElement("button", undefined, "Hello!");
}
var Greeting = {
make: _none_$Greeting
};
ReactDOMRe.renderToElementWithId(React.createElement(_none_$Greeting, { }), "preview");
exports.Greeting = Greeting;
/* Not a pure module */
https://reasonml.github.io
Language-to-language
ReasonML
MicheleRivaCode
Language-to-language
TypeScript
import { VFC } from 'react';
enum UserType {
ADMIN = "admin",
EDITOR = "editor",
USER = "user",
ANONYMOUS = "guest"
}
type MyProps = {
userType: UserType;
}
const MyComponent: VFC<MyProps> = ({ userType }) => {
return (
<div>
User is of type: {props.userType}
</div>
)
}
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
var UserType;
(function (UserType) {
UserType["ADMIN"] = "admin";
UserType["EDITOR"] = "editor";
UserType["USER"] = "user";
UserType["ANONYMOUS"] = "guest";
})(UserType || (UserType = {}));
const MyComponent = ({ userType }) => {
return /*#__PURE__*/ React.createElement(
"div",
null,
"User is of type: ",
props.userType
);
};
MicheleRivaCode
Every language has its own transpiler
TypeScript TSC
data:image/s3,"s3://crabby-images/63fd0/63fd009389b398a0bf869c00f9c2628ab650f9f4" alt=""
Fable
ClojureScript
BuckleScript
data:image/s3,"s3://crabby-images/5a661/5a661324e74a71da1ed72dc23470e548ae6f13b3" alt=""
JavaScript (Babel)
MicheleRivaCode
Problem #1
transpilation time
Really fast
Average
data:image/s3,"s3://crabby-images/63fd0/63fd009389b398a0bf869c00f9c2628ab650f9f4" alt=""
Fast
Slow on large codebases
Slow on large codebases
MicheleRivaCode
Problem #2
optimized output
Beautifully optimized
Awful
data:image/s3,"s3://crabby-images/63fd0/63fd009389b398a0bf869c00f9c2628ab650f9f4" alt=""
Beautifully optimized
Quite optimized
Well optimized*
MicheleRivaCode
Let's focus on the most popular ones
Quite slow, quite optimized
Quite fast, well optimized*
MicheleRivaCode
Bundling time grows
Quite slow, quite optimized
Quite fast, well optimized
MicheleRivaCode
MicheleRivaCode
How does a transpiler work?
MicheleRivaCode
Parsing
Transformation
Codegen
MicheleRivaCode
Parsing
Step 1
Tokenization
var foo = 10
var
foo
=
10
input
tokens
MicheleRivaCode
Parsing
Step 2
Syntactic Analysis
var foo = 10
=
/ \
/ \
var 10
|
foo
input
parse tree
(aka concrete syntax tree)
MicheleRivaCode
Parsing
Step 2
Syntactic Analysis
var foo = 10
=
/ \
/ \
var 10
|
foo
input
variableDeclaration
|
|
vairiableDeclarator
/ \
/ \
Identifier NumericLiteral
abstract syntax tree (AST)
MicheleRivaCode
MicheleRivaCode
{
"body":[
{
"type":"VariableDeclaration",
"declarations":[
{
"type":"VariableDeclarator",
"id":{
"type":"Identifier",
"name":"foo",
"loc":{
"identifierName":"foo"
}
},
"init":{
"type":"NumericLiteral",
"extra":{
"rawValue":10,
"raw":"10"
},
"value":10
}
}
],
"kind":"var"
}
]
}
MicheleRivaCode
data:image/s3,"s3://crabby-images/2a4ee/2a4ee77f22808cd0b77b8679a9d107b013f8fa35" alt=""
MicheleRivaCode
export default function () {
return {
visitor: {
VariableDeclaration(path) {
console.log({ VariableDeclaration: path.node });
// { type: "VariableDeclaration", kind: "var", ... }
},
Identifier(path) {
console.log({ Identifier: path.node });
// { type: "Identifier", name: "foo", ... }
},
NumericLiteral(path) {
console.log({ NumericLiteral: path.node });
// { type: "NumericLiteral", value: 10, ... }
}
}
};
}
Traversing the AST
Babel example implementing the visitor pattern
MicheleRivaCode
export default function() {
return {
visitor: {
VariableDeclaration(path) {
if (path.node.kind === "var") {
path.node.kind = "let"
}
}
}
};
}
Transforming
Babel example
MicheleRivaCode
Codegen
Babel example
input
output
var foo = 10
const bar = true
let foo = 10;
const bar = true;
MicheleRivaCode
ESLint
jscodeshift
Prettier
MicheleRivaCode
ESLint
export default function(context) {
return {
TemplateLiteral(node) {
context.report({
node,
message: 'Do not use template literals',
fix(fixer) {
if (node.expressions.length) {
// Can't auto-fix template literal with expressions
return;
}
return [
fixer.replaceTextRange([node.start, node.start + 1], '"'),
fixer.replaceTextRange([node.end - 1, node.end], '"'),
];
},
});
}
};
};
Example code taken from astexplorer.net
MicheleRivaCode
ESLint
const myString = `Hello, World!`;
const myString = "Hello, World!";
input
output
MicheleRivaCode
ttypescript (https://github.com/cevek/ttypescript)
export default function (program) {
const checker = program.getTypeChecker();
return (context) => {
return (sourceFile) => {
const visitor = (node) => {
// This branch evaluates '2 + 2' like expressions and replaces the node with the result (in this case '4')
if (ts.isBinaryExpression(node)) {
if (ts.isNumericLiteral(node.left) && ts.isNumericLiteral(node.right)) {
// We could parse `node.text` as a number, or we can use the typechecker to get type info for nodes
const lhs = checker.getTypeAtLocation(node.left);
const rhs = checker.getTypeAtLocation(node.right);
switch (node.operatorToken.kind) {
case ts.SyntaxKind.PlusToken:
return context.factory.createNumericLiteral(lhs.value + rhs.value);
}
}
}
//
if (ts.isIdentifier(node) && node.text === 'printTips' || node.text === 'tips') {
return context.factory.createIdentifier(node.text.split('').reverse().join(''));
}
return ts.visitEachChild(node, visitor, context);
};
return ts.visitNode(sourceFile, visitor);
};
};
};
Example code taken from https://github.com/cevek/ttypescript#program
MicheleRivaCode
ttypescript (https://github.com/cevek/ttypescript)
input
output
const mySum = 10 + 20;
const mySum = 30;
MicheleRivaCode
export const parser = "babel";
export default function transformer(file, api) {
const j = api.jscodeshift;
return j(file.source)
.find(j.Identifier)
.forEach((path) => {
if (j(path).get().value.name.includes("oldName")) {
j(path).replaceWith(
j.identifier(path.node.name.replace("oldName", "newName"))
);
}
})
.toSource();
}
jscodeshift
MicheleRivaCode
jscodeshift
const oldNameFactory = () => {/* */};
const newNameFactory = () => {/* */};
input
output
MicheleRivaCode
Desired output
input
output
import MyHeader from 'components/MyHeader';
export function MyApp(props) {
return (
<div>
<MyHeader {...props.headerProps} />
<p> Hello, {props.name}!</p>
</div>
)
}
export function MyApp(props) {
return (
<div>
<p> Hello, {props.name}!</p>
</div>
)
}
MicheleRivaCode
export const parser = 'babel'
export default function transformer(file, api) {
const j = api.jscodeshift;
const withoutElement = j(file.source)
.find(j.JSXElement)
.forEach(function (path) {
if (path.value.openingElement.name.name === "MyHeader") {
path.prune();
}
})
.toSource();
const withoutImport = j(withoutElement)
.find(j.ImportDefaultSpecifier)
.forEach(function (path) {
if (path.value.local.name === "MyHeader") {
path.parentPath.parentPath.prune();
}
})
.toSource();
return withoutImport;
};
MicheleRivaCode
const user = {
data: {
name: {
first: "Michele",
last: "Riva"
}
}
}
const middleName = user.data
&& user.data.name
&& user.data.name.middle
|| "No middle name";
console.log(middleName);
Node 12.x
No middle name
MicheleRivaCode
const user = {
data: {
name: {
first: "Michele",
last: "Riva"
}
}
}
const middleName = user.data?.name?.middle ?? "No middle name";
console.log(middleName);
Node 12.x
SyntaxError: Unexpected token '.'
MicheleRivaCode
const user = {
data: {
name: {
first: "Michele",
last: "Riva"
}
}
}
const middleName = user.data?.name?.middle ?? "No middle name";
console.log(middleName);
Node >14.x
"No middle name"
MicheleRivaCode
import get from 'lodash/get';
const user = {
data: {
name: {
first: "Michele",
last: "Riva"
}
}
}
const middleName = get(user, "data.name.middle", "No middle name");
console.log(middleName);
"No middle name"
MicheleRivaCode
import reverse from 'lodash/reverse';
import isString from 'lodash/isString';
import isFunction from 'lodash/isFunction';
import isNull from 'lodash/isNull';
import split from 'lodash/split';
import filter from 'lodash/filter';
import map from 'lodash/map';
import keys from 'lodash/keys';
// and so on...
MicheleRivaCode
import reverse from 'lodash/reverse';
// Array.reverse()
import isString from 'lodash/isString';
// typeof
import isFunction from 'lodash/isFunction';
// typeof
import isNull from 'lodash/isNull';
// typeof
import split from 'lodash/split';
// Array.split()
import filter from 'lodash/filter';
// Array.filter
import map from 'lodash/map';
// Array.map
import keys from 'lodash/keys';
// Object.keys
// and so on...
MicheleRivaCode
What if we have thousands of files to refactor?
data:image/s3,"s3://crabby-images/ca50a/ca50a801a741c8eac4bd8f22f01cd6de5ff5d514" alt=""
MicheleRivaCode
data:image/s3,"s3://crabby-images/97de9/97de939cb5c78a1dec7029f3366f18534ed50ce3" alt=""
MicheleRivaCode
data:image/s3,"s3://crabby-images/66a06/66a06413f887de16bb482cd84b9232926b8a21cb" alt=""
MicheleRivaCode
in depths
bundling
WebPack
data:image/s3,"s3://crabby-images/09641/096411d569dcc7c571295dbf5d63f91600a0606e" alt=""
Parcel
Rollup
MicheleRivaCode
Hard
Easier
Easiest
WebPack
data:image/s3,"s3://crabby-images/09641/096411d569dcc7c571295dbf5d63f91600a0606e" alt=""
Parcel
Rollup
MicheleRivaCode
Hard
Easier
Easiest
Slowest
Slower
Fast
WebPack
data:image/s3,"s3://crabby-images/09641/096411d569dcc7c571295dbf5d63f91600a0606e" alt=""
Parcel
Rollup
MicheleRivaCode
king of configurations
MicheleRivaCode
king of configurations
Is it still worth it?
MicheleRivaCode
data:image/s3,"s3://crabby-images/d4b05/d4b058d9baa9da6be753d7713d49119a1eb539f7" alt=""
ESBuild
data:image/s3,"s3://crabby-images/1f593/1f593de545b3655a8dc57f78700ad3671e24fd69" alt=""
SWC
Vite
data:image/s3,"s3://crabby-images/3608b/3608b0939c7237d08ae33e5615704850993f912f" alt=""
Snowpack
is there any better alternative?
(rest in pepperoni)
MicheleRivaCode
esbuild src/myEntry.js --bundle --sourcemap --minify --outfile=dist/mybundle.js
data:image/s3,"s3://crabby-images/d4b05/d4b058d9baa9da6be753d7713d49119a1eb539f7" alt=""
ESBuild
MicheleRivaCode
data:image/s3,"s3://crabby-images/1f593/1f593de545b3655a8dc57f78700ad3671e24fd69" alt=""
MicheleRivaCode
data:image/s3,"s3://crabby-images/39e08/39e08947c8bd3593f8b3a0fa554b52ced26164fd" alt=""
data:image/s3,"s3://crabby-images/88553/8855302ee0230e947688cedd38c7233fa1caa989" alt=""
ES2019
ES2020
data:image/s3,"s3://crabby-images/1f593/1f593de545b3655a8dc57f78700ad3671e24fd69" alt=""
{
"jsc": {
"parser": {
"syntax": "ecmascript",
"jsx": false,
"dynamicImport": false,
"privateMethod": false,
"functionBind": false,
"exportDefaultFrom": false,
"exportNamespaceFrom": false,
"decorators": false,
"decoratorsBeforeExport": false,
"topLevelAwait": false,
"importMeta": false
},
"transform": null,
"target": "es5",
"loose": false,
"externalHelpers": false,
// Requires v1.2.50 or upper and requires target to be es2016 or upper.
"keepClassNames": false
}
}
MicheleRivaCode
data:image/s3,"s3://crabby-images/1f593/1f593de545b3655a8dc57f78700ad3671e24fd69" alt=""
SWC can run in a browser thanks to WASM
MicheleRivaCode
Vite
MicheleRivaCode
Vite
MicheleRivaCode
Vite
data:image/s3,"s3://crabby-images/d4b05/d4b058d9baa9da6be753d7713d49119a1eb539f7" alt=""
MicheleRivaCode
// ESM
export default function greet() {
return "Hello Serbia!";
}
// ESM
import foo, { bar } from "foobar";
// CJS
module.exports = function greet() {
return "Hello Serbia!";
}
// CJS
const foo, { bar } = require("foobar");
MicheleRivaCode
Vite
MicheleRivaCode
data:image/s3,"s3://crabby-images/9246b/9246bc6be0829dd792f5b8244686fe26187b98e3" alt=""
MicheleRivaCode
rest in pepperoni
data:image/s3,"s3://crabby-images/3608b/3608b0939c7237d08ae33e5615704850993f912f" alt=""
Snowpack
MicheleRivaCode
data:image/s3,"s3://crabby-images/3608b/3608b0939c7237d08ae33e5615704850993f912f" alt=""
Snowpack
data:image/s3,"s3://crabby-images/d4b05/d4b058d9baa9da6be753d7713d49119a1eb539f7" alt=""
rest in pepperoni
MicheleRivaCode
data:image/s3,"s3://crabby-images/3608b/3608b0939c7237d08ae33e5615704850993f912f" alt=""
Snowpack
data:image/s3,"s3://crabby-images/d4b05/d4b058d9baa9da6be753d7713d49119a1eb539f7" alt=""
rest in pepperoni
MicheleRivaCode
MicheleRivaCode
/*
* Skypack CDN - canvas-confetti@1.4.0
*
* Learn more:
* 📙 Package Documentation: https://www.skypack.dev/view/canvas-confetti
* 📘 Skypack Documentation: https://www.skypack.dev/docs
*
* Pinned URL: (Optimized for Production)
* ▶️ Normal: https://cdn.skypack.dev/pin/canvas-confetti@v1.4.0-POmgSMO0U5q84otJfYlN/mode=imports/optimized/canvas-confetti.js
* ⏩ Minified: https://cdn.skypack.dev/pin/canvas-confetti@v1.4.0-POmgSMO0U5q84otJfYlN/mode=imports,min/optimized/canvas-confetti.js
*
*/
// Browser-Optimized Imports (Don't directly import the URLs below in your application!)
export * from '/-/canvas-confetti@v1.4.0-POmgSMO0U5q84otJfYlN/dist=es2020,mode=imports/optimized/canvas-confetti.js';
export {default} from '/-/canvas-confetti@v1.4.0-POmgSMO0U5q84otJfYlN/dist=es2020,mode=imports/optimized/canvas-confetti.js';
MicheleRivaCode
data:image/s3,"s3://crabby-images/63067/63067beb7e3018ef792ee0d858d6e033e402e313" alt=""
MicheleRivaCode
The future is no-bundle
MicheleRivaCode
The future is no-bundle*
*maybe
MicheleRivaCode
The future is bright
MicheleRivaCode
data:image/s3,"s3://crabby-images/dd571/dd571dc3ce045660727b1a1c7da1a5193e7e9c34" alt=""
https://kdy1.dev/posts/2022/1/tsc-go
MicheleRivaCode
MicheleRivaCode
data:image/s3,"s3://crabby-images/e975f/e975fc359894c4bffe2d440c5eca727c28e872b3" alt=""
MicheleRivaCode
@MicheleRiva
@MicheleRivaCode
/in/MicheleRiva95
www.micheleriva.dev
data:image/s3,"s3://crabby-images/5d3d4/5d3d4d0cde216cb157c4e24640e4de0bb1bff3ac" alt=""
Copy of Compiling and bundling JS, the painless way - 1hr
By Michele Riva
Copy of Compiling and bundling JS, the painless way - 1hr
- 583