Skip to content
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
151 changes: 151 additions & 0 deletions test/packages/storage/secure_storage_static_test.dart
Original file line number Diff line number Diff line change
@@ -0,0 +1,151 @@
import 'dart:typed_data';

import 'package:flutter_test/flutter_test.dart';
import 'package:realunit_wallet/packages/storage/secure_storage.dart';

void main() {
group('$SecureStorage static helpers', () {
group('generatePinSalt', () {
test('returns a 16-byte Uint8List', () {
final salt = SecureStorage.generatePinSalt();

expect(salt, isA<Uint8List>());
expect(salt, hasLength(16));
});

test('consecutive calls return distinct salts (CSPRNG)', () {
final a = SecureStorage.generatePinSalt();
final b = SecureStorage.generatePinSalt();

// Probability of a collision on 128-bit random is negligible.
expect(a, isNot(equals(b)));
});

test('every byte falls in the valid 0..255 range', () {
final salt = SecureStorage.generatePinSalt();

for (final b in salt) {
expect(b, inInclusiveRange(0, 255));
}
});
});

group('hashPin', () {
test('returns a 64-char hex string (32 bytes × 2 nibbles)', () {
final salt = SecureStorage.generatePinSalt();

// Pass a small iteration count so the synchronous version stays fast.
final hash = SecureStorage.hashPin('123456', salt, iterations: 1);

expect(hash, hasLength(64));
expect(RegExp(r'^[0-9a-f]+$').hasMatch(hash), isTrue);
});

test('same pin + salt → same hash (deterministic)', () {
final salt = SecureStorage.generatePinSalt();

final a = SecureStorage.hashPin('123456', salt, iterations: 1);
final b = SecureStorage.hashPin('123456', salt, iterations: 1);

expect(a, b);
});

test('different pin → different hash', () {
final salt = SecureStorage.generatePinSalt();

final a = SecureStorage.hashPin('123456', salt, iterations: 1);
final b = SecureStorage.hashPin('999999', salt, iterations: 1);

expect(a, isNot(b));
});

test('different salt → different hash for the same pin', () {
final saltA = SecureStorage.generatePinSalt();
final saltB = SecureStorage.generatePinSalt();

final a = SecureStorage.hashPin('123456', saltA, iterations: 1);
final b = SecureStorage.hashPin('123456', saltB, iterations: 1);

expect(a, isNot(b));
});

test('more iterations → different output (covers the legacy migration path)', () {
final salt = SecureStorage.generatePinSalt();

final a = SecureStorage.hashPin('123456', salt, iterations: 1);
final b = SecureStorage.hashPin('123456', salt, iterations: 10);

expect(a, isNot(b));
});
});

group('encryptSeed / decryptSeed round-trip', () {
Uint8List aesKey() {
// 32-byte AES-256 key.
return Uint8List.fromList(List.generate(32, (i) => i));
}

test('round-trips a typical 12-word mnemonic', () {
const seed = 'test test test test test test test test test test test junk';
final key = aesKey();

final encoded = SecureStorage.encryptSeed(key, seed);
final decoded = SecureStorage.decryptSeed(key, encoded);

expect(decoded, seed);
});

test('round-trips an empty string', () {
final key = aesKey();

final encoded = SecureStorage.encryptSeed(key, '');

expect(SecureStorage.decryptSeed(key, encoded), '');
});

test('round-trips unicode content', () {
const seed = 'Schloß Größe → über 1k';
final key = aesKey();

final encoded = SecureStorage.encryptSeed(key, seed);

expect(SecureStorage.decryptSeed(key, encoded), seed);
});

test('different IVs → different ciphertexts for the same plaintext (GCM nonce)', () {
final key = aesKey();
const seed = 'fixed-plaintext';

final a = SecureStorage.encryptSeed(key, seed);
final b = SecureStorage.encryptSeed(key, seed);

// Each call generates a fresh 12-byte IV, so the ciphertext halves
// (after the `iv:ciphertext` split) differ.
expect(a, isNot(b));
});

test('decryptSeed with the wrong key throws', () {
final correctKey = aesKey();
final wrongKey = Uint8List.fromList(List.generate(32, (i) => 0xFF - i));
final encoded = SecureStorage.encryptSeed(correctKey, 'secret');

expect(
() => SecureStorage.decryptSeed(wrongKey, encoded),
throwsA(anything),
);
});

test('output format is "<base64-iv>:<base64-ciphertext>"', () {
final key = aesKey();

final encoded = SecureStorage.encryptSeed(key, 'x');

expect(encoded, contains(':'));
final parts = encoded.split(':');
expect(parts, hasLength(2));
// IV is 12 bytes → base64 length 16 (with padding).
expect(parts[0], hasLength(16));
});
});
});
}
Loading