Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(665)

Unified Diff: pkg/gcloud/test/service_scope_test.dart

Issue 804973002: Add appengine/gcloud/mustache dependencies. (Closed) Base URL: git@github.com:dart-lang/pub-dartlang-dart.git@master
Patch Set: Added AUTHORS/LICENSE/PATENTS files Created 6 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « pkg/gcloud/test/db_all_e2e_test.dart ('k') | pkg/gcloud/test/storage/e2e_test.dart » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: pkg/gcloud/test/service_scope_test.dart
diff --git a/pkg/gcloud/test/service_scope_test.dart b/pkg/gcloud/test/service_scope_test.dart
new file mode 100644
index 0000000000000000000000000000000000000000..afaa3be3912f5566866136277a48623ca9ec1349
--- /dev/null
+++ b/pkg/gcloud/test/service_scope_test.dart
@@ -0,0 +1,221 @@
+// Copyright (c) 2014, the Dart project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+
+library gcloud.test.service_scope_test;
+
+import 'dart:async';
+
+import 'package:gcloud/service_scope.dart' as ss;
+import 'package:unittest/unittest.dart';
+
+main() {
+ test('no-service-scope', () {
+ expect(() => ss.register(1, 'foobar'), throwsA(isStateError));
+ expect(() => ss.registerScopeExitCallback(() {}), throwsA(isStateError));
+ expect(() => ss.lookup(1), throwsA(isStateError));
+
+ var c = new Completer.sync();
+ ss.fork(expectAsync(() {
+ c.complete();
+ return new Future.value();
+ }));
+
+ // Assert that after fork()ing we still don't have a service scope outside
+ // of the zone created by the fork()ing.
+ c.future.then(expectAsync((_) {
+ expect(() => ss.register(1, 'foobar'), throwsA(isStateError));
+ expect(() => ss.registerScopeExitCallback(() {}), throwsA(isStateError));
+ expect(() => ss.lookup(1), throwsA(isStateError));
+ }));
+ });
+
+ test('non-existent-key', () {
+ return ss.fork(expectAsync(() {
+ expect(ss.lookup(1), isNull);
+ return new Future.value();
+ }));
+ });
+
+ test('fork-callback-returns-non-future', () {
+ // The closure passed to fork() must return a future.
+ expect(() => ss.fork(expectAsync(() => null)),
+ throwsA(isArgumentError));
+ });
+
+ test('error-on-double-insert', () {
+ // Ensure that inserting twice with the same key results in an error.
+ return ss.fork(expectAsync(() => new Future.sync(() {
+ ss.register(1, 'firstValue');
+ expect(() => ss.register(1, 'firstValue'), throwsA(isArgumentError));
+ })));
+ });
+
+ test('only-cleanup', () {
+ return ss.fork(expectAsync(() => new Future.sync(() {
+ ss.registerScopeExitCallback(expectAsync(() {}));
+ })));
+ });
+
+ test('correct-insertion-and-cleanup-order', () {
+ // Ensure cleanup functions are called in the reverse order of inserting
+ // their entries.
+ int insertions = 0;
+ return ss.fork(expectAsync(() => new Future.value(() {
+ int NUM = 10;
+
+ for (int i = 0; i < NUM; i++) {
+ var key = i;
+
+ insertions++;
+ ss.register(key, 'value$i');
+ ss.registerScopeExitCallback(expectAsync(() {
+ expect(insertions, equals(i + 1));
+ insertions--;
+ }));
+
+ for (int j = 0; j <= NUM; j++) {
+ if (j <= i) {
+ expect(ss.lookup(key), 'value$i');
+ } else {
+ expect(ss.lookup(key), isNull);
+ }
+ }
+ }
+ })));
+ });
+
+ test('onion-cleanup', () {
+ // Ensures that a cleanup method can look up things registered before it.
+ return ss.fork(expectAsync(() {
+ ss.registerScopeExitCallback(expectAsync(() {
+ expect(ss.lookup(1), isNull);
+ expect(ss.lookup(2), isNull);
+ }));
+ ss.register(1, 'value1');
+ ss.registerScopeExitCallback(expectAsync(() {
+ expect(ss.lookup(1), equals('value1'));
+ expect(ss.lookup(2), isNull);
+ }));
+ ss.register(2, 'value2', onScopeExit: expectAsync(() {
+ expect(ss.lookup(1), equals('value1'));
+ expect(ss.lookup(2), isNull);
+ }));
+ ss.registerScopeExitCallback(expectAsync(() {
+ expect(ss.lookup(1), 'value1');
+ expect(ss.lookup(2), 'value2');
+ }));
+ return new Future.value();
+ }));
+ });
+
+ test('correct-insertion-and-cleanup-order--errors', () {
+ // Ensure that all cleanup functions will be called - even if some of them
+ // result in an error.
+ // Ensure the fork() error message contains all error messages from the
+ // failed cleanup() calls.
+ int insertions = 0;
+ return ss.fork(() => new Future.sync(() {
+ for (int i = 0; i < 10; i++) {
+ insertions++;
+ ss.register(i, 'value$i');
+ ss.registerScopeExitCallback(() {
+ expect(insertions, equals(i + 1));
+ insertions--;
+ if (i.isEven) throw 'xx${i}yy';
+ });
+ }
+ })).catchError(expectAsync((e, _) {
+ for (int i = 0; i < 10; i++) {
+ expect('$e'.contains('xx${i}yy'), equals(i.isEven));
+ }
+ }));
+ });
+
+ test('service-scope-destroyed-after-callback-completes', () {
+ // Ensure that once the closure passed to fork() completes, the service
+ // scope is destroyed.
+ return ss.fork(expectAsync(() => new Future.sync(() {
+ var key = 1;
+ ss.register(key, 'firstValue');
+ ss.registerScopeExitCallback(Zone.current.bindCallback(() {
+ // Spawn an async task which will be run after the cleanups to ensure
+ // the service scope got destroyed.
+ Timer.run(expectAsync(() {
+ expect(() => ss.lookup(key), throwsA(isStateError));
+ expect(() => ss.register(2, 'value'), throwsA(isStateError));
+ expect(() => ss.registerScopeExitCallback(() {}),
+ throwsA(isStateError));
+ }));
+ }));
+ expect(ss.lookup(key), equals('firstValue'));
+ })));
+ });
+
+ test('override-parent-value', () {
+ // Ensure that once the closure passed to fork() completes, the service
+ // scope is destroyed.
+ return ss.fork(expectAsync(() => new Future.sync(() {
+ var key = 1;
+ ss.register(key, 'firstValue');
+ expect(ss.lookup(key), equals('firstValue'));
+
+ return ss.fork(expectAsync(() => new Future.sync(() {
+ ss.register(key, 'secondValue');
+ expect(ss.lookup(key), equals('secondValue'));
+ })));
+ })));
+ });
+
+ test('fork-onError-handler', () {
+ // Ensure that once the closure passed to fork() completes, the service
+ // scope is destroyed.
+ ss.fork(expectAsync(() {
+ Timer.run(() => throw new StateError('foobar'));
+ return new Future.value();
+ }), onError: expectAsync((error, _) {
+ expect(error, isStateError);
+ }));
+ });
+
+ test('nested-fork-and-insert', () {
+ // Ensure that independently fork()ed serice scopes can insert keys
+ // independently and they cannot see each others values but can see parent
+ // service scope values.
+ var rootKey = 1;
+ var subKey = 2;
+ var subKey1 = 3;
+ var subKey2 = 4;
+
+ return ss.fork(expectAsync(() {
+ int cleanupFork1 = 0;
+ int cleanupFork2 = 0;
+
+ ss.register(rootKey, 'root');
+ ss.registerScopeExitCallback(expectAsync(() {
+ expect(cleanupFork1, equals(2));
+ expect(cleanupFork2, equals(2));
+ }));
+ expect(ss.lookup(rootKey), equals('root'));
+
+ Future spawnChild(ownSubKey, otherSubKey, int i, cleanup) {
+ return ss.fork(expectAsync(() => new Future.sync(() {
+ ss.register(subKey, 'fork$i');
+ ss.registerScopeExitCallback(cleanup);
+ ss.register(ownSubKey, 'sub$i');
+ ss.registerScopeExitCallback(cleanup);
+
+ expect(ss.lookup(rootKey), equals('root'));
+ expect(ss.lookup(subKey), equals('fork$i'));
+ expect(ss.lookup(ownSubKey), equals('sub$i'));
+ expect(ss.lookup(otherSubKey), isNull);
+ })));
+ }
+
+ return Future.wait([
+ spawnChild(subKey1, subKey2, 1, () => cleanupFork1++),
+ spawnChild(subKey2, subKey1, 2, () => cleanupFork2++),
+ ]);
+ }));
+ });
+}
« no previous file with comments | « pkg/gcloud/test/db_all_e2e_test.dart ('k') | pkg/gcloud/test/storage/e2e_test.dart » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698