2022-02-07 02:18:42 +00:00
|
|
|
import 'dart:async';
|
2022-02-09 14:47:36 +00:00
|
|
|
import 'dart:ffi';
|
2022-02-07 02:18:42 +00:00
|
|
|
import 'dart:io';
|
2022-02-09 14:47:36 +00:00
|
|
|
import 'dart:isolate';
|
|
|
|
import 'dart:convert';
|
2022-10-01 02:37:55 +00:00
|
|
|
import 'dart:typed_data';
|
2022-02-07 02:18:42 +00:00
|
|
|
|
|
|
|
import 'package:ffi/ffi.dart';
|
2022-02-09 14:47:36 +00:00
|
|
|
|
|
|
|
import 'veilid.dart';
|
2022-02-07 02:18:42 +00:00
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
// Load the veilid_flutter library once
|
|
|
|
const _base = 'veilid_flutter';
|
|
|
|
final _path = Platform.isWindows
|
|
|
|
? '$_base.dll'
|
|
|
|
: Platform.isMacOS
|
|
|
|
? 'lib$_base.dylib'
|
|
|
|
: 'lib$_base.so';
|
2022-11-16 17:49:53 +00:00
|
|
|
final _dylib =
|
2022-02-09 14:47:36 +00:00
|
|
|
Platform.isIOS ? DynamicLibrary.process() : DynamicLibrary.open(_path);
|
2022-02-07 02:18:42 +00:00
|
|
|
|
|
|
|
// Linkage for initialization
|
2022-02-14 02:09:43 +00:00
|
|
|
typedef _DartPostCObject
|
2022-02-09 14:47:36 +00:00
|
|
|
= NativeFunction<Int8 Function(Int64, Pointer<Dart_CObject>)>;
|
2022-02-07 02:18:42 +00:00
|
|
|
// fn free_string(s: *mut std::os::raw::c_char)
|
2022-02-14 02:09:43 +00:00
|
|
|
typedef _FreeStringC = Void Function(Pointer<Utf8>);
|
|
|
|
typedef _FreeStringDart = void Function(Pointer<Utf8>);
|
2022-02-07 02:18:42 +00:00
|
|
|
// fn initialize_veilid_flutter(dart_post_c_object_ptr: ffi::DartPostCObjectFnType)
|
2022-02-14 02:09:43 +00:00
|
|
|
typedef _InitializeVeilidFlutterC = Void Function(Pointer<_DartPostCObject>);
|
|
|
|
typedef _InitializeVeilidFlutterDart = void Function(Pointer<_DartPostCObject>);
|
2022-07-01 20:20:43 +00:00
|
|
|
// fn initialize_veilid_core(platform_config: FfiStr)
|
|
|
|
typedef _InitializeVeilidCoreC = Void Function(Pointer<Utf8>);
|
|
|
|
typedef _InitializeVeilidCoreDart = void Function(Pointer<Utf8>);
|
2022-07-01 16:13:52 +00:00
|
|
|
// fn change_log_level(layer: FfiStr, log_level: FfiStr)
|
|
|
|
typedef _ChangeLogLevelC = Void Function(Pointer<Utf8>, Pointer<Utf8>);
|
|
|
|
typedef _ChangeLogLevelDart = void Function(Pointer<Utf8>, Pointer<Utf8>);
|
2022-02-07 02:18:42 +00:00
|
|
|
// fn startup_veilid_core(port: i64, config: FfiStr)
|
2022-09-09 20:27:13 +00:00
|
|
|
typedef _StartupVeilidCoreC = Void Function(Int64, Int64, Pointer<Utf8>);
|
|
|
|
typedef _StartupVeilidCoreDart = void Function(int, int, Pointer<Utf8>);
|
2022-02-07 02:18:42 +00:00
|
|
|
// fn get_veilid_state(port: i64)
|
2022-02-14 02:09:43 +00:00
|
|
|
typedef _GetVeilidStateC = Void Function(Int64);
|
|
|
|
typedef _GetVeilidStateDart = void Function(int);
|
2022-09-06 22:59:41 +00:00
|
|
|
// fn attach(port: i64)
|
|
|
|
typedef _AttachC = Void Function(Int64);
|
|
|
|
typedef _AttachDart = void Function(int);
|
|
|
|
// fn detach(port: i64)
|
|
|
|
typedef _DetachC = Void Function(Int64);
|
|
|
|
typedef _DetachDart = void Function(int);
|
2022-11-26 19:16:02 +00:00
|
|
|
|
|
|
|
// fn routing_context(port: i64)
|
|
|
|
typedef _RoutingContextC = Void Function(Int64);
|
|
|
|
typedef _RoutingContextDart = void Function(int);
|
|
|
|
// fn release_routing_context(id: u32)
|
|
|
|
typedef _ReleaseRoutingContextC = Int32 Function(Uint32);
|
|
|
|
typedef _ReleaseRoutingContextDart = int Function(int);
|
|
|
|
// fn routing_context_with_privacy(id: u32) -> u32
|
|
|
|
typedef _RoutingContextWithPrivacyC = Uint32 Function(Uint32);
|
|
|
|
typedef _RoutingContextWithPrivacyDart = int Function(int);
|
|
|
|
// fn routing_context_with_custom_privacy(id: u32, stability: FfiStr)
|
|
|
|
typedef _RoutingContextWithCustomPrivacyC = Uint32 Function(
|
|
|
|
Uint32, Pointer<Utf8>);
|
|
|
|
typedef _RoutingContextWithCustomPrivacyDart = int Function(int, Pointer<Utf8>);
|
|
|
|
// fn routing_context_with_sequencing(id: u32, sequencing: FfiStr)
|
|
|
|
typedef _RoutingContextWithSequencingC = Uint32 Function(Uint32, Pointer<Utf8>);
|
|
|
|
typedef _RoutingContextWithSequencingDart = int Function(int, Pointer<Utf8>);
|
|
|
|
// fn routing_context_app_call(port: i64, id: u32, target: FfiStr, request: FfiStr)
|
|
|
|
typedef _RoutingContextAppCallC = Void Function(
|
|
|
|
Int64, Uint32, Pointer<Utf8>, Pointer<Utf8>);
|
|
|
|
typedef _RoutingContextAppCallDart = void Function(
|
|
|
|
int, int, Pointer<Utf8>, Pointer<Utf8>);
|
|
|
|
// fn routing_context_app_message(port: i64, id: u32, target: FfiStr, request: FfiStr)
|
|
|
|
typedef _RoutingContextAppMessageC = Void Function(
|
|
|
|
Int64, Uint32, Pointer<Utf8>, Pointer<Utf8>);
|
|
|
|
typedef _RoutingContextAppMessageDart = void Function(
|
|
|
|
int, int, Pointer<Utf8>, Pointer<Utf8>);
|
|
|
|
|
|
|
|
// fn new_private_route(port: i64)
|
|
|
|
typedef _NewPrivateRouteC = Void Function(Int64);
|
|
|
|
typedef _NewPrivateRouteDart = void Function(int);
|
|
|
|
// fn new_custom_private_route(port: i64, stability: FfiStr, sequencing: FfiStr)
|
|
|
|
typedef _NewCustomPrivateRouteC = Void Function(
|
|
|
|
Int64, Pointer<Utf8>, Pointer<Utf8>);
|
|
|
|
typedef _NewCustomPrivateRouteDart = void Function(
|
|
|
|
int, Pointer<Utf8>, Pointer<Utf8>);
|
|
|
|
// fn import_remote_private_route(port: i64, blob: FfiStr)
|
|
|
|
typedef _ImportRemotePrivateRouteC = Void Function(Int64, Pointer<Utf8>);
|
|
|
|
typedef _ImportRemotePrivateRouteDart = void Function(int, Pointer<Utf8>);
|
|
|
|
// fn release_private_route(port:i64, key: FfiStr)
|
|
|
|
typedef _ReleasePrivateRouteC = Void Function(Int64, Pointer<Utf8>);
|
|
|
|
typedef _ReleasePrivateRouteDart = void Function(int, Pointer<Utf8>);
|
|
|
|
|
2022-10-01 02:37:55 +00:00
|
|
|
// fn app_call_reply(port: i64, id: FfiStr, message: FfiStr)
|
|
|
|
typedef _AppCallReplyC = Void Function(Int64, Pointer<Utf8>, Pointer<Utf8>);
|
|
|
|
typedef _AppCallReplyDart = void Function(int, Pointer<Utf8>, Pointer<Utf8>);
|
|
|
|
|
2022-11-26 19:16:02 +00:00
|
|
|
// fn debug(port: i64, log_level: FfiStr)
|
|
|
|
typedef _DebugC = Void Function(Int64, Pointer<Utf8>);
|
|
|
|
typedef _DebugDart = void Function(int, Pointer<Utf8>);
|
2022-02-07 02:18:42 +00:00
|
|
|
// fn shutdown_veilid_core(port: i64)
|
2022-02-14 02:09:43 +00:00
|
|
|
typedef _ShutdownVeilidCoreC = Void Function(Int64);
|
|
|
|
typedef _ShutdownVeilidCoreDart = void Function(int);
|
2022-02-07 02:18:42 +00:00
|
|
|
// fn veilid_version_string() -> *mut c_char
|
2022-02-14 02:09:43 +00:00
|
|
|
typedef _VeilidVersionStringC = Pointer<Utf8> Function();
|
|
|
|
typedef _VeilidVersionStringDart = Pointer<Utf8> Function();
|
2022-02-09 14:47:36 +00:00
|
|
|
|
2022-02-07 02:18:42 +00:00
|
|
|
// fn veilid_version() -> VeilidVersion
|
2022-02-09 14:47:36 +00:00
|
|
|
class VeilidVersionFFI extends Struct {
|
2022-02-07 02:18:42 +00:00
|
|
|
@Uint32()
|
|
|
|
external int major;
|
|
|
|
@Uint32()
|
|
|
|
external int minor;
|
|
|
|
@Uint32()
|
|
|
|
external int patch;
|
|
|
|
}
|
2022-02-09 14:47:36 +00:00
|
|
|
|
2022-02-14 02:09:43 +00:00
|
|
|
typedef _VeilidVersionC = VeilidVersionFFI Function();
|
|
|
|
typedef _VeilidVersionDart = VeilidVersionFFI Function();
|
2022-02-09 14:47:36 +00:00
|
|
|
|
|
|
|
// Async message types
|
2022-02-14 02:09:43 +00:00
|
|
|
const int messageOk = 0;
|
|
|
|
const int messageErr = 1;
|
|
|
|
const int messageOkJson = 2;
|
|
|
|
const int messageErrJson = 3;
|
|
|
|
const int messageStreamItem = 4;
|
|
|
|
const int messageStreamItemJson = 5;
|
|
|
|
const int messageStreamAbort = 6;
|
|
|
|
const int messageStreamAbortJson = 7;
|
|
|
|
const int messageStreamClose = 8;
|
2022-02-07 02:18:42 +00:00
|
|
|
|
|
|
|
// Interface factory for high level Veilid API
|
|
|
|
Veilid getVeilid() => VeilidFFI(_dylib);
|
|
|
|
|
2022-02-09 14:47:36 +00:00
|
|
|
// Parse handle async returns
|
2022-03-04 01:45:39 +00:00
|
|
|
Future<T> processFuturePlain<T>(Future<dynamic> future) {
|
2022-02-09 14:47:36 +00:00
|
|
|
return future.then((value) {
|
|
|
|
final list = value as List<dynamic>;
|
|
|
|
switch (list[0] as int) {
|
2022-02-14 02:09:43 +00:00
|
|
|
case messageOk:
|
2022-02-09 14:47:36 +00:00
|
|
|
{
|
2022-02-10 02:40:01 +00:00
|
|
|
if (list[1] == null) {
|
|
|
|
throw VeilidAPIExceptionInternal("Null MESSAGE_OK value");
|
2022-02-09 14:47:36 +00:00
|
|
|
}
|
|
|
|
return list[1] as T;
|
|
|
|
}
|
2022-02-14 02:09:43 +00:00
|
|
|
case messageErr:
|
2022-02-09 14:47:36 +00:00
|
|
|
{
|
2022-02-10 02:40:01 +00:00
|
|
|
throw VeilidAPIExceptionInternal("Internal API Error: ${list[1]}");
|
|
|
|
}
|
2022-02-14 02:09:43 +00:00
|
|
|
case messageErrJson:
|
2022-02-10 02:40:01 +00:00
|
|
|
{
|
|
|
|
throw VeilidAPIException.fromJson(jsonDecode(list[1]));
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
throw VeilidAPIExceptionInternal(
|
|
|
|
"Unexpected async return message type: ${list[0]}");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}).catchError((e) {
|
|
|
|
// Wrap all other errors in VeilidAPIExceptionInternal
|
|
|
|
throw VeilidAPIExceptionInternal(e.toString());
|
|
|
|
}, test: (e) {
|
|
|
|
// Pass errors that are already VeilidAPIException through without wrapping
|
|
|
|
return e is! VeilidAPIException;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<T> processFutureJson<T>(
|
2022-02-14 02:09:43 +00:00
|
|
|
T Function(Map<String, dynamic>) jsonConstructor, Future<dynamic> future) {
|
2022-02-10 02:40:01 +00:00
|
|
|
return future.then((value) {
|
|
|
|
final list = value as List<dynamic>;
|
|
|
|
switch (list[0] as int) {
|
2022-02-14 02:09:43 +00:00
|
|
|
case messageErr:
|
2022-02-10 02:40:01 +00:00
|
|
|
{
|
|
|
|
throw VeilidAPIExceptionInternal("Internal API Error: ${list[1]}");
|
2022-02-09 14:47:36 +00:00
|
|
|
}
|
2022-02-14 02:09:43 +00:00
|
|
|
case messageOkJson:
|
2022-02-09 14:47:36 +00:00
|
|
|
{
|
2022-02-10 02:40:01 +00:00
|
|
|
if (list[1] == null) {
|
|
|
|
throw VeilidAPIExceptionInternal("Null MESSAGE_OK_JSON value");
|
2022-02-09 14:47:36 +00:00
|
|
|
}
|
2022-02-10 02:40:01 +00:00
|
|
|
var ret = jsonDecode(list[1] as String);
|
|
|
|
return jsonConstructor(ret);
|
2022-02-09 14:47:36 +00:00
|
|
|
}
|
2022-02-14 02:09:43 +00:00
|
|
|
case messageErrJson:
|
2022-02-09 14:47:36 +00:00
|
|
|
{
|
2022-02-10 02:40:01 +00:00
|
|
|
throw VeilidAPIException.fromJson(jsonDecode(list[1]));
|
2022-02-09 14:47:36 +00:00
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
throw VeilidAPIExceptionInternal(
|
2022-02-10 02:40:01 +00:00
|
|
|
"Unexpected async return message type: ${list[0]}");
|
2022-02-09 14:47:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}).catchError((e) {
|
|
|
|
// Wrap all other errors in VeilidAPIExceptionInternal
|
|
|
|
throw VeilidAPIExceptionInternal(e.toString());
|
|
|
|
}, test: (e) {
|
|
|
|
// Pass errors that are already VeilidAPIException through without wrapping
|
|
|
|
return e is! VeilidAPIException;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-02-10 02:40:01 +00:00
|
|
|
Future<void> processFutureVoid(Future<dynamic> future) {
|
2022-02-09 14:47:36 +00:00
|
|
|
return future.then((value) {
|
|
|
|
final list = value as List<dynamic>;
|
|
|
|
switch (list[0] as int) {
|
2022-02-14 02:09:43 +00:00
|
|
|
case messageOk:
|
2022-02-09 14:47:36 +00:00
|
|
|
{
|
|
|
|
if (list[1] != null) {
|
|
|
|
throw VeilidAPIExceptionInternal(
|
|
|
|
"Unexpected MESSAGE_OK value '${list[1]}' where null expected");
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2022-02-14 02:09:43 +00:00
|
|
|
case messageErr:
|
2022-02-09 14:47:36 +00:00
|
|
|
{
|
2022-02-10 02:40:01 +00:00
|
|
|
throw VeilidAPIExceptionInternal("Internal API Error: ${list[1]}");
|
2022-02-09 14:47:36 +00:00
|
|
|
}
|
2022-02-14 02:09:43 +00:00
|
|
|
case messageOkJson:
|
2022-02-09 14:47:36 +00:00
|
|
|
{
|
|
|
|
var ret = jsonDecode(list[1] as String);
|
|
|
|
if (ret != null) {
|
|
|
|
throw VeilidAPIExceptionInternal(
|
|
|
|
"Unexpected MESSAGE_OK_JSON value '$ret' where null expected");
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2022-02-14 02:09:43 +00:00
|
|
|
case messageErrJson:
|
2022-02-09 14:47:36 +00:00
|
|
|
{
|
2022-02-10 02:40:01 +00:00
|
|
|
throw VeilidAPIException.fromJson(jsonDecode(list[1] as String));
|
2022-02-09 14:47:36 +00:00
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
throw VeilidAPIExceptionInternal(
|
2022-02-10 02:40:01 +00:00
|
|
|
"Unexpected async return message type: ${list[0]}");
|
2022-02-09 14:47:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}).catchError((e) {
|
|
|
|
// Wrap all other errors in VeilidAPIExceptionInternal
|
|
|
|
throw VeilidAPIExceptionInternal(e.toString());
|
|
|
|
}, test: (e) {
|
|
|
|
// Pass errors that are already VeilidAPIException through without wrapping
|
|
|
|
return e is! VeilidAPIException;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-09-09 20:27:13 +00:00
|
|
|
Future<Stream<T>> processFutureStream<T>(
|
|
|
|
Stream<T> returnStream, Future<dynamic> future) {
|
|
|
|
return future.then((value) {
|
|
|
|
final list = value as List<dynamic>;
|
|
|
|
switch (list[0] as int) {
|
|
|
|
case messageOk:
|
|
|
|
{
|
|
|
|
if (list[1] != null) {
|
|
|
|
throw VeilidAPIExceptionInternal(
|
|
|
|
"Unexpected MESSAGE_OK value '${list[1]}' where null expected");
|
|
|
|
}
|
|
|
|
return returnStream;
|
|
|
|
}
|
|
|
|
case messageErr:
|
|
|
|
{
|
|
|
|
throw VeilidAPIExceptionInternal("Internal API Error: ${list[1]}");
|
|
|
|
}
|
|
|
|
case messageOkJson:
|
|
|
|
{
|
|
|
|
var ret = jsonDecode(list[1] as String);
|
|
|
|
if (ret != null) {
|
|
|
|
throw VeilidAPIExceptionInternal(
|
|
|
|
"Unexpected MESSAGE_OK_JSON value '$ret' where null expected");
|
|
|
|
}
|
|
|
|
return returnStream;
|
|
|
|
}
|
|
|
|
case messageErrJson:
|
|
|
|
{
|
|
|
|
throw VeilidAPIException.fromJson(jsonDecode(list[1] as String));
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
throw VeilidAPIExceptionInternal(
|
|
|
|
"Unexpected async return message type: ${list[0]}");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}).catchError((e) {
|
|
|
|
// Wrap all other errors in VeilidAPIExceptionInternal
|
|
|
|
throw VeilidAPIExceptionInternal(e.toString());
|
|
|
|
}, test: (e) {
|
|
|
|
// Pass errors that are already VeilidAPIException through without wrapping
|
|
|
|
return e is! VeilidAPIException;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-02-14 02:09:43 +00:00
|
|
|
Stream<T> processStreamJson<T>(
|
2022-03-09 03:32:12 +00:00
|
|
|
T Function(Map<String, dynamic>) jsonConstructor, ReceivePort port) async* {
|
|
|
|
try {
|
|
|
|
await for (var value in port) {
|
|
|
|
final list = value as List<dynamic>;
|
|
|
|
switch (list[0] as int) {
|
|
|
|
case messageStreamItemJson:
|
|
|
|
{
|
|
|
|
if (list[1] == null) {
|
|
|
|
throw VeilidAPIExceptionInternal(
|
|
|
|
"Null MESSAGE_STREAM_ITEM_JSON value");
|
|
|
|
}
|
|
|
|
var ret = jsonDecode(list[1] as String);
|
|
|
|
yield jsonConstructor(ret);
|
|
|
|
break;
|
2022-02-14 02:09:43 +00:00
|
|
|
}
|
2022-03-09 03:32:12 +00:00
|
|
|
case messageStreamAbort:
|
|
|
|
{
|
|
|
|
port.close();
|
|
|
|
throw VeilidAPIExceptionInternal("Internal API Error: ${list[1]}");
|
|
|
|
}
|
|
|
|
case messageStreamAbortJson:
|
|
|
|
{
|
|
|
|
port.close();
|
|
|
|
throw VeilidAPIException.fromJson(jsonDecode(list[1]));
|
|
|
|
}
|
|
|
|
case messageStreamClose:
|
|
|
|
{
|
|
|
|
port.close();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
throw VeilidAPIExceptionInternal(
|
|
|
|
"Unexpected async return message type: ${list[0]}");
|
|
|
|
}
|
|
|
|
}
|
2022-02-14 02:09:43 +00:00
|
|
|
}
|
2022-06-16 01:51:38 +00:00
|
|
|
} catch (e, s) {
|
2022-02-14 02:09:43 +00:00
|
|
|
// Wrap all other errors in VeilidAPIExceptionInternal
|
2022-06-16 01:51:38 +00:00
|
|
|
throw VeilidAPIExceptionInternal(
|
|
|
|
"${e.toString()}\nStack Trace:\n${s.toString()}");
|
2022-03-09 03:32:12 +00:00
|
|
|
}
|
2022-02-14 02:09:43 +00:00
|
|
|
}
|
|
|
|
|
2022-11-26 19:16:02 +00:00
|
|
|
// FFI implementation of VeilidRoutingContext
|
|
|
|
class VeilidRoutingContextFFI implements VeilidRoutingContext {
|
|
|
|
final int _id;
|
|
|
|
final VeilidFFI _ffi;
|
|
|
|
|
|
|
|
VeilidRoutingContextFFI._(this._id, this._ffi);
|
|
|
|
@override
|
|
|
|
VeilidRoutingContextFFI withPrivacy() {
|
|
|
|
final newId = _ffi._routingContextWithPrivacy(_id);
|
|
|
|
return VeilidRoutingContextFFI._(newId, _ffi);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
VeilidRoutingContextFFI withCustomPrivacy(Stability stability) {
|
|
|
|
final newId = _ffi._routingContextWithCustomPrivacy(
|
|
|
|
_id, stability.json.toNativeUtf8());
|
|
|
|
return VeilidRoutingContextFFI._(newId, _ffi);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
VeilidRoutingContextFFI withSequencing(Sequencing sequencing) {
|
|
|
|
final newId =
|
|
|
|
_ffi._routingContextWithSequencing(_id, sequencing.json.toNativeUtf8());
|
|
|
|
return VeilidRoutingContextFFI._(newId, _ffi);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<Uint8List> appCall(String target, Uint8List request) async {
|
|
|
|
var nativeEncodedTarget = target.toNativeUtf8();
|
|
|
|
var nativeEncodedRequest = base64UrlEncode(request).toNativeUtf8();
|
|
|
|
|
|
|
|
final recvPort = ReceivePort("routing_context_app_call");
|
|
|
|
final sendPort = recvPort.sendPort;
|
|
|
|
_ffi._routingContextAppCall(
|
|
|
|
sendPort.nativePort, _id, nativeEncodedTarget, nativeEncodedRequest);
|
|
|
|
final out = await processFuturePlain(recvPort.first);
|
|
|
|
return base64Decode(out);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<void> appMessage(String target, Uint8List message) async {
|
|
|
|
var nativeEncodedTarget = target.toNativeUtf8();
|
|
|
|
var nativeEncodedMessage = base64UrlEncode(message).toNativeUtf8();
|
|
|
|
|
|
|
|
final recvPort = ReceivePort("routing_context_app_call");
|
|
|
|
final sendPort = recvPort.sendPort;
|
|
|
|
_ffi._routingContextAppCall(
|
|
|
|
sendPort.nativePort, _id, nativeEncodedTarget, nativeEncodedMessage);
|
|
|
|
return processFutureVoid(recvPort.first);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-07 02:18:42 +00:00
|
|
|
// FFI implementation of high level Veilid API
|
2022-02-09 14:47:36 +00:00
|
|
|
class VeilidFFI implements Veilid {
|
2022-02-07 02:18:42 +00:00
|
|
|
// veilid_core shared library
|
|
|
|
final DynamicLibrary _dylib;
|
|
|
|
|
|
|
|
// Shared library functions
|
2022-02-14 02:09:43 +00:00
|
|
|
final _FreeStringDart _freeString;
|
2022-07-01 20:20:43 +00:00
|
|
|
final _InitializeVeilidCoreDart _initializeVeilidCore;
|
2022-07-01 16:13:52 +00:00
|
|
|
final _ChangeLogLevelDart _changeLogLevel;
|
2022-02-14 02:09:43 +00:00
|
|
|
final _StartupVeilidCoreDart _startupVeilidCore;
|
|
|
|
final _GetVeilidStateDart _getVeilidState;
|
2022-09-06 22:59:41 +00:00
|
|
|
final _AttachDart _attach;
|
|
|
|
final _DetachDart _detach;
|
2022-02-14 02:09:43 +00:00
|
|
|
final _ShutdownVeilidCoreDart _shutdownVeilidCore;
|
2022-11-26 19:16:02 +00:00
|
|
|
|
|
|
|
final _RoutingContextDart _routingContext;
|
|
|
|
final _ReleaseRoutingContextDart _releaseRoutingContext;
|
|
|
|
final _RoutingContextWithPrivacyDart _routingContextWithPrivacy;
|
|
|
|
final _RoutingContextWithCustomPrivacyDart _routingContextWithCustomPrivacy;
|
|
|
|
final _RoutingContextWithSequencingDart _routingContextWithSequencing;
|
|
|
|
final _RoutingContextAppCallDart _routingContextAppCall;
|
|
|
|
final _RoutingContextAppMessageDart _routingContextAppMessage;
|
|
|
|
|
|
|
|
final _NewPrivateRouteDart _newPrivateRoute;
|
|
|
|
final _NewCustomPrivateRouteDart _newCustomPrivateRoute;
|
|
|
|
final _ImportRemotePrivateRouteDart _importRemotePrivateRoute;
|
|
|
|
final _ReleasePrivateRouteDart _releasePrivateRoute;
|
|
|
|
|
2022-10-01 02:37:55 +00:00
|
|
|
final _AppCallReplyDart _appCallReply;
|
2022-11-26 19:16:02 +00:00
|
|
|
|
|
|
|
final _DebugDart _debug;
|
2022-02-14 02:09:43 +00:00
|
|
|
final _VeilidVersionStringDart _veilidVersionString;
|
|
|
|
final _VeilidVersionDart _veilidVersion;
|
2022-02-07 02:18:42 +00:00
|
|
|
|
2022-02-09 14:47:36 +00:00
|
|
|
VeilidFFI(DynamicLibrary dylib)
|
|
|
|
: _dylib = dylib,
|
2022-02-14 02:09:43 +00:00
|
|
|
_freeString =
|
|
|
|
dylib.lookupFunction<_FreeStringC, _FreeStringDart>('free_string'),
|
2022-07-01 20:20:43 +00:00
|
|
|
_initializeVeilidCore = dylib.lookupFunction<_InitializeVeilidCoreC,
|
|
|
|
_InitializeVeilidCoreDart>('initialize_veilid_core'),
|
2022-07-01 16:13:52 +00:00
|
|
|
_changeLogLevel =
|
|
|
|
dylib.lookupFunction<_ChangeLogLevelC, _ChangeLogLevelDart>(
|
|
|
|
'change_log_level'),
|
2022-02-14 02:09:43 +00:00
|
|
|
_startupVeilidCore =
|
|
|
|
dylib.lookupFunction<_StartupVeilidCoreC, _StartupVeilidCoreDart>(
|
|
|
|
'startup_veilid_core'),
|
2022-02-09 14:47:36 +00:00
|
|
|
_getVeilidState =
|
2022-02-14 02:09:43 +00:00
|
|
|
dylib.lookupFunction<_GetVeilidStateC, _GetVeilidStateDart>(
|
2022-02-09 14:47:36 +00:00
|
|
|
'get_veilid_state'),
|
2022-09-06 22:59:41 +00:00
|
|
|
_attach = dylib.lookupFunction<_AttachC, _AttachDart>('attach'),
|
|
|
|
_detach = dylib.lookupFunction<_DetachC, _DetachDart>('detach'),
|
2022-02-14 02:09:43 +00:00
|
|
|
_shutdownVeilidCore =
|
|
|
|
dylib.lookupFunction<_ShutdownVeilidCoreC, _ShutdownVeilidCoreDart>(
|
|
|
|
'shutdown_veilid_core'),
|
2022-11-26 19:16:02 +00:00
|
|
|
_routingContext =
|
|
|
|
dylib.lookupFunction<_RoutingContextC, _RoutingContextDart>(
|
|
|
|
'routing_context'),
|
|
|
|
_releaseRoutingContext = dylib.lookupFunction<_ReleaseRoutingContextC,
|
|
|
|
_ReleaseRoutingContextDart>('release_routing_context'),
|
|
|
|
_routingContextWithPrivacy = dylib.lookupFunction<
|
|
|
|
_RoutingContextWithPrivacyC,
|
|
|
|
_RoutingContextWithPrivacyDart>('routing_context_with_privacy'),
|
|
|
|
_routingContextWithCustomPrivacy = dylib.lookupFunction<
|
|
|
|
_RoutingContextWithCustomPrivacyC,
|
|
|
|
_RoutingContextWithCustomPrivacyDart>(
|
|
|
|
'routing_context_with_custom_privacy'),
|
|
|
|
_routingContextWithSequencing = dylib.lookupFunction<
|
|
|
|
_RoutingContextWithSequencingC,
|
|
|
|
_RoutingContextWithSequencingDart>(
|
|
|
|
'routing_context_with_sequencing'),
|
|
|
|
_routingContextAppCall = dylib.lookupFunction<_RoutingContextAppCallC,
|
|
|
|
_RoutingContextAppCallDart>('routing_context_app_call'),
|
|
|
|
_routingContextAppMessage = dylib.lookupFunction<
|
|
|
|
_RoutingContextAppMessageC,
|
|
|
|
_RoutingContextAppMessageDart>('routing_context_app_message'),
|
|
|
|
_newPrivateRoute =
|
|
|
|
dylib.lookupFunction<_NewPrivateRouteC, _NewPrivateRouteDart>(
|
|
|
|
'new_private_route'),
|
|
|
|
_newCustomPrivateRoute = dylib.lookupFunction<_NewCustomPrivateRouteC,
|
|
|
|
_NewCustomPrivateRouteDart>('new_custom_private_route'),
|
|
|
|
_importRemotePrivateRoute = dylib.lookupFunction<
|
|
|
|
_ImportRemotePrivateRouteC,
|
|
|
|
_ImportRemotePrivateRouteDart>('import_remote_private_route'),
|
|
|
|
_releasePrivateRoute = dylib.lookupFunction<_ReleasePrivateRouteC,
|
|
|
|
_ReleasePrivateRouteDart>('release_private_route'),
|
2022-10-01 02:37:55 +00:00
|
|
|
_appCallReply = dylib.lookupFunction<_AppCallReplyC, _AppCallReplyDart>(
|
|
|
|
'app_call_reply'),
|
2022-11-26 19:16:02 +00:00
|
|
|
_debug = dylib.lookupFunction<_DebugC, _DebugDart>('debug'),
|
2022-02-14 02:09:43 +00:00
|
|
|
_veilidVersionString = dylib.lookupFunction<_VeilidVersionStringC,
|
|
|
|
_VeilidVersionStringDart>('veilid_version_string'),
|
2022-02-09 14:47:36 +00:00
|
|
|
_veilidVersion =
|
2022-02-14 02:09:43 +00:00
|
|
|
dylib.lookupFunction<_VeilidVersionC, _VeilidVersionDart>(
|
2022-02-09 14:47:36 +00:00
|
|
|
'veilid_version') {
|
|
|
|
// Get veilid_flutter initializer
|
|
|
|
var initializeVeilidFlutter = _dylib.lookupFunction<
|
2022-02-14 02:09:43 +00:00
|
|
|
_InitializeVeilidFlutterC,
|
|
|
|
_InitializeVeilidFlutterDart>('initialize_veilid_flutter');
|
2022-02-07 02:18:42 +00:00
|
|
|
initializeVeilidFlutter(NativeApi.postCObject);
|
2022-02-09 14:47:36 +00:00
|
|
|
}
|
|
|
|
|
2022-06-16 01:51:38 +00:00
|
|
|
@override
|
2022-07-01 20:20:43 +00:00
|
|
|
void initializeVeilidCore(Map<String, dynamic> platformConfigJson) {
|
2022-06-16 01:51:38 +00:00
|
|
|
var nativePlatformConfig =
|
|
|
|
jsonEncode(platformConfigJson, toEncodable: veilidApiToEncodable)
|
|
|
|
.toNativeUtf8();
|
|
|
|
|
2022-07-01 20:20:43 +00:00
|
|
|
_initializeVeilidCore(nativePlatformConfig);
|
2022-06-16 01:51:38 +00:00
|
|
|
|
|
|
|
malloc.free(nativePlatformConfig);
|
|
|
|
}
|
|
|
|
|
2022-07-01 16:13:52 +00:00
|
|
|
@override
|
|
|
|
void changeLogLevel(String layer, VeilidConfigLogLevel logLevel) {
|
|
|
|
var nativeLogLevel =
|
|
|
|
jsonEncode(logLevel.json, toEncodable: veilidApiToEncodable)
|
|
|
|
.toNativeUtf8();
|
|
|
|
var nativeLayer = layer.toNativeUtf8();
|
|
|
|
_changeLogLevel(nativeLayer, nativeLogLevel);
|
|
|
|
malloc.free(nativeLayer);
|
|
|
|
malloc.free(nativeLogLevel);
|
|
|
|
}
|
|
|
|
|
2022-02-09 14:47:36 +00:00
|
|
|
@override
|
2022-09-09 20:27:13 +00:00
|
|
|
Future<Stream<VeilidUpdate>> startupVeilidCore(VeilidConfig config) {
|
2022-02-14 02:09:43 +00:00
|
|
|
var nativeConfig =
|
|
|
|
jsonEncode(config.json, toEncodable: veilidApiToEncodable)
|
|
|
|
.toNativeUtf8();
|
2022-09-09 20:27:13 +00:00
|
|
|
final recvStreamPort = ReceivePort("veilid_api_stream");
|
|
|
|
final sendStreamPort = recvStreamPort.sendPort;
|
2022-02-10 02:40:01 +00:00
|
|
|
final recvPort = ReceivePort("startup_veilid_core");
|
|
|
|
final sendPort = recvPort.sendPort;
|
2022-09-09 20:27:13 +00:00
|
|
|
_startupVeilidCore(
|
|
|
|
sendPort.nativePort, sendStreamPort.nativePort, nativeConfig);
|
2022-02-10 02:40:01 +00:00
|
|
|
malloc.free(nativeConfig);
|
2022-09-09 20:27:13 +00:00
|
|
|
return processFutureStream(
|
|
|
|
processStreamJson(VeilidUpdate.fromJson, recvStreamPort),
|
|
|
|
recvPort.first);
|
2022-02-10 02:40:01 +00:00
|
|
|
}
|
2022-02-07 02:18:42 +00:00
|
|
|
|
2022-02-09 14:47:36 +00:00
|
|
|
@override
|
|
|
|
Future<VeilidState> getVeilidState() async {
|
2022-02-10 02:40:01 +00:00
|
|
|
final recvPort = ReceivePort("get_veilid_state");
|
|
|
|
final sendPort = recvPort.sendPort;
|
2022-02-14 02:09:43 +00:00
|
|
|
_getVeilidState(sendPort.nativePort);
|
2022-03-09 03:32:12 +00:00
|
|
|
return processFutureJson(VeilidState.fromJson, recvPort.first);
|
2022-02-07 02:18:42 +00:00
|
|
|
}
|
|
|
|
|
2022-09-06 22:59:41 +00:00
|
|
|
@override
|
|
|
|
Future<void> attach() async {
|
|
|
|
final recvPort = ReceivePort("attach");
|
|
|
|
final sendPort = recvPort.sendPort;
|
|
|
|
_attach(sendPort.nativePort);
|
|
|
|
return processFutureVoid(recvPort.first);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<void> detach() async {
|
|
|
|
final recvPort = ReceivePort("detach");
|
|
|
|
final sendPort = recvPort.sendPort;
|
|
|
|
_detach(sendPort.nativePort);
|
|
|
|
return processFutureVoid(recvPort.first);
|
|
|
|
}
|
|
|
|
|
2022-02-09 14:47:36 +00:00
|
|
|
@override
|
|
|
|
Future<void> shutdownVeilidCore() async {
|
2022-02-10 02:40:01 +00:00
|
|
|
final recvPort = ReceivePort("shutdown_veilid_core");
|
|
|
|
final sendPort = recvPort.sendPort;
|
|
|
|
_shutdownVeilidCore(sendPort.nativePort);
|
2022-03-09 03:32:12 +00:00
|
|
|
return processFutureVoid(recvPort.first);
|
2022-02-09 14:47:36 +00:00
|
|
|
}
|
|
|
|
|
2022-03-04 01:45:39 +00:00
|
|
|
@override
|
2022-11-26 19:16:02 +00:00
|
|
|
Future<VeilidRoutingContext> routingContext() async {
|
|
|
|
final recvPort = ReceivePort("routing_context");
|
2022-03-04 01:45:39 +00:00
|
|
|
final sendPort = recvPort.sendPort;
|
2022-11-26 19:16:02 +00:00
|
|
|
_routingContext(sendPort.nativePort);
|
|
|
|
final id = await processFuturePlain(recvPort.first);
|
|
|
|
return VeilidRoutingContextFFI._(id, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<KeyBlob> newPrivateRoute() async {
|
|
|
|
final recvPort = ReceivePort("new_private_route");
|
|
|
|
final sendPort = recvPort.sendPort;
|
|
|
|
_newPrivateRoute(sendPort.nativePort);
|
|
|
|
return processFutureJson(KeyBlob.fromJson, recvPort.first);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<KeyBlob> newCustomPrivateRoute(
|
|
|
|
Stability stability, Sequencing sequencing) async {
|
|
|
|
final recvPort = ReceivePort("new_custom_private_route");
|
|
|
|
final sendPort = recvPort.sendPort;
|
|
|
|
_newCustomPrivateRoute(sendPort.nativePort, stability.json.toNativeUtf8(),
|
|
|
|
sequencing.json.toNativeUtf8());
|
|
|
|
final keyblob = await processFutureJson(KeyBlob.fromJson, recvPort.first);
|
|
|
|
return keyblob;
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<String> importRemotePrivateRoute(Uint8List blob) async {
|
|
|
|
var nativeEncodedBlob = base64UrlEncode(blob).toNativeUtf8();
|
|
|
|
|
|
|
|
final recvPort = ReceivePort("import_remote_private_route");
|
|
|
|
final sendPort = recvPort.sendPort;
|
|
|
|
_importRemotePrivateRoute(sendPort.nativePort, nativeEncodedBlob);
|
2022-03-09 03:32:12 +00:00
|
|
|
return processFuturePlain(recvPort.first);
|
2022-03-04 01:45:39 +00:00
|
|
|
}
|
|
|
|
|
2022-11-26 19:16:02 +00:00
|
|
|
@override
|
|
|
|
Future<void> releasePrivateRoute(String key) async {
|
|
|
|
var nativeEncodedKey = key.toNativeUtf8();
|
|
|
|
|
|
|
|
final recvPort = ReceivePort("release_private_route");
|
|
|
|
final sendPort = recvPort.sendPort;
|
|
|
|
_releasePrivateRoute(sendPort.nativePort, nativeEncodedKey);
|
|
|
|
return processFutureVoid(recvPort.first);
|
|
|
|
}
|
|
|
|
|
2022-10-01 02:37:55 +00:00
|
|
|
@override
|
|
|
|
Future<void> appCallReply(String id, Uint8List message) async {
|
|
|
|
var nativeId = id.toNativeUtf8();
|
|
|
|
var nativeEncodedMessage = base64UrlEncode(message).toNativeUtf8();
|
|
|
|
final recvPort = ReceivePort("app_call_reply");
|
|
|
|
final sendPort = recvPort.sendPort;
|
|
|
|
_appCallReply(sendPort.nativePort, nativeId, nativeEncodedMessage);
|
|
|
|
return processFutureVoid(recvPort.first);
|
|
|
|
}
|
|
|
|
|
2022-11-26 19:16:02 +00:00
|
|
|
@override
|
|
|
|
Future<String> debug(String command) async {
|
|
|
|
var nativeCommand = command.toNativeUtf8();
|
|
|
|
final recvPort = ReceivePort("debug");
|
|
|
|
final sendPort = recvPort.sendPort;
|
|
|
|
_debug(sendPort.nativePort, nativeCommand);
|
|
|
|
return processFuturePlain(recvPort.first);
|
|
|
|
}
|
|
|
|
|
2022-02-09 14:47:36 +00:00
|
|
|
@override
|
2022-03-17 14:31:10 +00:00
|
|
|
String veilidVersionString() {
|
2022-02-09 14:47:36 +00:00
|
|
|
final versionString = _veilidVersionString();
|
|
|
|
String ret = versionString.toDartString();
|
|
|
|
_freeString(versionString);
|
|
|
|
return ret;
|
2022-02-07 02:18:42 +00:00
|
|
|
}
|
|
|
|
|
2022-02-09 14:47:36 +00:00
|
|
|
@override
|
2022-03-17 14:31:10 +00:00
|
|
|
VeilidVersion veilidVersion() {
|
2022-02-09 14:47:36 +00:00
|
|
|
final version = _veilidVersion();
|
|
|
|
return VeilidVersion(
|
|
|
|
version.major,
|
|
|
|
version.minor,
|
|
|
|
version.patch,
|
|
|
|
);
|
2022-02-07 02:18:42 +00:00
|
|
|
}
|
|
|
|
}
|