Skip to content

Cluster Types

Cbjs introduces type safety for KV operations. You can opt-in during the connection to your cluster :

ts
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 :

ts
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.

ts
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 :

ts
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.

ts
// 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 :

ts
// 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 :

ts
// 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.

ts
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 :

ts
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

ts
// ❌ 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:

ts
// ✅ 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.

ts
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 :

ts
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:

ts
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 :

ts
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.

ts
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.