dart_monty
Monty is a restricted, sandboxed Python interpreter built in Rust by the Pydantic team. It runs a safe subset of Python designed for embedding.
This package is co-designed by human and AI — nearly all code is AI-generated.
dart_monty provides pure Dart bindings for the Monty interpreter, bringing sandboxed Python execution to Dart and Flutter apps — on desktop, web, and mobile — with resource limits, iterative execution, and snapshot/restore support.
Fork notice: dart_monty currently builds against
runyaga/monty(branchrunyaga/main), a fork ofpydantic/monty. The fork carries patches required for embedding that are not yet upstream. We intend to upstream all changes and return topydantic/montyonce accepted.
Patch Fork PR Upstream PR Status Why needed Fix partial future resolution panics in mixed asyncio.gather()— pydantic/monty#251 Submitted, awaiting review Two panics in async_exec.rswhen a gather mixes coroutine tasks with direct external calls — blocks any async host function useCancellableTracker (preemptive script cancellation) runyaga/monty#3 Not yet submitted Merged in fork Cooperative cancel via Arc<AtomicBool>checked in bytecode loop — required fordispose()to not hang on stuck FFI callscpu: wasm32restriction inmonty-wasm32-wasinpm package— runyaga/monty#4 Open issue npm refuses install on non-wasm hosts, blocking CI and local dev
Platform Support
| Platform | Status |
|---|---|
| macOS | Supported |
| Linux | Supported |
| Web (browser) | Supported |
| Windows | Planned |
| iOS | Planned |
| Android | Planned |
Installation
flutter pub add dart_monty
Usage
import 'dart:convert';
import 'package:http/http.dart' as http;
import 'package:dart_monty/dart_monty.dart';
final monty = MontyPlatform.instance;
// Simple execution
final result = await monty.run('2 + 2');
print(result.value); // 4
// With resource limits
final limited = await monty.run(
'fib(30)',
limits: MontyLimits(timeoutMs: 5000, memoryBytes: 10 * 1024 * 1024),
);
External Functions
When Python calls a function listed in externalFunctions, execution
pauses and Dart handles the call. The function name in Python maps 1:1
to the name you provide — when Python calls fetch(...), Dart receives
a MontyPending with functionName == 'fetch' and the arguments Python
passed.
// Python calls fetch() → execution pauses → Dart handles it → resumes
var progress = await monty.start(
'fetch("https://api.example.com/users")',
externalFunctions: ['fetch'],
);
// Dispatch loop: match functionName to your Dart implementation
while (progress is MontyPending) {
final pending = progress as MontyPending;
final name = pending.functionName; // 'fetch'
final args = pending.arguments; // ['https://api.example.com/users']
switch (name) {
case 'fetch':
final url = args.first as String;
final response = await http.get(Uri.parse(url));
progress = await monty.resume(jsonDecode(response.body));
default:
progress = await monty.resumeWithError(
'Unknown function: $name',
);
}
}
final complete = progress as MontyComplete;
print(complete.result.value);
await monty.dispose();
Error Handling
dart_monty uses a sealed MontyError hierarchy for structured error handling:
import 'package:dart_monty_platform_interface/dart_monty_platform_interface.dart';
try {
final result = await monty.run('1 / 0');
} on MontyError catch (e) {
switch (e) {
case MontyScriptError(:final exception):
print('Python error: ${exception.message}');
print('Type: ${exception.excType}');
for (final frame in exception.traceback) {
print(' ${frame.filename}:${frame.lineNumber} in ${frame.name}');
}
case MontyCancelledError():
print('Execution was cancelled');
case MontyResourceError(:final exception):
print('Resource limit exceeded: ${exception.message}');
case MontyPanicError(:final message):
print('Interpreter panic: $message');
case MontyCrashError(:final message):
print('Interpreter crash: $message');
case MontyDisposedError():
print('Interpreter was disposed');
}
}
Cancellation
Cancel a running interpreter from any isolate using MontyCancelToken:
final token = MontyCancelToken(handleId);
token.cancel(); // sets atomic flag in bytecode loop
Stateful Sessions
MontySession persists Python globals across multiple run() calls using
snapshot/restore under the hood:
import 'package:dart_monty_platform_interface/dart_monty_platform_interface.dart';
final session = MontySession(platform: MontyPlatform.instance);
// Globals persist across run() calls via snapshot/restore
await session.run('x = 42');
await session.run('y = x * 2');
final result = await session.run('x + y');
print(result.value); // 126
// Session also supports start/resume (same dispatch pattern)
await session.clearState();
await session.dispose();
Monty API Coverage (~75%)
dart_monty wraps the Monty Rust API (fork of pydantic/monty). The table below shows current coverage and what's planned.
| API Area | Status | Notes |
|---|---|---|
Core execution (run, start, resume, dispose) |
Covered | Full iterative execution loop |
| External functions (host-provided callables) | Covered | start() / resume() / resumeWithError() |
| Resource limits (time, memory, recursion depth) | Covered | MontyLimits on run() and start() |
Print capture (print() output collection) |
Covered | MontyResult.printOutput |
Snapshot / restore (MontyRun::dump/load) |
Covered | Compile-once, run-many pattern |
| Exception model (excType, traceback, stack frames) | Covered | Full MontyException with StackFrame list |
| Call metadata (kwargs, callId, methodCall, scriptName) | Covered | Structured external call context |
| Cancellation (cooperative abort via atomic flag) | Covered | MontyCancelToken, cancel(), terminate() with zombie tracking |
Error hierarchy (sealed MontyError with 6 subtypes) |
Covered | Script, Cancel, Panic, Crash, Disposed, Resource |
| Multi-session (WASM Worker pool) | Covered | createSession/disposeSession, 16 MB per session |
Async / futures (asyncio.gather, concurrent calls) |
Covered | Native only — WASM upstream lacks FutureSnapshot API |
| Rich types (tuple, set, bytes, dataclass, namedtuple) | Planned | Currently collapsed to List/Map |
REPL (stateful sessions, feed(), persistence) |
Planned | MontyRepl multi-step sessions |
OS calls (os.getenv, os.environ, os.stat) |
Planned | OsCall progress variant |
| Print streaming (real-time callback) | Planned | Currently batch-only after execution |
Advanced limits (allocations, GC interval, runNoLimits) |
Planned | Extended ResourceTracker surface |
| Type checking (static analysis before execution) | Planned | ty / Red Knot integration |
| Progress serialization (suspend/resume across restarts) | Planned | RunProgress::dump/load |
| Platform expansion (Windows, iOS, Android) | Planned | macOS + Linux + Web today |
Architecture
See docs/architecture.md for detailed architecture documentation including state machine contracts, memory management, error handling, and cross-backend parity guarantees.
Federated plugin with six packages:
| Package | Type | Description |
|---|---|---|
dart_monty |
Flutter plugin | App-facing API |
dart_monty_platform_interface |
Pure Dart | Abstract contract — no Flutter dependency |
dart_monty_ffi |
Pure Dart | Native FFI bindings (dart:ffi -> Rust) |
dart_monty_wasm |
Pure Dart | WASM bindings (dart:js_interop -> Web Worker) |
dart_monty_native |
Flutter plugin | Native platform registration shim (delegates to dart_monty_ffi) |
dart_monty_web |
Flutter plugin | Web platform (browser, script injection) |
The three pure-Dart packages can be used without Flutter (e.g. in CLI tools or server-side Dart).
Native Path (desktop)
Dart app -> MontyNative (Isolate)
-> MontyFfi (dart:ffi)
-> libdart_monty_native.{dylib,so}
-> Monty Rust interpreter
Web Path (browser)
Dart app (compiled to JS) -> DartMontyWeb
-> MontyWasm (dart:js_interop)
-> Web Worker -> @pydantic/monty WASM
The Web Worker architecture bypasses Chrome's 8 MB synchronous WASM compilation limit.
Web Setup
The web backend requires COOP/COEP HTTP headers for SharedArrayBuffer support:
Cross-Origin-Opener-Policy: same-origin
Cross-Origin-Embedder-Policy: require-corp
Contributing
See CONTRIBUTING.md for development setup, gate scripts, and CI details.
License
MIT License. See LICENSE.
Libraries
- dart_monty
- Flutter plugin exposing the Monty sandboxed Python interpreter.