MongoDB
 sql >> Base de Dados >  >> NoSQL >> MongoDB

Como usar eventos para manter a lógica do mongodb fora dos manipuladores de solicitação node.js


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;