@livestore/livestore 0.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +108 -0
- package/dist/.tsbuildinfo +1 -0
- package/dist/LiveRiffleStore.d.ts +42 -0
- package/dist/LiveRiffleStore.d.ts.map +1 -0
- package/dist/LiveRiffleStore.js +36 -0
- package/dist/LiveRiffleStore.js.map +1 -0
- package/dist/QueryCache.d.ts +20 -0
- package/dist/QueryCache.d.ts.map +1 -0
- package/dist/QueryCache.js +71 -0
- package/dist/QueryCache.js.map +1 -0
- package/dist/__tests__/react/fixture.d.ts +141 -0
- package/dist/__tests__/react/fixture.d.ts.map +1 -0
- package/dist/__tests__/react/fixture.js +72 -0
- package/dist/__tests__/react/fixture.js.map +1 -0
- package/dist/__tests__/react/useLiveStoreComponent.test.d.ts +2 -0
- package/dist/__tests__/react/useLiveStoreComponent.test.d.ts.map +1 -0
- package/dist/__tests__/react/useLiveStoreComponent.test.js +78 -0
- package/dist/__tests__/react/useLiveStoreComponent.test.js.map +1 -0
- package/dist/__tests__/react/useRiffleComponent.test.d.ts +2 -0
- package/dist/__tests__/react/useRiffleComponent.test.d.ts.map +1 -0
- package/dist/__tests__/react/useRiffleComponent.test.js +78 -0
- package/dist/__tests__/react/useRiffleComponent.test.js.map +1 -0
- package/dist/__tests__/reactive.test.d.ts +2 -0
- package/dist/__tests__/reactive.test.d.ts.map +1 -0
- package/dist/__tests__/reactive.test.js +167 -0
- package/dist/__tests__/reactive.test.js.map +1 -0
- package/dist/backends/base.d.ts +13 -0
- package/dist/backends/base.d.ts.map +1 -0
- package/dist/backends/base.js +53 -0
- package/dist/backends/base.js.map +1 -0
- package/dist/backends/index.d.ts +41 -0
- package/dist/backends/index.d.ts.map +1 -0
- package/dist/backends/index.js +38 -0
- package/dist/backends/index.js.map +1 -0
- package/dist/backends/noop.d.ts +18 -0
- package/dist/backends/noop.d.ts.map +1 -0
- package/dist/backends/noop.js +21 -0
- package/dist/backends/noop.js.map +1 -0
- package/dist/backends/tauri.d.ts +24 -0
- package/dist/backends/tauri.d.ts.map +1 -0
- package/dist/backends/tauri.js +48 -0
- package/dist/backends/tauri.js.map +1 -0
- package/dist/backends/utils/idb.d.ts +10 -0
- package/dist/backends/utils/idb.d.ts.map +1 -0
- package/dist/backends/utils/idb.js +58 -0
- package/dist/backends/utils/idb.js.map +1 -0
- package/dist/backends/web-in-memory.d.ts +24 -0
- package/dist/backends/web-in-memory.d.ts.map +1 -0
- package/dist/backends/web-in-memory.js +46 -0
- package/dist/backends/web-in-memory.js.map +1 -0
- package/dist/backends/web-worker.d.ts +17 -0
- package/dist/backends/web-worker.d.ts.map +1 -0
- package/dist/backends/web-worker.js +139 -0
- package/dist/backends/web-worker.js.map +1 -0
- package/dist/backends/web.d.ts +28 -0
- package/dist/backends/web.d.ts.map +1 -0
- package/dist/backends/web.js +64 -0
- package/dist/backends/web.js.map +1 -0
- package/dist/bounded-collections.d.ts +34 -0
- package/dist/bounded-collections.d.ts.map +1 -0
- package/dist/bounded-collections.js +103 -0
- package/dist/bounded-collections.js.map +1 -0
- package/dist/componentKey.d.ts +20 -0
- package/dist/componentKey.d.ts.map +1 -0
- package/dist/componentKey.js +3 -0
- package/dist/componentKey.js.map +1 -0
- package/dist/effect/LiveStore.d.ts +42 -0
- package/dist/effect/LiveStore.d.ts.map +1 -0
- package/dist/effect/LiveStore.js +36 -0
- package/dist/effect/LiveStore.js.map +1 -0
- package/dist/effect/index.d.ts +2 -0
- package/dist/effect/index.d.ts.map +1 -0
- package/dist/effect/index.js +2 -0
- package/dist/effect/index.js.map +1 -0
- package/dist/events.d.ts +7 -0
- package/dist/events.d.ts.map +1 -0
- package/dist/events.js +2 -0
- package/dist/events.js.map +1 -0
- package/dist/inMemoryDatabase.d.ts +65 -0
- package/dist/inMemoryDatabase.d.ts.map +1 -0
- package/dist/inMemoryDatabase.js +241 -0
- package/dist/inMemoryDatabase.js.map +1 -0
- package/dist/index.d.ts +20 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +10 -0
- package/dist/index.js.map +1 -0
- package/dist/otel.d.ts +5 -0
- package/dist/otel.d.ts.map +1 -0
- package/dist/otel.js +17 -0
- package/dist/otel.js.map +1 -0
- package/dist/react/LiveStoreContext.d.ts +11 -0
- package/dist/react/LiveStoreContext.d.ts.map +1 -0
- package/dist/react/LiveStoreContext.js +10 -0
- package/dist/react/LiveStoreContext.js.map +1 -0
- package/dist/react/LiveStoreProvider.d.ts +21 -0
- package/dist/react/LiveStoreProvider.d.ts.map +1 -0
- package/dist/react/LiveStoreProvider.js +48 -0
- package/dist/react/LiveStoreProvider.js.map +1 -0
- package/dist/react/RiffleProvider.d.ts +21 -0
- package/dist/react/RiffleProvider.d.ts.map +1 -0
- package/dist/react/RiffleProvider.js +48 -0
- package/dist/react/RiffleProvider.js.map +1 -0
- package/dist/react/StoreContext.d.ts +11 -0
- package/dist/react/StoreContext.d.ts.map +1 -0
- package/dist/react/StoreContext.js +10 -0
- package/dist/react/StoreContext.js.map +1 -0
- package/dist/react/index.d.ts +7 -0
- package/dist/react/index.d.ts.map +1 -0
- package/dist/react/index.js +6 -0
- package/dist/react/index.js.map +1 -0
- package/dist/react/useGlobalQuery.d.ts +3 -0
- package/dist/react/useGlobalQuery.d.ts.map +1 -0
- package/dist/react/useGlobalQuery.js +25 -0
- package/dist/react/useGlobalQuery.js.map +1 -0
- package/dist/react/useGraphQL.d.ts +11 -0
- package/dist/react/useGraphQL.d.ts.map +1 -0
- package/dist/react/useGraphQL.js +68 -0
- package/dist/react/useGraphQL.js.map +1 -0
- package/dist/react/useLiveStoreComponent.d.ts +70 -0
- package/dist/react/useLiveStoreComponent.d.ts.map +1 -0
- package/dist/react/useLiveStoreComponent.js +261 -0
- package/dist/react/useLiveStoreComponent.js.map +1 -0
- package/dist/react/useRiffleComponent.d.ts +70 -0
- package/dist/react/useRiffleComponent.d.ts.map +1 -0
- package/dist/react/useRiffleComponent.js +261 -0
- package/dist/react/useRiffleComponent.js.map +1 -0
- package/dist/react/useRiffleJsonHook.d.ts +4 -0
- package/dist/react/useRiffleJsonHook.d.ts.map +1 -0
- package/dist/react/useRiffleJsonHook.js +21 -0
- package/dist/react/useRiffleJsonHook.js.map +1 -0
- package/dist/react/utils/useStateRefWithReactiveInput.d.ts +13 -0
- package/dist/react/utils/useStateRefWithReactiveInput.d.ts.map +1 -0
- package/dist/react/utils/useStateRefWithReactiveInput.js +38 -0
- package/dist/react/utils/useStateRefWithReactiveInput.js.map +1 -0
- package/dist/reactive.d.ts +140 -0
- package/dist/reactive.d.ts.map +1 -0
- package/dist/reactive.js +301 -0
- package/dist/reactive.js.map +1 -0
- package/dist/reactiveQueries/base-class.d.ts +24 -0
- package/dist/reactiveQueries/base-class.d.ts.map +1 -0
- package/dist/reactiveQueries/base-class.js +22 -0
- package/dist/reactiveQueries/base-class.js.map +1 -0
- package/dist/reactiveQueries/graphql.d.ts +25 -0
- package/dist/reactiveQueries/graphql.d.ts.map +1 -0
- package/dist/reactiveQueries/graphql.js +14 -0
- package/dist/reactiveQueries/graphql.js.map +1 -0
- package/dist/reactiveQueries/js.d.ts +19 -0
- package/dist/reactiveQueries/js.d.ts.map +1 -0
- package/dist/reactiveQueries/js.js +13 -0
- package/dist/reactiveQueries/js.js.map +1 -0
- package/dist/reactiveQueries/sql.d.ts +31 -0
- package/dist/reactiveQueries/sql.d.ts.map +1 -0
- package/dist/reactiveQueries/sql.js +28 -0
- package/dist/reactiveQueries/sql.js.map +1 -0
- package/dist/schema.d.ts +163 -0
- package/dist/schema.d.ts.map +1 -0
- package/dist/schema.js +92 -0
- package/dist/schema.js.map +1 -0
- package/dist/store.d.ts +175 -0
- package/dist/store.d.ts.map +1 -0
- package/dist/store.js +546 -0
- package/dist/store.js.map +1 -0
- package/dist/util.d.ts +24 -0
- package/dist/util.d.ts.map +1 -0
- package/dist/util.js +51 -0
- package/dist/util.js.map +1 -0
- package/package.json +52 -0
- package/src/QueryCache.ts +81 -0
- package/src/__tests__/react/fixture.tsx +106 -0
- package/src/__tests__/react/useLiveStoreComponent.test.tsx +111 -0
- package/src/__tests__/reactive.test.ts +227 -0
- package/src/ambient.d.ts +7 -0
- package/src/backends/base.ts +67 -0
- package/src/backends/index.ts +94 -0
- package/src/backends/noop.ts +32 -0
- package/src/backends/tauri.ts +74 -0
- package/src/backends/utils/idb.ts +71 -0
- package/src/backends/web-in-memory.ts +65 -0
- package/src/backends/web-worker.ts +176 -0
- package/src/backends/web.ts +96 -0
- package/src/bounded-collections.ts +112 -0
- package/src/componentKey.ts +9 -0
- package/src/effect/LiveStore.ts +123 -0
- package/src/effect/index.ts +7 -0
- package/src/events.ts +8 -0
- package/src/inMemoryDatabase.ts +347 -0
- package/src/index.ts +47 -0
- package/src/otel.ts +20 -0
- package/src/react/LiveStoreContext.ts +23 -0
- package/src/react/LiveStoreProvider.tsx +93 -0
- package/src/react/index.ts +11 -0
- package/src/react/useGlobalQuery.ts +40 -0
- package/src/react/useGraphQL.ts +113 -0
- package/src/react/useLiveStoreComponent.ts +493 -0
- package/src/react/utils/useStateRefWithReactiveInput.ts +51 -0
- package/src/reactive.ts +538 -0
- package/src/reactiveQueries/base-class.ts +49 -0
- package/src/reactiveQueries/graphql.ts +52 -0
- package/src/reactiveQueries/js.ts +38 -0
- package/src/reactiveQueries/sql.ts +65 -0
- package/src/schema.ts +219 -0
- package/src/store.ts +889 -0
- package/src/util.ts +59 -0
- package/tsconfig.json +15 -0
- package/vitest.config.js +13 -0
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import type * as otel from '@opentelemetry/api';
|
|
2
|
+
import type { Scope } from '@overtone/utils/effect';
|
|
3
|
+
import { Context, Deferred, Effect, Layer, Otel } from '@overtone/utils/effect';
|
|
4
|
+
import type { GraphQLSchema } from 'graphql';
|
|
5
|
+
import type { Backend, BackendOptions } from './backends/index.js';
|
|
6
|
+
import type { InMemoryDatabase } from './inMemoryDatabase.js';
|
|
7
|
+
import type { Schema } from './schema.js';
|
|
8
|
+
import type { BaseGraphQLContext, GraphQLOptions, RiffleQuery, Store } from './store.js';
|
|
9
|
+
export type StoreContext = {
|
|
10
|
+
store: Store<any>;
|
|
11
|
+
globalQueries: RiffleQueryTypes;
|
|
12
|
+
};
|
|
13
|
+
export type QueryDefinition = (store: Store<any>) => RiffleQuery;
|
|
14
|
+
export type GlobalQueryDefs = {
|
|
15
|
+
[key: string]: QueryDefinition;
|
|
16
|
+
};
|
|
17
|
+
export type RiffleCreateStoreOptions<GraphQLContext extends BaseGraphQLContext> = {
|
|
18
|
+
schema: Schema;
|
|
19
|
+
globalQueryDefs: GlobalQueryDefs;
|
|
20
|
+
backendOptions: BackendOptions;
|
|
21
|
+
graphQLOptions?: GraphQLOptions<GraphQLContext>;
|
|
22
|
+
otelTracer?: otel.Tracer;
|
|
23
|
+
otelRootSpanContext?: otel.Context;
|
|
24
|
+
boot?: (backend: Backend, parentSpan: otel.Span) => Promise<void>;
|
|
25
|
+
};
|
|
26
|
+
export declare const StoreContext: Context.Tag<StoreContext, StoreContext>;
|
|
27
|
+
export type DeferredStoreContext = Deferred.Deferred<never, StoreContext>;
|
|
28
|
+
export declare const DeferredStoreContext: Context.Tag<DeferredStoreContext, DeferredStoreContext>;
|
|
29
|
+
export type RiffleStoreContextProps<GraphQLContext extends BaseGraphQLContext> = {
|
|
30
|
+
schema: Schema;
|
|
31
|
+
globalQueryDefs?: Effect.Effect<Otel.Tracer | Otel.Span, never, GlobalQueryDefs>;
|
|
32
|
+
backendOptions: Effect.Effect<Otel.Tracer | Otel.Span, never, BackendOptions>;
|
|
33
|
+
graphQLOptions?: {
|
|
34
|
+
schema: Effect.Effect<Otel.Tracer, never, GraphQLSchema>;
|
|
35
|
+
makeContext: (db: InMemoryDatabase) => GraphQLContext;
|
|
36
|
+
};
|
|
37
|
+
boot?: (backend: Backend) => Effect.Effect<Otel.Tracer, never, void>;
|
|
38
|
+
};
|
|
39
|
+
export declare const LiveRiffleStoreContext: <GraphQLContext extends BaseGraphQLContext>(props: RiffleStoreContextProps<GraphQLContext>) => Layer.Layer<Otel.Tracer, never, StoreContext>;
|
|
40
|
+
export declare const LiveRiffleStoreContextDeferred: Layer.Layer<never, never, DeferredStoreContext>;
|
|
41
|
+
export declare const makeRiffle: <GraphQLContext extends BaseGraphQLContext>({ globalQueryDefs, schema, backendOptions: backendOptions_, graphQLOptions: graphQLOptions_, boot: boot_, }: RiffleStoreContextProps<GraphQLContext>) => Effect.Effect<Otel.Tracer | Otel.Span | DeferredStoreContext | Scope.Scope, never, StoreContext>;
|
|
42
|
+
//# sourceMappingURL=LiveRiffleStore.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"LiveRiffleStore.d.ts","sourceRoot":"","sources":["../src/LiveRiffleStore.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { Context, Deferred, Duration, Effect, Layer, Otel, pipe } from '@overtone/utils/effect';
|
|
2
|
+
import { mapValues } from 'lodash-es';
|
|
3
|
+
import { createStore } from './store.js';
|
|
4
|
+
export const StoreContext = Context.Tag('@overtone/riffle/StoreContext');
|
|
5
|
+
export const DeferredStoreContext = Context.Tag(Symbol.for('@overtone/riffle/DeferredStoreContext'));
|
|
6
|
+
export const LiveRiffleStoreContext = (props) => Layer.provide(LiveRiffleStoreContextDeferred, pipe(Layer.scoped(StoreContext, makeRiffle(props)), Otel.withSpanLayer('Riffle')));
|
|
7
|
+
export const LiveRiffleStoreContextDeferred = Layer.effect(DeferredStoreContext, Deferred.make());
|
|
8
|
+
export const makeRiffle = ({ globalQueryDefs, schema, backendOptions: backendOptions_, graphQLOptions: graphQLOptions_, boot: boot_, }) => pipe(Effect.gen(function* ($) {
|
|
9
|
+
const ctx = yield* $(Effect.context());
|
|
10
|
+
const otelRootSpanContext = yield* $(Otel.activeContext);
|
|
11
|
+
const { tracer: otelTracer } = yield* $(Otel.Tracer);
|
|
12
|
+
const graphQLOptions = yield* $(graphQLOptions_
|
|
13
|
+
? Effect.all({ schema: graphQLOptions_.schema, makeContext: Effect.succeed(graphQLOptions_.makeContext) })
|
|
14
|
+
: Effect.succeed(undefined));
|
|
15
|
+
const backendOptions = yield* $(backendOptions_ ?? Effect.succeed(undefined));
|
|
16
|
+
const boot = boot_
|
|
17
|
+
? (db) => pipe(boot_(db), Effect.provideContext(ctx), Effect.tapCauseLogPretty, Effect.runPromise)
|
|
18
|
+
: undefined;
|
|
19
|
+
const store = yield* $(Effect.tryPromise(() => createStore({
|
|
20
|
+
schema,
|
|
21
|
+
backendOptions,
|
|
22
|
+
graphQLOptions,
|
|
23
|
+
otelTracer,
|
|
24
|
+
otelRootSpanContext,
|
|
25
|
+
boot,
|
|
26
|
+
})), Effect.acquireRelease((store) => Effect.sync(() => store.destroy())));
|
|
27
|
+
window.__debugRiffleStore = store;
|
|
28
|
+
const globalQueries = yield* $(globalQueryDefs ?? Effect.succeed({}), Effect.map((defs) => mapValues(defs, (queryDef) => queryDef(store))), Otel.withSpan('riffle:makeGlobalQueries', {}, store.otel.queriesSpanContext));
|
|
29
|
+
// NOTE give main thread a chance to render
|
|
30
|
+
yield* $(Effect.yieldNow());
|
|
31
|
+
return { store, globalQueries };
|
|
32
|
+
}), Effect.tap((storeCtx) => Effect.flatMap(DeferredStoreContext, (def) => Deferred.succeed(def, storeCtx))), Effect.timeoutFail({
|
|
33
|
+
onTimeout: () => new Error('Timed out while creating Riffle store after 10sec'),
|
|
34
|
+
duration: Duration.seconds(10),
|
|
35
|
+
}), Effect.orDie);
|
|
36
|
+
//# sourceMappingURL=LiveRiffleStore.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"LiveRiffleStore.js","sourceRoot":"","sources":["../src/LiveRiffleStore.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import type { Bindable } from './util.js';
|
|
2
|
+
type Opaque<BaseType, BrandType = unknown> = BaseType & {
|
|
3
|
+
readonly [Symbols.base]: BaseType;
|
|
4
|
+
readonly [Symbols.brand]: BrandType;
|
|
5
|
+
};
|
|
6
|
+
declare namespace Symbols {
|
|
7
|
+
const base: unique symbol;
|
|
8
|
+
const brand: unique symbol;
|
|
9
|
+
}
|
|
10
|
+
export type CacheKey = Opaque<string, string>;
|
|
11
|
+
export default class QueryCache {
|
|
12
|
+
#private;
|
|
13
|
+
getKey: (sql: string, bindValues?: Bindable) => CacheKey;
|
|
14
|
+
get: (key: CacheKey) => any;
|
|
15
|
+
set: (queriedTables: string[], key: CacheKey, results: any) => void;
|
|
16
|
+
ignoreQuery: (query: string) => boolean;
|
|
17
|
+
invalidate: (queriedTables: string[]) => void;
|
|
18
|
+
}
|
|
19
|
+
export {};
|
|
20
|
+
//# sourceMappingURL=QueryCache.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"QueryCache.d.ts","sourceRoot":"","sources":["../src/QueryCache.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAA;AAEzC,KAAK,MAAM,CAAC,QAAQ,EAAE,SAAS,GAAG,OAAO,IAAI,QAAQ,GAAG;IACtD,QAAQ,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,QAAQ,CAAA;IACjC,QAAQ,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,SAAS,CAAA;CACpC,CAAA;AAED,kBAAU,OAAO,CAAC;IACD,MAAM,IAAI,EAAE,OAAO,MAAM,CAAA;IACzB,MAAM,KAAK,EAAE,OAAO,MAAM,CAAA;CAC1C;AAED,MAAM,MAAM,QAAQ,GAAG,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;AAO7C,MAAM,CAAC,OAAO,OAAO,UAAU;;IAI7B,MAAM,QAAS,MAAM,eAAe,QAAQ,KAAG,QAAQ,CAUtD;IAED,GAAG,QAAS,QAAQ,SAEnB;IAED,GAAG,kBAAmB,MAAM,EAAE,OAAO,QAAQ,WAAW,GAAG,UAW1D;IAMD,WAAW,UAAW,MAAM,aAE3B;IAUD,UAAU,kBAAmB,MAAM,EAAE,UAUpC;CACF"}
|
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, state, kind, f) {
|
|
2
|
+
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter");
|
|
3
|
+
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
|
|
4
|
+
return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
|
|
5
|
+
};
|
|
6
|
+
var _QueryCache_entries, _QueryCache_dependencies, _QueryCache_dependencyTrackerEvicted;
|
|
7
|
+
import BoundMap, { BoundSet } from './bounded-collections.js';
|
|
8
|
+
var Symbols;
|
|
9
|
+
(function (Symbols) {
|
|
10
|
+
})(Symbols || (Symbols = {}));
|
|
11
|
+
const ignore = ['begin', 'rollback', 'commit', 'savepoint', 'release'];
|
|
12
|
+
// TODO: profile to see how big we need this cache to be.
|
|
13
|
+
const cacheSize = 200;
|
|
14
|
+
class QueryCache {
|
|
15
|
+
constructor() {
|
|
16
|
+
_QueryCache_entries.set(this, new BoundMap(cacheSize));
|
|
17
|
+
_QueryCache_dependencies.set(this, new Map());
|
|
18
|
+
this.getKey = (sql, bindValues) => {
|
|
19
|
+
if (bindValues == null) {
|
|
20
|
+
return sql;
|
|
21
|
+
}
|
|
22
|
+
if (Array.isArray(bindValues)) {
|
|
23
|
+
return (sql + '\n' + bindValues.join('\n'));
|
|
24
|
+
}
|
|
25
|
+
return (sql + '\n' + Object.values(bindValues).join('\n'));
|
|
26
|
+
};
|
|
27
|
+
this.get = (key) => {
|
|
28
|
+
return __classPrivateFieldGet(this, _QueryCache_entries, "f").get(key);
|
|
29
|
+
};
|
|
30
|
+
this.set = (queriedTables, key, results) => {
|
|
31
|
+
__classPrivateFieldGet(this, _QueryCache_entries, "f").set(key, results);
|
|
32
|
+
for (const table of queriedTables) {
|
|
33
|
+
let keys = __classPrivateFieldGet(this, _QueryCache_dependencies, "f").get(table);
|
|
34
|
+
if (keys == null) {
|
|
35
|
+
keys = new BoundSet(cacheSize);
|
|
36
|
+
keys.onEvict = __classPrivateFieldGet(this, _QueryCache_dependencyTrackerEvicted, "f");
|
|
37
|
+
__classPrivateFieldGet(this, _QueryCache_dependencies, "f").set(table, keys);
|
|
38
|
+
}
|
|
39
|
+
keys.add(key);
|
|
40
|
+
}
|
|
41
|
+
};
|
|
42
|
+
_QueryCache_dependencyTrackerEvicted.set(this, (key) => {
|
|
43
|
+
__classPrivateFieldGet(this, _QueryCache_entries, "f").delete(key);
|
|
44
|
+
});
|
|
45
|
+
this.ignoreQuery = (query) => {
|
|
46
|
+
return ignore.some((prefix) => query.startsWith(prefix));
|
|
47
|
+
};
|
|
48
|
+
// The next simplest step is to create a specific implementation for invalidating
|
|
49
|
+
// the expensive track list queries only when constraints data in a write overlaps with read constraints.
|
|
50
|
+
//
|
|
51
|
+
// As well as either:
|
|
52
|
+
// a. removeing the big view (since we'll have our cache)
|
|
53
|
+
// b. incrementally updating the view on insert by the EventImporter
|
|
54
|
+
//
|
|
55
|
+
// We'll not try to tackle any generalized approach until we have a proof of concept working.
|
|
56
|
+
this.invalidate = (queriedTables) => {
|
|
57
|
+
for (const table of queriedTables) {
|
|
58
|
+
const keys = __classPrivateFieldGet(this, _QueryCache_dependencies, "f").get(table);
|
|
59
|
+
if (keys == null) {
|
|
60
|
+
continue;
|
|
61
|
+
}
|
|
62
|
+
for (const k of keys) {
|
|
63
|
+
__classPrivateFieldGet(this, _QueryCache_entries, "f").delete(k);
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
};
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
_QueryCache_entries = new WeakMap(), _QueryCache_dependencies = new WeakMap(), _QueryCache_dependencyTrackerEvicted = new WeakMap();
|
|
70
|
+
export default QueryCache;
|
|
71
|
+
//# sourceMappingURL=QueryCache.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"QueryCache.js","sourceRoot":"","sources":["../src/QueryCache.ts"],"names":[],"mappings":";;;;;;AAAA,OAAO,QAAQ,EAAE,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAA;AAQ7D,IAAU,OAAO,CAGhB;AAHD,WAAU,OAAO;AAGjB,CAAC,EAHS,OAAO,KAAP,OAAO,QAGhB;AAKD,MAAM,MAAM,GAAG,CAAC,OAAO,EAAE,UAAU,EAAE,QAAQ,EAAE,WAAW,EAAE,SAAS,CAAC,CAAA;AAEtE,yDAAyD;AACzD,MAAM,SAAS,GAAG,GAAG,CAAA;AACrB,MAAqB,UAAU;IAA/B;QACE,8BAAW,IAAI,QAAQ,CAAgB,SAAS,CAAC,EAAA;QACjD,mCAAgB,IAAI,GAAG,EAAiC,EAAA;QAExD,WAAM,GAAG,CAAC,GAAW,EAAE,UAAqB,EAAY,EAAE;YACxD,IAAI,UAAU,IAAI,IAAI,EAAE;gBACtB,OAAO,GAAe,CAAA;aACvB;YAED,IAAI,KAAK,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE;gBAC7B,OAAO,CAAC,GAAG,GAAG,IAAI,GAAG,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAa,CAAA;aACxD;YAED,OAAO,CAAC,GAAG,GAAG,IAAI,GAAG,MAAM,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAa,CAAA;QACxE,CAAC,CAAA;QAED,QAAG,GAAG,CAAC,GAAa,EAAE,EAAE;YACtB,OAAO,uBAAA,IAAI,2BAAS,CAAC,GAAG,CAAC,GAAG,CAAC,CAAA;QAC/B,CAAC,CAAA;QAED,QAAG,GAAG,CAAC,aAAuB,EAAE,GAAa,EAAE,OAAY,EAAE,EAAE;YAC7D,uBAAA,IAAI,2BAAS,CAAC,GAAG,CAAC,GAAG,EAAE,OAAO,CAAC,CAAA;YAC/B,KAAK,MAAM,KAAK,IAAI,aAAa,EAAE;gBACjC,IAAI,IAAI,GAAG,uBAAA,IAAI,gCAAc,CAAC,GAAG,CAAC,KAAK,CAAC,CAAA;gBACxC,IAAI,IAAI,IAAI,IAAI,EAAE;oBAChB,IAAI,GAAG,IAAI,QAAQ,CAAC,SAAS,CAAC,CAAA;oBAC9B,IAAI,CAAC,OAAO,GAAG,uBAAA,IAAI,4CAA0B,CAAA;oBAC7C,uBAAA,IAAI,gCAAc,CAAC,GAAG,CAAC,KAAK,EAAE,IAAI,CAAC,CAAA;iBACpC;gBACD,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAA;aACd;QACH,CAAC,CAAA;QAED,+CAA4B,CAAC,GAAa,EAAE,EAAE;YAC5C,uBAAA,IAAI,2BAAS,CAAC,MAAM,CAAC,GAAG,CAAC,CAAA;QAC3B,CAAC,EAAA;QAED,gBAAW,GAAG,CAAC,KAAa,EAAE,EAAE;YAC9B,OAAO,MAAM,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,KAAK,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,CAAA;QAC1D,CAAC,CAAA;QAED,iFAAiF;QACjF,yGAAyG;QACzG,EAAE;QACF,qBAAqB;QACrB,yDAAyD;QACzD,oEAAoE;QACpE,EAAE;QACF,6FAA6F;QAC7F,eAAU,GAAG,CAAC,aAAuB,EAAE,EAAE;YACvC,KAAK,MAAM,KAAK,IAAI,aAAa,EAAE;gBACjC,MAAM,IAAI,GAAG,uBAAA,IAAI,gCAAc,CAAC,GAAG,CAAC,KAAK,CAAC,CAAA;gBAC1C,IAAI,IAAI,IAAI,IAAI,EAAE;oBAChB,SAAQ;iBACT;gBACD,KAAK,MAAM,CAAC,IAAI,IAAI,EAAE;oBACpB,uBAAA,IAAI,2BAAS,CAAC,MAAM,CAAC,CAAC,CAAC,CAAA;iBACxB;aACF;QACH,CAAC,CAAA;IACH,CAAC;CAAA;;eA5DoB,UAAU"}
|
|
@@ -0,0 +1,141 @@
|
|
|
1
|
+
import React from 'react';
|
|
2
|
+
import * as LiveStore from '../../index.js';
|
|
3
|
+
export type Todo = {
|
|
4
|
+
id: string;
|
|
5
|
+
text: string | null;
|
|
6
|
+
completed: boolean;
|
|
7
|
+
};
|
|
8
|
+
export type Filter = 'all' | 'active' | 'completed';
|
|
9
|
+
export type AppState = {
|
|
10
|
+
newTodoText: string;
|
|
11
|
+
filter: Filter;
|
|
12
|
+
};
|
|
13
|
+
export declare const globalQueryDefs: {
|
|
14
|
+
appState: LiveStore.QueryDefinition;
|
|
15
|
+
};
|
|
16
|
+
export declare const schema: {
|
|
17
|
+
tables: {
|
|
18
|
+
todos: {
|
|
19
|
+
columns: {
|
|
20
|
+
id: {
|
|
21
|
+
type: "text";
|
|
22
|
+
primaryKey: true;
|
|
23
|
+
};
|
|
24
|
+
text: {
|
|
25
|
+
type: "text";
|
|
26
|
+
default: string;
|
|
27
|
+
nullable: false;
|
|
28
|
+
};
|
|
29
|
+
completed: {
|
|
30
|
+
type: "boolean";
|
|
31
|
+
default: false;
|
|
32
|
+
nullable: false;
|
|
33
|
+
};
|
|
34
|
+
};
|
|
35
|
+
};
|
|
36
|
+
app: {
|
|
37
|
+
columns: {
|
|
38
|
+
id: {
|
|
39
|
+
type: "text";
|
|
40
|
+
primaryKey: true;
|
|
41
|
+
};
|
|
42
|
+
newTodoText: {
|
|
43
|
+
type: "text";
|
|
44
|
+
default: string;
|
|
45
|
+
nullable: true;
|
|
46
|
+
};
|
|
47
|
+
filter: {
|
|
48
|
+
type: "text";
|
|
49
|
+
default: string;
|
|
50
|
+
nullable: false;
|
|
51
|
+
};
|
|
52
|
+
};
|
|
53
|
+
};
|
|
54
|
+
};
|
|
55
|
+
materializedViews: {};
|
|
56
|
+
actions: {
|
|
57
|
+
addTodo: {
|
|
58
|
+
statement: {
|
|
59
|
+
sql: string;
|
|
60
|
+
writeTables: string[];
|
|
61
|
+
};
|
|
62
|
+
};
|
|
63
|
+
completeTodo: {
|
|
64
|
+
statement: {
|
|
65
|
+
sql: string;
|
|
66
|
+
writeTables: string[];
|
|
67
|
+
};
|
|
68
|
+
};
|
|
69
|
+
uncompleteTodo: {
|
|
70
|
+
statement: {
|
|
71
|
+
sql: string;
|
|
72
|
+
writeTables: string[];
|
|
73
|
+
};
|
|
74
|
+
};
|
|
75
|
+
deleteTodo: {
|
|
76
|
+
statement: {
|
|
77
|
+
sql: string;
|
|
78
|
+
writeTables: string[];
|
|
79
|
+
};
|
|
80
|
+
};
|
|
81
|
+
clearCompleted: {
|
|
82
|
+
statement: {
|
|
83
|
+
sql: string;
|
|
84
|
+
writeTables: string[];
|
|
85
|
+
};
|
|
86
|
+
};
|
|
87
|
+
updateNewTodoText: {
|
|
88
|
+
statement: {
|
|
89
|
+
sql: string;
|
|
90
|
+
writeTables: string[];
|
|
91
|
+
};
|
|
92
|
+
};
|
|
93
|
+
setFilter: {
|
|
94
|
+
statement: {
|
|
95
|
+
sql: string;
|
|
96
|
+
writeTables: string[];
|
|
97
|
+
};
|
|
98
|
+
};
|
|
99
|
+
};
|
|
100
|
+
} & {
|
|
101
|
+
tables: {
|
|
102
|
+
__events: {
|
|
103
|
+
readonly columns: {
|
|
104
|
+
readonly id: {
|
|
105
|
+
readonly type: "text";
|
|
106
|
+
readonly primaryKey: true;
|
|
107
|
+
};
|
|
108
|
+
readonly type: {
|
|
109
|
+
readonly type: "text";
|
|
110
|
+
readonly nullable: false;
|
|
111
|
+
};
|
|
112
|
+
readonly args: {
|
|
113
|
+
readonly type: "text";
|
|
114
|
+
readonly nullable: false;
|
|
115
|
+
};
|
|
116
|
+
};
|
|
117
|
+
};
|
|
118
|
+
livestore__event_cursor: {
|
|
119
|
+
readonly columns: {
|
|
120
|
+
readonly id: {
|
|
121
|
+
readonly type: "text";
|
|
122
|
+
readonly primaryKey: true;
|
|
123
|
+
};
|
|
124
|
+
readonly cursor: {
|
|
125
|
+
readonly type: "text";
|
|
126
|
+
readonly nullable: false;
|
|
127
|
+
};
|
|
128
|
+
};
|
|
129
|
+
};
|
|
130
|
+
};
|
|
131
|
+
};
|
|
132
|
+
export declare const makeTodoMvc: () => Promise<{
|
|
133
|
+
wrapper: ({ children }: any) => React.JSX.Element;
|
|
134
|
+
AppSchema: import("../../schema.js").ComponentStateSchema<{
|
|
135
|
+
username: string;
|
|
136
|
+
} & {
|
|
137
|
+
id: string;
|
|
138
|
+
}>;
|
|
139
|
+
store: LiveStore.Store<LiveStore.BaseGraphQLContext>;
|
|
140
|
+
}>;
|
|
141
|
+
//# sourceMappingURL=fixture.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fixture.d.ts","sourceRoot":"","sources":["../../../src/__tests__/react/fixture.tsx"],"names":[],"mappings":"AAEA,OAAO,KAAK,MAAM,OAAO,CAAA;AAEzB,OAAO,KAAK,SAAS,MAAM,gBAAgB,CAAA;AAM3C,MAAM,MAAM,IAAI,GAAG;IACjB,EAAE,EAAE,MAAM,CAAA;IACV,IAAI,EAAE,MAAM,GAAG,IAAI,CAAA;IACnB,SAAS,EAAE,OAAO,CAAA;CACnB,CAAA;AAED,MAAM,MAAM,MAAM,GAAG,KAAK,GAAG,QAAQ,GAAG,WAAW,CAAA;AAEnD,MAAM,MAAM,QAAQ,GAAG;IACrB,WAAW,EAAE,MAAM,CAAA;IACnB,MAAM,EAAE,MAAM,CAAA;CACf,CAAA;AAcD,eAAO,MAAM,eAAe;;CAE3B,CAAA;AAED,eAAO,MAAM,MAAM;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAmCjB,CAAA;AAEF,eAAO,MAAM,WAAW;4BAwBS,GAAG;;kBAvBQ,MAAM;;;;;EA4BjD,CAAA"}
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
import { mapObjectValues } from '@livestore/utils';
|
|
2
|
+
import * as otel from '@opentelemetry/api';
|
|
3
|
+
import React from 'react';
|
|
4
|
+
import * as LiveStore from '../../index.js';
|
|
5
|
+
import { sql } from '../../index.js';
|
|
6
|
+
import * as LiveStoreReact from '../../react/index.js';
|
|
7
|
+
const mockOtelCtx = otel.context.active();
|
|
8
|
+
const appState = (store) => store
|
|
9
|
+
.querySQL(() => `select newTodoText, filter from app;`, ['app'], undefined, undefined, undefined, mockOtelCtx)
|
|
10
|
+
.getFirstRow();
|
|
11
|
+
export const globalQueryDefs = {
|
|
12
|
+
appState,
|
|
13
|
+
};
|
|
14
|
+
export const schema = LiveStore.defineSchema({
|
|
15
|
+
tables: {
|
|
16
|
+
todos: {
|
|
17
|
+
columns: {
|
|
18
|
+
id: { type: 'text', primaryKey: true },
|
|
19
|
+
text: { type: 'text', default: '', nullable: false },
|
|
20
|
+
completed: { type: 'boolean', default: false, nullable: false },
|
|
21
|
+
},
|
|
22
|
+
},
|
|
23
|
+
app: {
|
|
24
|
+
columns: {
|
|
25
|
+
id: { type: 'text', primaryKey: true },
|
|
26
|
+
newTodoText: { type: 'text', default: '', nullable: true },
|
|
27
|
+
filter: { type: 'text', default: 'all', nullable: false },
|
|
28
|
+
},
|
|
29
|
+
},
|
|
30
|
+
},
|
|
31
|
+
materializedViews: {},
|
|
32
|
+
actions: {
|
|
33
|
+
// TODO: fix these actions to make them have write annotatinos
|
|
34
|
+
addTodo: {
|
|
35
|
+
statement: {
|
|
36
|
+
sql: sql `INSERT INTO todos (id, text, completed) VALUES ($id, $text, false);`,
|
|
37
|
+
writeTables: ['app'],
|
|
38
|
+
},
|
|
39
|
+
},
|
|
40
|
+
completeTodo: { statement: { sql: sql `UPDATE todos SET completed = true WHERE id = $id;`, writeTables: ['app'] } },
|
|
41
|
+
uncompleteTodo: {
|
|
42
|
+
statement: { sql: sql `UPDATE todos SET completed = false WHERE id = $id;`, writeTables: ['app'] },
|
|
43
|
+
},
|
|
44
|
+
deleteTodo: { statement: { sql: sql `DELETE FROM todos WHERE id = $id;`, writeTables: ['app'] } },
|
|
45
|
+
clearCompleted: { statement: { sql: sql `DELETE FROM todos WHERE completed = true;`, writeTables: ['app'] } },
|
|
46
|
+
updateNewTodoText: { statement: { sql: sql `UPDATE app SET newTodoText = $text;`, writeTables: ['app'] } },
|
|
47
|
+
setFilter: { statement: { sql: sql `UPDATE app SET filter = $filter;`, writeTables: ['app'] } },
|
|
48
|
+
},
|
|
49
|
+
});
|
|
50
|
+
export const makeTodoMvc = async () => {
|
|
51
|
+
const AppSchema = LiveStore.defineComponentStateSchema({
|
|
52
|
+
componentType: 'UserInfo',
|
|
53
|
+
columns: {
|
|
54
|
+
username: { type: 'text', default: '' },
|
|
55
|
+
},
|
|
56
|
+
});
|
|
57
|
+
const store = await LiveStore.createStore({
|
|
58
|
+
schema,
|
|
59
|
+
backendOptions: { type: 'web-in-memory' },
|
|
60
|
+
boot: async (backend) => {
|
|
61
|
+
backend.execute(sql `INSERT INTO app (newTodoText, filter) VALUES ('', 'all');`);
|
|
62
|
+
// NOTE we can't insert into components__UserInfo yet because the table doesn't exist yet
|
|
63
|
+
// backend.execute(sql`INSERT INTO components__UserInfo (id, username) VALUES ('u1', 'username_u1');`)
|
|
64
|
+
// backend.execute(sql`INSERT INTO components__UserInfo (id, username) VALUES ('u2', 'username_u2');`)
|
|
65
|
+
},
|
|
66
|
+
});
|
|
67
|
+
const globalQueries = mapObjectValues(globalQueryDefs, (_, queryDef) => queryDef(store));
|
|
68
|
+
const storeContext = { store, globalQueries };
|
|
69
|
+
const wrapper = ({ children }) => (React.createElement(LiveStoreReact.LiveStoreContext.Provider, { value: storeContext }, children));
|
|
70
|
+
return { wrapper, AppSchema, store };
|
|
71
|
+
};
|
|
72
|
+
//# sourceMappingURL=fixture.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fixture.js","sourceRoot":"","sources":["../../../src/__tests__/react/fixture.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"useLiveStoreComponent.test.d.ts","sourceRoot":"","sources":["../../../src/__tests__/react/useLiveStoreComponent.test.tsx"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
import { act, renderHook } from '@testing-library/react';
|
|
2
|
+
import { describe, expect, it } from 'vitest';
|
|
3
|
+
import { sql } from '../../index.js';
|
|
4
|
+
import * as LiveStoreReact from '../../react/index.js';
|
|
5
|
+
import { makeTodoMvc } from './fixture.js';
|
|
6
|
+
describe('useLiveStoreComponent', () => {
|
|
7
|
+
it('should update the data based on component key', async () => {
|
|
8
|
+
let renderCount = 0;
|
|
9
|
+
const { wrapper, AppSchema, store } = await makeTodoMvc();
|
|
10
|
+
const { result, rerender } = renderHook((userId) => {
|
|
11
|
+
renderCount++;
|
|
12
|
+
return LiveStoreReact.useLiveStoreComponent({
|
|
13
|
+
stateSchema: AppSchema,
|
|
14
|
+
componentKey: { name: 'UserInfo', id: userId },
|
|
15
|
+
queries: () => ({}),
|
|
16
|
+
});
|
|
17
|
+
}, { wrapper, initialProps: 'u1' });
|
|
18
|
+
expect(result.current.state.id).toBe('u1');
|
|
19
|
+
expect(result.current.state.username).toBe('');
|
|
20
|
+
expect(renderCount).toBe(1);
|
|
21
|
+
act(() => {
|
|
22
|
+
void store.execute(sql `INSERT INTO components__UserInfo (id, username) VALUES ('u2', 'username_u2');`);
|
|
23
|
+
});
|
|
24
|
+
rerender('u2');
|
|
25
|
+
expect(result.current.state.id).toBe('u2');
|
|
26
|
+
expect(result.current.state.username).toBe('username_u2');
|
|
27
|
+
expect(renderCount).toBe(2);
|
|
28
|
+
});
|
|
29
|
+
it('should update the data reactively - via setState', async () => {
|
|
30
|
+
let renderCount = 0;
|
|
31
|
+
const { wrapper, AppSchema } = await makeTodoMvc();
|
|
32
|
+
const { result } = renderHook((userId) => {
|
|
33
|
+
renderCount++;
|
|
34
|
+
return LiveStoreReact.useLiveStoreComponent({
|
|
35
|
+
stateSchema: AppSchema,
|
|
36
|
+
componentKey: { name: 'UserInfo', id: userId },
|
|
37
|
+
queries: () => ({}),
|
|
38
|
+
});
|
|
39
|
+
}, { wrapper, initialProps: 'u1' });
|
|
40
|
+
expect(result.current.state.id).toBe('u1');
|
|
41
|
+
expect(result.current.state.username).toBe('');
|
|
42
|
+
expect(renderCount).toBe(1);
|
|
43
|
+
act(() => {
|
|
44
|
+
result.current.setState.username('username_u1_hello');
|
|
45
|
+
});
|
|
46
|
+
// act(() => {
|
|
47
|
+
// store.execute(sql`UPDATE components__UserInfo SET username = 'username_u1_hello' WHERE id = 'u1';`)
|
|
48
|
+
// })
|
|
49
|
+
expect(result.current.state.id).toBe('u1');
|
|
50
|
+
expect(result.current.state.username).toBe('username_u1_hello');
|
|
51
|
+
expect(renderCount).toBe(2);
|
|
52
|
+
});
|
|
53
|
+
it('should update the data reactively - via raw store update', async () => {
|
|
54
|
+
let renderCount = 0;
|
|
55
|
+
const { wrapper, AppSchema, store } = await makeTodoMvc();
|
|
56
|
+
const { result } = renderHook((userId) => {
|
|
57
|
+
renderCount++;
|
|
58
|
+
return LiveStoreReact.useLiveStoreComponent({
|
|
59
|
+
stateSchema: AppSchema,
|
|
60
|
+
componentKey: { name: 'UserInfo', id: userId },
|
|
61
|
+
queries: () => ({}),
|
|
62
|
+
});
|
|
63
|
+
}, { wrapper, initialProps: 'u1' });
|
|
64
|
+
expect(result.current.state.id).toBe('u1');
|
|
65
|
+
expect(result.current.state.username).toBe('');
|
|
66
|
+
expect(renderCount).toBe(1);
|
|
67
|
+
act(() => {
|
|
68
|
+
result.current.setState.username('username_u1_hello');
|
|
69
|
+
});
|
|
70
|
+
act(() => {
|
|
71
|
+
void store.execute(sql `UPDATE components__UserInfo SET username = 'username_u1_hello' WHERE id = 'u1';`);
|
|
72
|
+
});
|
|
73
|
+
expect(result.current.state.id).toBe('u1');
|
|
74
|
+
expect(result.current.state.username).toBe('username_u1_hello');
|
|
75
|
+
expect(renderCount).toBe(2);
|
|
76
|
+
});
|
|
77
|
+
});
|
|
78
|
+
//# sourceMappingURL=useLiveStoreComponent.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"useLiveStoreComponent.test.js","sourceRoot":"","sources":["../../../src/__tests__/react/useLiveStoreComponent.test.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"useRiffleComponent.test.d.ts","sourceRoot":"","sources":["../../../src/__tests__/react/useRiffleComponent.test.tsx"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
import { act, renderHook } from '@testing-library/react';
|
|
2
|
+
import { describe, expect, it } from 'vitest';
|
|
3
|
+
import { sql } from '../../index.js';
|
|
4
|
+
import * as RiffleReact from '../../react/index.js';
|
|
5
|
+
import { makeTodoMvc } from './fixture.js';
|
|
6
|
+
describe('useRiffleComponent', () => {
|
|
7
|
+
it('should update the data based on component key', async () => {
|
|
8
|
+
let renderCount = 0;
|
|
9
|
+
const { wrapper, AppSchema, store } = await makeTodoMvc();
|
|
10
|
+
const { result, rerender } = renderHook((userId) => {
|
|
11
|
+
renderCount++;
|
|
12
|
+
return RiffleReact.useRiffleComponent({
|
|
13
|
+
stateSchema: AppSchema,
|
|
14
|
+
componentKey: { name: 'UserInfo', id: userId },
|
|
15
|
+
queries: () => ({}),
|
|
16
|
+
});
|
|
17
|
+
}, { wrapper, initialProps: 'u1' });
|
|
18
|
+
expect(result.current.state.id).toBe('u1');
|
|
19
|
+
expect(result.current.state.username).toBe('');
|
|
20
|
+
expect(renderCount).toBe(1);
|
|
21
|
+
act(() => {
|
|
22
|
+
void store.execute(sql `INSERT INTO components__UserInfo (id, username) VALUES ('u2', 'username_u2');`);
|
|
23
|
+
});
|
|
24
|
+
rerender('u2');
|
|
25
|
+
expect(result.current.state.id).toBe('u2');
|
|
26
|
+
expect(result.current.state.username).toBe('username_u2');
|
|
27
|
+
expect(renderCount).toBe(2);
|
|
28
|
+
});
|
|
29
|
+
it('should update the data reactively - via setState', async () => {
|
|
30
|
+
let renderCount = 0;
|
|
31
|
+
const { wrapper, AppSchema } = await makeTodoMvc();
|
|
32
|
+
const { result } = renderHook((userId) => {
|
|
33
|
+
renderCount++;
|
|
34
|
+
return RiffleReact.useRiffleComponent({
|
|
35
|
+
stateSchema: AppSchema,
|
|
36
|
+
componentKey: { name: 'UserInfo', id: userId },
|
|
37
|
+
queries: () => ({}),
|
|
38
|
+
});
|
|
39
|
+
}, { wrapper, initialProps: 'u1' });
|
|
40
|
+
expect(result.current.state.id).toBe('u1');
|
|
41
|
+
expect(result.current.state.username).toBe('');
|
|
42
|
+
expect(renderCount).toBe(1);
|
|
43
|
+
act(() => {
|
|
44
|
+
result.current.setState.username('username_u1_hello');
|
|
45
|
+
});
|
|
46
|
+
// act(() => {
|
|
47
|
+
// store.execute(sql`UPDATE components__UserInfo SET username = 'username_u1_hello' WHERE id = 'u1';`)
|
|
48
|
+
// })
|
|
49
|
+
expect(result.current.state.id).toBe('u1');
|
|
50
|
+
expect(result.current.state.username).toBe('username_u1_hello');
|
|
51
|
+
expect(renderCount).toBe(2);
|
|
52
|
+
});
|
|
53
|
+
it('should update the data reactively - via raw store update', async () => {
|
|
54
|
+
let renderCount = 0;
|
|
55
|
+
const { wrapper, AppSchema, store } = await makeTodoMvc();
|
|
56
|
+
const { result } = renderHook((userId) => {
|
|
57
|
+
renderCount++;
|
|
58
|
+
return RiffleReact.useRiffleComponent({
|
|
59
|
+
stateSchema: AppSchema,
|
|
60
|
+
componentKey: { name: 'UserInfo', id: userId },
|
|
61
|
+
queries: () => ({}),
|
|
62
|
+
});
|
|
63
|
+
}, { wrapper, initialProps: 'u1' });
|
|
64
|
+
expect(result.current.state.id).toBe('u1');
|
|
65
|
+
expect(result.current.state.username).toBe('');
|
|
66
|
+
expect(renderCount).toBe(1);
|
|
67
|
+
act(() => {
|
|
68
|
+
result.current.setState.username('username_u1_hello');
|
|
69
|
+
});
|
|
70
|
+
act(() => {
|
|
71
|
+
void store.execute(sql `UPDATE components__UserInfo SET username = 'username_u1_hello' WHERE id = 'u1';`);
|
|
72
|
+
});
|
|
73
|
+
expect(result.current.state.id).toBe('u1');
|
|
74
|
+
expect(result.current.state.username).toBe('username_u1_hello');
|
|
75
|
+
expect(renderCount).toBe(2);
|
|
76
|
+
});
|
|
77
|
+
});
|
|
78
|
+
//# sourceMappingURL=useRiffleComponent.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"useRiffleComponent.test.js","sourceRoot":"","sources":["../../../src/__tests__/react/useRiffleComponent.test.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"reactive.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/reactive.test.ts"],"names":[],"mappings":""}
|