Resource Class
Resource Class
The Resource class is designed to model different data resources within HarperDB. The Resource class can be extended to create new data sources. Resources can be exported to define endpoints. Tables themselves extend the Resource class, and can be extended by users.
Conceptually, a Resource class provides an interface for accessing, querying, modifying, and monitoring a set of entities or records. Instances of a Resource class can represent a single record or entity, or a collection of records, at a given point in time, that you can interact with through various methods or queries. Resource instances can represent an atomic transactional view of a resource and facilitate transactional interaction. A Resource instance holds the primary key/identifier, context information, and any pending updates to the record, so any instance methods can act on the record and have full access to this information to during execution. Therefore, there are distinct resource instances created for every record or query that is accessed, and the instance methods are used for interaction with the data.
The RESTful HTTP server and other server interfaces will instantiate/load resources to fulfill incoming requests so resources can be defined as endpoints for external interaction. When resources are used by the server interfaces, they will be executed in transaction and the access checks will be performed before the method is executed.
Paths (URL, MQTT topics) are mapped to different resource instances. Using a path that does specify an ID like /MyResource/3492
will be mapped to a Resource instance where the instance's ID will be 3492
, and interactions will use the instance methods like get()
, put()
, and post()
. Using the root path (/MyResource/
) will map to a Resource instance with an ID of null
.
You can create classes that extend Resource to define your own data sources, typically to interface with external data sources (the Resource base class is available as a global variable in the HarperDB JS environment). In doing this, you will generally be extending and providing implementations for the instance methods below. For example:
You can also extend table classes in the same way, overriding the instance methods for custom functionality. The tables
object is a global variable in the HarperDB JavaScript environment, along with Resource
:
Make sure that if are extending and export
ing your table with this class, that you remove the @export
directive in the your schema, so that you aren't exporting the same table/class twice.
Global Variables
tables
tables
This is an object with all the tables in the default database (the default database is "data"). Each table that has been declared or created will be available as a (standard) property on this object, and the value will be the table class that can be used to interact with that table. The table classes implement the Resource API.
databases
databases
This is an object with all the databases that have been defined in HarperDB (in the running instance). Each database that has been declared or created will be available as a (standard) property on this object. The property values are an object with the tables in that database, where each property is a table, like the tables
object. In fact, databases.data === tables
should always be true.
Resource
Resource
This is the Resource base class. This can be directly extended for custom resources, and is the base class for all tables.
server
server
This object provides extension points for extension components that wish to implement new server functionality (new protocols, authentication, etc.). See the extensions documentation for more information.
transaction
transaction
This provides a function for starting transactions. See the transactions section below for more information.
contentTypes
contentTypes
This provides an interface for defining new content type handlers. See the content type extensions documentation for more information.
TypeScript Support
While these objects/methods are all available as global variables, it is easier to get TypeScript support (code assistance, type checking) for these interfaces by explicitly import
ing them. This can be done by setting up a package link to the main HarperDB package in your app:
And then you can import any of the main HarperDB APIs you will use, and your IDE should understand the full typings associated with them:
Resource Class (Instance) Methods
Properties/attributes declared in schema
Properties that have been defined in your table's schema can be accessed and modified as direct properties on the Resource instances.
get(queryOrProperty?)
: Resource|AsyncIterable
get(queryOrProperty?)
: Resource|AsyncIterableThis is called to return the record or data for this resource, and is called by HTTP GET requests. This may be optionally called with a query
object to specify a query should be performed, or a string to indicate that the specified property value should be returned. When defining Resource classes, you can define or override this method to define exactly what should be returned when retrieving a record. The default get
method (super.get()
) returns the current record as a plain object.
The query object can be used to access any query parameters that were included in the URL. For example, with a request to /my-resource/some-id?param1=value
, we can access URL/request information:
If get
is called for a single record (for a request like /Table/some-id
), the default action is to return this
instance of the resource. If get
is called on a collection (/Table/?name=value
), the default action is to search
and return an AsyncIterable of results.
It is important to note that this
is the resource instance for a specific record, specified by the primary key. Therefore, calling super.get(query)
performs a get
on this specific record/resource, not on the whole table. If you wish to access a different record, you should use the static get
method on the table class, like Table.get(otherId, context)
.
search(query: Query)
: AsyncIterable
search(query: Query)
: AsyncIterableThis performs a query on this resource, searching for records that are descendants. By default, this is called by get(query)
from a collection resource. When this is called for the root resource (like /Table/
) it searches through all records in the table. However, if you call search from an instance with a specific ID like 1
from a path like Table/1
, it will only return records that are descendants of that record, like [1, 1]
(path of Table/1/1) and [1, 2]
(path of Table/1/2). If you want to do a standard search of the table, make you call the static method like Table.search(...)
. You can define or override this method to define how records should be queried. The default search
method on tables (super.search(query)
) will perform a query and return an AsyncIterable of results. The query object can be used to specify the desired query.
getId(): string|number|Array<string|number>
getId(): string|number|Array<string|number>
Returns the primary key value for this resource.
put(data: object)
put(data: object)
This will assign the provided record or data to this resource, and is called for HTTP PUT requests. You can define or override this method to define how records should be updated. The default put
method on tables (super.put(data)
) writes the record to the table (updating or inserting depending on if the record previously existed) as part of the current transaction for the resource instance.
It is important to note that this
is the resource instance for a specific record, specified by the primary key. Therefore, calling super.put(data)
updates this specific record/resource, not another records in the table. If you wish to update a different record, you should use the static put
method on the table class, like Table.put(data, context)
.
patch(data: object)
patch(data: object)
This will update the existing record with the provided data's properties, and is called for HTTP PATCH requests. You can define or override this method to define how records should be updated. The default patch
method on tables (super.patch(data)
) updates the record. The properties will be applied to the existing record, overwriting the existing records properties, and preserving any properties in the record that are not specified in the data
object. This is performed as part of the current transaction for the resource instance.
update(data: object, fullUpdate: boolean?)
update(data: object, fullUpdate: boolean?)
This is called by the default put
and patch
handlers to update a record. put
calls with fullUpdate
as true
to indicate a full record replacement (patch
calls it with the second argument as false
). Any additional property changes that are made before the transaction commits will also be persisted.
delete(queryOrProperty?)
delete(queryOrProperty?)
This will delete this record or resource, and is called for HTTP DELETE requests. You can define or override this method to define how records should be deleted. The default delete
method on tables (super.put(record)
) deletes the record from the table as part of the current transaction.
publish(message)
publish(message)
This will publish a message to this resource, and is called for MQTT publish commands. You can define or override this method to define how messages should be published. The default publish
method on tables (super.publish(message)
) records the published message as part of the current transaction; this will not change the data in the record but will notify any subscribers to the record/topic.
post(data)
post(data)
This is called for HTTP POST requests. You can define this method to provide your own implementation of how POST requests should be handled. Generally POST
provides a generic mechanism for various types of data updates, and is a good place to define custom functionality for updating records.
invalidate()
invalidate()
This method is available on tables. This will invalidate the current record in the table. This can be used with a caching table and is used to indicate that the source data has changed, and the record needs to be reloaded when next accessed.
subscribe(subscriptionRequest): Promise<Subscription>
subscribe(subscriptionRequest): Promise<Subscription>
This will subscribe to the current resource, and is called for MQTT subscribe commands. You can define or override this method to define how subscriptions should be handled. The default subscribe
method on tables (super.publish(message)
) will set up a listener that will be called for any changes or published messages to this resource.
The returned (promise resolves to) Subscription object is an AsyncIterable
that you can use a for await
to iterate through. It also has a queue
property which holds (an array of) any messages that are ready to be delivered immediately (if you have specified a start time, previous count, or there is a message for the current or "retained" record, these may be immediately returned).
The subscriptionRequest
object supports the following properties (all optional):
id
- The primary key of the record (or topic) that you want to subscribe to. If omitted, this will be a subscription to the whole table.isCollection
- If this is enabled and theid
was included, this will create a subscription to all the record updates/messages that are prefixed with the id. For example, a subscription request of{id:'sub', isCollection: true}
would return events for any update with an id/topic of the form sub/* (likesub/1
).startTime
- This will begin the subscription at a past point in time, returning all updates/messages since the start time (a catch-up of historical messages). This can be used to resume a subscription, getting all messages since the last subscription.previousCount
- This specifies the number of previous updates/messages to deliver. For example,previousCount: 10
would return the last ten messages. Note thatpreviousCount
can not be used in conjunction withstartTime
.omitCurrent
- Indicates that the current (or retained) record should not be immediately sent as the first update in the subscription (if nostartTime
orpreviousCount
was used). By default, the current record is sent as the first update.
connect(incomingMessages?: AsyncIterable<any>): AsyncIterable<any>
connect(incomingMessages?: AsyncIterable<any>): AsyncIterable<any>
This is called when a connection is received through WebSockets or Server Sent Events (SSE) to this resource path. This is called with incomingMessages
as an iterable stream of incoming messages when the connection is from WebSockets, and is called with no arguments when the connection is from a SSE connection. This can return an asynchronous iterable representing the stream of messages to be sent to the client.
set(property, value)
set(property, value)
This will assign the provided value to the designated property in the resource's record. During a write operation, this will indicate that the record has changed and the changes will be saved during commit. During a read operation, this will modify the copy of the record that will be serialized during serialization (converted to the output format of JSON, MessagePack, etc.).
allowCreate(user)
allowCreate(user)
This is called to determine if the user has permission to create the current resource. This is called as part of external incoming requests (HTTP). The default behavior for a generic resource is that this requires super-user permission and the default behavior for a table is to check the user's role's insert permission to the table.
allowRead(user)
allowRead(user)
This is called to determine if the user has permission to read from the current resource. This is called as part of external incoming requests (HTTP GET). The default behavior for a generic resource is that this requires super-user permission and the default behavior for a table is to check the user's role's read permission to the table.
allowUpdate(user)
allowUpdate(user)
This is called to determine if the user has permission to update the current resource. This is called as part of external incoming requests (HTTP PUT). The default behavior for a generic resource is that this requires super-user permission and the default behavior for a table is to check the user's role's update permission to the table.
allowDelete(user)
allowDelete(user)
This is called to determine if the user has permission to delete the current resource. This is called as part of external incoming requests (HTTP DELETE). The default behavior for a generic resource is that this requires super-user permission and the default behavior for a table is to check the user's role's delete permission to the table.
addTo(property, value)
addTo(property, value)
This adds to provided value to the specified property using conflict-free data type (CRDT) incrementation. This ensures that even if multiple calls are simultaneously made to increment a value, the resulting merge of data changes from different threads and nodes will properly sum all the added values.
getUpdatedTime(): number
getUpdatedTime(): number
This returns the last updated time of the resource (timestamp of last commit). This is returned as milliseconds from epoch.
wasLoadedFromSource(): boolean
wasLoadedFromSource(): boolean
Indicates if the record had been loaded from source. When using caching tables, this indicates that there was a cache miss and the data had to be loaded from the source (or waiting on an inflight request from the source to finish).
getContext(): Context
getContext(): Context
Returns the context for this resource. The context contains information about the current transaction, the user that initiated this action, and other metadata that should be retained through the life of an action.
Context
Context
The Context
object has the following (potential) properties:
user
- This is the user object, which includes information about the username, role, and authorizations.transaction
- The current transaction If the current method was triggered by an HTTP request, the following properties are available:lastModified
- This value is used to indicate the last modified or updated timestamp of any resource(s) that are accessed and will inform the response'sETag
(orLast-Modified
) header. This can be updated by application code if it knows that modification should cause this timestamp to be updated.
When a resource gets a request through HTTP, the request object is the context, which has the following properties:
url
- The local path/URL of the request (this will not include the protocol or host name, but will start at the path and includes the query string).method
- The method of the HTTP request.headers
- This is an object with the headers that were included in the HTTP request. You can access headers by callingcontext.headers.get(headerName)
.responseHeaders
- This is an object with the headers that will be included in the HTTP response. You can set headers by callingcontext.responseHeaders.set(headerName, value)
.pathname
- This provides the path part of the URL (no querystring).host
- This provides the host name of the request (from theHost
header).ip
- This provides the ip address of the client that made the request.body
- This is the request body as a raw NodeJS Readable stream, if there is a request body.data
- If the HTTP request had a request body, this provides a promise to the deserialized data from the request body. (Note that for methods that normally have a request body likePOST
andPUT
, the resolved deserialized data is passed in as the main argument, but accessing the data from the context provides access to this for requests that do not traditionally have a request body likeDELETE
).
When a resource is accessed as a data source:
requestContext
- For resources that are acting as a data source for another resource, this provides access to the context of the resource that is making a request for data from the data source resource. Note that it is generally not recommended to rely on this context. The resolved data may be used fulfilled many different requests, and relying on this first request context may not be representative of future requests. Also, source resolution may be triggered by various actions, not just specified endpoints (for example queries, operations, studio, etc.), so make sure you are not relying on specific request context information.
operation(operationObject: Object, authorize?: boolean): Promise<any>
operation(operationObject: Object, authorize?: boolean): Promise<any>
This method is available on tables and will execute a HarperDB operation, using the current table as the target of the operation (the table
and database
do not need to be specified). See the operations API for available operations that can be performed. You can set the second argument to true
if you want the current user to be checked for authorization for the operation (if true
, will throw an error if they are not authorized).
allowStaleWhileRevalidate(entry: { version: number, localTime: number, expiresAt: number, value: object }, id): boolean
allowStaleWhileRevalidate(entry: { version: number, localTime: number, expiresAt: number, value: object }, id): boolean
For caching tables, this can be defined to allow stale entries to be returned while revalidation is taking place, rather than waiting for revalidation. The version
is the timestamp/version from the source, the localTime
is when the resource was last refreshed, the expiresAt
is when the resource expired and became stale, and the value
is the last value (the stale value) of the record/resource. All times are in milliseconds since epoch. Returning true
will allow the current stale value to be returned while revalidation takes place concurrently. Returning false
will cause the response to wait for the data source or origin to revalidate or provide the latest value first, and then return the latest value.
Resource Static Methods and Properties
The Resource class also has static methods that mirror the instance methods with an initial argument that is the id of the record to act on. The static methods are generally the preferred and most convenient method for interacting with tables outside of methods that are directly extending a table. Whereas instances methods are bound to a specific record, the static methods allow you to specify any record in the table to act on.
The get, put, delete, subscribe, and connect methods all have static equivalents. There is also a static search()
method for specifically handling searching a table with query parameters. By default, the Resource static methods default to creating an instance bound to the record specified by the arguments, and calling the instance methods. Again, generally static methods are the preferred way to interact with resources and call them from application code. These methods are available on all user Resource classes and tables.
get(id: Id, context?: Resource|Context)
get(id: Id, context?: Resource|Context)
This will retrieve a resource instance by id. For example, if you want to retrieve comments by id in the retrieval of a blog post you could do:
Type definition for Id
:
get(query: Query, context?: Resource|Context)
get(query: Query, context?: Resource|Context)
This can be used to retrieve a resource instance by a query. If the query can be used to specify a single/unique record by an id
property, and can be combined with a select
:
This method may also be used to retrieve a collection of records by a query. If the query is not for a specific record id, this will call the search
method, described above.
put(record: object, context?: Resource|Context): Promise<void>
put(record: object, context?: Resource|Context): Promise<void>
put(id: Id, record: object, context?: Resource|Context): Promise<void>
put(id: Id, record: object, context?: Resource|Context): Promise<void>
This will save the provided record or data to this resource. This will fully replace the existing record. Make sure to await
this function to ensure it finishes execution within the surrounding transaction.
patch(recordUpdate: object, context?: Resource|Context): Promise<void>
patch(recordUpdate: object, context?: Resource|Context): Promise<void>
patch(id: Id, recordUpdate: object, context?: Resource|Context): Promise<void>
patch(id: Id, recordUpdate: object, context?: Resource|Context): Promise<void>
This will save the provided updates to the record. The recordUpdate
object's properties will be applied to the existing record, overwriting the existing records properties, and preserving any properties in the record that are not specified in the recordUpdate
object. Make sure to await
this function to ensure it finishes execution within the surrounding transaction.
delete(id: Id, context?: Resource|Context): Promise<void>
delete(id: Id, context?: Resource|Context): Promise<void>
Deletes this resource's record or data. Make sure to await
this function to ensure it finishes execution within the surrounding transaction.
publish(message: object, context?: Resource|Context): Promise<void>
publish(message: object, context?: Resource|Context): Promise<void>
publish(topic: Id, message: object, context?: Resource|Context): Promise<void>
publish(topic: Id, message: object, context?: Resource|Context): Promise<void>
Publishes the given message to the record entry specified by the id in the context. Make sure to await
this function to ensure it finishes execution within the surrounding transaction.
subscribe(subscriptionRequest, context?: Resource|Context): Promise<Subscription>
subscribe(subscriptionRequest, context?: Resource|Context): Promise<Subscription>
Subscribes to a record/resource.
search(query: Query, context?: Resource|Context): AsyncIterable
search(query: Query, context?: Resource|Context): AsyncIterable
This will perform a query on this table or collection. The query parameter can be used to specify the desired query.
primaryKey
primaryKey
This property indicates the name of the primary key attribute for a table. You can get the primary key for a record using this property name. For example:
There are additional methods that are only available on table classes (which are a type of resource).
Table.sourcedFrom(Resource, options)
Table.sourcedFrom(Resource, options)
This defines the source for a table. This allows a table to function as a cache for an external resource. When a table is configured to have a source, any request for a record that is not found in the table will be delegated to the source resource to retrieve (via get
) and the result will be cached/stored in the table. All writes to the table will also first be delegated to the source (if the source defines write functions like put
, delete
, etc.). The options
parameter can include an expiration
property that will configure the table with a time-to-live expiration window for automatic deletion or invalidation of older entries. The options
parameter (also) supports:
expiration
- Default expiration time for records in seconds.eviction
- Eviction time for records in seconds.scanInterval
- Time period for scanning the table for records to evict.
If the source resource implements subscription support, real-time invalidation can be performed to ensure the cache is guaranteed to be fresh (and this can eliminate or reduce the need for time-based expiration of data).
parsePath(path, context, query) {
parsePath(path, context, query) {
This is called by static methods when they are responding to a URL (from HTTP request, for example), and translates the path to an id. By default, this will convert a multi-segment path to multipart id (an array), which facilitates hierarchical id-based data access, and also parses .property
suffixes for accessing properties and specifying preferred content type in the URL. However, in some situations you may wish to preserve the path directly as a string. You can override parsePath
for simpler path to id preservation:
isCollection(resource: Resource): boolean
isCollection(resource: Resource): boolean
This returns a boolean indicating if the provide resource instance represents a collection (can return a query result) or a single record/entity.
Context and Transactions
Whenever you implement an action that is calling other resources, it is recommended that you provide the "context" for the action. This allows a secondary resource to be accessed through the same transaction, preserving atomicity and isolation.
This also allows timestamps that are accessed during resolution to be used to determine the overall last updated timestamp, which informs the header timestamps (which facilitates accurate client-side caching). The context also maintains user, session, and request metadata information that is communicated so that contextual request information (like headers) can be accessed and any writes are properly attributed to the correct user.
When using an export resource class, the REST interface will automatically create a context for you with a transaction and request metadata, and you can pass this to other actions by simply including this
as the source argument (second argument) to the static methods.
For example, if we had a method to post a comment on a blog, and when this happens we also want to update an array of comment IDs on the blog record, but then add the comment to a separate comment table. We might do this:
Please see the transaction documentation for more information on how transactions work in HarperDB.
Query
The get
/search
methods accept a Query object that can be used to specify a query for data. The query is an object that has the following properties, which are all optional:
conditions
conditions
This is an array of objects that specify the conditions to use the match records (if conditions are omitted or it is an empty array, this is a search for everything in the table). Each condition object can have the following properties:
attribute
: Name of the property/attribute to match on.value
: The value to match.comparator
: This can specify how the value is compared. This defaults to "equals", but can also be "greater_than", "greater_than_equal", "less_than", "less_than_equal", "starts_with", "contains", "ends_with", "between", and "not_equal".conditions
: An array of conditions, which follows the same structure as above.operator
: Specifies the operator to apply to this set of conditions (and
oror
. This is optional and defaults toand
). For example, a complex query might look like:
For example, a more complex query might look like:
Chained Attributes/Properties
Chained attribute/property references can be used to search on properties within related records that are referenced by relationship properties (in addition to the schema documentation, see the REST documentation for more of overview of relationships and querying). Chained property references are specified with an array, with each entry in the array being a property name for successive property references. For example, if a relationship property called brand
has been defined that references a Brand
table, we could search products by brand name:
This effectively executes a join, searching on the Brand
table and joining results with matching records in the Product
table. Chained array properties can be used in any condition, as well nested/grouped conditions. The chain of properties may also be more than two entries, allowing for multiple relationships to be traversed, effectively joining across multiple tables. An array of chained properties can also be used as the attribute
in the sort
property, allowing for sorting by an attribute in a referenced joined tables.
operator
operator
Specifies if the conditions should be applied as an "and"
(records must match all conditions), or as an "or" (records must match at least one condition). This is optional and defaults to "and"
.
limit
limit
This specifies the limit of the number of records that should be returned from the query.
offset
offset
This specifies the number of records that should be skipped prior to returning records in the query. This is often used with limit
to implement "paging" of records.
select
select
This specifies the specific properties that should be included in each record that is returned. This can be an array, to specify a set of properties that should be included in the returned objects. The array can specify an select.asArray = true
property and the query results will return a set of arrays of values of the specified properties instead of objects; this can be used to return more compact results. Each of the elements in the array can be a property name, or can be an object with a name
and select
array itself that specifies properties that should be returned by the referenced sub-object or related record. For example, a select
can defined:
Or nested/joined properties from referenced objects can be specified, here we are including the referenced related
records, and returning the description
and id
from each of the related objects:
The select properties can also include certain special properties:
$id
- This will specifically return the primary key of the record (regardless of name, even if there is no defined primary key attribute for the table).$updatedtime
- This will return the last updated timestamp/version of the record (regardless of whether there is an attribute for the updated time).
Alternately, the select value can be a string value, to specify that the value of the specified property should be returned for each iteration/element in the results. For example to just return an iterator of the id
s of object:
sort
sort
This defines the sort order, and should be an object that can have the following properties:
attributes
: The attribute to sort on.descending
: If true, will sort in descending order (optional and defaults tofalse
).next
: Specifies the next sort order to resolve ties. This is an object that follows the same structure assort
.
explain
explain
This will return the conditions re-ordered as HarperDB will execute them. HarperDB will estimate the number of the matching records for each condition and apply the narrowest condition applied first.
enforceExecutionOrder
enforceExecutionOrder
This will force the conditions to be executed in the order they were supplied, rather than using query estimation to re-order them.
The query results are returned as an AsyncIterable
. In order to access the elements of the query results, you must use a for await
loop (it does not return an array, you can not access the results by index).
For example, we could do a query like:
AsyncIterable
s can be returned from resource methods, and will be properly serialized in responses. When a query is performed, this will open/reserve a read transaction until the query results are iterated, either through your own for await
loop or through serialization. Failing to iterate the results this will result in a long-lived read transaction which can degrade performance (including write performance), and may eventually be aborted.
Interacting with the Resource Data Model
When extending or interacting with table resources, when a resource instance is retrieved and instantiated, it will be loaded with the record data from its table. You can interact with this record through the resource instance. For any properties that have been defined in the table's schema, you can direct access or modify properties through standard property syntax. For example, let's say we defined a product schema:
If we have extended this table class with our get() we can interact with any these specified attributes/properties:
Likewise, we can interact with resource instances in the same way when retrieving them through the static methods:
If there are additional properties on (some) products that aren't defined in the schema, we can still access them through the resource instance, but since they aren't declared, there won't be getter/setter definition for direct property access, but we can access properties with the get(propertyName)
method and modify properties with the set(propertyName, value)
method:
And likewise, we can do this in an instance method, although you will probably want to use super.get()/set() so you don't have to write extra logic to avoid recursion:
Note that you may also need to use get
/set
for properties that conflict with existing method names. For example, your schema defines an attribute called getId
(not recommended), you would need to access that property through get('getId')
and set('getId', value)
.
If you want to save the changes you make, you can call the `update()`` method:
Updates are automatically saved inside modifying methods like put and post:
We can also interact with properties in nested objects and arrays, following the same patterns. For example we could define more complex types on our product:
We can interact with these nested properties:
If you need to delete a property, you can do with the delete
method:
You can also get "plain" object representation of a resource instance by calling toJSON
, which will return a simple object with all the properties (whether defined in the schema) as direct normal properties:
Throwing Errors
You may throw errors (and leave them uncaught) from the response methods and these should be caught and handled by protocol the handler. For REST requests/responses, this will result in an error response. By default the status code will be 500. You can assign a property of statusCode
to errors to indicate the HTTP status code that should be returned. For example:
Last updated