// ignore_for_file: non_constant_identifier_names, prefer_typing_uninitialized_variables, constant_identifier_names, no_leading_underscores_for_local_identifiers import 'dart:convert' as cvt; import 'dart:ffi' as ffi; import 'dart:io'; import 'package:ffi/ffi.dart'; export 'jsonrpc.dart'; class Bindings { static ffi.DynamicLibrary _load() { String _libName = "pocketpy"; if (Platform.isIOS) { return ffi.DynamicLibrary.process(); } if (Platform.isAndroid || Platform.isLinux) { return ffi.DynamicLibrary.open('lib$_libName.so'); } if (Platform.isWindows) { return ffi.DynamicLibrary.open('$_libName.dll'); } throw UnsupportedError('Unknown platform: ${Platform.operatingSystem}'); } static final _lib = _load(); static final pkpy_delete = _lib.lookupFunction("pkpy_delete"); static final pkpy_new_repl = _lib.lookupFunction("pkpy_new_repl"); static final pkpy_repl_input = _lib.lookupFunction line), int Function(ffi.Pointer r, ffi.Pointer line)>("pkpy_repl_input"); static final pkpy_new_tvm = _lib.lookupFunction("pkpy_new_tvm"); static final pkpy_tvm_exec_async = _lib.lookupFunction source), bool Function(ffi.Pointer vm, ffi.Pointer source)>("pkpy_tvm_exec_async"); static final pkpy_tvm_get_state = _lib.lookupFunction("pkpy_tvm_get_state"); static final pkpy_tvm_read_jsonrpc_request = _lib.lookupFunction Function(ffi.Pointer vm), ffi.Pointer Function(ffi.Pointer vm)>("pkpy_tvm_read_jsonrpc_request"); static final pkpy_tvm_reset_state = _lib.lookupFunction("pkpy_tvm_reset_state"); static final pkpy_tvm_terminate = _lib.lookupFunction("pkpy_tvm_terminate"); static final pkpy_tvm_write_jsonrpc_response = _lib.lookupFunction value), void Function(ffi.Pointer vm, ffi.Pointer value)>("pkpy_tvm_write_jsonrpc_response"); static final pkpy_new_vm = _lib.lookupFunction("pkpy_new_vm"); static final pkpy_vm_add_module = _lib.lookupFunction name, ffi.Pointer source), bool Function(ffi.Pointer vm, ffi.Pointer name, ffi.Pointer source)>("pkpy_vm_add_module"); static final pkpy_vm_eval = _lib.lookupFunction Function(ffi.Pointer vm, ffi.Pointer source), ffi.Pointer Function(ffi.Pointer vm, ffi.Pointer source)>("pkpy_vm_eval"); static final pkpy_vm_exec = _lib.lookupFunction source), bool Function(ffi.Pointer vm, ffi.Pointer source)>("pkpy_vm_exec"); static final pkpy_vm_get_global = _lib.lookupFunction Function(ffi.Pointer vm, ffi.Pointer name), ffi.Pointer Function(ffi.Pointer vm, ffi.Pointer name)>("pkpy_vm_get_global"); static final pkpy_vm_read_output = _lib.lookupFunction Function(ffi.Pointer vm), ffi.Pointer Function(ffi.Pointer vm)>("pkpy_vm_read_output"); } class PyOutput { final String stdout; final String stderr; PyOutput(this.stdout, this.stderr); PyOutput.fromJson(Map json) : stdout = json['stdout'], stderr = json['stderr']; } class Str { static final Finalizer> finalizer = Finalizer((p) => calloc.free(p)); late final ffi.Pointer _p; Str(String s) { _p = s.toNativeUtf8(); finalizer.attach(this, _p); } ffi.Pointer get p => _p; } class VM { late final ffi.Pointer pointer; VM() { if (this is ThreadedVM) { pointer = Bindings.pkpy_new_tvm(false); } else { pointer = Bindings.pkpy_new_vm(false); } } void dispose() { Bindings.pkpy_delete(pointer); } PyOutput read_output() { var _o = Bindings.pkpy_vm_read_output(pointer); String _j = _o.toDartString(); var ret = PyOutput.fromJson(cvt.jsonDecode(_j)); Bindings.pkpy_delete(_o); return ret; } /// Add a source module into a virtual machine. Return `true` if there is no complie error. bool add_module(String name, String source) { var ret = Bindings.pkpy_vm_add_module(pointer, Str(name).p, Str(source).p); return ret; } /// Evaluate an expression. Return a json representing the result. If there is any error, return `nullptr`. String? eval(String source) { var ret = Bindings.pkpy_vm_eval(pointer, Str(source).p); if (ret == ffi.nullptr) return null; String s = ret.toDartString(); calloc.free(ret); return s; } /// Run a given source on a virtual machine. Return `true` if there is no compile error. bool exec(String source) { var ret = Bindings.pkpy_vm_exec(pointer, Str(source).p); return ret; } /// Get a global variable of a virtual machine. Return a json representing the result. If the variable is not found, return `nullptr`. String? get_global(String name) { var ret = Bindings.pkpy_vm_get_global(pointer, Str(name).p); if (ret == ffi.nullptr) return null; String s = ret.toDartString(); calloc.free(ret); return s; } } enum ThreadState { ready, running, suspended, finished } class ThreadedVM extends VM { ThreadState get state => ThreadState.values[Bindings.pkpy_tvm_get_state(pointer)]; /// Run a given source on a threaded virtual machine. The excution will be started in a new thread. Return `true` if there is no compile error. bool exec_async(String source) { var ret = Bindings.pkpy_tvm_exec_async(pointer, Str(source).p); return ret; } /// Read the current JSONRPC request from shared string buffer. String? read_jsonrpc_request() { var ret = Bindings.pkpy_tvm_read_jsonrpc_request(pointer); if (ret == ffi.nullptr) return null; String s = ret.toDartString(); calloc.free(ret); return s; } /// Set the state of a threaded virtual machine to `THREAD_READY`. The current state should be `THREAD_FINISHED`. void reset_state() { Bindings.pkpy_tvm_reset_state(pointer); } /// Emit a KeyboardInterrupt signal to stop a running threaded virtual machine. void terminate() { Bindings.pkpy_tvm_terminate(pointer); } /// Write a JSONRPC response to shared string buffer. void write_jsonrpc_response(String value) { Bindings.pkpy_tvm_write_jsonrpc_response(pointer, Str(value).p); } } class REPL { late final ffi.Pointer pointer; REPL(VM vm) { pointer = Bindings.pkpy_new_repl(vm.pointer); } void dispose() { Bindings.pkpy_delete(pointer); } /// Input a source line to an interactive console. Return `0` if need more lines, `1` if execution happened, `2` if execution skipped (compile error or empty input). int input(String line) { var ret = Bindings.pkpy_repl_input(pointer, Str(line).p); return ret; } }