Aqui está a solução que eu encontrei.
Eu usei mongojs o que simplifica bastante a interface do mongodb -- ao custo da flexibilidade na configuração -- mas oculta os retornos de chamada aninhados que o driver mongodb requer. Também torna a sintaxe muito mais parecida com o cliente mongo.
Em seguida, envolvo o objeto HTTP Response em um encerramento e passo esse encerramento para o método de consulta mongodb em um retorno de chamada.
var MongoProvider = require('./MongoProvider');
MongoProvider.setCollection('things');
exports.index = function(request, response){
function sendResponse(err, data) {
if (err) {
response.send(500, err);
}
response.send(data);
};
MongoProvider.fetchAll(things, sendResponse);
};
Ele ainda está essencialmente apenas passando o objeto de resposta para o provedor de banco de dados, mas ao envolvê-lo em um encerramento que sabe como lidar com a resposta, ele mantém essa lógica fora do meu módulo de banco de dados.
Uma pequena melhoria é usar uma função para criar um encerramento do manipulador de resposta fora do meu manipulador de solicitação:
function makeSendResponse(response){
return function sendResponse(err, data) {
if (err) {
console.warn(err);
response.send(500, {error: err});
return;
}
response.send(data);
};
}
Então agora meu manipulador de solicitação se parece com isso:
exports.index = function(request, response) {
response.send(makeSendResponse(response));
}
E meu MongoProvider está assim:
var mongojs = require('mongojs');
MongoProvider = function(config) {
this.configure(config);
this.db = mongojs.connect(this.url, this.collections);
}
MongoProvider.prototype.configure = function(config) {
this.url = config.host + "/" + config.name;
this.collections = config.collections;
}
MongoProvider.prototype.connect = function(url, collections) {
return mongojs.connect(this.url, this.collections);
}
MongoProvider.prototype.fetchAll = function fetchAll(collection, callback) {
this.db(collection).find(callback);
}
MongoProvider.prototype.fetchById = function fetchById(id, collection, callback) {
var objectId = collection.db.bson_serializer.ObjectID.createFromHexString(id.toString());
this.db(collection).findOne({ "_id": objectId }, callback);
}
MongoProvider.prototype.fetchMatches = function fetchMatches(json, collection, callback) {
this.db(collection).find(Json.parse(json), callback);
}
module.exports = MongoProvider;
Também posso estender o MongoProvider para coleções específicas para simplificar a API e fazer validação adicional:
ThingsProvider = function(config) {
this.collection = 'things';
this.mongoProvider = new MongoProvider(config);
things = mongoProvider.db.collection('things');
}
ThingsProvider.prototype.fetchAll = function(callback) {
things.fetchAll(callback);
}
//etc...
module.exports = ThingsProvider;