达特的未来是我生存的祸根。
我有一个类,它调用async(Future)函数来启动数据库实例,如下所示:
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();
});
然后这是调用登录功能的调用,该函数最终调用了失败的attemptLogin函数:
var req = new MockRequest("/user/login", contentType: 'JSON', method:'POST',
body:JSON.encode(
{"username" : 'test',
"password" : 'test' }));
如何处理数据库init的异步性质并仍然进行模拟测试?具体来说,有没有办法强制attemptLogin()Future以某种方式等待DataManager类初始化的完成?
感谢您的帮助, 格雷格
答案 0 :(得分: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));
}
...
};
我不知道你是如何将它挂钩到你的依赖注入框架中的,但我想它可以用静态方法做些什么。
答案 1 :(得分:1)
如何使用@lrn的解决方案
setUp(() {
return DataManager.createNew().then((dm) {
app.addModule(new Module()
..bind(DataManager, toValue: dm);
app.setUp();
}
});
这样您就已经将初始化的DataManager
实例传递到DI中。如果您稍后提出要求,则始终可以确定它已经初始化。