Released LiteStore.
@@ -5466,6 +5466,8 @@ <li><a href="#Document-Tagging">Document Tagging</a></li>
<li><a href="#Enhanced-Querying-of-JSON-documents">Enhanced Querying of JSON documents</a></li> <li><a href="#Full-text-Search">Full-text Search</a></li> <li><a href="#RESTful-HTTP-API">RESTful HTTP API</a></li> + <li><a href="#JWT-based-Authorization">JWT-based Authorization</a></li> + <li><a href="#Middleware">Middleware</a></li> <li><a href="#Nim-API">Nim API</a></li> <li><a href="#Command-line-API">Command-line API</a></li> <li><a href="#Directory-Bulk-Import/Export">Directory Bulk Import/Export</a></li>@@ -5488,6 +5490,7 @@ <li><a href="#Database-Schema">Database Schema</a>
<ul> <li><a href="#info-Table">info Table</a></li> <li><a href="#documents-Table">documents Table</a></li> + <li><a href="#system_documents-Table">system_documents Table</a></li> <li><a href="#tags-Table">tags Table</a></li> <li><a href="#searchdata-Table">searchdata Table</a> </li>@@ -5518,6 +5521,7 @@ <li><a href="#Examples">Examples</a>
<ul> <li><a href="#Starting-the-HTTP-Server">Starting the HTTP Server</a></li> <li><a href="#Importing-a-directory">Importing a directory</a></li> + <li><a href="#Importing-system-documents-from-a-directory">Importing system documents from a directory</a></li> <li><a href="#Exporting-a-directory">Exporting a directory</a></li> <li><a href="#Deleting-documents-within-a-directory">Deleting documents within a directory</a></li> <li><a href="#Performing-maintenance-operations">Performing maintenance operations</a></li>@@ -5528,6 +5532,38 @@ </li>
</ul> </li> <li><a href="#Authorization">Authorization</a></li> + <li><a href="#Configuration-File">Configuration File</a> + <ul> + <li><a href="#settings">settings</a></li> + <li><a href="#resources">resources</a></li> + <li><a href="#signature">signature</a></li> + </ul> + </li> + <li><a href="#Middleware">Middleware</a> + <ul> + <li><a href="#How-middleware-works">How middleware works</a></li> + <li><a href="#Creating-a-JavaScript-Middleware-Function">Creating a JavaScript Middleware Function</a></li> + <li><a href="#Passing-data-to-another-middleware">Passing data to another middleware</a></li> + </ul> + </li> + <li><a href="#Global-JavaScript-Objects">Global JavaScript Objects</a> + <ul> + <li><a href="#$ctx">$ctx</a></li> + <li><a href="#$req">$req</a> + <ul> + <li><a href="#Properties">Properties</a></li> + </ul> + </li> + <li><a href="#$res">$res</a></li> + <li><a href="#$store">$store</a></li> + </ul> + </li> + <li><a href="#System-Documents">System Documents</a> + <ul> + <li><a href="#Importing,-exporting-and-deleting-System-Documents">Importing, exporting and deleting System Documents</a></li> + <li><a href="#How-LiteStore-uses-System-Documents">How LiteStore uses System Documents</a></li> + </ul> + </li> <li><a href="#Administration-App">Administration App</a> <ul> <li><a href="#Obtaining-and-Running-the-Administration-App">Obtaining and Running the Administration App</a>@@ -5686,6 +5722,7 @@ </li>
<li><a href="#GET-tags">GET tags</a> <ul> <li><a href="#<code>like</code>-option"><code>like</code> option</a></li> + <li><a href="#<code>limit</code>-and-<code>offset</code>-options"><code>limit</code> and <code>offset</code> options</a></li> <li><a href="#Example">Example</a></li> </ul> </li>@@ -5694,6 +5731,43 @@ <ul>
<li><a href="#Example">Example</a> </li> </ul> + </li> + </ul> + </li> + <li><a href="#indexes-(LiteStore-Indexes)">indexes (LiteStore Indexes)</a> + <ul> + <li><a href="#OPTIONS-indexes">OPTIONS indexes</a> + <ul> + <li><a href="#Example">Example</a></li> + </ul> + </li> + <li><a href="#OPTIONS-indexes/:id">OPTIONS indexes/:id</a> + <ul> + <li><a href="#Example">Example</a></li> + </ul> + </li> + <li><a href="#GET-indexes">GET indexes</a> + <ul> + <li><a href="#<code>like</code>-option"><code>like</code> option</a></li> + <li><a href="#<code>limit</code>-and-<code>offset</code>-options"><code>limit</code> and <code>offset</code> options</a></li> + <li><a href="#Example">Example</a></li> + </ul> + </li> + <li><a href="#GET-indexes/:id">GET indexes/:id</a> + <ul> + <li><a href="#Example">Example</a></li> + </ul> + </li> + <li><a href="#PUT-indexes/:id">PUT indexes/:id</a> + <ul> + <li><a href="#Example">Example</a></li> + </ul> + </li> + <li><a href="#DELETE-indexes/:id">DELETE indexes/:id</a> + <ul> + <li><a href="#Example">Example</a> +</li> + </ul> </li> </ul>@@ -5843,7 +5917,7 @@
<a name="Enhanced-Querying-of-JSON-documents"></a> <h4>Enhanced Querying of JSON documents<a href="#document-top" title="Go to top"></a></h4> -<p>By leveraging the <a href="https://www.sqlite.org/json1.html">SQLite JSON1 extension</a> and implementing custom query string parsing, LiteStore provides enhanced filtering, ordering, and custom field selection of JSON documents.</p> +<p>By leveraging the <a href="https://www.sqlite.org/json1.html">SQLite JSON1 extension</a> and implementing custom query string parsing, LiteStore provides enhanced filtering, ordering, and custom field selection of JSON documents. Additionally, you can also configure custom indexes specifying JSON fields for extra speed!</p> <a name="Full-text-Search"></a> <h4>Full-text Search<a href="#document-top" title="Go to top"></a></h4>@@ -5855,6 +5929,16 @@ <h4>RESTful HTTP API<a href="#document-top" title="Go to top"></a></h4>
<p>Every operation can be performed on the data store using a simple but powerful RESTful HTTP API, perfect for client-side, single-page applications.</p> +<a name="JWT-based-Authorization"></a> +<h4>JWT-based Authorization<a href="#document-top" title="Go to top"></a></h4> + +<p>LiteStore can be configure validate <a href="https://jwt.io/">JWT</a> tokens and configure access to specific resources based on specific <a href="https://oauth.net/2/scope/">OAuth2 scopes</a>.</p> + +<a name="Middleware"></a> +<h4>Middleware<a href="#document-top" title="Go to top"></a></h4> + +<p>By leveraging the <a href="https://duktape.org/">duktape</a> library, you can create your own middleware functions in JavaScript to perform additional tasks (validation, logging, data aggregation…) before accessing data.</p> + <a name="Nim-API"></a> <h4>Nim API<a href="#document-top" title="Go to top"></a></h4>@@ -5889,7 +5973,7 @@ <p>As a document store, LiteStore provides the following features</p>
<ul> <li>You can save and retrieve data as arbitrary JSON documents but also as arbitrary documents of virtually any content type.</li> -<li>You can query data using user-specified and system tags and/or via the native full-text search functionality (available only for textual documents).</li> +<li>You can query data using user-specified and system tags, via the native full-text search functionality (available only for textual documents) or writing SQL-like queries on JSON documents.</li> <li>You can access data by the means of a RESTful API.</li> </ul>@@ -5961,7 +6045,7 @@ <h4>Database Schema<a href="#document-top" title="Go to top"></a></h4>
<p>The database schema of LiteStore data store file is very simple, as shown in the following diagram:</p> -<p><img src="" alt="LiteStore Database" /></p> +<p><img src="" alt="LiteStore Database" /></p> <a name="info-Table"></a> <h5>info Table<a href="#document-top" title="Go to top"></a></h5>@@ -5981,7 +6065,7 @@ <p>The <span class="kwd">documents</span> table is the most important table of the data store, as it contains all the documents stored in it. The following information is stored for each document:</p>
<ul> <li><strong>docid</strong> – The internal unique document identifier.</li> -<li><strong>id</strong> – The public unique document identifier, used to access the document via the HTTP API.</li> +<li><strong>id</strong> – The public unique document identifier, used to access the document via the HTTP API or Nim API.</li> <li><strong>data</strong> – The contents of the document (or their base64-encoded representation in case of binary documents).</li> <li><strong>binary</strong> – Whether the document is binary (1) or textual (0).</li> <li><strong>searchable</strong> – Whether the document is searchable (1) or not (0). Currently, textual documents are searchable and binary documents are not.</li>@@ -5990,6 +6074,28 @@ <li><strong>modified</strong> – When the document was last modified.</li>
</ul> +<a name="system_documents-Table"></a> +<h5>system_documents Table<a href="#document-top" title="Go to top"></a></h5> + +<p>The <span class="kwd">system_documents</span> table has a structure similar to the <span class="kwd">documents</span> table, but it is used for well-known system documents that are used to provide additional functionalities, such as authorization or custom resources.</p> + +<p>Unlike ordinary documents, system documents: +* cannot be accessed via the HTTP or Nim API, they can only be imported, exported, or deleted with the corresponding commands. +* are not searchable. +* cannot be tagged.</p> + +<p>The following information is stored for each system document:</p> + +<ul> +<li><strong>docid</strong> – The internal unique document identifier.</li> +<li><strong>id</strong> – The public unique document identifier.</li> +<li><strong>data</strong> – The contents of the document (or their base64-encoded representation in case of binary documents).</li> +<li><strong>binary</strong> – Whether the document is binary (1) or textual (0).</li> +<li><strong>created</strong> – When the document was created.</li> +<li><strong>modified</strong> – When the document was last modified.</li> +</ul> + + <a name="tags-Table"></a> <h5>tags Table<a href="#document-top" title="Go to top"></a></h5>@@ -6009,9 +6115,9 @@
<p>The easiest way to get LiteStore is by downloading one of the prebuilt binaries from the [Github Release Page][release]:</p> <ul> -<li><a href="https://github.com/h3rald/litestore/releases/download/1.6.0litestore_1.6.0_macosx_x64.zip">LiteStore for Mac OS X (x64)</a></li> -<li><a href="https://github.com/h3rald/litestore/releases/download/1.6.0/litestore_1.6.0_windows_x64.zip">LiteStore for Windows (x64)</a></li> -<li><a href="https://github.com/h3rald/litestore/releases/download/1.6.0/litestore_1.6.0_linux_x64.zip">LiteStore for Linux (x64)</a></li> +<li><a href="https://github.com/h3rald/litestore/releases/download/1.8.0/litestore_1.8.0_macosx_x64.zip">LiteStore for Mac OS X (x64)</a></li> +<li><a href="https://github.com/h3rald/litestore/releases/download/1.8.0/litestore_1.8.0_windows_x64.zip">LiteStore for Windows (x64)</a></li> +<li><a href="https://github.com/h3rald/litestore/releases/download/1.8.0/litestore_1.8.0_linux_x64.zip">LiteStore for Linux (x64)</a></li> </ul>@@ -6042,7 +6148,7 @@
<p>To get the app up and running (assuming that you have the <span class="cmd">litestore</span> executable in your path):</p> <ol> -<li>Download the default <a href="https://github.com/h3rald/litestore/releases/download/1.6.0/data.db">data.db</a> file. This file is a LiteStore data store file containing the sample app.</li> +<li>Extract the default <strong>data.db</strong> file included in the LiteStore release package. This file is a LiteStore data store file containing the sample app.</li> <li>Go to the local directory in which you downloaded the <span class="cmd">data.db</span> file.</li> <li>Run <span class="cmd">litestore -s:data.db</span></li> <li>Go to <a href="http://localhost:9500/docs/admin/index.html">localhost:9500/docs/admin/index.html</a>.</li>@@ -6078,6 +6184,7 @@ <ul>
<li><strong>-a</strong>, <strong>--address</strong> — Specify server address (default: 127.0.0.1).</li> <li><strong>–auth</strong> — Specify an authorization configuration file.</li> <li><strong>-b</strong>, <strong>–body</strong> — Specify a string containing input data for an operation to be executed.</li> +<li><strong>-c</strong>, <strong>–config</strong> — Specify a configuration file.</li> <li><strong>-d</strong>, <strong>--directory</strong> — Specify a directory to serve, import, export, delete, or mount.</li> <li><strong>-f</strong>, <strong>–file</strong> — Specify a file containing input data for an operation to be executed.</li> <li><strong>-h</strong>, <strong>--help</strong> — Display program usage.</li>@@ -6087,9 +6194,11 @@ <li><strong>-o</strong>, <strong>–operation</strong> — Specify an operation to execute via the execute command: get, put, delete, patch, post, head, options.</li>
<li><strong>-p</strong>, <strong>--port</strong> —Specify server port number (default: 9500).</li> <li><strong>-r</strong>, <strong>--readonly</strong> — Allow only data retrieval operations.</li> <li><strong>-s</strong>, <strong>--store</strong> — Specify a datastore file (default: data.db)</li> +<li><strong>–system</strong> — Set the system flag for import, export, and delete operations</li> <li><strong>-t</strong>, <strong>–type</strong> — Specify a content type for the body an operation to be executed via the execute command.</li> <li><strong>-u</strong>, <strong>–uri</strong> — Specify an uri to execute an operation through the execute command.</li> <li><strong>-v</strong>, <strong>--version</strong> — Display the program version.</li> +<li><strong>-w</strong>, <strong>–middleware</strong> — Specify a path to a folder containing middleware definitions</li> </ul>@@ -6103,6 +6212,12 @@ <ul>
<li><p>with default settings:</p> <p><span class="cmd">litestore</span></p></li> +<li><p>loading configuration from a configuration file called <strong>config.json</strong>:</p> + +<p><span class="cmd">litestore -c:config.json</span></p></li> +<li><p>loading middleware definition files stored in a directory called <strong>myMiddleware</strong>:</p> + +<p><span class="cmd">litestore -w:myMiddleware</span></p></li> <li><p>with custom port (<strong>9700</strong>) and address (<strong>0.0.0.0</strong>):</p> <p><span class="cmd">litestore -p:9700 -a:0.0.0.0</span></p></li>@@ -6124,6 +6239,13 @@
<p>Import a directory called <strong>admin</strong>:</p> <p><span class="cmd">litestore import -d:admin</span></p> + +<a name="Importing-system-documents-from-a-directory"></a> +<h4>Importing system documents from a directory<a href="#document-top" title="Go to top"></a></h4> + +<p>Import all documents stored in a directory called <strong>system</strong> as system documents:</p> + +<p><span class="cmd">litestore import -d:system –system</span></p> <a name="Exporting-a-directory"></a> <h4>Exporting a directory<a href="#document-top" title="Go to top"></a></h4>@@ -6170,6 +6292,10 @@ <h2>Authorization<a href="#document-top" title="Go to top"></a></h2>
<p>LiteStore can be configured to automatically validate <a href="https://jwt.io/">JWT</a> tokens and authorize authenticated users on specific resources (and specific resource verbs even) based on their <a href="https://oauth.net/2/scope/">OAuth2 scopes</a> specified in the token itself.</p> +<div class="note"><p>auth.json vs. config.json</p> + +<p>As of version 1.8.0, it is recommended to use the LiteStore configuration file to configure authorization. This specialized <strong>auth.json</strong> configuration file format will however be maintained for compatibility reasons.</p></div> + <p>To configure authorization, create an <strong>auth.json</strong> file like the following:</p> <pre><code>{@@ -6208,10 +6334,431 @@ <p><code>litestore --auth:auth.json</code></p>
<p>Once enabled, LiteStore will return HTTP 401 error codes if an invalid token or no token is included in the HTTP Authorization header of the request accessing the resource or HTTP 403 error codes in case an authenticated user does not have a valid scope to access a specified resource.</p> +<a name="Configuration-File"></a> +<h2>Configuration File<a href="#document-top" title="Go to top"></a></h2> + +<p>As of version 1.8.0, you can specify a configuration file containing settings, middleware and authorization configuration using the <strong>–config</strong> or <strong>-c</strong> command line option:</p> + +<p><span class="cmd">litestore -c:config.json</span></p> + +<p>A typical configuration file looks like this:</p> + +<pre><code>{ + "settings": { + "log": "debug", + "port": 9200 + }, + "resources": { + "/docs/vehicles/*": { + "GET": { + "middleware": ["validate", "log"] + }, + "HEAD": { + "middleware": ["validate", "log"] + }, + "POST": { + "allowed": false + }, + "PATCH": { + "auth": ["admin:vehicles"], + "middleware": ["validate", "log"] + }, + "PUT": { + "auth": ["admin:vehicles"], + "middleware": ["validate", "log"] + }, + "DELETE": { + "auth": ["admin:vehicles"], + "middleware": ["validate", "log"] + } + }, + "/docs/logs/*": { + "GET": { + "auth": ["admin:server"] + }, + "POST": { + "allowed": false + }, + "PUT": { + "allowed": false + }, + "PATCH": { + "allowed": false + }, + "DELETE": { + "allowed": false + } + } + }, + "signature": "\n-----BEGIN CERTIFICATE-----\n<certificate text goes here>\n-----END CERTIFICATE-----\n" +} +</code></pre> + +<p>At present, it contains a <span class="kwd">settings</span>, a <span class="kwd">resources</span>, and a <span class="kwd">signature</span> section.</p> + +<a name="settings"></a> +<h3>settings<a href="#document-top" title="Go to top"></a></h3> + +<p>This section contains some of the most common command-line options, i.e.:</p> + +<ul> +<li>address</li> +<li>port</li> +<li>store</li> +<li>directory</li> +<li>mount</li> +<li>readonly</li> +<li>middleware</li> +<li>log</li> +</ul> + + +<p>If a configuration file is specified and some of these settings are configured, they will be recognized as if they were specified via command line. However, if you also specify the same settings via command line, the command line settings will take precedence over the settings defined in the configuration file.</p> + +<a name="resources"></a> +<h3>resources<a href="#document-top" title="Go to top"></a></h3> + +<p>This section can contain any number of resource paths, like <span class="kwd">/docs/</span>, <span class="kwd">/info/</span>, <span class="kwd">/docs/vehicles/AA456CC</span> or <span class="kwd">/docs/logs/*</span>. If a wildcard is specified after a resource or folder path, the rules defined within that section will match any document within the specified path. So for examople <span class="kwd">/docs/vehicles/*</span> will match both <span class="kwd">/docs/vehicles/AB547QV</span> and <span class="kwd">/docs/vehicles/BB326CZ</span>, but <em>not</em> <span class="kwd">/docs/vehicles/</span>.</p> + +<p>Within each resource path, you can specify different HTTP methods (all uppercase) and within each method any of the following properties:</p> + +<ul> +<li><strong>auth</strong> — A list of JWT scopes necessary to access the specified resource with the specified method.</li> +<li><strong>middleware</strong> — A list of middleware function definitions that will be executed in sequence when the resource is accessed with the specified method.</li> +<li><strong>allowed</strong> — If set to <strong>false</strong>, LiteStore will return a <span class="kwd">405 - Method not allowed</span> error code when accessing the resource with the specified method.</li> +</ul> + + +<a name="signature"></a> +<h3>signature<a href="#document-top" title="Go to top"></a></h3> + +<p>This section must be set to a valid certificate used validate JWT tokens. Note that the certificate must follow a specific format and start with the appropriate begin/end blocks.</p> + +<a name="Middleware"></a> +<h2>Middleware<a href="#document-top" title="Go to top"></a></h2> + +<p>As of version 1.8.0, you can define your own custom middleware using JavaScript executed on the server side.</p> + +<p>LiteStore embeds the <a href="https://duktape.org/">duktape</a> lightweight JavaScript engine and therefore you can use all functionalities exposed by duktape in your code, plus access some LiteStore-specific properties and method through some special global object.</p> + +<p>Although writing extremely complex logic in a JavaScript handler may not be appropriate, it can be useful for certain use cases, such as: +* validating data before it is saved +* manipulate data before it is saved +* aggregating different types of data not accessible via a single query +* perform additional operation when accessing data, such as logging who requested it</p> + +<a name="How-middleware-works"></a> +<h3>How middleware works<a href="#document-top" title="Go to top"></a></h3> + +<p>Potentially, each resource could have one or more middleware functions associated to it. Association is done through the LiteStore configuration file in a similar way as authentication is configured:</p> + +<pre><code>{ + "settings": { + "middleware": "test/middleware" + }, + "resources": { + "/docs/vehicles/*": { + "GET": { + "middleware": ["validate", "log"] + }, + "PUT": { + "auth": ["admin:vehicles"], + "middleware": ["validate", "log"] + } + } + } +} +</code></pre> + +<p>This simple configuration file shows how to configure middleware to be executed when a resources it requested via GET or PUT. In both cases, first the <em>validate</em> middleware function is executed, and then the <em>log</em>. These functions must reside in separate files named <em>validate.js</em> and <em>log.js</em> respectively, and placed into a folder (<strong>test/middleware</strong> in this case) referenced via the <strong>middleware</strong> setting (which is also exposed as a command line option, with <strong>-w</strong> as shorthand option).</p> + +<p>Middleware functions are executed sequentially until one of them explicitly stops the execution chain or the execution completes (by requesting the original resource).</p> + +<p>Considering the previous configuration example, if a PUT request is made to an item under <strong>/docs/vehicles</strong>:</p> + +<ol> +<li>The <em>validate</em> middleware function is executed.</li> +<li>The <em>log</em> middleware function is executed.</li> +<li>The request is processed as normal.</li> +</ol> + + +<p>Note that, for example, the <em>validate</em> middleware function may cause the execution to stop before it reaches the <em>log</em> middleware, thereby effectively implementing server-side validation.</p> + +<a name="Creating-a-JavaScript-Middleware-Function"></a> +<h3>Creating a JavaScript Middleware Function<a href="#document-top" title="Go to top"></a></h3> + +<p>Let’s say you want to keep records of Italian vehicles identified by their number plate, which is in the following format:</p> + +<p>\[two-uppercase-letters\][three-digits][two-uppercase-letters]</p> + +<p>For example: AB467DX (OK, in reality there’s a space between each set of digits/letters, but spaces in identifiers are ugly, so let’s remove them!)</p> + +<p>Let’s also say that Italian vehicle data will be managed within a folder called <strong>vehicles</strong>, therefore vehicles will be accessible at URLs similar to the following:</p> + +<ul> +<li>http://localhost:9500/docs/vehicles/AB467DX</li> +<li>http://localhost:9500/docs/vehicles/CD569BW</li> +<li>http://localhost:9500/docs/vehicles/EF981DE</li> +</ul> + + +<p>To make sure that valid IDs are used, we can create a file called <strong>vehicles.js</strong> and write the following code:</p> + +<pre><code>(function() { + var id = $req.path.replace(/^\/docs\//, ""); + var valid = /[A-Z]{2}[0-9]{3}[A-Z]{2}/; + if (!id.match(valid)) { + $res.content = { + error: "Invalid number plate" + }; + $res.code = 400; + return true; + } + $ctx.existing = !!($store.get("docs", id).code == 200); +})(); +</code></pre> + +<p>Note that middleware must be coded in the form of an <a href="https://en.wikipedia.org/wiki/Immediately_invoked_function_expression">IIFE</a>. In this case, the function: +* retrieves the ID of the vehicle. +* checks if it’s valid +* if it’s invalid, prepares a 400 - Bad Request response and stops the execution of additional middleware (and ultimately the request itself) by returning <strong>true</strong>. +* otherwise, it checks whether the vehicle already exists and stores this information on a context, so that it will be accessible to other middleware functions down the execution chain.</p> + +<a name="Passing-data-to-another-middleware"></a> +<h3>Passing data to another middleware<a href="#document-top" title="Go to top"></a></h3> + +<p>Although you can technically add additional properties to the <strong>$req</strong> and <strong>$res</strong> objects, you should use <strong>$ctx</strong> instead. <strong>$ctx</strong> is a global objects.</p> + +<p>In the <em>validate</em> middleware described in the previous section, the <strong>$ctx.existing</strong> property was set. This property can then be accessed and/or modified in additional middleware down the execution chain.</p> + +<p>Consider for example the following, very basic <em>log</em> middleware function:</p> + +<pre><code>(function(){ + var doc = { + user: $req.jwt.claims && $req.jwt.claims.sub || null, + agent: $req.headers['user-agent'], + language: $req.headers['accept-language'] && $req.headers['accept-language'].replace(/,.+$/, ''), + path: $req.path, + existing: !!$ctx.existing, + method: $req.method, + timestamp: Date.now() + } + $store.post('docs', 'logs', JSON.stringify(doc), 'application/json'); +}()) +</code></pre> + +<p>This middleware function simply logs the current request to another folder within the data store, and gathers some stats and whether the request was performed on an existing object or not. In this case, <strong>$ctx.existing</strong> should be set by another middleware up the chain (<em>validate</em>).</p> + +<a name="Global-JavaScript-Objects"></a> +<h2>Global JavaScript Objects<a href="#document-top" title="Go to top"></a></h2> + +<p>When creating JavaScript handlers for middleware, you can use some special $-prefixed global objects to access the HTTP request to the resource, the HTTP response, and also access other LiteStore resources.</p> + +<a name="$ctx"></a> +<h3>$ctx<a href="#document-top" title="Go to top"></a></h3> + +<p>An empty object that can be used to temporarily store data to pass across different middleware handlers.</p> + +<a name="$req"></a> +<h3>$req<a href="#document-top" title="Go to top"></a></h3> + +<p>The current HTTP request sent to access the current resource.</p> + +<a name="Properties"></a> +<h4>Properties<a href="#document-top" title="Go to top"></a></h4> + +<dl> +<dt>method: string</dt> +<dd>The HTTP method used by the request, all uppercase (GET, POST, DELETE, PUT, PATCH, OPTIOONS, or HEAD).</dd> +<dt>jwt: object</dt> +<dd>An object containing a parsed JWT token, if present. It exposes two properties: +<ul> +<li><strong>headers</strong>, an object typically containing the <strong>alg</strong> (algorithm) and <strong>typ</strong> (type) keys.</li> +<li><strong>claims</strong>, an object containing the claims included in the token (see the <a href="https://www.iana.org/assignments/jwt/jwt.xhtml#claims">IANA JSON Web Token Claims Registry</a> for a list of possible claims).</li> +</ul></dd> +<dt>headers: object</dt> +<dd>An object containing the request headers, as keys and values.</dd> +<dt>protocol: string</dt> +<dd>The request protocol and version.</dd> +<dt>hostname: string</dt> +<dd>The hostname target of the request.</dd> +<dt>port: number</dt> +<dd>The port used for the request.</dd> +<dt>path: string</dt> +<dd>The path to the resource requested.</dd> +<dt>query: string</dt> +<dd>The contents of the request query string.</dd> +<dt>content: string</dt> +<dd>When applicable, the content that was sent as body of the request.</dd> +</dl> + + +<a name="$res"></a> +<h3>$res<a href="#document-top" title="Go to top"></a></h3> + +<p>The HTTP response to return to the client.</p> + +<dl> +<dt>code: number</dt> +<dd>The HTTP return code, by default set to `200`.</dd> +<dt>content: string</dt> +<dd>The response content, by default set to `""`.</dd> +<dt>headers: object</dt> +<dd>The response headers, by default set to: +<pre><code> +{ + "Access-Control-Allow-Origin": "*", + "Access-Control-Allow-Headers": "Authorization, Content-Type", + "Server": "LiteStore/<version>", + "Content-Type": "application/json", + "Content-Length": "<Automatically set to the length of the <b>content</b> property.>" +} +</code></pre></dd> +</dl> + + +<a name="$store"></a> +<h3>$store<a href="#document-top" title="Go to top"></a></h3> + +<p>Simple synchronous API to access LiteStore resources in a RESTful way, mimicking HTTP methods.</p> + +<p>All methods return a response object containing two String properties, <strong>code</strong> and <strong>content</strong>.</p> + +<dl> +<dt>function get(resource: string, id: string, parameters: string): object</dt> +<dd>Retrieves the specified resource(s.). +<p> +Examples: +<ul> +<li><code>$store.get('docs', 'test-folder/test.json')</code></li> +<li><code>$store.get('docs', '', 'search=test&limit=20&offset=0')</code></li> +</ul> +</p> +</dd> +<dt>function post(resource: string, folder: string, body: string, contentType: string): object</dt> +<dd>Creates a new resource. +<p> +Examples: +<ul> +<li><code>$store.post('docs', 'test-folder', 'test!', 'text/plain')</code></li> +<li><code>$store.post('docs', '', '{"a": 1}', ?application/json')</code></li> +</ul> +</p> +</dd> +<dt>function put(resource: string, id: string, body: string, contentType: string): object</dt> +<dd>Creates or updates a specific resource. +<p> +Examples: +<ul> +<li><code>$store.put('docs', 'test-folder/test1.txt', 'Another Test.', 'text/plain')</code></li> +<li><code>$store.put('docs', 'test.json', '{"a": 2}', 'application/json')</code></li> +</ul> +</p> +</dd> +<dt>function patch(resource: string, id: string, body: string): object</dt> +<dd>Patches one or more fields of an existing resource. +<p> +Examples: +<ul> +<li><code>$store.patch('docs', 'test-folder/test1.txt', '{"op":"add", "path":"/tags/3", "value":"test1"}')</code></li> +</ul> +</p> +</dd> +<dt>function delete(resource: string, id: string): object</dt> +<dd>Deletes a specific resource. +<p> +Examples: +<ul> +<li><code>$store.delete('docs', 'test-folder/test1.txt')</code></li> +<li><code>$store.delete('docs', 'test.json')</code></li> +</ul> +</p> +</dd> +<dt>function head(resource: string, id: string): object</dt> +<dd>Retrieves the metadata of one or more resources, without retrieving their contents. +<p> +Examples: +<ul> +<li><code>$store.head('docs', 'test-folder/test1.txt')</code></li> +<li><code>$store.head('docs')</code></li> +</ul> +</p> +</dd> +</dl> + + +<a name="System-Documents"></a> +<h2>System Documents<a href="#document-top" title="Go to top"></a></h2> + +<p>As of version 1.8.0, it is possible to import, export, or delete <em>system documents</em> besides ordinary documents. Such documents are different from ordinary documents, because:</p> + +<ul> +<li>they are only intended to be used internally by LiteStore.</li> +<li>they cannot be accessed via any of the public APIs except for the <span class="kwd">import</span>, <span class="kwd">export</span> and <span class="kwd">delete</span> commands.</li> +<li>they must have a well-known name and/or folder structure.</li> +</ul> + + +<p>At present, only the following system documents are recognized by LiteStore:</p> + +<ul> +<li><strong>auth.json</strong> — The LiteStore authorization configuration file.</li> +<li><strong>config.json</strong> — The main LiteStore configuration file.</li> +<li><strong>middleware/*.js</strong> — Any <span class="ext">.js</span> file containing the definition of a middleware function, placed within a <span class="dir">middleware</span> folder.</li> +</ul> + + +<a name="Importing,-exporting-and-deleting-System-Documents"></a> +<h3>Importing, exporting and deleting System Documents<a href="#document-top" title="Go to top"></a></h3> + +<p>You can import, export, and delete system documents with the respective commands, but you must specify the <span class="kwd">–system</span> command line flag.</p> + +<p>For example, suppose you have a <span class="dir">sysdocs</span> folder containing the following file hierarchy:</p> + +<ul> +<li>sysdocs/ + +<ul> +<li>auth.jsom</li> +<li>config.json</li> +<li>middleware/ + +<ul> +<li>log.js</li> +<li>req.js</li> +<li>validate.js</li> +</ul> +</li> +</ul> +</li> +</ul> + + +<p>To import all the documents stored within the <span class="dir">sysdocs</span> folder, you must run the following command:</p> + +<p><span class="kwd">litestore -d:sysdocs –system import</span></p> + +<p>Similarly, the <span class="kwd">export</span> and <span class="kwd">delete</span> commands can be used to export and delete system documents respectively, always specifying the <span class="kwd">–system</span> flag.</p> + +<a name="How-LiteStore-uses-System-Documents"></a> +<h3>How LiteStore uses System Documents<a href="#document-top" title="Go to top"></a></h3> + +<p>While at development time you may want to be able to edit your system documents and therefore keep them outside your data store as ordinary text files (and load them using the <strong>–auth</strong>, <strong>–config</strong> and <strong>–middleware</strong> options), in production you may want to ship them within the data store along with your application data.</p> + +<p>At run time, LiteStore will attempt to retrieve settings/middleware/authorization configuration using the following order of precedence (first listed have higher precedence):</p> + +<ol> +<li>Command line options</li> +<li>Configuration files specified via command line options</li> +<li>Configuration files loaded as system documents</li> +</ol> + + <a name="Administration-App"></a> <h2>Administration App<a href="#document-top" title="Go to top"></a></h2> -<p>A simple but fully-functional administration app can be used to manage LiteStore data stores. This app can simply be imported into a data store file and then run via LiteStore.</p> +<p>A simple, <em>slightly</em> dated, but fully-functional administration app can be used to manage LiteStore data stores. This app can simply be imported into a data store file and then run via LiteStore.</p> <a name="Obtaining-and-Running-the-Administration-App"></a> <h3>Obtaining and Running the Administration App<a href="#document-top" title="Go to top"></a></h3>@@ -6675,10 +7222,14 @@ <p>Filter expressions can be composed by one or more clauses joined together through <strong>or</strong> or <strong>and</strong> operators. Each clause must be composed exactly by:</p>
<ul> <li>A path expression indicating a field or array item within the JSON document.</li> -<li>One operator among the following: eq, not eq, gt, gte, lt, lte, contains.</li> +<li>One operator among the following: eq, not eq, gt, gte, lt, lte, contains, and like.</li> <li>A value that can be a number, string, <strong>true</strong>, <strong>false</strong> or <strong>nil</strong></li> </ul> + +<div class="note"><p>API v5 Required</p> + +<p>Support for the <strong>like</strong> operator has been added in version 5 of the LiteStore API.</p></div> <div class="warning"><p>Limitations</p>@@ -6694,6 +7245,7 @@
<ul> <li>http://127.0.0.1:9500/docs/?filter=$.age%20gte%2018%20or%20$.skills%20contains%20"maths"</li> <li>http://127.0.0.1:9500/docs/?filter=$.name.first&20eq%20"Jack"%20or%20$.fav_food[0]%20eq%20"pizza"</li> +<li>http://127.0.0.1:9500/docs/?filter=$.name.first&20like%20"J*"</li> </ul>@@ -7087,6 +7639,17 @@ <div class="tip"><p>Wildcards</p>
<p>You can use asterisks (*) as wildcards.</p></div> +<a name="<code>limit</code>-and-<code>offset</code>-options"></a> +<h5><code>limit</code> and <code>offset</code> options<a href="#document-top" title="Go to top"></a></h5> + +<p>Provide a way to implement pagination:</p> + +<ul> +<li><strong>limit</strong> causes the query to retrieve only the first <em>n</em> results.</li> +<li><strong>offset</strong> causes the query to skip the first <em>n</em> results.</li> +</ul> + + <a name="Example"></a> <h5>Example<a href="#document-top" title="Go to top"></a></h5>@@ -7140,6 +7703,172 @@
{"id":"$type:text","documents":32} </code></pre> +<a name="indexes-(LiteStore-Indexes)"></a> +<h3>indexes (LiteStore Indexes)<a href="#document-top" title="Go to top"></a></h3> + +<div class="note"><p>API v5 Required</p> + +<p>This resource has been introduced in version 5 of the LiteStore API.</p></div> + +<p>LiteStore Indexes are special indexes used to optimize the performance of queries on JSON documents.</p> + +<div class="warning"><p>JSON-only Documents Required!</p> + +<p>Indexes can be created <em>only</em> if the entire database is composed by JSON documents. If not, LiteStore will return an error when attempting to create the first index.</p></div> + +<a name="OPTIONS-indexes"></a> +<h4>OPTIONS indexes<a href="#document-top" title="Go to top"></a></h4> + +<p>Returns the allowed HTTP verbs for this resource.</p> + +<a name="Example"></a> +<h5>Example<a href="#document-top" title="Go to top"></a></h5> + +<pre><code>$ curl -i -X OPTIONS http://127.0.0.1:9500/indexes +HTTP/1.1 200 OK +server: LiteStore/1.7.0 +access-control-allow-origin: http://localhost:9500 +access-control-allow-headers: Content-Type +allow: GET,OPTIONS +access-control-allow-methods: GET,OPTIONS +content-length: 0 +</code></pre> + +<a name="OPTIONS-indexes/:id"></a> +<h4>OPTIONS indexes/:id<a href="#document-top" title="Go to top"></a></h4> + +<p>Returns the allowed HTTP verbs for this resource.</p> + +<a name="Example"></a> +<h5>Example<a href="#document-top" title="Go to top"></a></h5> + +<pre><code>$ curl -i -X OPTIONS http://127.0.0.1:9500/indexes/name +HTTP/1.1 200 OK +server: LiteStore/1.7.0 +access-control-allow-origin: http://localhost:9500 +access-control-allow-headers: Content-Type +allow: GET,OPTIONS,PUT,DELETE +access-control-allow-methods: GET,OPTIONS,PUT,DELETE +Content-Length: 0 +</code></pre> + +<a name="GET-indexes"></a> +<h4>GET indexes<a href="#document-top" title="Go to top"></a></h4> + +<p>Retrieves all indexes and their respective JSON fields.</p> + +<a name="<code>like</code>-option"></a> +<h5><code>like</code> option<a href="#document-top" title="Go to top"></a></h5> + +<p>If this option is specified, retrieves all indexes matching the specified string.</p> + +<div class="tip"><p>Wildcards</p> + +<p>You can use asterisks (*) as wildcards.</p></div> + +<a name="<code>limit</code>-and-<code>offset</code>-options"></a> +<h5><code>limit</code> and <code>offset</code> options<a href="#document-top" title="Go to top"></a></h5> + +<p>Provide a way to implement pagination:</p> + +<ul> +<li><strong>limit</strong> causes the query to retrieve only the first <em>n</em> results.</li> +<li><strong>offset</strong> causes the query to skip the first <em>n</em> results.</li> +</ul> + + +<a name="Example"></a> +<h5>Example<a href="#document-top" title="Go to top"></a></h5> + +<pre><code>$ curl -i http://localhost:9500/indexes/?like=%2Aname%2A +HTTP/1.1 200 OK +server: LiteStore/1.7.0 +access-control-allow-origin: http://localhost:9500 +content-type: application/json +vary: Origin +access-control-allow-headers: Content-Type +Content-Length: 244 + +{ + "like": "*name*", + "total": 2, + "execution_time": 0.0006140000000000001, + "results": [ + { + "id": "name", + "field": "$.name" + }, + { + "id": "document.name", + "field": "$.document.name" + } + ] +} +</code></pre> + +<a name="GET-indexes/:id"></a> +<h4>GET indexes/:id<a href="#document-top" title="Go to top"></a></h4> + +<p>Retrieves the specified index and corresponding JSON field.</p> + +<a name="Example"></a> +<h5>Example<a href="#document-top" title="Go to top"></a></h5> + +<pre><code>$ curl -i http://localhost:9500/indexes/name +HTTP/1.1 200 OK +server: LiteStore/1.7.0 +access-control-allow-origin: http://localhost:9500 +content-type: application/json +vary: Origin +access-control-allow-headers: Content-Type +Content-Length: 30 + +{"id":"name","field":"$.name"} +</code></pre> + +<a name="PUT-indexes/:id"></a> +<h4>PUT indexes/:id<a href="#document-top" title="Go to top"></a></h4> + +<p>Creates a new index with the specified ID.</p> + +<p>Note that: +* Index IDs can only contain letters, numbers, and underscores. +* Index fields must be valid paths to JSON fields.</p> + +<div class="warning"><p>No updates</p> + +<p>It is not possible to update an existing index. Delete it and re-create it instead.</p></div> + +<a name="Example"></a> +<h5>Example<a href="#document-top" title="Go to top"></a></h5> + +<pre><code>$ curl -i -X PUT -d '{"field": "$.name"}' 'http://127.0.0.1:9500/indexes/name' --header "Content-Type:application/json" +HTTP/1.1 201 Created +Content-Length: 31 +Content-Type: application/json +Access-Control-Allow-Headers: Content-Type +Access-Control-Allow-Origin: http://localhost:9500 +Server: LiteStore/1.7.0 + +{"id":"name", "field":"$.name"} +</code></pre> + +<a name="DELETE-indexes/:id"></a> +<h4>DELETE indexes/:id<a href="#document-top" title="Go to top"></a></h4> + +<p>Deletes the specified index.</p> + +<a name="Example"></a> +<h5>Example<a href="#document-top" title="Go to top"></a></h5> + +<pre><code>$ curl -i -X DELETE 'http://127.0.0.1:9500/indexes/name' +HTTP/1.1 204 No Content +Content-Length: 0 +Access-Control-Allow-Headers: Content-Type +Access-Control-Allow-Origin: http://localhost:9500 +Server: LiteStore/1.7.0 +</code></pre> + <a name="Nim-API-Reference"></a> <h2>Nim API Reference<a href="#document-top" title="Go to top"></a></h2>@@ -7578,7 +8307,7 @@ </ul>
</div> <div id="footer"> - <p><span class="copy"></span> Fabio Cevasco – December 29, 2019</p> + <p><span class="copy"></span> Fabio Cevasco – March 7, 2020</p> <p><span>Powered by</span> <a href="https://h3rald.com/hastyscribe"><span class="hastyscribe"></span></a></p> </div> </div>
@@ -5,7 +5,7 @@ github: litestore
home: /litestore/ active: true docs: /litestore/LiteStore_UserGuide.htm -version: 1.6.0 +version: 1.8.0 subtitle: "A tiny NoSQL database for rapid prototyping" summary: "A lightweight, self-contained, RESTful, multi-format NoSQL document store server written in Nim and powered by a SQLite backend for storage." download: "https://github.com/h3rald/litestore/releases/download/"