Pedro Cattori · @pcattori
data:image/s3,"s3://crabby-images/f3f24/f3f2492cd9ce45dbfaaadc67d9b36ab6b3f9846a" alt=""
data:image/s3,"s3://crabby-images/163c9/163c950042954d44326babcb4c89da0795349496" alt=""
data:image/s3,"s3://crabby-images/3e613/3e6133990445fe14b150d292a3ba64ca1a16ecc8" alt=""
Legendary DX
The Quest for
data:image/s3,"s3://crabby-images/15320/15320b805d5de9afd9d9a56fa989cf5091705d92" alt=""
Part I
<Link to="the past"/>
1: File Server 🗡️
🤩
Browser
📚
html, css, js
🫥
📚
🫥
📚
📚
🫥
📚
📚
🤩
📚
2: Build + Watch 🗡️
📁
Source code
ESM
📁
Source code
ESM
📁
Source code
🤓
Compiler
🫥
🤓
📁
🤓
📁
🫥
📚
🫥
📚
🤓
📁
📚
🫥
📚
📚
🤓
📁
🤩
📚
🤓
📁
🤩
📚
✍️
🤓
📁
👀
📁
🤩
📚
✍️
✍️
🤓
📁
🤩
📚
✍️
✍️
🤓
📁
🤩
📚
✍️
✍️
data:image/s3,"s3://crabby-images/5b4a7/5b4a7222a5079017969bf8f6a867656a1f52ab06" alt=""
3: Live Reload 🗡️
🤩
📚
✍️
🤓
📁
👀
📁
🤩
📚
✍️
✍️
🤓
📁
🤩
📚
✍️
✍️
🔁
🤩
📚
🤓
📁
🤓
📁
🔁
✍️
✍️
🔁
📚
🤓
📁
🤓
📁
✍️
✍️
🫥
📚
🤓
📁
🤓
📁
📚
✍️
✍️
🫥
📚
🤓
📁
🤓
📁
📚
🤩
📚
🤓
📁
🤓
📁
✍️
data:image/s3,"s3://crabby-images/8c405/8c4051464afc470c3134ff2fa9fddc526743a969" alt=""
data:image/s3,"s3://crabby-images/28cca/28ccaea57cf12e474fed4efd8873ceabbe21bcb5" alt=""
4: HMR 🗡️
data:image/s3,"s3://crabby-images/5af20/5af207b5a1cbaaf4c4e3ad49b0e1100cb19361c7" alt=""
data:image/s3,"s3://crabby-images/cd169/cd169eee049f84b83bacf68fb4d2ed99026148a2" alt=""
data:image/s3,"s3://crabby-images/7d2a1/7d2a104f6b77891ff1bceb41615446ed629a53ea" alt=""
data:image/s3,"s3://crabby-images/82652/826522d95f0ea5aca3e9b93f004b5a74e8a4fed6" alt=""
data:image/s3,"s3://crabby-images/44e3c/44e3c5645b1d7ddf35be9fb521c933271f74a60d" alt=""
🤠
hot-swap.js
🤠
hot-swap.js
(aka HMR runtime)
🫥
🤓
📁
🤓
📁
🫥
📚
🤠
🫥
📚
🤓
📁
📚
🤠
🤠
🫥
📚
📚
🤓
📁
🤠
🤠
🤩
📚
🤓
📁
🤠
🤠
🤩
📚
✍️
🤓
📁
🤠
🤠
👀
📁
🤩
📚
✍️
✍️
🤠
🤠
🤓
📁
🤩
📚
✍️
✍️
✍️
🤠
🤠
🤩
📚
🤓
📁
🤓
📁
✍️
✍️
✍️
🤠
🤠
🤩
📚
🤓
📁
🤓
📁
✍️
✍️
✍️
🔁
🤠
🤩
📚
🤓
📁
🤓
📁
✍️
✍️
✍️
🤠
🤠
DEMO #1
Tears of the Server
Part II
😎
App Server
📙
server.js
🤓
🫥
📁
🤓
🫥
📁
📚
📙
🤠
🤓
🫥
📁
📚
📙
🤠
🏃♂️
📙
🤓
🫥
📁
📚
📙
🤠
🏃♂️
📙
🤓
🫥
📁
📚
📙
🤠
😎
localhost:3000
🤓
🫥
📁
📚
📙
🤠
😎
🤓
🫥
📁
📚
📙
🤠
😎
📚
🤠
🤓
🫥
📁
📚
📙
🤠
😎
📚
🤠
{"place": "world"}
🤓
🫥
📁
📚
📙
🤠
😎
📚
🤠
{"place": "world"}
🤓
🤩
📁
📚
📙
🤠
😎
🤠
🌇
{"place": "world"}
import { json } from "@remix-run/node"
import { useLoaderData } from "@remix-run/react"
export let loader = () => {
return json({ place: "world" })
}
export default Component() {
let { place } = useLoaderData<typeof loader>()
return (
<main>
<h1>Hey there, Remix Conf 👋</h1>
<h2>And hello, {place}!</h2>
</main>
)
}
✍️
import { json } from "@remix-run/node"
import { useLoaderData } from "@remix-run/react"
export let loader = () => {
return json({ place: "world" })
}
export default Component() {
let { place } = useLoaderData<typeof loader>()
return (
<main>
<h1>Hey there, friends 👋</h1>
<h2>And hello, {place}!</h2>
</main>
)
}
✍️
import { json } from "@remix-run/node"
import { useLoaderData } from "@remix-run/react"
export let loader = () => {
return json({ location: "Salt Lake City" })
}
export default Component() {
let { place } = useLoaderData<typeof loader>()
return (
<main>
<h1>Hey there, friends 👋</h1>
<h2>And hello, {place}!</h2>
</main>
)
}
✍️
import { json } from "@remix-run/node"
import { useLoaderData } from "@remix-run/react"
export let loader = () => {
return json({ location: "Salt Lake City" })
}
export default Component() {
let { location } = useLoaderData<typeof loader>()
return (
<main>
<h1>Hey there, friends 👋</h1>
<h2>And hello, {location}!</h2>
</main>
)
}
✍️
SAVE
✍️
import { json } from "@remix-run/node"
import { useLoaderData } from "@remix-run/react"
export let loader = () => {
return json({ location: "Salt Lake City" })
}
export default Component() {
let { location } = useLoaderData<typeof loader>()
return (
<main>
<h1>Hey there, friends 👋</h1>
<h2>And hello, {location}!</h2>
</main>
)
}
🤓
🤩
📁
📚
📙
🤠
😎
🤠
🌇
✍️
{"place": "world"}
👀
🤩
📁
📚
📙
🤠
😎
🤠
🌇
✍️
✍️
✍️
{"place": "world"}
🤓
🤩
📁
📚
📙
🤠
😎
🤠
🌇
✍️
✍️
✍️
✍️
{"place": "world"}
🤓
🤩
📁
📚
📙
🤠
😎
🤠
🌇
✍️
✍️
✍️
✍️
{"place": "world"}
🤓
🤩
📁
📚
📙
🤠
😎
🔁
🌇
✍️
✍️
✍️
✍️
{"place": "world"}
🤓
🤯
📁
📚
📙
🤠
😎
🤠
✍️
✍️
✍️
✍️
{"place": "world"}
let { location } =
Demo #2
🤓
😎
🥸
?
🥸
!=
😎
🥸
!=
😎
🐞
🐛
🥸
!=
😎
🐞
🐛
🐰
🐆
🥸
!=
😎
❌
🥸
~=
😎
🥸
~=
😎
data:image/s3,"s3://crabby-images/f44e1/f44e103ebe3f4aec50fd81bd9c32ce93a94c60d2" alt=""
data:image/s3,"s3://crabby-images/76ce3/76ce375fe0aa6a3ef7687cbd30b109646cd1d258" alt=""
data:image/s3,"s3://crabby-images/a14f2/a14f201dd8d9c7a894446b4ae4378f154421bbb8" alt=""
🥸
~=
😎
data:image/s3,"s3://crabby-images/f44e1/f44e103ebe3f4aec50fd81bd9c32ce93a94c60d2" alt=""
data:image/s3,"s3://crabby-images/76ce3/76ce375fe0aa6a3ef7687cbd30b109646cd1d258" alt=""
data:image/s3,"s3://crabby-images/a14f2/a14f201dd8d9c7a894446b4ae4378f154421bbb8" alt=""
❌
💡
💡
🤓
👂
📣
😎
"Ready!"
Part III
A between Worlds
fetch
🤓
🤩
📁
😎
🤓
🤩
📁
😎
💪
🤓
🫥
📁
📚
📙
🤠
🤓
🫥
📁
📚
📙
🤠
📙
🏃♂️
🤓
🫥
📁
📚
📙
🤠
📙
🏃♂️
🤓
🫥
📁
📚
📙
🤠
😎
🤠
📚
🤓
🫥
📁
📚
📙
🤠
😎
🤠
📚
🤓
🫥
📁
📚
📙
🤠
😎
🤠
📚
📚
🤓
🫥
📁
📚
📙
🤠
😎
🤠
📚
📚
🤓
🤩
📁
📚
📙
🤠
😎
🤠
📚
🤓
🤩
📁
📚
📙
🤠
😎
🤠
📚
✍️
👀
🤩
📁
📚
📙
🤠
😎
🤠
📚
✍️
✍️
✍️
🤓
🤩
📁
📚
📙
🤠
😎
🤠
📚
✍️
✍️
✍️
🤓
🤩
📁
📚
📙
🤠
😎
🤠
📚
✍️
✍️
✍️
⏳
👂
🤓
🤩
📁
📚
📙
🤠
👀
🤠
📚
✍️
✍️
✍️
⏳
👂
🤓
🤩
📁
📚
📙
🤠
💀
🤠
📚
✍️
✍️
✍️
⏳
👂
🤓
🤩
📁
📚
🤠
🏃♂️
🤠
📚
✍️
✍️
⏳
👂
✍️
📙
📙
✍️
🤓
🤩
📁
📚
🤠
😎
🤠
📚
✍️
✍️
⏳
👂
✍️
📙
✍️
✅
🤓
🤩
📁
📚
🤠
😎
🤠
📚
✍️
✍️
⏳
👂
✍️
📙
✍️
✅
🤓
🤩
📁
📚
🤠
😎
🤠
📚
✍️
✍️
✍️
📙
✍️
✍️
🤓
🤩
📁
📚
🤠
😎
🤠
📚
✍️
✍️
✍️
📙
✍️
✍️
🤓
🤩
📁
📚
🤠
😎
⏳
📚
✍️
✍️
✍️
📙
✍️
✍️
{"location": "world"}
🤓
🤩
📁
📚
🤠
😎
🔁
📚
✍️
✍️
✍️
📙
✍️
{"location": "world"}
✍️
🤓
🤩
📁
📚
🤠
😎
🔁
📚
✍️
✍️
✍️
📙
✍️
{"location": "world"}
✍️
🤓
🤩
📁
📚
🤠
😎
🤠
📚
✍️
✍️
✍️
📙
✍️
✍️
let { location } =
{"location": "world"}
Hot Data Revalidation
🔥
🔥
😎
💀
🏃♂️
😎
✍️
😎
💀
🏃♂️
😎
✍️
downtime
😎
💀
🏃♂️
😎
✍️
downtime
👎
Part IV
Ocarina of Downtime
😎
😎
✍️
💀
🏃♂️
🤠
hot-swap-server.js
😎
🤠
📙
😎
🤠
📙
✍️
😎
👀
📙
✍️
✍️
😎
🤠
📙
✍️
✍️
😎
🔁
📙
✍️
✍️
😎
🤠
📙
✍️
✍️
const BUILD_PATH = path.resolve(__dirname, "build");
function purgeRequireCache() {
for (const key in require.cache) {
if (key.startsWith(BUILD_DIR)) {
delete require.cache[key];
}
}
}
const watcher = chokidar.watch(BUILD_PATH);
watcher.on("all", () => {
// 1. purge require cache
purgeRequireCache();
// 2. load updated server build
const build = require(BUILD_PATH);
// 3. tell dev server that this app server is now ready
broadcastDevReady(build);
});
let createDevRequestHandler = (req, res, next) => {
return createRequestHandler({
build: require(BUILD_DIR),
mode: "development",
})(req, res, next);
}
CJS
function createDevRequestHandler() {
let devBuild = build;
const watcher = chokidar.watch(BUILD_PATH, { ignoreInitial: true });
watcher.on("all", async () => {
// 1. purge require cache && load updated server build
const stat = fs.statSync(BUILD_PATH);
devBuild = import(BUILD_PATH + "?t=" + stat.mtimeMs);
// 2. tell dev server that this app server is now ready
broadcastDevReady(await devBuild);
});
return async (req, res, next) => {
return createRequestHandler({
build: await devBuild,
mode: "development",
})(req, res, next);
};
}
ESM
🤓
🤩
😎
🤓
🤩
😎
🤓
🤩
😎
🤓
🤩
😎
🤓
🤩
😎
🤓
🤩
😎
https://remix.run/blog/remix-data-flow
data:image/s3,"s3://crabby-images/74b55/74b551db3ab6970960e0c0dc501fecd399102cc1" alt=""
🤓
🤩
😎
data:image/s3,"s3://crabby-images/dce1d/dce1dccdde9e71fbf2cfc34a2e548e7e25a57078" alt=""
https://redux.js.org/tutorials/fundamentals/part-2-concepts-data-flow
🤓
🤩
😎
🤓
🤩
😎
🤓
🤩
😎
Legendary DX
By pcattori
Legendary DX
- 255