Skip to content

Latest commit

 

History

History
199 lines (158 loc) · 11.3 KB

README.md

File metadata and controls

199 lines (158 loc) · 11.3 KB

SupaScript: NodeJS for PostgreSQL and Supabase

  • JavaScript-based, NodeJS-like, Deno-inspired extensions for Supabase.
  • Use require() to import node js modules into plv8 PostgreSQL from any location on the web, similar to Deno.
  • sql() helper function to give easy access to PostgreSQL databases
  • exec() helper function to make it easy to call other functions written in SupaScript (PLV8), PLPGSQL, SQL or any other supported language
  • console emulation for console.log() and other console functions See the docs
  • (See SupaScriptConsole client library for viewing console.log output in your browser console in real time)
  • Built-in support (like NodeJS built-ins) for:
    • require("http") see documentation: easy interface to make web calls and send headers, etc. for GET, POST, PUT, PATCH, DELETE, HEAD, HEADER
  • Polyfills
    • Base64: btoa(), atob()
  • Packaged as a PostgreSQL extension or just use the EASY-INSTALL-Vx.y.SQL script to install it

Make writing PostgreSQL modules fun again

Can you write JavaScript in your sleep? Me too. Can you write PlpgSql queries to save your life? Me either.

Enter our masked hero from heaven: PLV8

So how do I write nifty JavaScript modules for Supabase / Postgres? Use SupaScript!

Installation

(If you're using Supabase, you can safely use method 1 below.)

Method 1: Copy the contents of the file EASY-INSTALL-Vx.yy.SQL into a SQL window in your database and run it.

  • The PLV8 extension must be available on your server
  • The HTTP extension must be available on your server

Method 2: Regular extension installation method (if SUPASCRIPT is already available on your server along with PLV8 and HTTP)

CREATE EXTENSION SUPASCRIPT CASCADE

Quick Sample:

Sample function:

create or replace function test_underscore()
returns json as $$
    const _ = require('https://cdn.jsdelivr.net/npm/[email protected]/underscore-min.js');
    const retval = _.map([1, 2, 3], function(num){ return num * 3; });
    return retval;
$$ language plv8;

The sample above uses the require() function to dynamically load the underscore library:

const module_name = require(<url-or-module-name>, <autoload>);

where

  • url-or-module-name: either the public url of the node js module or a module name you've put into the plv8_js_modules table manually.

  • autoload: (optional) boolean: true if you want this module to be loaded automatically when the plv8 extension starts up, otherwise false

Writing SupaScript functions on the server using JavaScript V8

Functions are stored in your PostgreSQL database, so you write them (create them) as SQL statements:

create or replace function hello_javascript(name text)
returns json as $$
    const retval = { message: `Hello, my good friend ${name}!` };
    return retval; 
$$ language plv8;

Let's break this down

create or replace function hello_javascript(name text)

Since we're creating a function (or updating it later using replace), all functions are written using the create/replace syntax. The name of the function here is simply hello_javascript and the parameter the function accepts is called name and it's type is text. You can, of course, accept multiple parameters separated by commas, but each must have a type, such as 'hello_javascript(name text, age int, big_blob_of_json json)`, etc.

returns json

You need to add a return type to your function. Here we are saying this function returns a JSON Object. You can return text, int, or even void if you don't really need a return value. Any valid data type can be returned from a function. Tip: if you have multiple items you want to return from a function, you probably want to return json -- this allows you the most flexibility.

as $$
// JAVASCRIPT CODE HERE
$$

Everything in between the $$ and $$ is pure JavaScript V8 code. Simple.

language plv8;

You need to end your function with language plv8; to tell PostgreSQL to use the PLV8 (JavaScript V8) engine to run this function. You can use uppercase or lowercase here (`LANGUAGE PLV8;').

Now, you've got all that JavaScript goodness flowing, and it hits you -- What? I can't access the huge world of Node JS libraries? What do you expect me to do -- write JavaScript from scratch like an animal? Forget it!

Enter SupaScript require().

Using require()

If you've used NodeJS before, you're in love with require(). Since you don't have access to a file system, though, you can't use npm install. So we need to have a way to load those neato node_modules. How do we do it?

Method 1: load from the web automatically

This is the easiest (and preferred) method.

const module = require('https://url-to-public-function');

Here's how we'd use the popular Moment JS library:

create or replace function test_momentjs()
returns json as $$
    const moment = require('https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.29.1/moment.js', false);
    const retval = moment().add(7, 'days');
    
    return retval; 
$$ language plv8;

Then just call this function from SQL:

select test_momentjs();

Where do I find the url for the function or library I want to use? Hunt around on the library documentation page to find a CDN version of the library or look for documentation that shows how to load the library in HTML with a <SCRIPT> command. Basically, the url you use must point to plain javascript code that can be executed inside a block of JavaScript.

Method 2: manually load the library into your plv8_js_modules table

This isn't the ideal method, but you can do this on your own if you want. Basically you load the source code for the module into the plv8_js_modules table. But you need to deal with escaping the single-quotes and all that fun stuff. Try Method 1 first, there's really no downside as long as you choose a compatible library and you can access it from the internet the first time you use it. See below for details on how all this works.

How require() works

The first time you call require(url) the following stuff happens:

  1. If your requested module is cached, we return it from the cache. Super fast! Woohoo! Otherwise...
  2. We check to see if the url (or module name if you loaded it manually) exists in the plv8_js_modules table. If it does, we load the source for the module from the database and then eval() it. Yes, we're using eval(), and that's how this is all possible. We know about the security vulnerabilities with eval() but in this case, it's a necessary evil. If you've got a better way, hit me up on GitHub.
  3. If the module isn't in our plv8_js_modules table, we use the http_get() function from pgsql-http to load the source into a variable, then we store it in the plv8_js_modules for later. Later when we need it, we can get it from the database, then cache it.

So it goes:

  1. Are you in the cache? Load you now!
  2. Are you in the database? Load you from the database and cache you for next time!
  3. First time being called, ever? We'll load you over http, write you to the database, and you're all set and loaded for next time!

If you call require(url, true) that "true" parameter means "autoload this module" so that it gets loaded into the cache when PLV8 starts up. Only do this with modules you need to have ready to go immediately. False essentially lazy-loads this module the first time it's called after startup.

Requirements:

  1. Supabase database (or any PostgreSQL database, probably, as long as it's a current-enough version).
  2. The PLV8 extension loaded. If you load the SupaScript extension, this will be loaded automatically with cascade. If you're installing manually, make sure you've loaded the PLV8 extension.
  3. The pgsql-http extension loaded. (Same issues as #2 above.)

BONUS FUNCTIONS

sql(sql_statement, arguments)

Accessing the PostgreSQL database from inside JavaScript

We've included a bonus function to streamline access to your PostgreSQL database.

<result> = sql(<sql_statement>, <optional array of arguments>);

This maps directly to plv8.execute() -- SEE: plv8 documentation here

Example usage:

var json_result = sql('SELECT * FROM tbl');
var num_affected = sql('DELETE FROM tbl WHERE price > $1 and type = $2', [ 1000, 'active' ]);

exec(function_name, arguments)

Execute a PostgreSQL function and return a result

<result> = exec(<function_name>, <optional array of arguments>);

To execute another PostgreSQL function that you've created, you need to call it via SQL with "select function_name(parm1, parm2, etc)". This can get ugly and unwieldy, as shown below:

the ugly way

const html_email = sql('select prepare_message(\'invitation to join org\', \'{"name": "Mark", "orgname": "Acme Corp", "url": "https://acme.com"}\')')[0].prepare_message;

Nobody should have to escape nested delimiters. Also notice the sql result is an array of results (with one result) with the result stuffed into a property with the name of the function. Ugh!

Calling with sql results in this JSON that requires that you add [0].function_name to the end of the call:

[{"prepare_message":"prepared message text goes here"}]

Too much work, and too ugly.

Enter "exec", so you can call it like this:

const html_email = exec('prepare_message', ['invitation to join org', '{"name": "Mark", "orgname": "Acme Corp", "url": "https://acme.com"}']);

The result is just the result of the function. Much cleaner, much easier.

Just note: exec calls exactly two parameters:

  1. the name of the function you want to call
  2. an optional array of parameters you want to pass to the function

console

We emulate the JavaScript console object so you can console.log('stuff') and do profiling console.time('my-timer'); console.timeEnd('my-timer'); See the docs

Troubleshooting

If you need to reload a module for some reason, just remove the module's entry from your plv8_js_modules table. Or just wipe it out: delete from plv8_js_modules;

Sometimes a module won't work. If you're using the minified version, try the non-minified version of the library. Or vice-versa. Not every library is going to work, especially anything that requires a DOM, or access to hardware, or things like socket.io. This is just basic JavsScript stuff -- it's not going dispense Pepsi and shoot out rainbows. But it's still very cool and will save you eons of programming time.

There is NO EVENT LOOP

There's no event loop here -- go back in your time machine to 1998, before you knew what Google was, and all programming was simple, and synchronous, and you could still keep your shoes on going through airport security. Don't try using promises, async / await, or anything else that's fancy. Code it like your grandpa would, on a brand new Pentium-based screamer with a big-honkin' CRT monitor that uses more energy than a Tesla Model S.

Credits

  • This is based on the great work of Ryan McGrath here: Deep Dive Into PLV8
  • Of course, the real work here was done by the fantastic PLV8 team.