A script author opening docs/stdlib-reference.md should see every function they can call without imports: the Rhai built-in stdlib (math, string, array, map, blob) plus the seven new PiCloud namespaces. Tight tables over prose — scannable rather than exhaustive. CLAUDE.md current-focus paragraph picks up a pointer to the new doc. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
8.0 KiB
Rhai stdlib reference
Everything in this document is callable from any user script without imports — Rhai's built-in standard library plus the seven PiCloud utility modules added in v1.1.0. Stateful service modules (KV, docs, HTTP, …) ship in subsequent v1.1.x releases and are documented separately.
For the architectural shape (why some modules are stateless and register at engine build, why others are per-call), see sdk-shape.md.
Conventions
- Throw on failure. Every function throws a Rhai runtime error on
bad input (invalid pattern, invalid encoding, out-of-range arg). Use
try { ... } catch (e) { ... }if you want to handle it. ()for absent. Functions that semantically may have no result (e.g.regex::findwhen nothing matches) return(). Test withif v == () { ... }.boolfor predicates. Yes/no questions returnbool.- UTC, milliseconds, lowercase hex, RFC 3986. Defaults chosen once, not per call.
Rhai built-ins (free with every script)
These come with the Rhai engine itself. See the Rhai book for full signatures.
Math: + - * / %, min, max, abs, sqrt, pow, floor,
ceil, round, to_int, to_float, sin, cos, tan, asin,
acos, atan, exp, ln, log, PI(), E().
String: len, is_empty, contains, starts_with, ends_with,
index_of, split, trim, to_lower, to_upper, replace, chars,
pad, sub_string, crop, + (concatenation).
Array: push, pop, shift, insert, remove, len, clear,
truncate, extend, filter, map, reduce, reduce_rev, find,
find_map, any, all, index_of, contains, sort, reverse,
dedup, chunks, splice, [] indexing.
Map: len, is_empty, contains, keys, values, mixin,
remove, clear, fill_with, + (merge), [] and . access.
Blob: len, push, pop, clear, as_string, parse_le_int,
write_*, [] indexing. Blobs are Vec<u8> at the Rust layer.
Logging: log::trace, log::info, log::warn, log::error —
each takes a message and optionally a structured-data map. (Documented
with the SDK contract; mentioned here for completeness.)
regex:: — regular expressions
Linear-time, no backtracking (powered by the Rust regex crate).
Patterns compile per call.
| Function | Description |
|---|---|
regex::is_match(pattern, text) -> bool |
Whether text contains a match. |
regex::find(pattern, text) -> String | () |
First match or () if none. |
regex::find_all(pattern, text) -> Array |
All matches as String array. |
regex::replace(pattern, text, replacement) -> String |
Replace first match only. |
regex::replace_all(pattern, text, replacement) -> String |
Replace every match. |
regex::split(pattern, text) -> Array |
Split text on matches. |
regex::captures(pattern, text) -> Array | () |
[full, group1, group2, ...] from the first match; unmatched optional groups appear as (). |
Invalid patterns throw. Use \\ to escape inside Rhai string literals
("\\d+") or backtick strings to skip escaping (`\d+`).
if regex::is_match(`^/api/v\d+/`, ctx.request.path) {
let cap = regex::captures(`/api/v(\d+)/(.+)`, ctx.request.path);
let version = cap[1]; // "1"
let rest = cap[2]; // "users"
}
random:: — cryptographically-secure randomness
All randomness comes from OsRng. There is deliberately no "fast
non-crypto" variant — scripts shouldn't have to pick.
| Function | Description |
|---|---|
random::int(min, max) -> i64 |
Uniform integer in [min, max] (inclusive). Throws if min > max. |
random::float() -> f64 |
Uniform float in [0.0, 1.0). |
random::bytes(n) -> Blob |
n random bytes. n in 0..=65536. |
random::string(n) -> String |
n random alphanumeric chars (A-Za-z0-9). n in 0..=4096. |
random::uuid() -> String |
UUID v4 in canonical 8-4-4-4-12 form. |
let token = random::uuid();
let salt = random::bytes(16);
let pin = random::int(100000, 999999);
time:: — UTC time
Canonical time value is milliseconds since the Unix epoch as i64.
ISO 8601 / RFC 3339 strings are for I/O. UTC only — no timezone support.
| Function | Description |
|---|---|
time::now() -> String |
Current UTC time as ISO 8601 with ms (e.g. "2026-05-30T20:15:00.123Z"). |
time::now_ms() -> i64 |
Current ms since Unix epoch. |
time::parse(iso) -> i64 |
Parse RFC 3339 / ISO 8601 string to ms. Throws on bad input. |
time::format(ms) -> String |
Format ms-since-epoch as ISO 8601 with ms precision. |
time::add_seconds(ms, secs) -> i64 |
ms + secs*1000, with overflow check. |
time::diff_seconds(a_ms, b_ms) -> i64 |
(b_ms - a_ms) / 1000, truncated. |
let started_at = time::now_ms();
// ... do work ...
let elapsed = time::diff_seconds(started_at, time::now_ms());
let deadline = time::format(time::add_seconds(time::now_ms(), 3600));
json:: — JSON parse and stringify
| Function | Description |
|---|---|
json::parse(s) -> Dynamic |
Parse a JSON string. Returns Rhai maps, arrays, scalars, or () for null. Throws on invalid JSON. |
json::stringify(v) -> String |
Compact JSON. |
json::stringify_pretty(v) -> String |
Pretty-printed (2-space indent). |
let payload = json::parse(ctx.request.body); // if body came in as a string
let body_str = json::stringify(#{ ok: true, items: [1, 2, 3] });
Note: ctx.request.body is already parsed when the request body is
Content-Type: application/json — only call json::parse on raw
strings.
base64:: — standard and URL-safe Base64
Two alphabets: standard (with = padding) and URL-safe (no padding).
Encoders accept both String and Blob; decoders always return Blob.
| Function | Description |
|---|---|
base64::encode(input) -> String |
Standard alphabet, padded. input is String or Blob. |
base64::decode(s) -> Blob |
Decode standard alphabet. Throws on invalid. |
base64::encode_url(input) -> String |
URL-safe alphabet, no padding. |
base64::decode_url(s) -> Blob |
Decode URL-safe alphabet. Throws on invalid. |
let token = base64::encode_url(random::bytes(32)); // URL-safe session token
let raw = base64::decode("aGVsbG8=");
hex:: — hexadecimal
Encode produces lowercase. Decode accepts mixed case.
| Function | Description |
|---|---|
hex::encode(input) -> String |
Lowercase hex. input is String or Blob. |
hex::decode(s) -> Blob |
Decode hex (case-insensitive). Throws on invalid. |
let fingerprint = hex::encode(random::bytes(20));
url:: — percent-encoding
Unreserved set per RFC 3986 (A-Z, a-z, 0-9, -, _, ., ~)
is preserved; everything else is percent-encoded.
| Function | Description |
|---|---|
url::encode(s) -> String |
Percent-encode a component value. |
url::decode(s) -> String |
Percent-decode. Throws on invalid UTF-8 in the decoded output. |
url::encode_query(map) -> String |
Build k1=v1&k2=v2 from a Map. Both keys and values are percent-encoded. Non-string values are coerced via to_string(). |
url::encode_query emits keys in the Map's natural order, which is
alphabetical (Rhai's Map is a BTreeMap). RFC 3986 leaves query
parameter ordering unspecified, so this is fine for any conforming
consumer; if you need a specific ordering, build the string by hand.
let qs = url::encode_query(#{ q: "rust regex", page: 2 });
// → "page=2&q=rust%20regex"
What's not here
- Crypto (sha256/hmac/argon2/encryption) — deferred to a focused later PR.
- Timezones — UTC only in v1.1.0. Format with an offset upstream if you need local time.
- JWT, YAML, XML, CSV, Markdown — not planned for v1.1.x.
- Stateful services (KV, docs, HTTP, cron, files, pubsub, secrets, email, users, queue, invoke) — land per the v1.1.x roadmap in the blueprint §12.