You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
svelte/test/store/index.js

462 lines
8.8 KiB

import * as assert from 'assert';
import { readable, writable, derived, get, readonly } from '../../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<unknown>} */
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('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('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);
});
});
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));
});
});
});