blob: ba963b9db9fa7270b54b373d0ed88a11fc676d5f [file] [log] [blame]
// Copyright 2015 The Vanadium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
library syncbase_database_test;
import 'dart:async';
import 'dart:convert' show UTF8;
import 'package:test/test.dart';
import 'package:syncbase/src/testing_instrumentation.dart' as testing;
import 'package:syncbase/syncbase_client.dart'
show SyncbaseClient, WatchChangeTypes, WatchChange, WatchGlobStreamImpl;
import './utils.dart' as utils;
Iterable<String> decodeStrs(List<List<int>> bufList) {
return bufList.map((x) => UTF8.decode(x));
}
runDatabaseTests(SyncbaseClient c) {
test('getting a handle to a database', () {
var app = c.app(utils.uniqueName('app'));
var dbName = utils.uniqueName('db');
var db = app.noSqlDatabase(dbName);
expect(db.name, equals(dbName));
expect(db.fullName, equals(app.fullName + '/' + dbName));
});
test('creating and destroying a database', () async {
var app = c.app(utils.uniqueName('app'));
await app.create(utils.emptyPerms());
var db = app.noSqlDatabase(utils.uniqueName('db'));
expect(await db.exists(), equals(false));
await db.create(utils.emptyPerms());
expect(await db.exists(), equals(true));
await db.destroy();
expect(await db.exists(), equals(false));
});
test('listing tables', () async {
var app = c.app(utils.uniqueName('app'));
await app.create(utils.emptyPerms());
var db = app.noSqlDatabase(utils.uniqueName('db'));
await db.create(utils.emptyPerms());
var want = [utils.uniqueName('table1'), utils.uniqueName('table2')];
want.sort();
for (var tableName in want) {
await db.table(tableName).create(utils.emptyPerms());
}
var got = await db.listTables();
got.sort((t1, t2) => t1.compareTo(t2));
expect(got.length, equals(want.length));
for (var i = 0; i < got.length; i++) {
expect(got[i], equals(want[i]));
}
});
test('basic watch', () async {
var app = c.app(utils.uniqueName('app'));
await app.create(utils.emptyPerms());
var db = app.noSqlDatabase(utils.uniqueName('db'));
await db.create(utils.emptyPerms());
var table = db.table(utils.uniqueName('table'));
await table.create(utils.emptyPerms());
// Perform some operations that we won't be watching.
await table.put('row1', UTF8.encode('value1'));
await table.delete('row1');
// Start watching everything from now.
var resumeMarker = await db.getResumeMarker();
var prefix = '';
var watchStream = db.watch(table.name, prefix, resumeMarker);
// Perform some operations while are watching.
var expectedChanges = new List<WatchChange>();
await table.put('row2', UTF8.encode('value2'));
resumeMarker = await db.getResumeMarker();
var expectedChange = SyncbaseClient.watchChange(
table.name, 'row2', resumeMarker, WatchChangeTypes.put,
valueBytes: UTF8.encode('value2'));
expectedChanges.add(expectedChange);
await table.delete('row2');
resumeMarker = await db.getResumeMarker();
expectedChange = SyncbaseClient.watchChange(
table.name, 'row2', resumeMarker, WatchChangeTypes.delete);
expectedChanges.add(expectedChange);
// Ensure we see all the expected changes in order in the watch stream.
var changeNum = 0;
await for (var change in watchStream) {
// Classes generated by mojom Dart compiler do not override == and
// hashCode but they do override toString to print all properties. So we
// use toString to assert equality.
expect(change.toString(), equals(expectedChanges[changeNum].toString()));
changeNum++;
// We need to break out of awaiting for watch stream values when we get
// everything we expected. because watch stream does not end until
// canceled by design and we don't have canceling mechanism yet.
if (changeNum == expectedChanges.length) {
break;
}
}
});
test('watch flow control', () async {
var app = c.app(utils.uniqueName('app'));
await app.create(utils.emptyPerms());
var db = app.noSqlDatabase(utils.uniqueName('db'));
await db.create(utils.emptyPerms());
var table = db.table(utils.uniqueName('table'));
await table.create(utils.emptyPerms());
var resumeMarker = await db.getResumeMarker();
var aFewMoments = new Duration(seconds: 1);
const int numOperations = 10;
var allOperations = [];
// Do several put operations in parallel and wait until they are all done.
for (var i = 0; i < numOperations; i++) {
allOperations.add(table.put('row $i', UTF8.encode('value$i')));
}
await Future.wait(allOperations);
// Reset testing instrumentations.
testing.DatabaseWatch.onChangeCounter.reset();
// Create a watch stream.
var watchStream = db.watch(table.name, '', resumeMarker);
// Listen for the data on the stream.
var allExpectedChangesReceived = new Completer();
onData(_) {
if (testing.DatabaseWatch.onChangeCounter.count == numOperations) {
allExpectedChangesReceived.complete();
}
}
var streamListener = watchStream.listen(onData);
// Pause the stream.
streamListener.pause();
// Wait a few moments.
await new Future.delayed(aFewMoments);
// Assert that we did not got any* events from server when paused.
// testing.DatabaseWatch.onChangeCounter instrumentation is used to ensure
// that client did not receive any updates from mojo server, guaranteeing
// flow control propagated properly all the way to the other end of the pipe
// *Note: We always get 1 change before we can tell the server to block by
// not acking that single change.
expect(testing.DatabaseWatch.onChangeCounter.count, equals(1));
// Resume the stream.
streamListener.resume();
// Wait until we get all expected changes.
await allExpectedChangesReceived.future;
// Assert we've got all the expected changes after resuming.
expect(testing.DatabaseWatch.onChangeCounter.count, equals(numOperations));
});
test('basic exec', () async {
var app = c.app(utils.uniqueName('app'));
await app.create(utils.emptyPerms());
var db = app.noSqlDatabase(utils.uniqueName('db'));
await db.create(utils.emptyPerms());
var table = db.table('airports');
await table.create(utils.emptyPerms());
await table.put('aӲ읔', UTF8.encode('ᚸӲ읔+קAل'));
await table.put('yyz', UTF8.encode('Toronto'));
var query = 'select k as code, v as cityname from airports';
var resultStream = db.exec(query);
var results = await resultStream.toList();
// Expect column headers plus two rows.
expect(results.length, 3);
// Check column headers.
expect(decodeStrs(results[0].values), equals(['code', 'cityname']));
// Check rows.
expect(decodeStrs(results[1].values), equals(['aӲ읔', 'ᚸӲ읔+קAل']));
expect(decodeStrs(results[2].values), equals(['yyz', 'Toronto']));
});
// TODO(nlacasse): Test database.get/setPermissions.
}