Liyad
Let's make your yet another DSL with Lisp S-expression!
Liyad (Lisp yet another DSL interpreter, or LIYAD is yum and delicious) is
very small Lisp interpreter written in JavaScript.
You can easily start making your new DSL using Lisp and S-expression.
Install
from NPM:
$ npm install liyad --save
or download UMD from release page.
NOTICE:
Use withwebpack >= 5
If you get the error:
Module not found: Error: Can't resolve '(importing/path/to/filename)' in '(path/to/node_modules/path/to/dirname)' Did you mean '(filename).js'?`
Add following setting to your
webpack.config.js
.test: /\.m?js/resolve:fullySpecified: falseOn
webpack >= 5
, the extension in the request is mandatory for it to be fully specified if the origin is a '.mjs' file or a '.js' file where the package.json contains '"type": "module"'.
Install CLI
See liyad-cli .
$ npm install -g liyad-cli$ liyad
Playground
https://shellyln.github.io/liyad/playground.html
Features
- APIs to customize all operators and macros
- Builtin S-expression parser
- Builtin minimal Lisp interpreter
- Reference implementation of LSX (alternative JSX notation using Lisp)
Real world examples
- Ménneu
Component-based extensible document processor - mdne - Markdown Neo Edit
A simple markdown and code editor powered by Markdown-it, Ace and Carlo. - Tynder
TypeScript friendly Data validator for JavaScript.
What is LSX
LSX is an alternative JSX notation using Lisp.
LSX and Liyad advantages:
-
No transpiler needed
- Liyad uses ES6 template literal syntax.
You don't pass the entire code to transpile and evaluate it.
Save your coding times.
- Liyad uses ES6 template literal syntax.
-
Secure execution for untrusted contents
- No host environment's symbols are accessible from evaluated user contents by default.
Malicious codes can not make a serious attack.
- No host environment's symbols are accessible from evaluated user contents by default.
-
Simple and powerful
- What you can do with JSX can be done with LSX.
Plus, LSX itself is a complete data description format and is a complete programming language,
so you can write more concise and powerful.
- What you can do with JSX can be done with LSX.
The LSX runtime directly calls React.createElement
(or a JSX Factory function such as
RedAgate,
Vue.js, etc.) as a Lisp function,
Convert a Lisp list to a renderer component object tree.
In order to resolve the renderer component, you must register the object's constructor with the LSX runtime in advance.
All unresolved lisp function symbols are dispatched to React.createElement('some_unresolved_name', ...)
.
You can declare HTML/XML standard tags.
As with JSX, LSX must always return a single component.
Using Template
Lisp function instead of JSX Fragment
tag will produce the same result.
Example:
lsx`(Template (select (@ (style (display "inline-block") (width "300px") ) (className "foo bar baz") (onChange ) ) ($=for ($=if (== (% $index 2) 1) (option (@ (value $index)) ($concat "odd: " ($get $data "name"))) ) ($=if (== (% $index 2) 0) (option (@ (value $index)) ($concat "even: " ($get $data "name"))) ))))`;
See also:
Playground's source code is written in LSX.
Usage
Output S-expression into JSON:
; console.log JSON.stringifyS` ($list 1 2 3 "a" "b" "C" ($list 4 5 6) )` // You can also parse by calling w/o template literal syntax as following: // S(' ... ') ;
Output:
Run minimal Lisp interpreter:
; console.log JSON.stringifylisp` ($defun fac (n) ($if (== n 0) 1 (* n ($self (- n 1))) )) ($list 1 2 (fac 3) "a" "b" "c" ($list 4 5 (fac 6) ) )` // You can also evaluate by calling w/o template literal syntax as following: // lisp(' ... ') ;
Output:
Render web page with LSX:
;;; ; ; ; ReactDOM.renderlsx`(App)`, document.getElementById'app';
Build your new DSL:
; ; ; ; ; console.log JSON.stringifyMyDSL`( ... )`;
Extended syntax
Comments
# This is a line comment (# ; <-- This is a object literal, not a line comment) ; This is a line comment #|This is a block comment |#
Here document:
lisp
preset interpreter:
"""Hello, Liyad!"""
is equivalent to:
($concat"Hello, Liyad!")
LSX
preset interpreter:
"""Hello, Liyad!"""
is equivalent to:
(Template"Hello, Liyad!")
Template
on theLSX
preset interpreter, it is mapped to the function passed byLsxConfig.JsxFragment
.
See also: Fragments (React), Template (RedAgate).
Here document with variable substitution:
"""Hello, %%%($get name)!"""
is equivalent to:
(Template"Hello, " ($get name) "!")
Here document with custom function:
"""divHello, %%%($get name)!"""
is equivalent to:
(div"Hello, " ($get name) "!")
Here document with custom function and LSX props:
"""div@{(id "123") (class "foo bar baz")}Hello, %%%($get name)!"""
is equivalent to:
(div (@ (id "123") (class "foo bar baz"))"Hello, " ($get name) "!")
Spread operator
($list 1 2 ...($concat (3 4) (5 6)) 7 8)
is equivalent to:
($list 1 2 ($spread ($concat (3 4) (5 6))) 7 8)
and is to be:
$spread
is NOT a macro. The list passed as a parameter is spliced after evaluation.
Splice macro
($list 1 2 3 4 ($splice (5 6 7 8)) 9 10)
is equivalent to:
($list 1 2 3 4 5 6 7 8 9 10)
(($splice ($call x add)) 5 7)
is equivalent to:
($call x add 5 7)
Shorthands
$set
(::foo:bar:baz= 7)
is equivalent to:
($set ("foo" "bar" "baz") 7)
$get
($list ::foo:bar:baz)
is equivalent to:
($list ($get "foo" "bar" "baz"))
$call
(::foo:bar@baz 3 5 7)
is equivalent to:
($call ($get "foo" "bar") baz 3 5 7)
Rest parameter
($defun f (x ...y) ($list x y) ) ($list (f 1) (f 1 2) (f 1 2 3) (f 1 2 3 4) (f 1 2 3 4 5) )
is to be:
Verbatim string literal
Verbatim string literal
($last @"c:\documents\files\u0066.txt")
is to be:
"c:\\documents\\files\\u0066.txt"
Normal string literal
($last "c:\documents\files\u0066.txt")
is to be:
"c:documents\filesf.txt"
Object literal
(# (foo "a") (bar 10) (baz) )
is to be:
nil, null, undefined
($list nil null undefined)
is to be:
null undefined
See this.
Refer the function
($defun fn(x) (+ x 1))($let x (<- fn))(x 3) ;; 4
Liyad is
Lisp-2
language.
Lambda and closure
Lambda
($let fn (-> (x y z) (+ x y z))) (fn 1 2 3) ;; 6
$lambda
is synonym of->
.
Closure
($let fn ($local ((a 1)(b 2)(c 3)) (|-> (x y z) use (a b c) ($set a (+ a x)) ($set b (+ b y)) ($set c (+ c z)) (+ a b c) ))) (fn 1 2 3) ;; 12(fn 1 2 3) ;; 18
$closure
is synonym of|->
.
is equivalent to:
($let fn ($local ((a 1)(b 2)(c 3)) ($capture (a b c) (-> (x y z) ($set a (+ a x)) ($set b (+ b y)) ($set c (+ c z)) (+ a b c) )))) (fn 1 2 3) ;; 12(fn 1 2 3) ;; 18
$capture
can also be used with$defun
.
Recursive call
($defun tarai(x y z) ($if (<= x y) y ($self ($self (- x 1) y z) ($self (- y 1) z x) ($self (- z 1) x y) )))
$self
refers to the function currently defined by$defun
or->
.
Macro
($defmacro FOR (!i <[> <FROM> s <TO> e <]> ...body) `($last ($local ((,i ,s)) ($while (<= ,i ,e) ,@body ($set ,i (+ ,i 1)) )))) ($let c1 0)($let c2 100)(FOR p [ FROM (+ 1) TO (+ 6 -3) ] ($set c1 (+ c1 p)) ($set c2 (+ c2 p)) )
Parameter type checking
formal parameter | constraint |
---|---|
! token |
parameter should be symbol |
< token> |
parameter should be symbol named token |
token:number |
parameter should be number |
token:string |
parameter should be string |
token:function |
parameter should be function |
token:list |
parameter should be list |
token:symbol |
parameter should be symbol |
Don't put spaces between
!
<
>
:type
and token.
Type checking checks formal parameter types before evaluation.
Macro can be overloaded with the same macro name but different numbers of formal parameters.
This object
($let fn (-> () $this))($let xx (# (a 3) (b 5) (f fn) ))($json-stringify (::xx@f)) ;; {"a":3,"b":5}
Compiling functions and lambdas (experimental)
interpreting | compiling | |
---|---|---|
$defun | $$defun | define the function |
$lambda | $$lambda | define the lambda |
-> | => | define the lambda |
$closure | $$closure | define the closure |
|-> | |=> | define the closure |
APIs
SExpression
/ SExpressionAsync
Create a new DSL.
- returns : Template literal function.
config
: Parser config.
S
Parse a S-expression.
- returns : S-expression parsing result as JSON object.
strings
: Template strings.values
: values.
lisp
Evaluate a Lisp code.
- returns : Evalueting result value of Lisp code.
- If input Lisp code has multiple top level parenthesis,
result value is last one.
- If input Lisp code has multiple top level parenthesis,
strings
: Template strings.values
: values.
lisp_async
Evaluate a Lisp code.
(asynchronous features are enabled.)
- returns : Promise that evalueting result value of Lisp code.
- If input Lisp code has multiple top level parenthesis,
result value is last one.
- If input Lisp code has multiple top level parenthesis,
strings
: Template strings.values
: values.
LM
Evaluate a Lisp code (returns multiple value).
- returns : Evalueting result value of lisp code.
- If input Lisp code has multiple top level parenthesis,
result value is array.
- If input Lisp code has multiple top level parenthesis,
strings
: Template strings.values
: values.
LM_async
Evaluate a Lisp code (returns multiple value).
(asynchronous features are enabled.)
- returns : Promise that evalueting result value of lisp code.
- If input Lisp code has multiple top level parenthesis,
result value is array.
- If input Lisp code has multiple top level parenthesis,
strings
: Template strings.values
: values.
LSX
Evaluate a Lisp code as LSX.
- returns : Template literal function.
lsxConf
: LSX config.
LSX_async
Evaluate a Lisp code as LSX.
(asynchronous features are enabled.)
- returns : Template literal function.
lsxConf
: LSX config.
lisp
| lisp_async
| LM
| LM_async
: SExpressionTemplateFn) methods
(evaluateAST
evaluateASTast: SxToken: SxToken;
- returns : evaluation result value.
ast
: AST to evaluate. it should be enclosed in[]
.lisp.evaluateAST; // 6
repl
repl: SExpressionTemplateFn;
- returns : Template literal function that will keep variables and states for each evaluation.
setGlobals
setGlobalsglobals: object: SExpressionTemplateFn;
- returns : myself (template literal function).
globals
: Global variables to preset.
appendGlobals
appendGlobalsglobals: object: SExpressionTemplateFn;
- returns : myself (template literal function).
globals
: Global variables to preset.
setStartup
setStartupstrings: TemplateStringsArray | string, ...values: any: SExpressionTemplateFn;
- returns : myself (template literal function).
strings
: Startup code that evaluate before each evaluation of user code.
setStartupAST
setStartupASTast: SxToken: SExpressionTemplateFn;
- returns : myself (template literal function).
ast
: Startup code AST that evaluate before each evaluation of user code.
appendStartup
appendStartupstrings: TemplateStringsArray | string, ...values: any: SExpressionTemplateFn;
- returns : myself (template literal function).
strings
: Startup code that evaluate before each evaluation of user code.
appendStartupAST
appendStartupASTast: SxToken: SExpressionTemplateFn;
- returns : myself (template literal function).
ast
: Startup code AST that evaluate before each evaluation of user code.
install
installinstaller:SxParserConfig: SExpressionTemplateFn;
- returns : myself (template literal function).
installer
: Installer function that register the operators, macros, constants to theconfig
object.
runScriptTags
Run script tags.
- returns : Evaluation result.
lisp
: Evaluater function.globals
: Global variables.contentType
: Content type attribute of script tags.
Usage:
Tree shaking (webpack)
You can benefit from tree shaking by importing ES module separated files.
Import path | Description |
---|---|
liyad/modules |
Entire library |
liyad/modules/s-exp/types |
Type definitions |
liyad/modules/s-exp/interpreter |
Interpreter DIY APIs SExpression , SExpressionAsync |
liyad/modules/s-exp/interpreter/presets/s-exp |
Preset s-expression parser S |
liyad/modules/s-exp/interpreter/presets/lisp |
Preset interpreters lisp , lisp_async , LM , LM_async |
liyad/modules/s-exp/interpreter/presets/lsx |
Preset interpreters LSX , LSX_async |
liyad/modules/s-exp/operators/core |
Core operators |
liyad/modules/s-exp/operators/arithmetic |
Arithmetic operators |
liyad/modules/s-exp/operators/sequence |
Sequence operators |
liyad/modules/s-exp/operators/concurrent |
Concurrent operators |
liyad/modules/s-exp/operators/jsx |
JSX (LSX) operators |
NOTICE:
liyad/modules/*
are not babelized. These are output asES2015
by tsc.
Operators
See core, arithmetic, sequence, concurrent, JSX (LSX) operators.
License
ISC
Copyright (c) 2018, 2019 Shellyl_N and Authors.