AskQL is a new query language that is also a programming language. I’ve written an introduction about it in the former post.
In this article we will see how it can be setup in a nodejs server.
Table of Contents
Quickly setting a server
To setup a server with AskQL, we first need a server. Let’s set it up quickly
- Clone the express demo repository:
git clone https://github.com/YonatanKra/askql-demo.git
- Enter into the folder:
cd askql-demo
- Checkout the express-quick-start tag:
git checkout express-quick-start
- npm i
- npm run dev
Now you have a working express repository.
The setup is pretty simple – in index.js
(Code snippet #1) the server is setup with 3 middleware: logRequestMiddleware
, logResultsMiddleware
and errorHandler
. They are being used in the /ask
endpoint.
There’s also a public html page that will serve as our client.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
|
/ask
entry point. Line 12: The server starts to listen to port 8080.If you go to the client (http://localhost:8080) and submit the form, you will see that the submit has been logged. The output should be something like:
The output shown in Figure 1 is not very exciting. You can see something that looks like code (actually, it is called AskScript
) but it doesn’t do anything.
Let’s install AskQL so it would be able to parse our script.
Setting up AskQL
First we need to install AskQL: npm i askql
Now that AskQL is installed, we can import its utilities and classes in our express app.
Let’s import an AskQL middleware:
import middlewareFactory from "askql/askExpressMiddleware/askExpressMiddleware.js";
const { askExpressMiddleware } = middlewareFactory;
The askExpressMiddleware is a function that accepts some resources and configuration objects and returns a middleware function:
const askMiddleware = askExpressMiddleware(
{ resources, values }, { callNext: true, passError: true}
);
Resources are dynamic AskQL resources that can be accessed in the AskScript.
Resources can be functions, data handlers (like filter, map etc.) mathematical operators and actually anything a function can do (even 3rd party npm modules you import).
Values are static values that exist in the server. The values
object is a regular object that holds numbers, strings, booleans, arrays or other objects.
Let’s take the resources that are coming from AskQL and define some values:
import askql from 'askql';
const {resources} = askql;
const values = {
names: ['Johnny', 'Rita', 'Jane', 'Martha', 'Clark'],
powers: ['Nerd', 'C#', 'Cool', 'Superman\'s mom', 'Superman']
};
The configuration object is a means to customise the middleware – for instance, how to handle errors.
Now that the middleware is ready, we can add it to the list of middleware between the two loggers and restart the server.
The final index.js
looks like this:
import express from 'express'; | |
import bodyParser from 'body-parser'; | |
import {logRequestMiddleware, logResultsMiddleware, logError} from "./logger/logger.js"; | |
import middlewareFactory from "askql/askExpressMiddleware/askExpressMiddleware.js"; | |
import askql from 'askql'; | |
const {resources} = askql; | |
const values = { | |
names: ['Johnny', 'Rita', 'Jane', 'Martha', 'Clark'], | |
powers: ['Nerd', 'C#', 'Cool', 'Superman\'s mom', 'Superman'] | |
}; | |
const { askExpressMiddleware } = middlewareFactory; | |
const askMiddleware = askExpressMiddleware({ resources, values }, { callNext: true, passError: true }); | |
const port = 8080; | |
const app = express(); | |
app.use(bodyParser.json()); | |
app.use(bodyParser.urlencoded({ extended: true })); | |
app.use(express.static('public')); | |
app.post('/ask', [logRequestMiddleware, askMiddleware, logResultsMiddleware, logError]); | |
app.listen(port, () => { | |
console.log(`Server listening at http://localhost:${port}`); | |
}); | |
Let’s see it in action
The client is a form that sends AskScript to the ask access point (a post request).
Let’s try out the first code we can possibly think of:
ask { 'Hello World!' }
Write it down in the text area and submit.
Surprisingly, it returns “Hello World!”
Now, instead of just asking for a value from the server, let’s ask for a value from a variable that exists on our server (via the values
we sent to our askMiddleware):
ask { names }
And we get all the names in the group.
Now let’s query for their powers (or maybe I should have used the word traits):
ask { powers }
That was wonderful.
Now we see how easy it is to access values we know about on the server. What about combining resources with values? Let’s do something more interesting:
ask {
let arr = [];
for (let i = 0; i < names:length; i = i + 1) {
arr = arr:set(i, {
name: names:at(i),
power:powers:at(i)
});
}
arr
}
What do you think the above piece of code will do? You probably guessed right – it is a map!
AskQL has us covered with yet another core resource:
ask {
names:map(fun(name, i) {
{name, power: powers:at(i)}
})
}
The two last snippets would both have the same result!
What Else?
AskQL has lots of resources.
Let’s say we want to find one of our heroes according to the hero’s power? We just use the find
resource:
ask {
names:find(fun(c, i) {
return powers:at(i):equals('Innocent')
})
}
There are a lot more resources for AskQL.
Summary
In this post, we’ve setup a nodejs server that allows a very flexible API without doing anything server side – just expose some data.
Anyone using microservices?
Let’s say there’s a microservice that serves some crucial data. How do we access it immediately?
ask {
fetch('https://swapi.dev/api/people'):at('results')
}
That script will return a list of Star Wars characters via the public StarWars API.
In order for the above script to run, one needs to install node-fetch: npm i node-fetch
With AskQL we are not limited by anything. We can not only use the built in resources in order to created flexible scripts. We can also create our own custom resources!
In the next post, I’ll expand on this example and show how to create a custom resource to manipulate MongoDB.
I invite you to the AskQL repository to contribute and be involved with a promising technology. There’s also the discord server where you can hang out with the core team.
Thanks a lot to Piotr from xFAANG and Yonatan Doron from Hodash.dev for the kind and thorough review.