Skip to content

Commit

Permalink
chore(native_storage): Various fixes
Browse files Browse the repository at this point in the history
- Ensure only one `NativeStorage` instance exists for any namespace/scope pair
- Buffer calls to `close`
- Ensure consistent validation of namespaces
  • Loading branch information
dnys1 committed Jun 8, 2024
1 parent 7188957 commit 625d042
Show file tree
Hide file tree
Showing 30 changed files with 378 additions and 196 deletions.
6 changes: 6 additions & 0 deletions packages/native/storage/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,9 @@
## 0.1.7

- fix: Ensure only one `NativeStorage` instance exists for any namespace/scope pair
- fix: Buffer calls to `close`
- chore: Ensure consistent validation of namespaces

## 0.1.6

- feat: Support absolute scopes in `NativeStorage.scoped`
Expand Down
225 changes: 168 additions & 57 deletions packages/native/storage/example/integration_test/storage_shared.dart
Original file line number Diff line number Diff line change
@@ -1,27 +1,83 @@
// ignore_for_file: invalid_use_of_internal_member

import 'dart:math';

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

void sharedTests(String name, NativeStorageExtendedFactory factory) {
group(name, () {
enum NativeStorageType {
memory,
secure,
local;

String get name => switch (this) {
memory => 'NativeMemoryStorage',
secure => 'NativeSecureStorage',
local => 'NativeLocalStorage',
};
}

void sharedTests(NativeStorageType type, NativeStorageFactory factory_) {
NativeStorageBase factory({
String? namespace,
String? scope,
}) =>
factory_(namespace: namespace, scope: scope) as NativeStorageBase;

const bool isWeb = bool.fromEnvironment('dart.library.js_interop');
final isMemoryStorage = type == NativeStorageType.memory;
final isSecureStorage = type == NativeStorageType.secure;
final hasIsolatedStorage =
isWeb ? (isSecureStorage || isMemoryStorage) : true;
final hasFullyIsolatedStorage = isWeb ? hasIsolatedStorage : isMemoryStorage;

// ignore: unused_element
void dumpInstances() {
String dump(Map<(String, String?), NativeStorage> instances) {
return instances
.map((k, v) => MapEntry(k, '${v.runtimeType}(${v.hashCode})'))
.toString();
}

print('Memory: ${dump(NativeMemoryStorage.instances)}');
print('Instances: ${dump(NativeStorage.instances)}');
}

void reset() {
for (final instance in List.of(NativeStorage.instances.values)) {
instance.close();
}
for (final instance in List.of(NativeMemoryStorage.instances.values)) {
instance.close();
}
}

group(type.name, () {
const invalidNamespaces = ['com.domain/myapp', 'com.domain.myapp/'];
for (final namespace in invalidNamespaces) {
test('throws for invalid namespace: $namespace', () {
expect(() => factory(namespace: namespace), throwsArgumentError);
});
}

const allowedNamespaces = [null, 'com.domain.myapp'];
const allowedScopes = [null, 'scope', 'scope1/scope2'];
for (final namespace in allowedNamespaces) {
for (final scope in allowedScopes) {
group('namespace=$namespace', () {
group('scope=$scope', () {
late String key;
late final storage = factory(namespace: namespace, scope: scope);
late NativeStorageBase storage;

setUp(() {
storage = factory(namespace: namespace, scope: scope);
storage.clear();
// Add some randomness to prevent overlap between concurrent tests.
key = _randomString(10);
});

tearDownAll(() {
tearDown(() {
storage.clear();
storage.close();
});
Expand Down Expand Up @@ -119,23 +175,19 @@ void sharedTests(String name, NativeStorageExtendedFactory factory) {

group('isolated', () {
late String key;
late final isolated = storage.isolated;
late IsolatedNativeStorage isolated;

setUp(() async {
isolated = storage.isolated;
await isolated.clear();
// Add some randomness to prevent overlap between concurrent tests.
key = _randomString(10);
});

tearDownAll(() async {
await isolated.clear();
await isolated.close();
});

test(
'shares with non-isolated storage',
// The NativeMemoryStorage does not share.
skip: storage is NativeMemoryStorage,
skip: hasFullyIsolatedStorage,
() async {
storage.write(key, 'value');
expect(await isolated.read(key), 'value');
Expand All @@ -148,6 +200,22 @@ void sharedTests(String name, NativeStorageExtendedFactory factory) {
},
);

test(
'does not share with non-isolated storage',
// The NativeMemoryStorage does not share.
skip: !hasFullyIsolatedStorage,
() async {
storage.write(key, 'value');
expect(await isolated.read(key), null);

await isolated.write(key, 'isolated');
expect(storage.read(key), 'value');

await isolated.clear();
expect(storage.read(key), 'value');
},
);

group('write', () {
test('writes a new key-value pair to storage', () async {
await isolated.write(key, 'value');
Expand Down Expand Up @@ -233,70 +301,113 @@ void sharedTests(String name, NativeStorageExtendedFactory factory) {
}
}

test('parent clears child', () {
final parent = factory(namespace: 'com.domain', scope: 'scope');
final child = parent.scoped('child');
group('fixes', () {
setUp(reset);

parent.write('key', 'parentValue');
child.write('key', 'childValue');
test('parent clears child', () {
final parent = factory(namespace: 'com.domain', scope: 'scope');
final child = parent.scoped('child');

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

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

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

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

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

test('child does not clear parent', () {
final parent = factory(namespace: 'com.domain', scope: 'scope');
final child = parent.scoped('child');
expect(parent.allKeys, isEmpty);
expect(child.allKeys, isEmpty);
});

parent.write('key', 'parentValue');
child.write('key', 'childValue');
test('child does not clear parent', () {
final parent = factory(namespace: 'com.domain', scope: 'scope');
final child = parent.scoped('child');

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

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

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

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

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

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

parent.clear();
});

test('rescope', () {
final nested = factory(
namespace: 'com.domain',
scope: 'scope/child/nested',
);
expect(nested.scope, 'scope/child/nested');

final root = nested.scoped('/');
expect(root.scope, null);

final child = nested.scoped('/scope/child');
expect(child.scope, 'scope/child');

final current = nested.scoped('');
expect(current.scope, 'scope/child/nested');

final drill = child.scoped('nested');
expect(drill.scope, 'scope/child/nested');
});

test('instance caching', () {
void compare(NativeStorage instance1, NativeStorage instance2) {
expect(instance1, same(instance2));

final secure1 = instance1.secure;
final secure2 = instance2.secure;
expect(secure1, same(secure2));

final isolated1 = instance1.isolated;
final isolated2 = instance2.isolated;
expect(isolated1, same(isolated2));
}

test('rescope', () {
final nested = factory(
namespace: 'com.domain',
scope: 'scope/child/nested',
);
expect(nested.scope, 'scope/child/nested');
final instance1 = factory();
final instance2 = factory();
compare(instance1, instance2);

final root = nested.scoped('/');
expect(root.scope, null);
final scope1 = factory(namespace: 'com.domain', scope: 'scope');
final scope2 = factory(namespace: 'com.domain', scope: 'scope');
compare(scope1, scope2);

final child = nested.scoped('/scope/child');
expect(child.scope, 'scope/child');
final child1 = scope1.scoped('child');
final child2 = scope2.scoped('child');
compare(child1, child2);
});

final current = nested.scoped('');
expect(current.scope, 'scope/child/nested');
test('buffers close calls', skip: !hasIsolatedStorage, () async {
final instance = factory();
final isolated = instance.isolated;
instance.close();
expect(isolated, same(instance.isolated));
await expectLater(isolated.read('key'), throwsStateError);

final drill = child.scoped('nested');
expect(drill.scope, 'scope/child/nested');
final newInstance = factory();
expect(instance, isNot(same(newInstance)));
expect(isolated, isNot(same(newInstance.isolated)));
await expectLater(newInstance.isolated.read('key'), completion(null));
});
});
});
}
Expand Down
Original file line number Diff line number Diff line change
@@ -1,15 +1,11 @@
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', NativeSecureStoragePlatform.new);
sharedTests('NativeLocalStorage', NativeLocalStoragePlatform.new);
sharedTests(NativeStorageType.memory, NativeMemoryStorage.new);
sharedTests(NativeStorageType.secure, NativeSecureStorage.new);
sharedTests(NativeStorageType.local, NativeLocalStorage.new);
}
2 changes: 1 addition & 1 deletion packages/native/storage/example/ios/Podfile.lock
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ EXTERNAL SOURCES:

SPEC CHECKSUMS:
Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7
integration_test: 13825b8a9334a850581300559b8839134b124670
integration_test: ce0a3ffa1de96d1a89ca0ac26fca7ea18a749ef4

PODFILE CHECKSUM: 819463e6a0290f5a72f145ba7cde16e8b6ef0796

Expand Down
2 changes: 1 addition & 1 deletion packages/native/storage/lib/native_storage.dart
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
export 'src/isolated/isolated_storage.dart';
export 'src/local/local_storage.dart';
export 'src/memory_storage.dart';
export 'src/native_memory_storage.dart';
export 'src/native_storage.dart';
export 'src/native_storage_exception.dart';
export 'src/secure/secure_storage.dart';
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import 'package:native_storage/src/isolated/isolated_storage_platform.unsupporte
if (dart.library.isolate) 'package:native_storage/src/isolated/isolated_storage_platform.vm.dart';

/// A [NativeStorage] constructor.
typedef NativeStorageFactory = NativeStorage Function({
typedef NativeStorageFactory<T extends NativeStorage> = T Function({
String? namespace,
String? scope,
});
Expand Down
Loading

0 comments on commit 625d042

Please sign in to comment.