description |
---|
Document Interface Reference Guide |
The document interface consists of two endpoints. The first endpoint, document
, is how we get documents into and out of TerminusDB. Since schemas consist of documents too, this is also how you'd update the schema.
The second endpoint, schema
, is how we can easily get schema information out of TerminusDB. While technically it is possible to get all schema information through the document interface, the schema interface is more convenient for this purpose, as it takes class inheritance into account to give a complete image of all the properties that are usable on a certain class.
All document retrieval is done through GET requests on the following endpoint:
GET /api/document/<resource path>
Where resource path is the usual strings like admin/foo
for database foo, or _system
for the system graph, or admin/foo/_meta
for the metadata graph of the foo database, etc.
By default, this will return a stream of all documents to be found at this location. What exactly is returned can be modified using parameters, which are to be provided as query parameters.
parameter | default | explanation |
---|---|---|
graph_type | either instance or schema. Used to switch between getting documents from the instance or the schema graph. | |
type | If given, only documents of the given type are returned. | |
id | If given, only the document with the given ID is returned. | |
prefixed | true | If true (the default), return IRIs using a prefixed notation wherever possible. If false, full IRIs are used. |
minimized | false | If true, forego pretty printing, and return the documents with very little whitespace. Each json document will be on its own line. |
unfold | true | If true (the default), any subdocuments contained in the returned document are returned too. If false, these are referred to by their ID instead. |
skip | 0 | How many results to skip |
count | How many results to return. If this option is absent, all results are returned. | |
as_list | false | If true, don't return a stream of json objects, but a list. This makes parsing the json easier in some environments. |
The above table shows parameters that are supposed to be provided as query parameters. There's however another mechanism, where instead, the parameters are passed in as a posted JSON document. In this calling style, an additional parameter is allowed, "query"
, by which the returned documents are filtered by matching against some template.
The alternative method uses a POST rather than a get, specifies the header X-HTTP-Method-Override: GET
, and posts a JSON document with the various query parameters instead:
{
"@rdf:type": "Person",
"count": 10,
"query": { "age": 42 },
}
The above example would find the first 10 documents of class Person
, whose age is 42.
This may provide a more convenient style for querying from a library, especially when a (large) query document has to be provided for filtering purposes. However, unlike a pure GET request with query parameters, a POST with a method override does not result in a page that can be bookmarked in a browser. If that is desirable, the GET style is better.
All new document submission is done through POST requests on the following endpoint:
POST /api/document/<resource path>
Where resource path is the usual strings like admin/foo
for database foo, or _system
for the system graph, or admin/foo/_meta
for the metadata graph of the foo database, etc.
The documents to be submitted are given as post data. Multiple documents can be specified at once, either as a stream of JSON objects or as a JSON list containing the documents to be inserted. If a document is specified that already exists, and overwrite is false (the default), an error is returned.
parameter | default | explanation |
---|---|---|
author | The commit author | |
message | The commit message | |
graph_type | instance | either instance or schema. Used to switch between submitting to the instance or the schema graph. |
full_replace | false | If true, all existing documents are deleted before inserting the posted documents. This allows the full replacement of the contents of a database. This is especially useful for replacing the schema. |
raw_json | false | If true, the input documents are treated as raw JSON , inserted as type sys:JSONDocument and are not subject to schema restrictions. |
After a successful post, the result will be a list of ids of the newly added documents.
Existing documents can be replaced through a PUT request on the following endpoint:
PUT /api/document/<resource path>
Where resource path is the usual strings like admin/foo
for database foo, or _system
for the system graph, or admin/foo/_meta
for the metadata graph of the foo database, etc.
The documents to be submitted are given as post data. Multiple documents can be specified at once, either as a stream of JSON objects or as a JSON list containing the documents to be replaced. If a document is specified that does not exist in the database, an error is returned unless create
is set to true
in which case it is inserted.
parameter | default | explanation |
---|---|---|
author | The commit author | |
message | The commit message | |
graph_type | instance | either instance or schema. Used to switch between submitting to the instance or the schema graph. |
create | false | insert if the document was not already in the database. |
raw_json | false | If true, the replaced documents are treated as raw JSON , they must be replacing a document of type sys:JSONDocument and they are not subject to schema restrictions. |
Existing documents can be deleted through a DELETE request on the following endpoint:
DELETE /api/document/<resource path>
Where resource path is the usual strings like admin/foo
for database foo, or _system
for the system graph, or admin/foo/_meta
for the metadata graph of the foo database, etc.
parameter | default | explanation |
---|---|---|
author | The commit author | |
message | The commit message | |
graph_type | instance | either instance or schema. Used to switch between submitting to the instance or the schema graph. |
id | If given, the document to delete. If not given, it is expected that the post data will contain a list of ids to delete. | |
nuke | false | If true, delete everything at this resource location (dangerous!). |
As shown above, deleting a single document can be done through query parameters alone. If multiple documents are to be deleted at once, a document has to be posted of the following format:
[ "..id 1..",
"..id 2..",
...
]
In other words, a JSON list of document IDs.
When inserting or replacing several documents at once, it may occur that some of these documents need to refer to each other. However, at insertion time, you may not know what the IDs of the new documents are going to be. This is especially the case for document types that generate their identifier randomly, but even for non-random key types, it may be convenient to rely on the server's ID generation algorithm, rather than trying to predict what IDs will get generated. Therefore, in order to support cross-references between newly inserted documents, the document interface allows you to capture newly generated document IDs in a variable, and then refer to that variable later in other documents.
When inserting or replacing a document that we want to refer to in another document inserted in the same operation, you can use a @capture
key in the document to associate the newly generated identifier with a variable. For example,
{ "@type": "Person",
"@capture": "Id_Tom",
"name": "Tom"
}
This will store the newly generated ID in a variable called ID_Tom
for the duration of the document insert/replace operation.
It is allowed to capture an ID and then never actually refer to it.
It is an error to capture the same variable twice. Doing so will result in a api:CaptureIdAlreadyBound
error with the following shape:
{"@type": "api:CaptureIdAlreadyBound",
"api:capture": "..capture id..",
"api:document": {..document where previously captured variable was captured again..}
When inserting or replacing a document that needs to refer to another document inserted in the same operation, you can use a json dictionary of the form {"@ref": "..id.."}
in place of an ordinary id. For example,
{ "@type": "Person",
"name": "Jerry",
"rival": {"@ref": "Id_Tom"}
}
It is an error to refer to a variable that is never captured. Doing so will result in a api:NotAllCapturesFound
error of the following shape:
{ "@type": "api:NotAllCapturesFound",
"api:captures": [..list of capture ids that were referenced but not found..]
}
ID captures and ID references can be done in any order. That means that when you are submitting several documents, you're allowed to refer to a captured ID in an earlier document. This also allows you to do cross-references, where two documents refer to each other:
{ "@type": "Person",
"@capture": "Id_Tom",
"name": "Tom",
"rival": {"@ref": "Id_Jerry"}
}
{ "@type": "Person",
"@capture": "Id_Jerry",
"name": "Jerry",
"rival": {"@ref": "Id_Tom"}
}
In this example, Tom refers to Jerry, even though at that point in the submitted document stream, Jerry has not yet been processed. This is not a problem - both Tom and Jerry will get inserted referring to each other.
Using ID capture, it is possible to create a document that refers to itself:
{ "@type": "Person",
"@capture": "Captured_Id",
"name": "Elmo",
"friend": {"@ref": "Captured_Id"}
}
This will make Elmo be his own friend.
It is important to keep in mind that the ID capture mechanism only works within a single call to the document api. It is not possible to capture an ID in one operation, and then refer to it in a second operation. The @capture
and @ref
instructions do not get saved into the database. They are processed immediately and are then forgotten.
If you need to refer to a document already in the database, the only way to do so is by referring to its ID.
The schema endpoint can be used to query information about classes in a resource. These queries happen through a GET on the following endpoint:
GET /api/schema/<resource path>
Where resource path is the usual strings like admin/foo
for database foo, or _system
for the system graph, or admin/foo/_meta
for the metadata graph of the foo database, etc.
The purpose of this endpoint is to quickly discover the supported fields of a particular type. The primary envisioned use case for this is the automatic generation of forms and other UI elements, as well as client code generation.
parameter | default | explanation |
---|---|---|
type | If given, the type to get information for. If omitted, information for all types is returned. |
The result of this GET is a stream of documents describing all the types in a particular resource.
The schema endpoint can also be used to switch between schema checking and schemaless mode.
Switching between checking or not checking does not delete the schema itself. After disabling schema checking it is still possible to update the schema or to query it through the schema endpoint. However, when disabled, it is possible to submit documents that do not match the schema. Re-enabling schema checking is only possible if all the documents in the given resource match the current schema.
POST /api/schema/<resource path>
parameter | default | explanation |
---|---|---|
author | The commit author | |
message | The commit message | |
schema_checking | Value should be either enabled or disabled |
The schema endpoint can be used to query information about classes in a resource. These queries happen through a GET on the following endpoint:
POST /api/schema/<resource path>
Where resource path is the usual strings like admin/foo
for database foo, or admin/foo/local/branch/dev
for the dev
branch of admin/foo
.
The purpose of this endpoint is to take the difference between any two commits and apply them to a branch.
parameter | default | explanation |
---|---|---|
before_commit | The first commit to compare in order to produce a diff | |
after_commit | The last commit to compare in order to produce a diff | |
commit_info | A JSON document with author and message | |
match_final_state | true | Ignores conflicts if the final state would remain the same |
type | squash | What type of application to perform - currently can only be squash |
The result of this POST request is either an updated branch with a successful application of the difference between two commits, or an error giving the reason for an unresolvable conflict.