import { describe, it, assert } from 'vitest'; import { readable, writable, derived, get, readonly } from 'svelte/store'; describe('store', () => { describe('writable', () => { it('creates a writable store', () => { const count = writable(0); const values = []; const unsubscribe = count.subscribe((value) => { values.push(value); }); count.set(1); count.update((n) => n + 1); unsubscribe(); count.set(3); count.update((n) => n + 1); assert.deepEqual(values, [0, 1, 2]); }); it('creates an undefined writable store', () => { const store = writable(); const values = []; const unsubscribe = store.subscribe((value) => { values.push(value); }); unsubscribe(); assert.deepEqual(values, [undefined]); }); it('calls provided subscribe handler', () => { let called = 0; const store = writable(0, () => { called += 1; return () => (called -= 1); }); const unsubscribe1 = store.subscribe(() => {}); assert.equal(called, 1); const unsubscribe2 = store.subscribe(() => {}); assert.equal(called, 1); unsubscribe1(); assert.equal(called, 1); unsubscribe2(); assert.equal(called, 0); }); it('does not assume immutable data', () => { const obj = {}; let called = 0; const store = writable(obj); store.subscribe(() => { called += 1; }); store.set(obj); assert.equal(called, 2); store.update((obj) => obj); assert.equal(called, 3); }); it('only calls subscriber once initially, including on resubscriptions', () => { let num = 0; const store = writable(num, (set) => set((num += 1))); let count1 = 0; let count2 = 0; store.subscribe(() => (count1 += 1))(); assert.equal(count1, 1); const unsubscribe = store.subscribe(() => (count2 += 1)); assert.equal(count2, 1); unsubscribe(); }); it('no error even if unsubscribe calls twice', () => { let num = 0; const store = writable(num, (set) => set((num += 1))); const unsubscribe = store.subscribe(() => {}); unsubscribe(); assert.doesNotThrow(() => unsubscribe()); }); }); describe('readable', () => { it('creates a readable store', () => { let running; /** @type {import('svelte/store').Subscriber} */ let tick; const store = readable(undefined, (set) => { tick = set; running = true; set(0); return () => { tick = () => {}; running = false; }; }); assert.ok(!running); const values = []; const unsubscribe = store.subscribe((value) => { values.push(value); }); assert.ok(running); tick(1); tick(2); unsubscribe(); assert.ok(!running); tick(3); tick(4); assert.deepEqual(values, [0, 1, 2]); }); it('passes an optional update function', () => { let running; /** @type {(value: any) => void} */ let tick; /** @type {(value: any) => void} */ let add; const store = readable(undefined, (set, update) => { tick = set; running = true; add = (n) => update((value) => value + n); set(0); return () => { tick = () => {}; add = (_) => {}; running = false; }; }); assert.ok(!running); const values = []; const unsubscribe = store.subscribe((value) => { values.push(value); }); assert.ok(running); tick(1); tick(2); add(3); add(4); tick(5); add(6); unsubscribe(); assert.ok(!running); tick(7); add(8); assert.deepEqual(values, [0, 1, 2, 5, 9, 5, 11]); }); it('creates an undefined readable store', () => { const store = readable(); const values = []; const unsubscribe = store.subscribe((value) => { values.push(value); }); unsubscribe(); assert.deepEqual(values, [undefined]); }); it('creates a readable store without updater', () => { const store = readable(100); const values = []; const unsubscribe = store.subscribe((value) => { values.push(value); }); unsubscribe(); assert.deepEqual(values, [100]); }); }); /** @type {any} */ const fake_observable = { subscribe(fn) { fn(42); return { unsubscribe: () => {} }; } }; describe('derived', () => { it('maps a single store', () => { const a = writable(1); const b = derived(a, (n) => n * 2); const values = []; const unsubscribe = b.subscribe((value) => { values.push(value); }); a.set(2); assert.deepEqual(values, [2, 4]); unsubscribe(); a.set(3); assert.deepEqual(values, [2, 4]); }); it('maps multiple stores', () => { const a = writable(2); const b = writable(3); const c = derived([a, b], ([a, b]) => a * b); const values = []; const unsubscribe = c.subscribe((value) => { values.push(value); }); a.set(4); b.set(5); assert.deepEqual(values, [6, 12, 20]); unsubscribe(); a.set(6); assert.deepEqual(values, [6, 12, 20]); }); it('passes optional set function', () => { const number = writable(1); const evens = derived( number, (n, set) => { if (n % 2 === 0) set(n); }, 0 ); const values = []; const unsubscribe = evens.subscribe((value) => { values.push(value); }); number.set(2); number.set(3); number.set(4); number.set(5); assert.deepEqual(values, [0, 2, 4]); unsubscribe(); number.set(6); number.set(7); number.set(8); assert.deepEqual(values, [0, 2, 4]); }); it('passes optional set and update functions', () => { const number = writable(1); const evensAndSquaresOf4 = derived( number, (n, set, update) => { if (n % 2 === 0) set(n); if (n % 4 === 0) update((n) => n * n); }, 0 ); const values = []; const unsubscribe = evensAndSquaresOf4.subscribe((value) => { values.push(value); }); number.set(2); number.set(3); number.set(4); number.set(5); number.set(6); assert.deepEqual(values, [0, 2, 4, 16, 6]); number.set(7); number.set(8); number.set(9); number.set(10); assert.deepEqual(values, [0, 2, 4, 16, 6, 8, 64, 10]); unsubscribe(); number.set(11); number.set(12); assert.deepEqual(values, [0, 2, 4, 16, 6, 8, 64, 10]); }); it('prevents glitches', () => { const lastname = writable('Jekyll'); const firstname = derived(lastname, (n) => (n === 'Jekyll' ? 'Henry' : 'Edward')); const fullname = derived([firstname, lastname], (names) => names.join(' ')); const values = []; const unsubscribe = fullname.subscribe((value) => { values.push(value); }); lastname.set('Hyde'); assert.deepEqual(values, ['Henry Jekyll', 'Edward Hyde']); unsubscribe(); }); it('prevents diamond dependency problem', () => { const count = writable(0); const values = []; const a = derived(count, ($count) => { return 'a' + $count; }); const b = derived(count, ($count) => { return 'b' + $count; }); const combined = derived([a, b], ([a, b]) => { return a + b; }); const unsubscribe = combined.subscribe((v) => { values.push(v); }); assert.deepEqual(values, ['a0b0']); count.set(1); assert.deepEqual(values, ['a0b0', 'a1b1']); unsubscribe(); }); it('derived dependency does not update and shared ancestor updates', () => { const root = writable({ a: 0, b: 0 }); const values = []; const a = derived(root, ($root) => { return 'a' + $root.a; }); const b = derived([a, root], ([$a, $root]) => { return 'b' + $root.b + $a; }); const unsubscribe = b.subscribe((v) => { values.push(v); }); assert.deepEqual(values, ['b0a0']); root.set({ a: 0, b: 1 }); assert.deepEqual(values, ['b0a0', 'b1a0']); unsubscribe(); }); it('is updated with safe_not_equal logic', () => { const arr = [0]; const number = writable(1); const numbers = derived(number, ($number) => { arr[0] = $number; return arr; }); const concatenated = []; const unsubscribe = numbers.subscribe((value) => { concatenated.push(...value); }); number.set(2); number.set(3); assert.deepEqual(concatenated, [1, 2, 3]); unsubscribe(); }); it('calls a cleanup function', () => { const num = writable(1); const values = []; const cleaned_up = []; const d = derived(num, ($num, set) => { set($num * 2); return function cleanup() { cleaned_up.push($num); }; }); num.set(2); const unsubscribe = d.subscribe((value) => { values.push(value); }); num.set(3); num.set(4); assert.deepEqual(values, [4, 6, 8]); assert.deepEqual(cleaned_up, [2, 3]); unsubscribe(); assert.deepEqual(cleaned_up, [2, 3, 4]); }); it('discards non-function return values', () => { const num = writable(1); const values = []; const d = derived(num, ($num, set) => { set($num * 2); return {}; }); num.set(2); const unsubscribe = d.subscribe((value) => { values.push(value); }); num.set(3); num.set(4); assert.deepEqual(values, [4, 6, 8]); unsubscribe(); }); it('allows derived with different types', () => { const a = writable('one'); const b = writable(1); const c = derived([a, b], ([a, b]) => `${a} ${b}`); assert.deepEqual(get(c), 'one 1'); a.set('two'); b.set(2); assert.deepEqual(get(c), 'two 2'); }); it('works with RxJS-style observables', () => { const d = derived(fake_observable, (_) => _); assert.equal(get(d), 42); }); it("doesn't restart when unsubscribed from another store with a shared ancestor", () => { const a = writable(true); let b_started = false; const b = derived(a, (_, __) => { b_started = true; return () => { assert.equal(b_started, true); b_started = false; }; }); const c = derived(a, ($a, set) => { if ($a) return b.subscribe(set); }); c.subscribe(() => {}); a.set(false); assert.equal(b_started, false); }); it('errors on undefined stores #1', () => { assert.throws(() => { derived(null, (n) => n); }); }); it('errors on undefined stores #2', () => { assert.throws(() => { const a = writable(1); derived([a, null, undefined], ([n]) => { return n * 2; }); }); }); }); describe('get', () => { it('gets the current value of a store', () => { const store = readable(42, () => {}); assert.equal(get(store), 42); }); it('works with RxJS-style observables', () => { assert.equal(get(fake_observable), 42); }); }); describe('readonly', () => { it('makes a store readonly', () => { const writableStore = writable(1); const readableStore = readonly(writableStore); assert.equal(get(readableStore), get(writableStore)); writableStore.set(2); assert.equal(get(readableStore), 2); assert.equal(get(readableStore), get(writableStore)); // @ts-ignore assert.throws(() => readableStore.set(3)); }); }); });