Skip to content

Commit

Permalink
Allow reading a store in its onUse callback
Browse files Browse the repository at this point in the history
  • Loading branch information
divdavem committed Nov 26, 2024
1 parent f4b0efa commit a5afe4f
Show file tree
Hide file tree
Showing 5 changed files with 42 additions and 24 deletions.
9 changes: 9 additions & 0 deletions src/index.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1279,6 +1279,15 @@ describe('stores', () => {
a.set(1);
expect(scopes).toEqual([undefined]);
});

it('should allow reading the store in onUse', () => {
const onUseValues: number[] = [];
const store = writable(0, () => {
onUseValues.push(store());
});
expect(store()).toBe(0);
expect(onUseValues).toEqual([0]);
});
});

describe('asWritable', () => {
Expand Down
7 changes: 3 additions & 4 deletions src/internal/store.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,11 +9,10 @@ export const enum RawStoreFlags {
// the following flags are used in RawStoreTrackingUsage and derived classes
HAS_VISIBLE_ONUSE = 1,
START_USE_CALLED = 1 << 1,
INSIDE_GET = 1 << 2,
FLUSH_PLANNED = 1 << 3,
FLUSH_PLANNED = 1 << 2,
// the following flags are used in RawStoreComputedOrDerived and derived classes
COMPUTING = 1 << 4,
DIRTY = 1 << 5,
COMPUTING = 1 << 3,
DIRTY = 1 << 4,
}

export interface BaseLink<T> {
Expand Down
4 changes: 3 additions & 1 deletion src/internal/storeComputed.ts
Original file line number Diff line number Diff line change
Expand Up @@ -77,7 +77,9 @@ export class RawStoreComputed<T>
link.producer.registerConsumer(link);
}
}
this.flags |= RawStoreFlags.DIRTY;
if (this.epoch !== epoch) {
this.flags |= RawStoreFlags.DIRTY;
}
}

override endUse(): void {
Expand Down
42 changes: 23 additions & 19 deletions src/internal/storeTrackingUsage.ts
Original file line number Diff line number Diff line change
Expand Up @@ -29,32 +29,35 @@ export const flushUnused = (): void => {
};

export abstract class RawStoreTrackingUsage<T> extends RawStoreWritable<T> {
private extraUsages = 0;
abstract startUse(): void;
abstract endUse(): void;

override updateValue(): void {
// Ignoring coverage for the following lines because, unless there is a bug in tansu (which would have to be fixed!)
// there should be no way to trigger this error.
/* v8 ignore next 3 */
if (!(this.flags & RawStoreFlags.START_USE_CALLED)) {
throw new Error('assert failed: untracked producer usage');
}
super.updateValue();
}

override checkUsed(): void {
const flags = this.flags;
if (!(flags & RawStoreFlags.START_USE_CALLED)) {
// Ignoring coverage for the following lines because, unless there is a bug in tansu (which would have to be fixed!)
// there should be no way to trigger this error.
/* v8 ignore next 3 */
if (!(flags & RawStoreFlags.INSIDE_GET) && !this.consumerLinks?.length) {
throw new Error('assert failed: untracked producer usage');
}
this.flags |= RawStoreFlags.START_USE_CALLED;
untrack(() => this.startUse());
}
}

override checkUnused(): void {
const flags = this.flags;
// Ignoring coverage for the following lines because, unless there is a bug in tansu (which would have to be fixed!)
// there should be no way to trigger this error.
/* v8 ignore next 3 */
if (flags & RawStoreFlags.INSIDE_GET) {
throw new Error('assert failed: INSIDE_GET flag in checkUnused');
}
if (flags & RawStoreFlags.START_USE_CALLED && !this.consumerLinks?.length) {
if (
flags & RawStoreFlags.START_USE_CALLED &&
!this.consumerLinks?.length &&
!this.extraUsages
) {
if (inFlushUnused || flags & RawStoreFlags.HAS_VISIBLE_ONUSE) {
this.flags &= ~RawStoreFlags.START_USE_CALLED;
untrack(() => this.endUse());
Expand All @@ -74,11 +77,9 @@ export abstract class RawStoreTrackingUsage<T> extends RawStoreWritable<T> {
if (activeConsumer) {
return activeConsumer.addProducer(this);
} else {
if (this.flags & RawStoreFlags.INSIDE_GET) {
throw new Error('recursive computed');
}
this.flags |= RawStoreFlags.INSIDE_GET;
this.extraUsages++;
try {
this.checkUsed();
this.updateValue();
// Ignoring coverage for the following lines because, unless there is a bug in tansu (which would have to be fixed!)
// there should be no way to trigger this error.
Expand All @@ -88,8 +89,11 @@ export abstract class RawStoreTrackingUsage<T> extends RawStoreWritable<T> {
}
return this.readValue();
} finally {
this.flags &= ~RawStoreFlags.INSIDE_GET;
this.checkUnused();
const extraUsages = this.extraUsages - 1;
this.extraUsages = extraUsages;
if (extraUsages === 0) {
this.checkUnused();
}
}
}
}
Expand Down
4 changes: 4 additions & 0 deletions src/internal/storeWritable.ts
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,9 @@ export class RawStoreWritable<T> implements RawStore<T, ProducerConsumerLink<T>>
const indexInProducer = consumerLinks.length;
link.indexInProducer = indexInProducer;
consumerLinks[indexInProducer] = link;
if (indexInProducer === 0) {
this.checkUsed();
}
return link;
}

Expand All @@ -102,6 +105,7 @@ export class RawStoreWritable<T> implements RawStore<T, ProducerConsumerLink<T>>
}
}

checkUsed(): void {}
checkUnused(): void {}
updateValue(): void {}

Expand Down

0 comments on commit a5afe4f

Please sign in to comment.