Skip to content

Commit

Permalink
fix(native_storage): Clearing of secure storage on macOS/iOS (#109)
Browse files Browse the repository at this point in the history
- The `kSecMatchLimit` flag must always be `kSecMatchLimitAll` to return
a list, but this was previously not set in release mode.
- Adds tests for all platforms to check that the keys to be cleared only
include those owned by native_storage
  • Loading branch information
dnys1 authored Apr 9, 2024
1 parent 294fff0 commit a5be178
Show file tree
Hide file tree
Showing 21 changed files with 300 additions and 137 deletions.
8 changes: 3 additions & 5 deletions .github/workflows/native_storage.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -123,8 +123,6 @@ jobs:
export DISPLAY=:99
sudo Xvfb -ac :99 -screen 0 1280x1024x24 > /dev/null 2>&1 &
flutter test -d linux integration_test/storage_test.dart
# TODO: Re-enable
# Need to fix this: Git error. Command: `git clone --mirror https://github.com/dart-lang/native /c/Users/runneradmin/.pub-cache\git\cache\native-647c69ed8027da6d6def6bc40efa87cf1a2f76aa`
test_windows:
runs-on: windows-latest
timeout-minutes: 15
Expand All @@ -138,9 +136,9 @@ jobs:
- name: Get Packages
working-directory: packages/native/storage
run: dart pub get --no-example
# - name: Test
# working-directory: packages/native/storage
# run: dart test
- name: Test
working-directory: packages/native/storage
run: dart test
- name: Get Packages (Example)
working-directory: packages/native/storage/example
run: flutter pub get
Expand Down
1 change: 1 addition & 0 deletions packages/native/storage/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
## 0.1.3

- chore: Migrate to jni 0.8.0 to enable isolated Android storage
- fix: Removal of secure storage values on macOS/iOS in release mode

## 0.1.2

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,10 @@ sealed class NativeStorage(
private val editor: SharedPreferences.Editor
get() = sharedPreferences.edit()

val allKeys: List<String>
get() = sharedPreferences.all.keys.filter { it.startsWith(prefix) }
.map { it.substring(prefix.length) }.toList()

fun write(key: String, value: String?) {
println("Writing: $prefix$key")
with(editor) {
Expand Down
Original file line number Diff line number Diff line change
@@ -1,9 +1,10 @@
import 'dart:math';

import 'package:native_storage/native_storage.dart';
import 'package:native_storage/src/native_storage_extended.dart';
import 'package:test/test.dart';

void sharedTests(String name, NativeStorageFactory factory) {
void sharedTests(String name, NativeStorageExtendedFactory factory) {
group(name, () {
const allowedNamespaces = [null, 'com.domain.myapp'];
const allowedScopes = [null, 'scope', 'scope1/scope2'];
Expand All @@ -29,6 +30,8 @@ void sharedTests(String name, NativeStorageFactory factory) {
test('writes a new key-value pair to storage', () {
storage.write(key, 'value');
expect(storage.read(key), 'value');

expect(storage.allKeys, equals([key]));
});

test('updates the value for an existing key', () {
Expand All @@ -38,12 +41,15 @@ void sharedTests(String name, NativeStorageFactory factory) {
storage.write(key, 'update');
expect(storage.read(key), 'update',
reason: 'Value was updated');

expect(storage.allKeys, equals([key]));
});
});

group('read', () {
test('can read a non-existent key', () {
expect(storage.read(key), isNull);
expect(storage.allKeys, isEmpty);
});
});

Expand All @@ -52,9 +58,11 @@ void sharedTests(String name, NativeStorageFactory factory) {
storage.write(key, 'delete');
expect(storage.read(key), 'delete',
reason: 'Value was written');
expect(storage.allKeys, equals([key]));

storage.delete(key);
expect(storage.read(key), isNull, reason: 'Value was deleted');
expect(storage.allKeys, isEmpty);
});

test('can delete a non-existent key', () {
Expand All @@ -74,6 +82,7 @@ void sharedTests(String name, NativeStorageFactory factory) {
storage.write(key2, value2);
expect(storage.read(key1), value1);
expect(storage.read(key2), value2);
expect(storage.allKeys, unorderedEquals([key1, key2]));

storage.clear();

Expand All @@ -87,6 +96,7 @@ void sharedTests(String name, NativeStorageFactory factory) {
isNull,
reason: 'Storage was cleared',
);
expect(storage.allKeys, isEmpty);
});

test('does not throw when no items present', () {
Expand Down Expand Up @@ -233,10 +243,16 @@ void sharedTests(String name, NativeStorageFactory factory) {
expect(parent.read('key'), 'parentValue');
expect(child.read('key'), 'childValue');

expect(parent.allKeys, unorderedEquals(['key', 'child/key']));
expect((child as NativeStorageExtended).allKeys, ['key']);

parent.clear();

expect(parent.read('key'), isNull);
expect(child.read('key'), isNull);

expect(parent.allKeys, isEmpty);
expect(child.allKeys, isEmpty);
});

test('child does not clear parent', () {
Expand All @@ -249,11 +265,17 @@ void sharedTests(String name, NativeStorageFactory factory) {
expect(parent.read('key'), 'parentValue');
expect(child.read('key'), 'childValue');

expect(parent.allKeys, unorderedEquals(['key', 'child/key']));
expect((child as NativeStorageExtended).allKeys, ['key']);

child.clear();

expect(parent.read('key'), 'parentValue');
expect(child.read('key'), isNull);

expect(parent.allKeys, ['key']);
expect(child.allKeys, isEmpty);

parent.clear();
});
});
Expand Down
Original file line number Diff line number Diff line change
@@ -1,11 +1,15 @@
import 'package:integration_test/integration_test.dart';
import 'package:native_storage/native_storage.dart';
import 'package:native_storage/src/local/local_storage_platform.vm.dart'
if (dart.library.js_interop) 'package:native_storage/src/local/local_storage_platform.web.dart';
import 'package:native_storage/src/secure/secure_storage_platform.vm.dart'
if (dart.library.js_interop) 'package:native_storage/src/secure/secure_storage_platform.web.dart';

import 'storage_shared.dart';

void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
sharedTests('NativeMemoryStorage', NativeMemoryStorage.new);
sharedTests('NativeSecureStorage', NativeSecureStorage.new);
sharedTests('NativeLocalStorage', NativeLocalStorage.new);
sharedTests('NativeSecureStorage', NativeSecureStoragePlatform.new);
sharedTests('NativeLocalStorage', NativeLocalStoragePlatform.new);
}
Original file line number Diff line number Diff line change
Expand Up @@ -42,4 +42,8 @@ final class LocalStoragePlatformAndroid extends NativeLocalStoragePlatform {
_storage.write(key.toJString(), value.toJString());
return value;
}

@override
List<String> get allKeys =>
_storage.getAllKeys().map((key) => key.toDartString()).toList();
}
Original file line number Diff line number Diff line change
Expand Up @@ -62,4 +62,10 @@ final class LocalStorageLinux extends NativeLocalStoragePlatform {
_writeData(data);
return value;
}

@override
List<String> get allKeys => [
for (final key in _readData().keys)
if (key.startsWith(_prefix)) key.substring(_prefix.length),
];
}
60 changes: 6 additions & 54 deletions packages/native/storage/lib/src/local/local_storage.windows.dart
Original file line number Diff line number Diff line change
@@ -1,72 +1,24 @@
import 'package:native_storage/src/local/local_storage_platform.vm.dart';
import 'package:native_storage/src/native/windows/windows.dart';
import 'package:native_storage/src/util/functional.dart';
import 'package:win32_registry/win32_registry.dart';

final class LocalStorageWindows extends NativeLocalStoragePlatform {
final class LocalStorageWindows extends NativeLocalStoragePlatform
with NativeStorageWindows {
LocalStorageWindows({
String? namespace,
super.scope,
}) : _namespace = namespace,
}) : namespaceOverride = namespace,
super.base();

final String? _namespace;

@override
late final String namespace = lazy(() {
if (_namespace != null) {
return _namespace;
}
if (windows.applicationInfo case (:final companyName, :final productName)) {
return '$companyName\\$productName';
}
return windows.applicationId;
});

late final _registry = lazy(() {
final hkcu = Registry.currentUser;
var key = hkcu
.createKey('SOFTWARE\\Classes\\Local Settings\\Software\\$namespace');
if (scope case final scope?) {
for (final path in scope.split('/')) {
key = key.createKey(path);
}
}
return key;
});

@override
String? delete(String key) {
final current = read(key);
if (current == null) {
return null;
}
_registry.deleteValue(key);
return current;
}
final String? namespaceOverride;

@override
String? read(String key) => _registry.getValueAsString(key);
String? read(String key) => registry.getValueAsString(key);

@override
String write(String key, String value) {
_registry.createValue(RegistryValue(key, RegistryValueType.string, value));
registry.createValue(RegistryValue(key, RegistryValueType.string, value));
return value;
}

@override
void clear() {
for (final value in List.of(_registry.values)) {
_registry.deleteValue(value.name);
}
for (final subkey in List.of(_registry.subkeyNames)) {
_registry.deleteKey(subkey, recursive: true);
}
}

@override
void close() {
_registry.close();
super.close();
}
}
24 changes: 17 additions & 7 deletions packages/native/storage/lib/src/local/local_storage_darwin.dart
Original file line number Diff line number Diff line change
Expand Up @@ -50,17 +50,27 @@ final class LocalStoragePlatformDarwin extends NativeLocalStoragePlatform {

@override
void clear() {
final allValues = _userDefaults.persistentDomainForName_(
for (final key in allKeys) {
_userDefaults.removeObjectForKey_(darwin.nsString('$_prefix$key'));
}
}

@override
List<String> get allKeys {
final domain = _userDefaults.persistentDomainForName_(
darwin.nsString(namespace),
);
if (allValues == null) {
return;
if (domain == null) {
return const [];
}
for (var i = 0; i < allValues.allKeys.count; i++) {
final key = NSString.castFrom(allValues.allKeys.objectAtIndex_(i));
if (scope == null || key.toString().startsWith(scope!)) {
_userDefaults.removeObjectForKey_(key);
final allKeys = <String>[];
final domainKeys = domain.allKeys;
for (var i = 0; i < domainKeys.count; i++) {
final key = NSString.castFrom(domainKeys.objectAtIndex_(i));
if (scope == null || key.toString().startsWith(_prefix)) {
allKeys.add(key.toString().substring(_prefix.length));
}
}
return allKeys;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,14 @@ import 'package:native_storage/src/local/local_storage.android.dart';
import 'package:native_storage/src/local/local_storage.linux.dart';
import 'package:native_storage/src/local/local_storage.windows.dart';
import 'package:native_storage/src/local/local_storage_darwin.dart';
import 'package:native_storage/src/native_storage_extended.dart';

/// The VM implementation of [NativeLocalStorage].
abstract base class NativeLocalStoragePlatform implements NativeLocalStorage {
abstract base class NativeLocalStoragePlatform
implements
NativeLocalStorage,
// ignore: invalid_use_of_visible_for_testing_member
NativeStorageExtended {
factory NativeLocalStoragePlatform({
String? namespace,
String? scope,
Expand Down
Original file line number Diff line number Diff line change
@@ -1,10 +1,15 @@
import 'package:native_storage/native_storage.dart';
import 'package:native_storage/src/isolated/isolated_storage_platform.unsupported.dart'
as unsupported;
import 'package:native_storage/src/native_storage_extended.dart';
import 'package:web/web.dart' as web;

/// The browser implementation of [NativeLocalStorage].
final class NativeLocalStoragePlatform implements NativeLocalStorage {
final class NativeLocalStoragePlatform
implements
NativeLocalStorage,
// ignore: invalid_use_of_visible_for_testing_member
NativeStorageExtended {
NativeLocalStoragePlatform({String? namespace, this.scope})
: namespace = namespace ?? web.window.location.hostname;

Expand All @@ -20,10 +25,8 @@ final class NativeLocalStoragePlatform implements NativeLocalStorage {

@override
void clear() {
for (final key in _storage.keys) {
if (key.startsWith(_prefix)) {
_storage.removeItem(key);
}
for (final key in allKeys) {
_storage.removeItem('$_prefix$key');
}
}

Expand All @@ -45,6 +48,12 @@ final class NativeLocalStoragePlatform implements NativeLocalStorage {
return value;
}

@override
List<String> get allKeys => [
for (final key in _storage.keys)
if (key.startsWith(_prefix)) key.substring(_prefix.length),
];

@override
void close() {
_secure?.close();
Expand Down
14 changes: 13 additions & 1 deletion packages/native/storage/lib/src/memory_storage.dart
Original file line number Diff line number Diff line change
@@ -1,9 +1,15 @@
import 'package:native_storage/src/isolated/isolated_storage.dart';
import 'package:native_storage/src/native_storage.dart';
import 'package:native_storage/src/native_storage_extended.dart';
import 'package:native_storage/src/secure/secure_storage.dart';

/// An in-memory implementation of [NativeStorage] and [NativeSecureStorage].
final class NativeMemoryStorage implements NativeStorage, NativeSecureStorage {
final class NativeMemoryStorage
implements
NativeStorage,
NativeSecureStorage,
// ignore: invalid_use_of_visible_for_testing_member
NativeStorageExtended {
NativeMemoryStorage({
String? namespace,
this.scope,
Expand Down Expand Up @@ -38,6 +44,12 @@ final class NativeMemoryStorage implements NativeStorage, NativeSecureStorage {
@override
String write(String key, String value) => _storage['$_prefix$key'] = value;

@override
List<String> get allKeys => [
for (final key in _storage.keys)
if (key.startsWith(_prefix)) key.substring(_prefix.length),
];

@override
void close() {
clear();
Expand Down

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Loading

0 comments on commit a5be178

Please sign in to comment.