Skip to content

Commit 9533728

Browse files
authored
Merge pull request #1121 from cardstack/cs-6457-implement-search-query-support-in-sqlite-adapter
SQLite implementation of query support for DB based index
2 parents 2f163c1 + 5feaf04 commit 9533728

File tree

12 files changed

+1092
-101
lines changed

12 files changed

+1092
-101
lines changed

packages/base/code-ref.gts

+15-13
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ import {
1212
type SerializeOpts,
1313
type JSONAPISingleResourceDocument,
1414
} from './card-api';
15+
import { ResolvedCodeRef } from '@cardstack/runtime-common';
1516

1617
class BaseView extends Component<typeof CodeRefField> {
1718
<template>
@@ -24,38 +25,39 @@ class BaseView extends Component<typeof CodeRefField> {
2425
</template>
2526
}
2627

27-
type CardId = { name: string; module: string };
28-
2928
export default class CodeRefField extends FieldDef {
30-
static [primitive]: CardId;
29+
static [primitive]: ResolvedCodeRef;
3130

3231
static [serialize](
33-
cardRef: CardId,
32+
codeRef: ResolvedCodeRef,
3433
_doc: JSONAPISingleResourceDocument,
3534
_visited?: Set<string>,
3635
opts?: SerializeOpts,
3736
) {
3837
return {
39-
...cardRef,
38+
...codeRef,
4039
...(opts?.maybeRelativeURL
41-
? { module: opts.maybeRelativeURL(cardRef.module) }
40+
? { module: opts.maybeRelativeURL(codeRef.module) }
4241
: {}),
4342
};
4443
}
4544
static async [deserialize]<T extends BaseDefConstructor>(
4645
this: T,
47-
cardRef: CardId,
46+
codeRef: ResolvedCodeRef,
4847
): Promise<BaseInstanceType<T>> {
49-
return { ...cardRef } as BaseInstanceType<T>; // return a new object so that the model cannot be mutated from the outside
48+
return { ...codeRef } as BaseInstanceType<T>; // return a new object so that the model cannot be mutated from the outside
5049
}
51-
static [queryableValue](cardRef: CardId | undefined, stack: CardDef[] = []) {
52-
if (cardRef) {
50+
static [queryableValue](
51+
codeRef: ResolvedCodeRef | undefined,
52+
stack: CardDef[] = [],
53+
) {
54+
if (codeRef) {
5355
// if a stack is passed in, use the containing card to resolve relative references
5456
let moduleHref =
5557
stack.length > 0
56-
? new URL(cardRef.module, stack[0][relativeTo]).href
57-
: cardRef.module;
58-
return `${moduleHref}/${cardRef.name}`;
58+
? new URL(codeRef.module, stack[0][relativeTo]).href
59+
: codeRef.module;
60+
return `${moduleHref}/${codeRef.name}`;
5961
}
6062
return undefined;
6163
}

packages/host/app/lib/SQLiteAdapter.ts

+4
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,10 @@ export default class SQLiteAdapter implements DBAdapter {
1414
private _sqlite: typeof SQLiteWorker | undefined;
1515
private _dbId: string | undefined;
1616

17+
// TODO: one difference that I'm seeing is that it looks like "json_each" is
18+
// actually similar to "json_each_text" in postgres. i think we might need to
19+
// transform the SQL we run to deal with this difference.-
20+
1721
constructor(private schemaSQL?: string) {}
1822

1923
async startClient() {

packages/host/tests/helpers/const.ts

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
import { type RealmInfo } from '@cardstack/runtime-common';
2+
export const testRealmURL = `http://test-realm/test/`;
3+
export const testRealmInfo: RealmInfo = {
4+
name: 'Unnamed Workspace',
5+
backgroundURL: null,
6+
iconURL: null,
7+
};

packages/host/tests/helpers/index.gts

+3-10
Original file line numberDiff line numberDiff line change
@@ -15,9 +15,9 @@ import {
1515
LooseSingleCardDocument,
1616
baseRealm,
1717
createResponse,
18-
RealmInfo,
1918
RealmPermissions,
2019
Deferred,
20+
type RealmInfo,
2121
type TokenClaims,
2222
} from '@cardstack/runtime-common';
2323

@@ -46,14 +46,14 @@ import {
4646
type FieldDef,
4747
} from 'https://cardstack.com/base/card-api';
4848

49+
import { testRealmInfo, testRealmURL } from './const';
4950
import percySnapshot from './percy-snapshot';
5051

5152
import { renderComponent } from './render-component';
5253
import { WebMessageStream, messageCloseHandler } from './stream';
5354
import visitOperatorMode from './visit-operator-mode';
5455

55-
export { percySnapshot };
56-
export { visitOperatorMode };
56+
export { visitOperatorMode, testRealmURL, testRealmInfo, percySnapshot };
5757
export * from './indexer';
5858

5959
const waiter = buildWaiter('@cardstack/host/test/helpers/index:onFetch-waiter');
@@ -144,13 +144,6 @@ export interface Dir {
144144
[name: string]: string | Dir;
145145
}
146146

147-
export const testRealmURL = `http://test-realm/test/`;
148-
export const testRealmInfo: RealmInfo = {
149-
name: 'Unnamed Workspace',
150-
backgroundURL: null,
151-
iconURL: null,
152-
};
153-
154147
export interface CardDocFiles {
155148
[filename: string]: LooseSingleCardDocument;
156149
}

packages/host/tests/helpers/indexer.ts

+138-14
Original file line numberDiff line numberDiff line change
@@ -1,27 +1,151 @@
1+
import isEqual from 'lodash/isEqual';
2+
13
import {
24
IndexerDBClient,
35
asExpressions,
46
addExplicitParens,
57
separatedByCommas,
8+
loaderFor,
9+
internalKeyFor,
10+
Deferred,
11+
identifyCard,
12+
apiFor,
13+
loadCard,
14+
baseCardRef,
15+
type CodeRef,
16+
type CardResource,
17+
type Expression,
618
type IndexedCardsTable,
719
type RealmVersionsTable,
820
} from '@cardstack/runtime-common';
921

22+
import type { CardDef } from 'https://cardstack.com/base/card-api';
23+
24+
import { testRealmURL } from './const';
25+
26+
let defaultIndexEntry = {
27+
realm_version: 1,
28+
realm_url: testRealmURL,
29+
};
30+
31+
let typesCache = new WeakMap<typeof CardDef, Promise<string[]>>();
32+
33+
// this leverages the logic from current-run.ts to generate the types for a card
34+
// that are serialized in the same manner as they appear in the index
35+
export async function getTypes(instance: CardDef): Promise<string[]> {
36+
let loader = loaderFor(instance);
37+
let card = Reflect.getPrototypeOf(instance)!.constructor as typeof CardDef;
38+
let cached = typesCache.get(card);
39+
if (cached) {
40+
return await cached;
41+
}
42+
let ref = identifyCard(card);
43+
if (!ref) {
44+
throw new Error(`could not identify card ${card.name}`);
45+
}
46+
let deferred = new Deferred<string[]>();
47+
typesCache.set(card, deferred.promise);
48+
let types: string[] = [];
49+
let fullRef: CodeRef = ref;
50+
while (fullRef) {
51+
let loadedCard, loadedCardRef;
52+
loadedCard = await loadCard(fullRef, { loader });
53+
loadedCardRef = identifyCard(loadedCard);
54+
if (!loadedCardRef) {
55+
throw new Error(`could not identify card ${loadedCard.name}`);
56+
}
57+
types.push(internalKeyFor(loadedCardRef, undefined));
58+
if (!isEqual(loadedCardRef, baseCardRef)) {
59+
fullRef = {
60+
type: 'ancestorOf',
61+
card: loadedCardRef,
62+
};
63+
} else {
64+
break;
65+
}
66+
}
67+
deferred.fulfill(types);
68+
return types;
69+
}
70+
71+
export async function serializeCard(card: CardDef): Promise<CardResource> {
72+
let api = await apiFor(card);
73+
return api.serializeCard(card).data as CardResource;
74+
}
75+
76+
type TestIndexRow =
77+
| (Pick<IndexedCardsTable, 'card_url'> &
78+
Partial<Omit<IndexedCardsTable, 'card_url'>>)
79+
| CardDef
80+
| {
81+
card: CardDef;
82+
data: Partial<
83+
Omit<IndexedCardsTable, 'card_url' | 'pristine_doc' | 'types'>
84+
>;
85+
};
86+
87+
// There are 3 ways to setup an index:
88+
// 1. provide the raw data for each row in the indexed_cards table
89+
// 2. provide a card instance for each row in the indexed_cards table
90+
// 3. provide an object { card, data } where the card instance is used for each
91+
// row in the indexed_cards table, as well as any additional fields that you
92+
// wish to set from the `data` object.
93+
//
94+
// the realm version table will default to version 1 of the testRealmURL if no
95+
// value is supplied
96+
export async function setupIndex(
97+
client: IndexerDBClient,
98+
indexRows: TestIndexRow[],
99+
): Promise<void>;
10100
export async function setupIndex(
11101
client: IndexerDBClient,
12102
versionRows: RealmVersionsTable[],
13-
// only assert that the non-null columns need to be present in rows objects
14-
indexRows: (Pick<
15-
IndexedCardsTable,
16-
'card_url' | 'realm_version' | 'realm_url'
17-
> &
18-
Partial<
19-
Omit<IndexedCardsTable, 'card_url' | 'realm_version' | 'realm_url'>
20-
>)[],
21-
) {
22-
let indexedCardsExpressions = indexRows.map((r) =>
23-
asExpressions(r, {
24-
jsonFields: ['deps', 'types', 'pristine_doc', 'error_doc', 'search_doc'],
103+
indexRows: TestIndexRow[],
104+
): Promise<void>;
105+
export async function setupIndex(
106+
client: IndexerDBClient,
107+
maybeVersionRows: RealmVersionsTable[] | TestIndexRow[],
108+
indexRows?: TestIndexRow[],
109+
): Promise<void> {
110+
let versionRows: RealmVersionsTable[];
111+
if (!indexRows) {
112+
versionRows = [{ realm_url: testRealmURL, current_version: 1 }];
113+
indexRows = maybeVersionRows as TestIndexRow[];
114+
} else {
115+
versionRows = maybeVersionRows as RealmVersionsTable[];
116+
}
117+
let indexedCardsExpressions = await Promise.all(
118+
indexRows.map(async (r) => {
119+
let row: Pick<IndexedCardsTable, 'card_url'> &
120+
Partial<Omit<IndexedCardsTable, 'card_url'>>;
121+
if ('card_url' in r) {
122+
row = r;
123+
} else if ('card' in r) {
124+
row = {
125+
card_url: r.card.id,
126+
pristine_doc: await serializeCard(r.card),
127+
types: await getTypes(r.card),
128+
...r.data,
129+
};
130+
} else {
131+
row = {
132+
card_url: r.id,
133+
pristine_doc: await serializeCard(r),
134+
types: await getTypes(r),
135+
};
136+
}
137+
return asExpressions(
138+
{ ...defaultIndexEntry, ...row },
139+
{
140+
jsonFields: [
141+
'deps',
142+
'types',
143+
'pristine_doc',
144+
'error_doc',
145+
'search_doc',
146+
],
147+
},
148+
);
25149
}),
26150
);
27151
let versionExpressions = versionRows.map((r) => asExpressions(r));
@@ -38,7 +162,7 @@ export async function setupIndex(
38162
addExplicitParens(separatedByCommas(row.valueExpressions)),
39163
),
40164
),
41-
]);
165+
] as Expression);
42166
}
43167

44168
if (versionExpressions.length > 0) {
@@ -53,6 +177,6 @@ export async function setupIndex(
53177
addExplicitParens(separatedByCommas(row.valueExpressions)),
54178
),
55179
),
56-
]);
180+
] as Expression);
57181
}
58182
}

packages/host/tests/unit/index-db-test.ts packages/host/tests/unit/indexer-test.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ const testRealmURL2 = `http://test-realm/test2/`;
1717

1818
let { sqlSchema } = ENV;
1919

20-
module('Unit | index-db', function (hooks) {
20+
module('Unit | indexer', function (hooks) {
2121
let adapter: SQLiteAdapter;
2222
let client: IndexerDBClient;
2323

0 commit comments

Comments
 (0)