To run a Koto script, instantiate koto::Koto
and call compile_and_run
:
use koto::prelude::*;
fn main() {
Koto::default()
.compile_and_run("print 'Hello, World!'")
.unwrap();
}
The result of calling compile_and_run
is a KValue
, which is Koto's main
value type.
KValue
is an enum that contains variants for each of the core Koto types,
like Number
, String
, etc.
The type of a KValue
as a string can be retrieved via KValue::type_as_string
,
and to render a KValue
, call Koto::value_to_string
.
use anyhow::{bail, Result};
use koto::prelude::*;
fn main() -> Result<()> {
let script = "1 + 2";
let mut koto = Koto::default();
match koto.compile_and_run(script)? {
KValue::Number(result) => {
println!("The result of '{script}' is {result}");
}
other => bail!(
"Expected a Number, found '{}': ({})",
other.type_as_string(),
koto.value_to_string(other)?
),
}
Ok(())
}
Values that are exported from the script are inserted in to the exports map,
which can be accessed by calling Koto::exports()
.
use koto::{prelude::*, Result};
fn main() -> Result<()> {
let script = "
export
exported_a: '42'.to_number()
exported_b: 'Hello from Koto'
";
let mut koto = Koto::default();
koto.compile_and_run(script)?;
let exports = koto.exports();
let exported_a = exports.get("exported_a").unwrap();
let exported_b = exports.get("exported_b").unwrap();
println!("exported_a: {}", koto.value_to_string(exported_a)?,);
println!("exported_b: {}", koto.value_to_string(exported_b)?,);
Ok(())
}
The runtime's prelude is a KMap
, which is Koto's standard hashmap type.
Values can be added to the prelude via KMap::insert
, taking any Rust value
that implements Into<KValue>
. Basic types like strings and numbers are
automatically converted to corresponding Koto types.
use koto::prelude::*;
fn main() {
let script = "
print 'name: {name}'
print 'how_many: {how_many}'
print 'yes_or_no: {if yes_or_no then 'yes' else 'no'}'
";
let mut koto = Koto::default();
let prelude = koto.prelude();
prelude.insert("name", "Alice");
prelude.insert("how_many", 99);
prelude.insert("yes_or_no", true);
koto.compile_and_run(script).unwrap();
}
The arguments that are accessible in a script from koto.args
can be set via
Koto::set_args
.
use koto::{prelude::*, Result};
fn main() -> Result<()> {
let script = "
from koto import args
if (size args) > 1
for i, arg in args.enumerate()
print '{i + 1}: {arg}'
else
print 'No arguments'
";
let mut koto = Koto::default();
koto.set_args(std::env::args())?;
koto.compile_and_run(script)?;
Ok(())
}
Any Rust function that implements KotoFunction
can be made available to the
Koto runtime.
use koto::{prelude::*, runtime};
fn main() {
let script = "
say_hello()
say_hello 'Alice'
print plus 10, 20
";
let mut koto = Koto::default();
let prelude = koto.prelude();
// Standalone functions can be inserted directly
prelude.insert("say_hello", say_hello);
// The add_fn helper avoids the need for type annotations
prelude.add_fn("plus", |ctx| match ctx.args() {
[KValue::Number(a), KValue::Number(b)] => Ok((a + b).into()),
unexpected => unexpected_args("|Number, Number|", unexpected),
});
koto.compile_and_run(script).unwrap();
}
fn say_hello(ctx: &mut CallContext) -> runtime::Result<KValue> {
match ctx.args() {
[] => println!("Hello?"),
[KValue::Str(name)] => println!("Hello, {name}"),
unexpected => return unexpected_args("||, or |String|", unexpected),
}
Ok(KValue::Null)
}
Koto::call_function
can be used to call Koto functions, or any other callable
Koto values.
use koto::{prelude::*, Result};
fn main() -> Result<()> {
let script = "
export my_fn = |a, b| '{a} + {b} is {a + b}'
";
let mut koto = Koto::default();
// Running the script exports the `my_fn` function
koto.compile_and_run(script).unwrap();
let my_fn = koto.exports().get("my_fn").unwrap();
assert!(my_fn.is_callable());
let result = koto.call_function(my_fn, &[1.into(), 2.into()])?;
println!("Result: {}", koto.value_to_string(result)?);
Ok(())
}
A module in Koto is simply a KMap
, conventionally with a defined
@type
.
use koto::prelude::*;
fn main() {
let script = "
from my_module import echo, square
print echo 'Hello'
print square 9
";
let mut koto = Koto::default();
koto.prelude().insert("my_module", make_module());
koto.compile_and_run(script).unwrap();
}
fn make_module() -> KMap {
// The `KMap::with_type` initializer sets up an empty map with a `@type` entry.
let module = KMap::with_type("my_module");
module.add_fn("echo", |ctx| match ctx.args() {
[KValue::Str(s)] => Ok(format!("{s}!").into()),
unexpected => unexpected_args("|String|", unexpected),
});
module.add_fn("square", |ctx| match ctx.args() {
[KValue::Number(n)] => Ok((n * n).into()),
unexpected => unexpected_args("|Number|", unexpected),
});
module
}
Any Rust type that implements KotoObject
can be used in the Koto runtime.
KotoObject
requires KotoType
, KotoCopy
, and KotoEntries
to be
implemented.
use koto::{derive::*, prelude::*, runtime};
fn main() {
let script = "
my_type = make_my_type 41
print my_type.get()
print my_type.set 99
";
let mut koto = Koto::default();
koto.prelude()
.add_fn("make_my_type", |ctx| match ctx.args() {
[KValue::Number(n)] => Ok(MyType::make_koto_object(*n).into()),
unexpected => unexpected_args("|Number|", unexpected),
});
koto.compile_and_run(script).unwrap();
}
// MyType is a type that we want to use in Koto
//
// The KotoCopy and KotoType traits are automatically derived.
#[derive(Clone, Copy, KotoCopy, KotoType)]
struct MyType(i64);
// The KotoEntries trait is implemented by the koto_impl macro,
// generating Koto functions for any impl function tagged with #[koto_method],
// and inserting them into a cached KMap.
#[koto_impl]
impl MyType {
fn make_koto_object(n: KNumber) -> KObject {
// From is available for any type that implements KotoObject
let my_type = Self(n.into());
KObject::from(my_type)
}
// A simple getter function
#[koto_method]
fn get(&self) -> runtime::Result<KValue> {
Ok(self.0.into())
}
// A function that returns the object instance as the result
#[koto_method]
fn set(ctx: MethodContext<Self>) -> runtime::Result<KValue> {
match ctx.args {
[KValue::Number(n)] => {
ctx.instance_mut()?.0 = n.into();
ctx.instance_result()
}
unexpected => unexpected_args("|Number|", unexpected),
}
}
}
impl KotoObject for MyType {
// KotoObject::Display allows mytype to be used with Koto's print function
fn display(&self, ctx: &mut DisplayContext) -> runtime::Result<()> {
ctx.append(format!("MyType({})", self.0));
Ok(())
}
}
Runtime type checks are enabled by default, the compiler can be prevented from
emitting type check instructions by disabling the enable_type_checks
flag.
use koto::{prelude::*, Result};
fn main() -> Result<()> {
let script = "
let x: String = 123
";
let mut koto = Koto::default();
// Type checks are enabled by default. Running the script will produce an error.
let result = koto.compile_and_run(script);
assert!(result.is_err());
// Type checks can disabled via `CompileArgs::enable_type_checks`.
// It should go without saying that checks should only be disabled if you're confident that your
// code is correct!
let result = koto.compile_and_run(CompileArgs::new(script).enable_type_checks(false));
assert!(result.is_ok());
Ok(())
}
By default, Koto's runtime is single-threaded, and many of its core types (e.g. KValue
) don't
implement Send
or Sync
.
For applications that need to support multi-threaded scripting, the arc
feature switches from an
Rc<RefCell<T>>
-based memory strategy to one using Arc<RwLock<T>>
.
Only one memory strategy can be enabled at a time, so default features need to be disabled.
# Cargo.toml
# ...
[dependencies.koto]
version = "0.15"
default-feautures = false
features = ["arc"]
Some applications (like REPLs) require assigned variables to persist between each script evaluation.
This can be achieved by enabling the export_top_level_ids
flag,
which will result in all top-level assignments being exported.
use koto::{prelude::*, Result};
fn main() -> Result<()> {
let mut koto = Koto::default();
// When using Koto in a REPL, variables from previous evaluations need to be made available
// for the next evaluation.
//
// This is achieved by telling the compiler to treat each top-level assignment as if it had been
// exported. i.e., the compiler turns the script `x = 1` into `export x = 1`.
koto.compile_and_run(CompileArgs::new("x = 1").export_top_level_ids(true))?;
assert!(koto.exports().get("x").is_some());
// The exports map gets reused by the Koto instance for each run.
match koto.compile_and_run(CompileArgs::new("x + x").export_top_level_ids(true))? {
KValue::Number(result) => assert_eq!(result, KNumber::from(2)),
unexpected => unexpected_type("Number", &unexpected)?,
}
Ok(())
}