Cluster Types
Cbjs introduces type safety for KV operations. You can opt-in during the connection to your cluster :
const cluster = await connect<MyClusterTypes>('...');
Definition of types
The definition of the types is done when connecting to the cluster, by passing a type with this shape :
type CouchbaseClusterTypes = {
[bucket: string]: {
[scope: string]: {
[collection: string]: DocDef[];
};
};
};
The following example describe the keyspace store.library.books
that can contain two types of documents.
TIP
Express documents keys as a template literal to enable key validation and more powerful type narrowing.
type MyClusterTypes = {
store: {
library: {
books: [
DocDef<`author::${string}`, { firstname: string; lastname: string }>,
DocDef<`book::${string}`, { title: string; authors: string[] }>,
]
};
};
};
Note that the cluster definition must always be wrapped with ClusterTypes
. Children definitions may use an object directly if they don't need to declare options.
Type safety
Given the previous definitions, the following type safety arise :
const cluster = await connect<MyClusterTypes>('...');
const collection = cluster.bucket('store').scope('library').collection('books');
const bookId = 'book::001';
const { content: book } = await collection.get(bookId);
const { content: [title] } = await collection.lookupIn(bookId).get('title');
const { content: [authors] } = await collection.lookupIn(bookId).get('authors');
const { content: [firstAuthor] } = await collection.lookupIn(bookId).get('authors[0]');
//
// Now let's see some mistakes being prevented by Cbjs
//
// Invalid document key
await collection.get('vegetable::001');
// Invalid document property
await collection.lookupIn(bookId).get('tite');
// Invalid property accessor : `quaterSales` has only 4 elements
await collection.lookupIn(bookId).get('quaterSales[4]');
// Property `title` is required, therefore it already exists. Use `upsert` instead.
await collection.mutateIn(bookId).insert('title', 'documentation');
// Invalid value : `quaterSales` is a tuple of numbers.
await collection.mutateIn(bookId).arrayInsert('quaterSales[2]', '3467');
Key Matching Strategy
When inferring the type of a document, Cbjs matches the given key with the string template of the document definitions of the targeted collection.
Three strategies are available : always
, delimiter
, firstMatch
.
Always default
This is the most basic strategy. It checks if the key extends the template.
// The result is either a book or the book reviews
const result = await collection.get('book::001::reviews');
type MyClusterTypes = {
store: {
library: {
books: [
DocDef<`book::${string}`, { title: string }>,
DocDef<`book::${string}::reviews`, Array<{ note: number; comment: string }>>,
];
};
};
};
Because the key 'book::001::reviews'
extends both string templates, both documents are matched.
This is the default key matching strategy.
Delimiter recommended
Now let's see another strategy named delimiter
, that uses the common delimiters to match the keys :
// The result has been narrowed using the delimiter to the book reviews
const result = await collection.get('book::001::reviews');
type ClusterTypesOptions = {
keyMatchingStrategy: 'delimiter';
keyDelimiter: '::';
};
type MyClusterTypes = {
'@options': ClusterTypesOptions;
store: {
library: {
books: [
DocDef<`book::${string}`, { title: string }>,
DocDef<`book::${string}::reviews`, Array<{ note: number; comment: string }>>,
];
};
};
};
First Match
Finally, if the previous strategy does not work for you, you can use the firstMatch
strategy, that simply uses the first declared document that matches the key :
// The reviews definition is the first to match, so it is used for the return type.
const result = await collection.get('book::001::reviews');
type ClusterTypesOptions = {
keyMatchingStrategy: 'firstMatch';
};
type MyClusterTypes = {
'@options': ClusterTypesOptions;
store: {
library: {
books: [
DocDef<`book::${string}::reviews`, Array<{ note: number; comment: string }>>,
DocDef<`book::${string}`, { title: string }>,
];
};
};
};
Picking a strategy
When declaring your cluster types, you can pass some options the special jey @options
.
Note that the cluster types options are inherited from the top down, but not merged.
type ClusterOptions = {
keyMatchingStrategy: 'firstMatch';
};
type BucketOptions = {
keyMatchingStrategy: 'delimiter';
keyDelimiter: '__';
};
type CollectionOptions = {
keyMatchingStrategy: 'always';
};
type MyClusterTypes = {
'@options': ClusterOptions;
store: {
'@options': BucketOptions;
library: {
books: CollectionOptions | [ DocDef ]
}
}
};
Incremental adoption
Using strict cluster types on an existing project can be overwhelming. You may want to adopt cluster types progressively, you can start by defining some part of your types :
import type { DefaultClusterTypes } from '@cbjsdev/cbjs';
// Note the use of the type helper `DefaultClusterTypes`
type MyClusterTypes = DefaultClusterTypes & {
store: {
library: {
books: [/* Document definitions */];
};
};
};
This will enable type safety only for the collection store.library.books
. Reference to other bucket, scope or collection will be treated as before.
Reference keyspace objects
When you expect some of your cluster's bucket, scope or collection, you cannot use a union type to define multiple collections for examples
// ❌ Doesn't work
type AcceptedScopes = Scope<MyClusterTypes, 'store', 'library' | 'groceries'>;
If you want to reference a range of bucket/scope/collection, the solution is to use the types provided by Cbjs:
// ✅ Works
type AcceptedScopes = ClusterScope<MyClusterTypes, 'store', 'library' | 'groceries'>;
declare function doSomethingWithScope(scope: AcceptedScopes): void;
// The scope we want to use.
declare const scope: Scope<MyClusterTypes, 'store', 'library'>;
// Our scope is accepted 👌
doSomethingWithScope(scope);
The types ClusterBucket
, ClusterScope
and ClusterCollection
simply generate a union type for all matching keyspaces. You can also use some kind of wildcard by passing any
or never
.
type BackendCustomersCollection = ClusterCollection<
MyClusterTypes,
'backend',
never,
'customers'
>;
Considerations
Document paths at runtime
The purpose of the path autocompletion is to prevent small mistakes and to write paths faster. It does not guarantee that the document path will exist at runtime. Consider the following :
type Book = { title: string; authors: string[] };
const result = await collection.lookupIn('book::01').get('authors[2]');
The path is valid because it may exist, but you may very well receive an error at runtime if the array index does not exist. The same logic applies when you use optional properties or union types:
type Book = {
title: string;
authors: string | string[];
};
const result = await collection.lookupIn('book::001').get('title').get('authors[0]');
IDE autocompletion
Because of IDEs current limitations, autocomplete will not be offered for array indexes. Using the previous example :
const result = await collection
.lookupIn('book::001')
.get('autho');
Because the path is expressed as a template literal, authors[${number}]
, your IDE will not offer authors[0]
but only authors
.
Nevertheless, authors[0]
is a valid path.
This does not apply to tuples, as their length is fixed.
const result = await collection
.lookupIn('book::001')
.get('quater_sal');
Tuples
Cbjs considers that tuples cannot change in length, regardless of them being readonly or not. The values themselves can be modified, unless the tuple is readonly
.