2

Dart 中的期货是我存在的祸根。

我有一个类,它调用异步(未来)函数来启动数据库实例,如下所示:

class DataManager {
  bool DbIsReady = false;
  Db _db;

  DataManager() {
    init_mongo_db();
  }

  void init_mongo_db() {
    print("Initializing MongoDB");
    _db = new Db("mongodb://127.0.0.1/test");
    _db.open().then((_) {
      DbIsReady = true;
    });
  }  

  Future<List> attemptLogin(String username, String password) {
    users = _db.collection("users");

    return // ... rest of code cut out for clarity
  }
}

这在服务器端工作正常,因为当服务器第一次启动时,数据库被初始化。当用户实际尝试登录时,数据库已经初始化并且一切正常。但是,当我尝试为此创建集成测试时,这会惨遭失败。当我创建类时,数据库尚未初始化,因此运行 AttemptLogin 例程失败。

DataManager db = new DataManager();
// This fails miserably, because db hasn't initialized yet.
db.AttemptLogin("test", "test"); 

更糟糕的是,代码使用了 Dart DI 框架,因此我实际上无法直接控制 DataManager 类的初始化。这是该类的实际设置:

setUp(() {
  app.addModule(new Module()
            ..bind(DataManager)
  app.setUp();
});

然后这是测试登录功能的调用,最终调用了失败的尝试登录函数:

var req = new MockRequest("/user/login", contentType: 'JSON', method:'POST',
    body:JSON.encode(
      {"username" : 'test',
       "password" : 'test' }));

如何处理数据库初始化的异步性质并仍然进行模拟测试?具体来说,有没有办法强制尝试登录()未来以某种方式等待 DataManager 类初始化的完成?

谢谢你的帮助,格雷格

4

2 回答 2

3

如果您的类具有异步设置,则不能使用普通的构造函数。在这种情况下,我只使用返回未来的工厂方法:

class DataManager {
  final Db _db;
  DataManager._(this._db);
  static Future<DataManager> createNew() {
    var db = new Db("mongodb://127.0.0.1/test");
    return db.open().then((_) => new DataManager._(db));
  }
  ...
};

我不知道你如何将它连接到你的依赖注入框架中,但我想它可以用静态方法做一些事情。

于 2014-07-06T10:02:26.863 回答
1

使用@lrn 的解决方案怎么样

setUp(() {
  return DataManager.createNew().then((dm) {
    app.addModule(new Module()
            ..bind(DataManager, toValue: dm);
    app.setUp();
  }
});

这样,您已经将一个初始化的DataManager实例传递给 DI。如果您稍后请求它,您始终可以确定它已经初始化。

于 2014-07-10T14:37:00.790 回答