rescript-embed-lang
v0.5.5
Published
A general purpose PPX and library for embedding other languages into ReScript, via code generation.
Downloads
144
Readme
rescript-embed-lang
A general purpose PPX and library for embedding other languages into ReScript, via code generation.
The PPX itself is very very simple - just swap out the embedded language string with a reference to the code generated for that embed. The code generation happens elsewhere. This way embedding languages is flexible and light weight.
This package will eventually ship with a set of utils for making the code generation part easy to set up as well.
Installation
npm i rescript-embed-lang
And then add the PPX to your bsconfig.json
:
"ppx-flags": ["rescript-embed-lang/ppx"]
There, all set!
Why one general PPX
PPXes can be complex and difficult to maintain, and costs at bit of performance. Therefore, this PPX is intended to be extended to support as many use cases around embedding other languages into ReScript as possible. This way, all language embeds built can use one central PPX rather than implementing their own. Maintenance becomes drastically easier, and performance is only hit once if you use several language embeds.
Supported extensions
EdgeQL
You can embed EdgeQL directly as an assignment to a let
binding:
// Movies.res
let findMovieQuery = %edgeql(`
# @name findMovieQuery
select Movie {
id
title
} filter .id = <uuid>$movieId
`)
Is transformed into:
// Movies.res
let findMovieQuery = Movies__edgedb.FindMovieQuery.query
You can also embed it via a module, in case you want easy access to all of the things emitted in the generated code:
// Movies.res
module FindMovieQuery = %edgeql(`
# @name findMovieQuery
select Movie {
id
title
} filter .id = <uuid>$movieId
`)
Is transformed into:
// Movies.res
module FindMovieQuery = Movies__edgedb.FindMovieQuery
Generic transform
rescript-embed-lang
ships with a generic transform, intended to make experimenting with writing new language embeds + generating code for them much easier in user land, without needing you to add a full transform to this PPX. It expects a specific structure (more below) in order to connect your generated code with your ReScript source.
You turn it on by passing -enable-generic-transform
in your PPX flags config:
"ppx-flags": [["rescript-embed-lang/ppx", "-enable-generic-transform"]]
It works like this:
// SomeFile.res
let myThing = %generated.css(`
.button {
color: blue;
}
`)
This will be transformed into:
// SomeFile.res
let myThing = SomeFile__css.M1.default
It also works with module references:
// SomeFile.res
module MyThing = %generated.css(`
.button {
color: blue;
}
`)
Is transformed into:
// SomeFile.res
module MyThing = SomeFile__css.M1
Notice that you can put anything to the right of
%generated
. The example showscss
, but you could use anything else as well. Example:%generated.openapi("...")
.
The formula for what code to refer to when transforming is be: <filename>__<generated-extension>.M<module-count-for-extension>.default
. When using module bindings, the last part .default
is omitted.
- We're in
SomeFile.res
and usinggenerated.css
, so the generated module is expected to be calledSomeFile__css
. - Each submodule in your generated file will be called
M
+ what number of transform for that extension it is, in the local file. So, the first%generated.css
module isM1
, the second in that same file isM2
, and so on. - Finally, we add a generic
default
a target value name, just to have something to refer to.
Remember, the actual codegen creating the module we're referring to here from the source
css
text isn't part of this package. This package is just about making it simple to tie together generated things with its source in ReScript.
SQL
Embedding for Postgres SQL via pgtyped-rescript.
// Movies.res
let findMovieQuery = %sql.one(`
/* @name findMovieQuery */
select id, title from movies where id = :id
`)
Is transformed into:
// Movies.res
let findMovieQuery = Movies__sql.FindMovieQuery.one
Adding more language embeds
Adding more embeds should be straight forward. Reach out if you're interested!