我有一个由随机失败的 1 节点 mongodb 支持的 Flask 集成测试:
pytest/test_webapi.py:59: in test_register_test
>           assert res.status_code == 302
E           assert <Response streamed [404 NOT FOUND]>.status_code == 302
失败率大约为 50%。
在 test_webapi.py 中测试如下:
def test_register_user(self):
    res = self.client.get("/logout")
    class MySMTPServer(smtpd.SMTPServer):
        mails = []
        def process_message(self, peer, mailfrom, rcpttos, data):
            self.mails.append((rcpttos[0], data))
    server = MySMTPServer(('localhost', 12345), None)
    t = threading.Thread(target=asyncore.loop, args=(1,))
    t.start()
    time.sleep(.1)
    try:
        res = self.client.post("/register", data=self.registration)
        assert res.status_code == 200
        mail, hash = server.mails[0]
        self.conn.fsync()
        time.sleep(.1)
        res = self.client.get('/activate/' + hash)
        assert res.status_code == 302
    finally:
        server.close()
webapi.py 中的相关 Flask 方法:
@app.route("/register", methods=["POST"])
def register_user():
    mail = flask.request.form['mail']
    user = flask.request.form["user"]
    pw = flask.request.form["pass"]
    hash = users.register(user, pw, mail=mail)
    return flask.jsonify({'_id': None}) # XXX
@app.route('/activate/<hash>', methods=['GET'])
def activate_user(hash):
    key = users.activate(hash=hash)
    if not key:
        flask.abort(404)
    return flask.redirect("/")
...由操作方法支持:
make_key = lambda : base64.encodestring(os.urandom(32)).strip()
def register(self, user, pw, **kw):
    hash = self.make_key()
    user = self.new(user, pw, activation=hash, **kw)
    self._send_mail(**user)
    return hash
def activate(self, hash):
    user = self.users.find_one({'activation': hash})
    if not user:
        return None
    key = self.make_key()
    activation = {
        '$unset': {'activation': 1},
        '$set': {'status': 'active', 'key': key} }
    self.users.update({'_id': user['_id']}, activation)
    return user
... self.users 是一个 mongodb 集合。
self.new() 使用 safe=True 持久化实体。
有趣的是,其他几个做类似事情的测试似乎从未遇到过这个问题。
我曾认为这足以确保持久对象对 pymongo 连接池中的其他线程可见。我应该更仔细地阅读 mongodb/pymongo 文档的哪一部分?还是与 asyncore 有一些奇怪的交互?