Tutorial
In this tutorial we'll run ws4sqlite for the first time, in order to serve a single database, and we'll run a couple of queries and statements against it.
We'll use Linux, but the information here is "portable" to MacOS and Windows as well.
If ws4sqlite offers a relevant capability that doesn't fit in a tutorial, it will be explained in an info box like this, and a link to the relevant documentation will be provided.
Let's start!
๐ง Installation
The installation is simple, as ws4sqlite "is" just an executable file. Download or build it (matching your OS and architecture), and put it somewhere on the filesystem.
๐ First Run & Configuration
Let's now start the application:
This tells ws4sqlite to serve a database, to be created because a file at the specified path doesn't exist, using default settings. It's now possible to access the database using its id, that is the filename minus the suffix (in this case, testDb
).
More than one database can be served from the same instance, and it's possible to create in-memory databases. Of course more options are possible: provide authentication, open the file as read only, specify some queries/statements on the server that can be referenced in requests, provide initialization statements to apply when creating a database, and several more. This is done by creating a companion YAML file at the same path, called like the database but with a .yaml
extension: testDb.yaml
in our example.
When the app starts, something like this will be printed; it gives information about what is now being served, and how.
The service is now active and serving requests. Use Ctrl-c
to exit, as usual.
๐ First Request
Let's now do something useful. Use a tool like postman to submit a POST call to http://localhost:12321/testDb
, with the following body:
Ensure that the header Content-Type
is set to application/json
!
Let's see what is in the request:
URL: in the connection URL, we specify the database ID to submit the request to, as defined in the config file;
Line 2: specify the transaction operation list: an array of requests (queries or statements) to submit;
Line 4: as the first (and only) request, specify a statement (i.e. a SQL command that doesn't return a resultset).
If all goes well, you should get a 200
response with the following body:
You did it! ๐ Going through the response:
Line 2: the array of results has the same size of the corresponding
transaction
array in the request, and lists the results of each request, in turn;Line 4: specifies that the statement completed with success;
Line 5: there were no updated rows (as reported by SQLite; this is a DDL command).
๐คน Multiple Requests
Let's say you want to run multiple SQLs in the same request. As you may suspect, this is just a matter of adding another item to the transaction
array:
As the name of the array implies, the queries/statements are run in a single transaction. It will be committed at the end of the call, and rolled back if an unmanaged error occours (see the relevant chapter).
The response now has 2 elements in the array:
Please notice Line 9: rowsUpdated
is 1, signaling that we affected one row with the INSERT
.
๐งฏ Managing Errors
Let's send over the last request again (don't remove the file!). The table already exists, so the response will now fail with 500 Internal Server Error
, and with a body of:
Line 2: the (0-based) index of the statement that failed in the
queries
array; an index of-1
would tell us that it's a generic error, not tied to a particular statement;Line 3: the reason of the failure, as reported by SQLite.
This is a general failure. As statements in the same request are run in transaction, the whole transaction is rolled back.
It's also possible to "allow" certain statements to fail. The transaction will be completed, and an error will be reported only on that statement. Send the following request, notice the "noFail"
in the first statement:
The following result is produced, signaling that the first statement failed; the transaction is committed anyway, so the second statement is actually persisted:
๐ Queries With a Result (Set)
Up to now, we tested only statements, that don't return results other than the number of affected rows. Let's see how to run a query.
In the next example we will create a table, insert two rows in it, and read them.
Please start from an empty database: stop ws4sqlite, remove the database file and start it again.
Request:
Line 10: notice that the key now is
query
, to signal that it will generate a result set.
Lines 7..8: we now have an array with two results, containing objects. Each object has several fields, with the key being the name of the database field and the value being... the value. The key/field name is as reported by the database, so
*
works well.
โ๏ธ Using Parameters
The last capability we'll cover is using parameters, either in a statement (e.g. an INSERT
) or in a query. They are specified using named placeholders, like the following.
Line 4: in the statement, we use named placeholders like
:id
;Line 5: we specify the actual values with a
values
object, containing a map with the keys being the placeholders;Same with queries, as at Line 9.
Using placeholders may seem more verbose than specifying the values in the SQL, but it is always the preferrable solution, allowing for example to avoid nasty SQL injection bugs.
For statements, it is also possible to specify multiple sets of values ('batches'); the statement will be cached and replayed for each set of the list. See valuesBatch
in the docs.
As you can see, the response is the same, with only one result in the resultset (since the query selects by primary key).
๐ฏ๏ธ Conclusions
Thanks for reading so far, I hope you liked it! There are many more topics of interest, among which:
Learn to protect your transactions with authentication;
Use a reverse proxy for HTTPS and additional security;
Use stored statements to avoid passing SQL from the client;
Perform scheduled activities, that is: sql statements,
VACUUM
s or backups;Configure CORS for more convenient access from a web page;
...and much more!
Have a nice day! โ๏ธ
Last updated