web.dart 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. // ignore_for_file: prefer_function_declarations_over_variables, non_constant_identifier_names, no_leading_underscores_for_local_identifiers
  2. import 'dart:convert' as cvt;
  3. import 'package:js/js.dart';
  4. import 'common.dart';
  5. @JS()
  6. @anonymous
  7. class Opt {
  8. external bool get async;
  9. external factory Opt({bool async});
  10. }
  11. @JS("Module.ccall")
  12. external dynamic ccall(String name, String? returnType, List<String> argTypes, List<dynamic> args, Opt opt);
  13. class _Bindings
  14. {
  15. static final pkpy_delete = (dynamic p) => ccall("pkpy_delete", null, ["number"], [p], Opt(async: false));
  16. static final pkpy_new_repl = (dynamic vm) => ccall("pkpy_new_repl", "number", ["number"], [vm], Opt(async: false));
  17. static final pkpy_repl_input = (dynamic r, String line) => ccall("pkpy_repl_input", null, ["number", "string"], [r, line], Opt(async: true));
  18. static final pkpy_repl_last_input_result = (dynamic r) => ccall("pkpy_repl_last_input_result", "number", ["number"], [r], Opt(async: false));
  19. static final pkpy_new_tvm = (bool use_stdio) => ccall("pkpy_new_tvm", "number", ["boolean"], [use_stdio], Opt(async: false));
  20. static final pkpy_tvm_exec_async = (dynamic vm, String source) => ccall("pkpy_tvm_exec_async", null, ["number", "string"], [vm, source], Opt(async: true));
  21. static final pkpy_tvm_get_state = (dynamic vm) => ccall("pkpy_tvm_get_state", "number", ["number"], [vm], Opt(async: false));
  22. static final pkpy_tvm_read_jsonrpc_request = (dynamic vm) => ccall("pkpy_tvm_read_jsonrpc_request", "string", ["number"], [vm], Opt(async: false));
  23. static final pkpy_tvm_reset_state = (dynamic vm) => ccall("pkpy_tvm_reset_state", null, ["number"], [vm], Opt(async: false));
  24. static final pkpy_tvm_terminate = (dynamic vm) => ccall("pkpy_tvm_terminate", null, ["number"], [vm], Opt(async: false));
  25. static final pkpy_tvm_write_jsonrpc_response = (dynamic vm, String value) => ccall("pkpy_tvm_write_jsonrpc_response", null, ["number", "string"], [vm, value], Opt(async: false));
  26. static final pkpy_new_vm = (bool use_stdio) => ccall("pkpy_new_vm", "number", ["boolean"], [use_stdio], Opt(async: false));
  27. static final pkpy_vm_add_module = (dynamic vm, String name, String source) => ccall("pkpy_vm_add_module", null, ["number", "string", "string"], [vm, name, source], Opt(async: false));
  28. static final pkpy_vm_eval = (dynamic vm, String source) => ccall("pkpy_vm_eval", "string", ["number", "string"], [vm, source], Opt(async: false));
  29. static final pkpy_vm_exec = (dynamic vm, String source) => ccall("pkpy_vm_exec", null, ["number", "string"], [vm, source], Opt(async: false));
  30. static final pkpy_vm_get_global = (dynamic vm, String name) => ccall("pkpy_vm_get_global", "string", ["number", "string"], [vm, name], Opt(async: false));
  31. static final pkpy_vm_read_output = (dynamic vm) => ccall("pkpy_vm_read_output", "string", ["number"], [vm], Opt(async: false));
  32. }
  33. class VM {
  34. late final dynamic pointer;
  35. VM() {
  36. if (this is ThreadedVM) {
  37. pointer = _Bindings.pkpy_new_tvm(false);
  38. } else {
  39. pointer = _Bindings.pkpy_new_vm(false);
  40. }
  41. }
  42. void dispose() {
  43. _Bindings.pkpy_delete(pointer);
  44. }
  45. PyOutput read_output() {
  46. var _o = _Bindings.pkpy_vm_read_output(pointer);
  47. String _j = _o;
  48. var ret = PyOutput.fromJson(cvt.jsonDecode(_j));
  49. return ret;
  50. }
  51. /// Add a source module into a virtual machine.
  52. void add_module(String name, String source)
  53. {
  54. _Bindings.pkpy_vm_add_module(pointer, name, source);
  55. }
  56. /// Evaluate an expression. Return a json representing the result. If there is any error, return `nullptr`.
  57. String? eval(String source)
  58. {
  59. var ret = _Bindings.pkpy_vm_eval(pointer, source);
  60. return ret;
  61. }
  62. /// Run a given source on a virtual machine.
  63. void exec(String source)
  64. {
  65. _Bindings.pkpy_vm_exec(pointer, source);
  66. }
  67. /// Get a global variable of a virtual machine. Return a json representing the result. If the variable is not found, return `nullptr`.
  68. String? get_global(String name)
  69. {
  70. var ret = _Bindings.pkpy_vm_get_global(pointer, name);
  71. return ret;
  72. }
  73. }
  74. enum ThreadState { ready, running, suspended, finished }
  75. class ThreadedVM extends VM {
  76. ThreadState get state => ThreadState.values[_Bindings.pkpy_tvm_get_state(pointer)];
  77. @override
  78. void dispose() {
  79. terminate();
  80. Future.delayed(Duration(milliseconds: 150)).then((_) => _Bindings.pkpy_delete(pointer));
  81. }
  82. /// Run a given source on a threaded virtual machine. The excution will be started in a new thread.
  83. void exec_async(String source)
  84. {
  85. _Bindings.pkpy_tvm_exec_async(pointer, source);
  86. }
  87. /// Read the current JSONRPC request from shared string buffer.
  88. String? read_jsonrpc_request()
  89. {
  90. var ret = _Bindings.pkpy_tvm_read_jsonrpc_request(pointer);
  91. return ret;
  92. }
  93. /// Set the state of a threaded virtual machine to `THREAD_READY`. The current state should be `THREAD_FINISHED`.
  94. void reset_state()
  95. {
  96. _Bindings.pkpy_tvm_reset_state(pointer);
  97. }
  98. /// Emit a KeyboardInterrupt signal to stop a running threaded virtual machine.
  99. void terminate()
  100. {
  101. _Bindings.pkpy_tvm_terminate(pointer);
  102. }
  103. /// Write a JSONRPC response to shared string buffer.
  104. void write_jsonrpc_response(String value)
  105. {
  106. _Bindings.pkpy_tvm_write_jsonrpc_response(pointer, value);
  107. }
  108. }
  109. class REPL {
  110. late final dynamic pointer;
  111. REPL(VM vm) {
  112. pointer = _Bindings.pkpy_new_repl(vm.pointer);
  113. }
  114. void dispose() {
  115. _Bindings.pkpy_delete(pointer);
  116. }
  117. /// Input a source line to an interactive console.
  118. void input(String line)
  119. {
  120. _Bindings.pkpy_repl_input(pointer, line);
  121. }
  122. /// Check if the REPL needs more lines.
  123. int last_input_result()
  124. {
  125. var ret = _Bindings.pkpy_repl_last_input_result(pointer);
  126. return ret;
  127. }
  128. }