| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229 | import 'dart:async' show Future;import 'dart:io';import 'dart:convert';import 'dart:math' show Random;import 'package:args/args.dart' show ArgParser;import 'package:mustache/mustache.dart' as mustache;import 'package:postgresql/postgresql.dart' as pg;import 'package:postgresql/postgresql_pool.dart' as pgpool;import 'package:yaml/yaml.dart' as yaml;/// Starts a new HTTP server that implements the tests to be benchmarked.  The/// address and port for incoming connections is configurable via command line/// arguments, as is the number of database connections to be maintained in the/// connection pool.main(List<String> args) {  var parser = new ArgParser();  parser.addOption('address', abbr: 'a', defaultsTo: '0.0.0.0');  parser.addOption('port', abbr: 'p', defaultsTo: '8080');  parser.addOption('dbconnections', abbr: 'd', defaultsTo: '256');  var arguments = parser.parse(args);  _startServer(      arguments['address'],      int.parse(arguments['port']),      int.parse(arguments['dbconnections']));}/// The entity used in the database query and update tests.class World {  int id;  int randomnumber;  World(this.id, this.randomnumber);  toJson() => { 'id': id, 'randomNumber': randomnumber };}/// The entity used in the fortunes test.class Fortune implements Comparable<Fortune> {  int id;  String message;  Fortune(this.id, this.message);  compareTo(Fortune other) => message.compareTo(other.message);}/// The number of rows in the world entity table.const _WORLD_TABLE_SIZE = 10000;/// A random number generator.final _RANDOM = new Random();/// The 'text/html; charset=utf-8' content type.final _TYPE_HTML = new ContentType('text', 'html', charset: 'utf-8');/// The 'application/json' content type.final _TYPE_JSON = new ContentType('application', 'json');/// The 'text/html; charset=utf-8' content type.final _TYPE_TEXT = new ContentType('text', 'plain', charset: 'utf-8');/// The PostgreSQL connection pool used by all the tests that require database/// connectivity.var _connectionPool;/// The mustache template which is rendered in the fortunes test.var _fortunesTemplate;/// Starts a benchmark server, which listens for connections from/// '[address] : [port]' and maintains [dbConnections] connections to the/// database._startServer(address, port, dbConnections) {  Future.wait([    new File('postgresql.yaml').readAsString().then((config) {      _connectionPool = new pgpool.Pool(          new pg.Settings.fromMap(yaml.loadYaml(config)).toUri(),          min: dbConnections,          max: dbConnections);      return _connectionPool.start();    }),    new File('fortunes.mustache').readAsString().then((template) {      _fortunesTemplate = mustache.parse(template);    })  ]).then((_) {    HttpServer.bind(address, port).then((server) {      server.serverHeader = 'dart';      server.listen((request) {        switch (request.uri.path) {          case '/json':            _jsonTest(request);            break;          case '/db':            _dbTest(request);            break;          case '/queries':            _queriesTest(request);            break;          case '/fortunes':            _fortunesTest(request);            break;          case '/updates':            _updatesTest(request);            break;          case '/plaintext':            _plaintextTest(request);            break;          default:            _sendResponse(request, HttpStatus.NOT_FOUND);            break;        }      });    });  });}/// Returns the given [text] parsed as a base 10 integer.  If the text is null/// or is an otherwise invalid representation of a base 10 integer, zero is/// returned._parseInt(text) =>    (text == null) ? 0 : int.parse(text, radix: 10, onError: ((_) => 0));/// Completes the given [request] by writing the [response] with the given/// [statusCode] and [type]._sendResponse(request, statusCode, [ type, response ]) {  request.response.statusCode = statusCode;  request.response.headers.date = new DateTime.now();  if (type != null) {    request.response.headers.contentType = type;  }  if (response != null) {    var data = UTF8.encode(response);    request.response.contentLength = data.length;    request.response.add(data);  } else {    request.response.contentLength = 0;  }  request.response.close();}/// Completes the given [request] by writing the [response] as HTML._sendHtml(request, response) {  _sendResponse(request, HttpStatus.OK, _TYPE_HTML, response);}/// Completes the given [request] by writing the [response] as JSON._sendJson(request, response) {  _sendResponse(request, HttpStatus.OK, _TYPE_JSON, JSON.encode(response));}/// Completes the given [request] by writing the [response] as plain text._sendText(request, response) {  _sendResponse(request, HttpStatus.OK, _TYPE_TEXT, response);}/// Responds with the JSON test to the [request]._jsonTest(request) {  _sendJson(request, { 'message': 'Hello, World!' });}_queryRandom() {  return _connectionPool.connect()      .then((connection) {        return connection.query(            'SELECT id, randomnumber FROM world WHERE id = @id;',            { 'id': _RANDOM.nextInt(_WORLD_TABLE_SIZE) + 1 })            //            // The benchmark's constraints tell us there is exactly one row.            //            .single            .then((row) => new World(row[0], row[1]))            .whenComplete(() { connection.close(); });      });}/// Responds with the database query test to the [request]._dbTest(request) {  _queryRandom().then((response) => _sendJson(request, response));}/// Responds with the database queries test to the [request]._queriesTest(request) {  var queries = _parseInt(request.uri.queryParameters['queries']).clamp(1, 500);  Future.wait(new List.generate(queries,                                (_) => _queryRandom(),                                growable: false))      .then((response) => _sendJson(request, response));}/// Responds with the fortunes test to the [request]._fortunesTest(request) {  _connectionPool.connect().then((connection) {    return connection.query('SELECT id, message FROM fortune;')        .map((row) => new Fortune(row[0], row[1]))        .toList()        .whenComplete(() { connection.close(); });  }).then((fortunes) {    fortunes.add(new Fortune(0, 'Additional fortune added at request time.'));    fortunes.sort();    _sendHtml(request, _fortunesTemplate.renderString({      'fortunes': fortunes.map((fortune) => {              'id': fortune.id, 'message': fortune.message          }).toList()    }));  });}/// Responds with the updates test to the [request]._updatesTest(request) {  var queries = _parseInt(request.uri.queryParameters['queries']).clamp(1, 500);  Future.wait(new List.generate(queries, (_) {    return _queryRandom()        .then((world) {          world.randomnumber = _RANDOM.nextInt(_WORLD_TABLE_SIZE) + 1;          return _connectionPool.connect().then((connection) {            return connection.execute(                'UPDATE world SET randomnumber = @randomnumber WHERE id = @id;',                { 'randomnumber': world.randomnumber, 'id': world.id })                .whenComplete(() { connection.close(); });          }).then((_) => world);        });  }, growable: false))      .then((worlds) => _sendJson(request, worlds));}/// Responds with the plaintext test to the [request]._plaintextTest(request) {  _sendText(request, 'Hello, World!');}
 |