Notebooks as offline-first single webfiles?!

This webpage is sort of a notebook that was exported from this actual notebook.

And the webpage has its own editor and its own exporter so you can mutate it and continue the chain of exporting. It serializes to a single file so you don’t need a webserver to open it from the local filesystem, and it bundles all its dependancies, even file attachments, so it works offline! 1MB!

I hope to explain how it works at HYTRADBOI 2025 and after a bit more filling in a few missing pieces I intend this to become my personal homepage.

3 Likes

I’m loving this path of self-hosting you’re on. First the exporter, now this. So fun.

Small note, the Editor panel doesn’t work in Safari. It shows “editor_view = RuntimeError: Importing a module script failed.”

Oh thanks, the editor is the hardest part. I can repro.

1 Like

ok the safari issue is fixed. Thanks for reporting. I think we will use GitHub - tomlarkworthy/lopecode: moldable substrate for web and offline as the primary place for issues long term (here is fine too).

Hah, is “implementing your own notebook development environment” is the 2020s edition of Greenspun’s Tenth Rule? Because I also took a stab at it a few years ago! :wink:

I called it NBD. It’s been stuck in “sort-of-usable but rough prototype” stage for two years now - I only got to work on it for a few months on it before my daughter was born, and then that took over my life (no complaints though). Here’s an improvised 7 minute silent screencapture I made just now to show what’s implemented:

If anyone is interested in fooling around with the prototype, it can be found here: NBD - notebook static site development

You always had really cool tech but a totally different approach to things than me Tom so I’m super curious and excited to see where our approaches differ :grin:.

In my case I tried to go full neo-luddite: you can open the html file and it just works with file://, no webserver needed. This is also why it uses Ace instead of more recent code editor libraries).

There’s no npm or webpack or anything resembling modern JS development. The libraries used are imported via <script> tags like we’re back in the pre-Require.js days. I only need ace, `filesaver.js I refuse to worry about “polluting the global namespace” or whatever because honestly it’s just for quick throwaway prototypes that should export to single static HTML files.

Because I’m grug-brained and couldn’t figure out how to detect if a name exist in another cell like Observable does it, I made it import cell names explicitly via a tiny header DSL “embedded” in a comment on the first line. On the plus side, the header DSL also replaces most UI elements so I can do almost everything via keyboard input.

You wanna know how I handle saving and loading notebooks? I export them as .js files, and “load” them via a script tag. It feels like such a dumb hack but it’s the only thing I could come up with that works without a server.

Same thing for variable name validation, I’m just trying to do the stupidest thing that works:

/**
 * Tests if `name` is a valid JavaScript identifier
 * @param {string} name
 * @returns {boolean}
 */
function validIdentifier(name) {
	if(typeof name !== 'string') return false;
	if(!/^([$_]|[^\0-/])+$/.test(name)) return false;
	// We're already allowing `new Function` anyway,
	// might as well leverage it for cheap variable name validation.
	try { new Function(`const ${name} = 0`) } catch(e) { return false }
	return true;
}

It’s so nasty, hahaha

Anyway, aside from a baby (now toddler) to look after, I basically hit a wall when I got to the reactivity part and couldn’t figure out how to do topological sorting and “swapping” out cells reactively, so right now it works like Jupyter notebooks do. Which is really annoying, but I can work around it.

Gonna explore your notebook now Tom!

Will you share the recording of the talk?

1 Like

is “implementing your own notebook development environment” is the 2020s edition of [Greenspun’s Tenth Rule]

yeah I tried to resist the urge for a long time. I guess my unique spin is I am trying to build it in userspace. So the means of production is not distinct from the program. Similarly to you, its a single file so it is also openable as a file:// without a webserver. It is just the vanilla runtime though, the framing is part of the runtime.

All the HYTRADBOI videos are online now at HYTRADBOI 2025

Mine just discusses how to get access to the runtime so I can do all this stuff in userspace

@yurivish 's work with Plot was also in the conference!

I think our serialization approaches are quite similar, I vomit all the modules into a script tags. I do some clever importmap magic so the normal runtime is able to import modules without knowing they come from inside the single file bundle.

Normal Observable Runtime is designed to register the “observer” factory before loading of modules. This is not good for userspace implementation as I want the notebook renderer to be implemented in userspace and not from the outside. E.g. the div the notebook is hung on is a normal HTML cell. The sync code to align runtime state with the div children is written in the runtime.

Anyway, to achieve this, I need to attach call observers after the runtime has started. The key tech for this is a bunch of hacks here under the “observe” function. It offers the same “observer” interface but one you can dynamically attach, for any variable, even those that are currently not in the reachable set and in a module dependancy. It flips them to actively computed if needed

My latest iteration is not working enough yet to be able to show it yet, but being able to attach to dependent modules lets you have all the full dependancy graph in view at the same time. It ends up removing the concept of a top-level notebook, which I think is actually quite nice coz you can have a test notebook and an implementation notebook and work on both in parallel and reactively without bloating the implementation notebook with tests.

2 Likes

I guess my unique spin is I am trying to build it in userspace.

Yeah the “the export is another site that can export itself” is a really cool angle :). I think that’s also the fundamental difference in our ambitions from where a lot of the other differences in design decisions follow. My “neo-Ludditist” goal is to make the notebook an interactive dev environment that can then be exported to a static html page, that can also be opened without a webserver, with only as much JS sprinkled on top as absolutely necessary but no more.

One litmus test/goal is that the exported website should be trivial to archive with archive.org in a way that lets people view the site decades from now. Hence, the export tries to be a single minimal selfcontained html file when possible (this is harder when importing a JS library for example).

I’m highlighting this because it makes the various similarities in our approach all the more interesting, no?

[explanation of Larkworthy magic]

You’re a wizard, Tom, seriously cool stuff

1 Like

finally sat down to watch all the talks. Yue’s and yours were both great! You managed to pack a ton into less than three minutes. Did you get any good responses and discussions out of it in the chat?

1 Like

My talk had reasonably middle-of-the-pack engagement and a few cross links made across communities. I am happy with the outcome. I am still learning how to present things and I do feel that was one of my better attempts at transmitting information. For this kind of stuff I am more entwined in the “Future of Coding” and the “local-first” community. Both these communities have a propensity of reinventing the wheel so my talk was meant to show a different way of development. I think you would enjoy these communities too.

Maybe directly relevant to Observable was a talk called

“A case for feminism in programming language design”

With much higher engagement. Felienne Hermans notes that the Programming language community does not value make things easy for users. Tagline “Spreadsheets are not real programming”, and that that ecosystem values publishing things that are hard-for-hard-sake. Really great talk and it matches my observations.

1 Like

Happy to hear you got something out of it!

I was aware of Future of Coding and Local First communities but admit I never gave either a closer look :see_no_evil:. Maybe I should change that!

And yes, Felienne is awesome, have been a fan of her work since her first excel presentations! I totally agree with both of you that we have a serious culture problem in programming.

1 Like