You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
141 lines
4.0 KiB
141 lines
4.0 KiB
import 'dart:convert'; |
|
import 'dart:ffi'; |
|
import 'dart:io'; |
|
import 'dart:typed_data'; |
|
|
|
import 'package:argon2_ffi_base/src/argon2_ffi_base.dart'; |
|
import 'package:argon2_ffi_base/src/utils.dart'; |
|
import 'package:ffi/ffi.dart'; |
|
import 'package:logging/logging.dart'; |
|
|
|
final _logger = Logger('argon2_ffi_base'); |
|
|
|
// ignore_for_file: non_constant_identifier_names, |
|
|
|
typedef Argon2HashNative = Pointer<Utf8> Function( |
|
Pointer<Uint8> key, |
|
Uint32 keyLen, |
|
Pointer<Uint8> salt, |
|
Uint32 saltlen, |
|
Uint32 m_cost, // memory cost |
|
Uint32 t_cost, // time cost (number iterations) |
|
Uint32 parallelism, |
|
IntPtr hashlen, |
|
Uint8 type, |
|
Uint32 version, |
|
); |
|
|
|
typedef Argon2Hash = Pointer<Utf8> Function( |
|
Pointer<Uint8> key, |
|
int keyLen, |
|
Pointer<Uint8> salt, |
|
int saltlen, |
|
int m_cost, // memory cost |
|
int t_cost, // time cost (number iterations) |
|
int parallelism, |
|
int hashlen, |
|
int type, |
|
int version, |
|
); |
|
|
|
abstract class Argon2Base extends Argon2 { |
|
// @protected |
|
Argon2Hash get argon2hash; |
|
|
|
@override |
|
Uint8List argon2(Argon2Arguments args) { |
|
final keyPtr = Uint8ArrayUtils.toPointer(args.key); |
|
final saltPtr = Uint8ArrayUtils.toPointer(args.salt); |
|
// final saltArray = allocate<Uint8>(count: args.salt.length); |
|
// final saltList = saltArray.asTypedList(args.length); |
|
// saltList.setAll(0, args.salt); |
|
// const memoryCost = 1 << 16; |
|
|
|
// _logger.fine('saltArray: ${ByteUtils.toHexList(saltArray.view)}'); |
|
|
|
final result = argon2hash( |
|
keyPtr, |
|
args.key.length, |
|
saltPtr, |
|
args.salt.length, |
|
args.memory, |
|
args.iterations, |
|
args.parallelism, |
|
args.length, |
|
args.type, |
|
args.version, |
|
); |
|
|
|
free(keyPtr); |
|
free(saltPtr); |
|
// free(saltArray); |
|
final resultString = Utf8.fromUtf8(result); |
|
return base64.decode(resultString); |
|
} |
|
|
|
@override |
|
Future<Uint8List> argon2Async(Argon2Arguments args) async { |
|
return argon2(args); |
|
} |
|
} |
|
|
|
class Argon2FfiFlutter extends Argon2Base { |
|
Argon2FfiFlutter({this.resolveLibrary}) { |
|
final argon2lib = _loadLib(); |
|
_nativeAdd = argon2lib |
|
.lookup<NativeFunction<Int32 Function(Int32, Int32)>>('native_add') |
|
.asFunction(); |
|
argon2hash = argon2lib |
|
.lookup<NativeFunction<Argon2HashNative>>('hp_argon2_hash') |
|
.asFunction(); |
|
} |
|
|
|
static ResolveLibrary defaultResolveLibrary = (name) => name; |
|
|
|
final ResolveLibrary resolveLibrary; |
|
|
|
int Function(int x, int y) _nativeAdd; |
|
@override |
|
Argon2Hash argon2hash; |
|
|
|
int addIt(int x, int y) => _nativeAdd(x, y); |
|
|
|
DynamicLibrary _loadLib() { |
|
final resolveLibrary = this.resolveLibrary ?? defaultResolveLibrary; |
|
|
|
if (!Argon2.resolveLibraryForceDynamic && |
|
(Platform.isIOS || Platform.isMacOS)) { |
|
return DynamicLibrary.executable(); |
|
} |
|
final libraryNames = [ |
|
[Platform.isAndroid, 'libargon2_ffi.so'], |
|
[Platform.isLinux, './libargon2_ffi_plugin.so'], |
|
[Platform.isWindows, 'argon2_ffi_plugin.dll'], |
|
[Platform.isMacOS, 'libargon2_ffi.dylib'], |
|
[Platform.isIOS, null], // only supports static linking. |
|
]; |
|
final libraryName = libraryNames.firstWhere((element) => element[0] == true, |
|
orElse: () => throw StateError( |
|
'Unsupported Operating System ${Platform.operatingSystem}'))[1] |
|
as String; |
|
final path = resolveLibrary(libraryName); |
|
try { |
|
return DynamicLibrary.open(libraryName); |
|
} on ArgumentError catch (e, stackTrace) { |
|
_logger.severe( |
|
'Error while loading dynamic library from $path ($libraryName)', |
|
e, |
|
stackTrace); |
|
if (e.message.toString().contains('hardened programs')) { |
|
final message = 'Unable to load argon2 library. On MacOS you have to ' |
|
'remove hardening from dart binary:\n\n' |
|
'codesign --remove-signature ${Platform.resolvedExecutable}\n\n' |
|
'https://github.com/dart-lang/sdk/issues/39231#issuecomment-579743656'; |
|
_logger.shout(message); |
|
throw ArgumentError('${e.message}\n\n$message'); |
|
} |
|
|
|
rethrow; |
|
} |
|
} |
|
}
|
|
|