Filo

May 28, 2026 · View on GitHub

Filo is a small scripting language I built to be embedded in Go applications. It's a Lisp with minimal syntax, deterministic execution, and explicit limits on every dimension that could break a host server.

I made it for a problem that comes up every time I try to give end users scripting power: the obvious paths are either too weak to be useful or too powerful to be safe. Filo is the third option -- small enough that an ordinary user can write short rules (validations, expressions, field logic) and closed enough that nothing can go down because of it.

Name and pronunciation

The name Filo comes from the Italian word filo ("thread").

It's pronounced like Italian filo:

  • IPA: /ˈfiː.lo/
  • Rough English approximation: "FEE-lo"

It is not "Filó", "FYE-lo" (/ˈfaɪ.loʊ/), or "fee-LOH" (/fiːˈloʊ/). Just "FEE-lo".

Why I built it

Three concrete cases that needed something like Filo:

  • a RAD-style generic application builder where end users write logic into fields;
  • an RPG platform with rules customizable per game and per character;
  • regular applications configurable by administrators without redeploys.

In all three the requirement is the same: let the user write logic, and never let that logic put the server at risk.

How it stays safe

The runtime ships with explicit constraints from day one:

  • StepLimit -- bounds the number of evaluation steps.
  • RecursionLimit -- bounds the call stack.
  • Timeout -- execution gets cancelled.
  • context.Context -- natural integration with Go cancellation.
  • recover() around the executor -- no script can cause a panic on the host.

There is no file access, no network access, no syscall, no "dangerous" calls of any kind. The only things scripts can touch are the Go functions the host explicitly registers as builtins.

Syntax

Lisp, minimal:

(+ 1 2)
(if (< age 18) "minor" "adult")
(map (fn (x) (* x x)) (list 1 2 3))

I picked Lisp because it's the cheapest syntax to implement and the most predictable for someone who has never programmed. There's nowhere to hide logic in syntactic ornament.

Go integration shape

  • Builtins are written in Go.
  • The global environment is a map[string]Value.
  • Scripts run inside whatever sandbox the host configures.
  • The host stays in control: only what you register is callable.

Language reference

Special forms

NameSyntaxDescription
if(if cond then [else])Conditional. Returns list() (empty/nil) if else is missing and cond is false.
do(do expr1 expr2 ...)Evaluates expressions in order, returns the last result.
let(let ((n v) ...) body)Local variables scoped to the body.
letv(letv (n1 n2) (values v1 v2) body)Destructures multi-value returns (tuples).
fn(fn (args) body)Anonymous function.
def(def name expr)Global variable or function (always in the root scope).
set(set name expr)Updates an existing variable in the nearest scope.
values(values v1 v2 ...)Returns multiple values (a tuple).
exit(exit [value])Terminates execution immediately.
return(return [value])Returns from the current function.

Core builtins

Note: NewEngine() includes the core math/logic/list/type builtins by default. Some sets are intentionally opt-in and must be registered explicitly.

CategoryFunctionDescription
Math+, -, *, /, %Basic arithmetic.
pow(pow x y)
Logic=, !=Equality.
<, <=, >, >=Numeric comparison.
and, or, notBoolean logic.
Typestype-ofReturns "number", "string", "list", etc.
is-emptyTrue for "" or empty list.
is-nilTrue for an empty list (closest thing to nil in the current runtime).
ListslistCreates a list (list 1 2 3).
lengthList length.
head, tailFirst element / rest of list.
nth(nth list index) 0-based access.
list-append(list-append list item) Returns new list with item appended.
list-concat(list-concat l1 l2 ...)
map(map fn list)
fold(fold fn init list)

String builtins

Not enabled by default. To use them:

eng := filo.NewEngine()
filo.RegisterStringBuiltins(eng)
FunctionDescription
str-fmt(str-fmt format args...) Safe fmt.Sprintf.
str-concatConcatenates arguments.
str-join(str-join sep list)
str-split(str-split sep str)
str-lenString length (runes).
str-sub(str-sub str start [end])
str-find(str-find sub str)
str-replace(str-replace old new str)
str-trimTrims whitespace.
str-upperUppercase.
str-lowerLowercase.

Extension: filomath

Requires explicit registration: filomath.RegisterMathBuiltins(eng).

FunctionDescription
abs, sqrtAbsolute value, square root.
floor, ceil, roundRounding.
to-int(to-int n) Truncates float to int.
sin, cos, tanTrig (radians).
log, log10, expLogarithms / exponential.
math-min, math-maxMin / max of arguments.
pi, eConstants.

Extension: filorand

Requires explicit registration: filorand.RegisterRandomBuiltins(eng). These are intentionally non-deterministic.

FunctionDescription
rand-floatRandom number [0.0, 1.0).
rand-int(rand-int n) Random integer [0, n).
uuid-v4Generates a UUID string.

Extension: filojson

JSON helpers for marshal/unmarshal: json-marshal, json-unmarshal, json-null.

Pre-parse / execute (template style)

For scripts run many times against different data, Filo supports pre-parsing, similar to Go's html/template:

// Parse once at startup
script := filo.Must(filo.ParseScript("calc", "(+ x y)"))

// Execute many times with different globals
for _, data := range items {
	globals := map[string]filo.Value{
		"x": filo.VNum(data.X),
		"y": filo.VNum(data.Y),
	}
	result, _, err := script.Execute(ctx, engine, globals, cfg)
	fmt.Println(result.Num)
}

API:

FunctionDescription
ParseScript(name, src)Creates and parses a reusable script.
script.Execute(ctx, eng, globals, cfg)Executes with explicit engine/config.
Must(script, err)Panics if error (for init).
Filo.Execute(script, overrides)Executes with Filo's globals + optional overrides.

Pre-parsing eliminates parsing overhead. Roughly 2x faster for repeated executions.

Marshal / Unmarshal

Convert Go values to Filo values and back:

type Config struct {
	Name string `filo:"name"`
	Port int    `filo:"port"`
}

cfg := Config{Name: "app", Port: 8080}
val, err := filo.Marshal(cfg)
// val = (list (tuple "name" "app") (tuple "port" 8080))

var cfg2 Config
err = filo.Unmarshal(val, &cfg2)
// cfg2 = {Name: "app", Port: 8080}

Type mapping:

Go TypeFilo Kind
boolKBool
int, float64, etcKNumber
stringKString
[]TKList
structKList of (key, value) tuples
map[K]VKList of (key, value) tuples
nilKTuple (empty)

Expressions are evaluated before reaching Unmarshal:

  • (list "port" (+ 8000 80))port = 8080
  • (list "port" "(+ 8000 80)")port = "(+ 8000 80)" (literal string)

Fuzz tests:

# A single fuzz target for 30 seconds
go test -fuzz=FuzzMarshalUnmarshalString -fuzztime=30s .

# Available targets:
# - FuzzMarshalUnmarshalInt
# - FuzzMarshalUnmarshalFloat
# - FuzzMarshalUnmarshalString
# - FuzzMarshalUnmarshalBool
# - FuzzMarshalUnmarshalBytes
# - FuzzMarshalSliceInt

Examples

1. Calculated field

(let ((forca field:for) (bonus field:bonus))
  (+ (* forca 2) bonus))

2. Dynamic configuration

(let ((env ENV))
  (set Address "http://localhost:3210")
  (if (= env "prod")
      (set Address "https://app.example.com")
      (set Address "http://localhost:3210")))

3. Controlled recursion

(let ()
  (def fact (fn (n)
    (if (<= n 1)
        1
        (* n (fact (- n 1))))))
  (fact 5)) ; 120

4. Auto-level helpers

(let ()
  (def thresholds (list 0 300 900 2700 6500 15000))

  (def auto-level (fn (xp thresholds)
    (fold (fn (lvl threshold)
            (if (>= xp threshold) (+ lvl 1) lvl))
         0
         thresholds)))

  (def auto-level-progress (fn (xp thresholds)
    (let ((lvl (auto-level xp thresholds))
          (total (length thresholds)))
      (if (>= lvl total)
          (values lvl 0)
          (let ((next (nth thresholds lvl)))
            (values lvl (- next xp))))))

  (auto-level-progress 1200 thresholds))

Returns (tuple 3 1500) -- character is at level three and needs 1,500 XP to reach the next tier.

Integrating with Go

Register a builtin

add-two sums two numbers passed positionally:

func addTwo(ctx context.Context, args []filo.Value) (filo.Value, error) {
	if len(args) != 2 {
		return filo.Value{}, errors.New("expected two numbers")
	}

	a, err := args[0].AsNumber()
	if err != nil {
		return filo.Value{}, err
	}

	b, err := args[1].AsNumber()
	if err != nil {
		return filo.Value{}, err
	}

	return filo.VNum(a + b), nil
}

func registerMathBuiltins(eng *filo.Engine) {
	eng.MustRegisterBuiltin("add-two", addTwo)
}

eng := filo.NewEngine()
registerMathBuiltins(eng)

res, _, err := eng.RunScript(ctx, "(add-two 10 32)", nil, cfg)
if err != nil {
	return err
}
// res = 42

Register a string formatter

full-name takes two strings, trims, joins:

func fullName(ctx context.Context, args []filo.Value) (filo.Value, error) {
	if len(args) != 2 {
		return filo.Value{}, errors.New("expected first and last name")
	}

	first, err := args[0].AsString()
	if err != nil {
		return filo.Value{}, err
	}

	last, err := args[1].AsString()
	if err != nil {
		return filo.Value{}, err
	}

	combined := strings.TrimSpace(first + " " + last)
	return filo.VString(combined), nil
}

func registerStringBuiltins(eng *filo.Engine) {
	eng.MustRegisterBuiltin("full-name", fullName)
}

eng := filo.NewEngine()
registerStringBuiltins(eng)

res, _, err := eng.RunScript(ctx, "(full-name \"Ada\" \"Lovelace\")", nil, cfg)
if err != nil {
	return err
}
// res = "Ada Lovelace"

Register an aggregator

min-max takes a list of numbers and returns a tuple (min max):

func minMax(ctx context.Context, args []filo.Value) (filo.Value, error) {
	if len(args) != 1 {
		return filo.Value{}, errors.New("expected one list")
	}

	list, err := args[0].AsList()
	if err != nil {
		return filo.Value{}, err
	}

	if len(list) == 0 {
		return filo.Value{}, errors.New("list cannot be empty")
	}

	minVal, err := list[0].AsNumber()
	if err != nil {
		return filo.Value{}, err
	}

	maxVal := minVal
	for i := 1; i < len(list); i++ {
		current, convErr := list[i].AsNumber()
		if convErr != nil {
			return filo.Value{}, convErr
		}

		if current < minVal {
			minVal = current
		}

		if current > maxVal {
			maxVal = current
		}
	}

	return filo.VList([]filo.Value{filo.VNum(minVal), filo.VNum(maxVal)}), nil
}

func registerAggregatorBuiltins(eng *filo.Engine) {
	eng.MustRegisterBuiltin("min-max", minMax)
}

eng := filo.NewEngine()
registerAggregatorBuiltins(eng)

res, _, err := eng.RunScript(ctx, "(min-max (list 4 7 1 9))", nil, cfg)
if err != nil {
	return err
}
// res = (list 1 9)

Passing globals

globals := map[string]filo.Value{
	"field:a": filo.VNum(10),
	"field:b": filo.VNum(5),
}

res, _, err := eng.RunScript(ctx, "(+ field:a field:b)", globals, cfg)
if err != nil {
	return err
}
// res = 15