npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

@autolib/auto

v1.40.0

Published

reactivity without side effects

Downloads

49

Readme

Note: the documentation is all over the place. See docs/ for all the (still to be fleshed out and organised) notes and diatribes.

manifesto

reactivity has immense potential but it is currently broken.

what is reactivity, reactivity is a game changer, reactivity is broken.

the solution is reactivity without side effects and auto is a javascript library for just that.

check out the next section for what this looks like in practice.

tiny tutorial

you use auto like this

let $ = auto({
    data: null,
    count: ($) => $.data ? $.data.length : 0,
    msg: ($) => $.data + " has " + $.count + " items"
})

see manual/syntax.md for a breakdown of the syntax.

now when you use $ everything updates automatically including nested relationships

$.data = [1,2,3];
console.log("msg =",$.msg);
msg = 1,2,3 has 3 items

this is much like other reactivity libraries but with the following it's quite different

let $ = auto({
    data: null,
    update: ($) => $.data = [1,2,3]
})
fatal: function update is trying to change value data

see the aforementioned discussions on why this is a really good idea.

features

besides no side affects what makes auto great is that it's design is really simple, it is really robust, and you can debug it.

really simple design

the auto library is 150 lines long, it uses no external libraries and the code is flat and clean

  • 7 state variables
  • 3 main functions and 4 helpers functions
  • an init block.

see manual/internals.md for details on these. also worth reading is the devlog docs/devlog of its development and design choices.

really robust

auto's internals are checked precisely by the tests. take a look at the tests/ which perform an exact match on the state including the call stack, function dependencies and fatal errors.

you can debug it

auto's internal variables are easy to interpret and are available at any time in the _ variable so

console.log($._)

will print something like

{
    fn: ['count'],
    subs: [],
    deps: { count: ['data'] },
    value: { data: [1,2,3], count: 3 },
    fatal: {}
}

it's worth looking at the tests in tests/ to see what different states are produced in various situations and also tutorial/explainability.md for a walkthrough of what these mean.

a lot of these documents have overlapping information which i should sort out

environments

each version of auto is in the root folder:

  • auto-commonjs.js
  • auto-es6.js
  • auto-no-export.js

they are all the same except for the last line (the export statement).

npm and node

run npm install @autolib/auto and then import with const auto = require('@autolib/auto');. see docs/npm-and-node.md for a walkthrough.

es6 module

simply use the right, i.e. const auto = import('auto-es6.js'); to test you could use deno i.e. deno run test.js. see docs/npm-and-node.md for a template - just replace the import statement.

browser

in a <script> tag link to auto-no-import.js. see docs/html.md for a walk-through.

integrations

auto was originally developed to be used with Svelte (see below) but other specific integrations with React, Vue, Mithril etc. still need to be done. however auto's subscribe method makes it easy to tie it into any existing framework.

subscribe

todo

svelte

todo

development

this library started as an attempt to create MobX-like observables in Svelte. that original version was based largely on a video by MobX's creator (see the docs/old-readme.md for more on this). then a new approach written up from scratch in a day which was documented with extreme detail in docs/devlog.

what's left out

what i can imagine some people might want in auto (because of what i've seen in other reactive libraries) but that i left out on purpose.

what about library/approach x

of course auto is very similar to other things i believe it has important distinctions. these are the ones i know about.

reactive programming

this i have covered in docs/discussion/bad-reactivity.md. auto falls under the category of reactive programming but it forces it into a particular shape which i believe makes it incalculably (ed: hmmm bit strong?) more powerful: docs/discussion/one-function-one-variable.md and docs/dicussion/no-side-effects.md.

svelte

the whole reason this library / these ideas were developed came out of using svelte's reactivity system and getting completely lost in a nightmare (ed: again too strong karl). see below for more on this. basically though this falls under the same as the above section, reactive programming: not having auto blocks, each function just returns a value for one variable, and no side effects ... (i should add the auto blocks article...)

rxjs

rxjs is actually what i wanted to start with even before svelte. it seemed like the right idea, in fact it seemed like a real revolution. i never went with it because svelte had similar things built into a ui library. so i don't know much about rxjs though i know they see themselves as the "lodash for events" which is cool... i should look into it further but i have a strong sense it's a different animal to what i'm doing here.

mobx

ok so really mobx is why auto looks the way it does.

this is getting a bit long winded - probably should put into it's own doc

as i mention below in the development section, it all started with this video the creator of mobx gave in russia (which bizarrely had very few views) that inspired me no end. he shows how the core mobx functionality can be done with a tiny amount of javascript. and it was clear from that that what that had done was ground breaking. it's such a bummer how a great idea needs to be looked after so carefully.

anyway, so auto looks a ton like mobx - at least the observable function. i changed it a bit: i don't use javascript property getters because it's cleaner to just pass in the state variable (just a bit shorter...). can't remember if there was another reason... however, after starting with the mobx guy's stuff (no internet can't get his name right now) i wanted to do derived values which he didn't cover ... and that broke and i ended up doing my own implementation from scratch ... which i found to be a lot clearer and stronger than his version ... and somewhere along the way i realised something about side effects ... i should check what happens in mobx if you try change one of the variables ...

but yes - my assumption is that .. let's not assume.

let observable = require('mobx').observable;

let state = observable({
    data: null,
    get count() { return this.data ? this.data.length : 0 }
})

console.log("count = "+state.count);
state.data = [1,2,3];
console.log("count = "+state.count);

(i just created an empty dir, ran npm install mobx and then put this into test.js and ran with node test.js).

C:\Users\karlp\tst\test-mobx>node test.js
count = 0
count = 3

ok works great. but what happens if i try to set a state variable from inside a function?

please don't work :|

let state = observable({
    data: null,
    amount: null,
    get count() { 
        this.amount = 100;
        return this.data ? this.data.length : 0 
    }
})

console.log("count = "+state.count);
console.log("amount = "+state.amount);
state.data = [1,2,3];
console.log("count = "+state.count);
console.log("amount = "+state.amount);
C:\Users\karlp\tst\test-mobx>node test.js
count = 0
amount = 100
count = 3
amount = 100

right. so it doesn't catch it. and why would it?

to be fair, i'm sure it would be easy enough to simply add this kind of check into mobx... perhaps with a parameter passed to observable (and it might even be there already...)

in this case you could replicate the primary reason for auto by restricting yourself to this approach - just using derived values i.e. no autorun blocks. as far as i can tell at the moment, that's the only real important difference between what you could do in mobx (if you made sure to restrict yourself to derived-values only) and auto - the fact that auto will break if you try to set a state value from within a derived function.

it does put the purpose of auto into question ... though to be fair, if i'm right about the side effects issue (and why autorun is so bad) mobx gives you a lot more ways to shoot yourself in the foot. so i think as an outsider if i were convinced by the correct-way-to-do-reactive arguments i'd rather use a library that was focused around that... what do you think?

side affects

this shouldn't be here cause i now deal with it up front

this is the big one - really the reason this library had to be made. it's a long story but basically all reactivity these days allows one to do this:

alwaysRun( () => {

    some_variable = 10;
    another_variable = [1,2,3];

    do_something();
    do_something_else();
})

the idea is that the whole block of code will re-run any time any of the referenced variables i.e. some_variable or another_variable change. this is a terrible idea, and auto was created specifically to do reactivity correctly, or rather to hone it down to one idea: automatic variables, not code.

i need to write this up properly.

nested reactivity

there was a version of the old library (mobx-svelte) that had nested reactivity (because i saw the MobX creator use it in the video that started this) but i didn't put it into the new one. i'm still haven't come across a convincing use-case. i'm using auto in a very large project and never needed nesting. i'd rather leave it out for now.