mirror of https://github.com/AMT-Cheif/drift.git
Merge branch 'ffi-dart-2-6' into develop
# Conflicts: # moor_ffi/CHANGELOG.md # moor_ffi/pubspec.yaml
This commit is contained in:
commit
f0dc307901
|
@ -1,7 +1,8 @@
|
|||
## unreleased
|
||||
## 0.2.0
|
||||
|
||||
- Remove the `background` flag from the moor apis provided by this package. Use the moor isolate api
|
||||
instead.
|
||||
- Support Dart 2.6, drop support for older versions
|
||||
|
||||
## 0.0.1
|
||||
|
||||
|
|
|
@ -15,7 +15,7 @@ import 'dart:ffi';
|
|||
/// is its destructor. There are many other interfaces (such as
|
||||
/// [sqlite3_prepare_v2()], [sqlite3_create_function()], and
|
||||
/// [sqlite3_busy_timeout()] to name but three) that are methods on an
|
||||
class Database extends Struct<Database> {}
|
||||
class Database extends Struct {}
|
||||
|
||||
/// SQL Statement Object
|
||||
///
|
||||
|
@ -38,7 +38,7 @@ class Database extends Struct<Database> {}
|
|||
///
|
||||
/// Refer to documentation on individual methods above for additional
|
||||
/// information.
|
||||
class Statement extends Struct<Statement> {}
|
||||
class Statement extends Struct {}
|
||||
|
||||
/// Dynamically Typed Value Object
|
||||
///
|
||||
|
@ -74,4 +74,4 @@ class Statement extends Struct<Statement> {}
|
|||
/// [sqlite3_result_value()] and [sqlite3_bind_value()].
|
||||
/// The [sqlite3_value_blob | sqlite3_value_type()] family of
|
||||
/// interfaces require protected sqlite3_value objects.
|
||||
class Value extends Struct<Value> {}
|
||||
class Value extends Struct {}
|
||||
|
|
|
@ -4,16 +4,14 @@ import 'dart:ffi';
|
|||
import 'dart:typed_data';
|
||||
|
||||
import 'package:moor_ffi/src/ffi/utils.dart';
|
||||
import 'package:ffi/ffi.dart' as ffi;
|
||||
|
||||
/// Pointer to arbitrary blobs in C.
|
||||
class CBlob extends Struct<CBlob> {
|
||||
@Uint8()
|
||||
int data;
|
||||
|
||||
class CBlob extends Struct {
|
||||
static Pointer<CBlob> allocate(Uint8List blob) {
|
||||
final str = Pointer<Uint8>.allocate(count: blob.length);
|
||||
final str = ffi.allocate<Uint8>(count: blob.length);
|
||||
|
||||
final asList = str.asExternalTypedData(count: blob.length) as Uint8List;
|
||||
final asList = str.asTypedList(blob.length);
|
||||
asList.setAll(0, blob);
|
||||
|
||||
return str.cast();
|
||||
|
@ -27,37 +25,38 @@ class CBlob extends Struct<CBlob> {
|
|||
..setAll(0, encoded);
|
||||
return CBlob.allocate(data);
|
||||
}
|
||||
}
|
||||
|
||||
/// Reads [bytesToRead] bytes from the current position.
|
||||
Uint8List read(int bytesToRead) {
|
||||
assert(bytesToRead >= 0);
|
||||
final str = addressOf.cast<Uint8>();
|
||||
if (isNullPointer(str)) return null;
|
||||
|
||||
final data = str.asExternalTypedData(count: bytesToRead) as Uint8List;
|
||||
return Uint8List.fromList(data);
|
||||
}
|
||||
|
||||
/// More efficient version of [readString] that doesn't have to find a nil-
|
||||
/// terminator. [length] is the amount of bytes to read. The string will be
|
||||
/// decoded via [utf8].
|
||||
String readAsStringWithLength(int length) {
|
||||
return utf8.decode(read(length));
|
||||
}
|
||||
|
||||
extension CBlobPointer on Pointer<CBlob> {
|
||||
/// Reads a 0-terminated string, decoded with utf8.
|
||||
///
|
||||
/// Warning: This method is very, very slow. If there is any way to know the
|
||||
/// length of the string to read, [readAsStringWithLength] will be orders of
|
||||
/// magnitude faster.
|
||||
String readString() {
|
||||
final str = addressOf;
|
||||
if (isNullPointer(str)) return null;
|
||||
if (isNullPointer) return null;
|
||||
|
||||
var len = 0;
|
||||
while (str.elementAt(++len).load<CBlob>().data != 0) {}
|
||||
final asUintPointer = cast<Uint8>();
|
||||
while (asUintPointer[++len] != 0) {}
|
||||
|
||||
final units = read(len);
|
||||
final units = readBytes(len);
|
||||
return utf8.decode(units);
|
||||
}
|
||||
|
||||
/// More efficient version of [readString] that doesn't have to find a nil-
|
||||
/// terminator. [length] is the amount of bytes to read. The string will be
|
||||
/// decoded via [utf8].
|
||||
String readAsStringWithLength(int length) {
|
||||
return utf8.decode(readBytes(length));
|
||||
}
|
||||
|
||||
/// Reads [length] bytes from this address.
|
||||
Uint8List readBytes(int length) {
|
||||
assert(length >= 0);
|
||||
if (isNullPointer) return null;
|
||||
|
||||
final data = cast<Uint8>().asTypedList(length);
|
||||
return Uint8List.fromList(data);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,3 +1,11 @@
|
|||
import 'dart:ffi';
|
||||
|
||||
bool isNullPointer<T extends NativeType>(Pointer<T> ptr) => ptr == nullptr;
|
||||
import 'package:ffi/ffi.dart' as ffi;
|
||||
|
||||
extension FreePointerExtension on Pointer {
|
||||
bool get isNullPointer => this == nullptr;
|
||||
|
||||
void free() {
|
||||
ffi.free(this);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -2,6 +2,7 @@ import 'dart:ffi';
|
|||
import 'dart:io';
|
||||
import 'dart:typed_data';
|
||||
|
||||
import 'package:ffi/ffi.dart';
|
||||
import 'package:moor_ffi/database.dart';
|
||||
import 'package:moor_ffi/src/api/result.dart';
|
||||
import 'package:moor_ffi/src/bindings/constants.dart';
|
||||
|
@ -31,12 +32,12 @@ class Database implements BaseDatabase {
|
|||
|
||||
/// Opens an sqlite3 database from a filename.
|
||||
factory Database.open(String fileName) {
|
||||
final dbOut = Pointer<Pointer<types.Database>>.allocate();
|
||||
final dbOut = allocate<Pointer<types.Database>>();
|
||||
final pathC = CBlob.allocateString(fileName);
|
||||
|
||||
final resultCode =
|
||||
bindings.sqlite3_open_v2(pathC, dbOut, _openingFlags, nullptr.cast());
|
||||
final dbPointer = dbOut.load<Pointer<types.Database>>();
|
||||
final dbPointer = dbOut.value;
|
||||
|
||||
dbOut.free();
|
||||
pathC.free();
|
||||
|
@ -88,19 +89,19 @@ class Database implements BaseDatabase {
|
|||
_ensureOpen();
|
||||
|
||||
final sqlPtr = CBlob.allocateString(sql);
|
||||
final errorOut = Pointer<Pointer<CBlob>>.allocate();
|
||||
final errorOut = allocate<Pointer<CBlob>>();
|
||||
|
||||
final result =
|
||||
bindings.sqlite3_exec(_db, sqlPtr, nullptr, nullptr, errorOut);
|
||||
|
||||
sqlPtr.free();
|
||||
|
||||
final errorPtr = errorOut.load<Pointer<CBlob>>();
|
||||
final errorPtr = errorOut.value;
|
||||
errorOut.free();
|
||||
|
||||
String errorMsg;
|
||||
if (!isNullPointer(errorPtr)) {
|
||||
errorMsg = errorPtr.load<CBlob>().readString();
|
||||
if (!errorPtr.isNullPointer) {
|
||||
errorMsg = errorPtr.readString();
|
||||
// the message was allocated from sqlite, we need to free it
|
||||
bindings.sqlite3_free(errorPtr.cast());
|
||||
}
|
||||
|
@ -114,14 +115,14 @@ class Database implements BaseDatabase {
|
|||
PreparedStatement prepare(String sql) {
|
||||
_ensureOpen();
|
||||
|
||||
final stmtOut = Pointer<Pointer<types.Statement>>.allocate();
|
||||
final stmtOut = allocate<Pointer<types.Statement>>();
|
||||
final sqlPtr = CBlob.allocateString(sql);
|
||||
|
||||
final resultCode =
|
||||
bindings.sqlite3_prepare_v2(_db, sqlPtr, -1, stmtOut, nullptr.cast());
|
||||
sqlPtr.free();
|
||||
|
||||
final stmt = stmtOut.load<Pointer<types.Statement>>();
|
||||
final stmt = stmtOut.value;
|
||||
stmtOut.free();
|
||||
|
||||
if (resultCode != Errors.SQLITE_OK) {
|
||||
|
|
|
@ -12,11 +12,11 @@ class SqliteException implements Exception {
|
|||
// hold the error message string is managed internally. The application does
|
||||
// not need to worry about freeing the result."
|
||||
// https://www.sqlite.org/c3ref/errcode.html
|
||||
final dbMessage = bindings.sqlite3_errmsg(db).load<CBlob>().readString();
|
||||
final dbMessage = bindings.sqlite3_errmsg(db).readString();
|
||||
|
||||
String explanation;
|
||||
if (code != null) {
|
||||
explanation = bindings.sqlite3_errstr(code).load<CBlob>().readString();
|
||||
explanation = bindings.sqlite3_errstr(code).readString();
|
||||
}
|
||||
|
||||
return SqliteException(dbMessage, explanation);
|
||||
|
|
|
@ -40,8 +40,7 @@ class PreparedStatement implements BasePreparedStatement {
|
|||
|
||||
for (var i = 0; i < columnCount; i++) {
|
||||
// name pointer doesn't need to be disposed, that happens when we finalize
|
||||
names[i] =
|
||||
bindings.sqlite3_column_name(_stmt, i).load<CBlob>().readString();
|
||||
names[i] = bindings.sqlite3_column_name(_stmt, i).readString();
|
||||
}
|
||||
|
||||
while (_step() == Errors.SQLITE_ROW) {
|
||||
|
@ -62,14 +61,10 @@ class PreparedStatement implements BasePreparedStatement {
|
|||
final length = bindings.sqlite3_column_bytes(_stmt, index);
|
||||
return bindings
|
||||
.sqlite3_column_text(_stmt, index)
|
||||
.load<CBlob>()
|
||||
.readAsStringWithLength(length);
|
||||
case Types.SQLITE_BLOB:
|
||||
final length = bindings.sqlite3_column_bytes(_stmt, index);
|
||||
return bindings
|
||||
.sqlite3_column_blob(_stmt, index)
|
||||
.load<CBlob>()
|
||||
.read(length);
|
||||
return bindings.sqlite3_column_blob(_stmt, index).readBytes(length);
|
||||
case Types.SQLITE_NULL:
|
||||
default:
|
||||
return null;
|
||||
|
|
|
@ -6,10 +6,11 @@ homepage: https://github.com/simolus3/moor/tree/develop/moor_ffi
|
|||
issue_tracker: https://github.com/simolus3/moor/issues
|
||||
|
||||
environment:
|
||||
sdk: ">=2.5.0-dev <2.6.0"
|
||||
sdk: ">=2.6.0 <3.0.0"
|
||||
|
||||
dependencies:
|
||||
moor: ">=1.7.0 <2.1.0"
|
||||
moor: ">=1.7.0 <3.0.0"
|
||||
ffi: ^0.1.3
|
||||
collection: ^1.0.0
|
||||
|
||||
dev_dependencies:
|
||||
|
|
|
@ -1,5 +1,6 @@
|
|||
import 'package:moor/moor.dart';
|
||||
import 'package:moor_ffi/src/ffi/blob.dart';
|
||||
import 'package:moor_ffi/src/ffi/utils.dart';
|
||||
import 'package:test/test.dart';
|
||||
|
||||
void main() {
|
||||
|
@ -7,7 +8,7 @@ void main() {
|
|||
final content = 'Hasta Mañana';
|
||||
final blob = CBlob.allocateString(content);
|
||||
|
||||
expect(blob.load<CBlob>().readString(), content);
|
||||
expect(blob.readString(), content);
|
||||
blob.free();
|
||||
});
|
||||
|
||||
|
@ -15,7 +16,7 @@ void main() {
|
|||
final data = List.generate(256, (x) => x);
|
||||
final blob = CBlob.allocate(Uint8List.fromList(data));
|
||||
|
||||
expect(blob.load<CBlob>().read(256), data);
|
||||
expect(blob.readBytes(256), data);
|
||||
blob.free();
|
||||
});
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue