2016-05-30 59 views
1

我知道this问题有几乎相同的标题,但问题是不同的。玉模板没有得到从快递传递的数据

我使用内置于最新的Keystone.js发行版中的Jade模板引擎v.1.11.0。在一个控制器中,我用两个view.on('init')回调查询数据。第一个回叫只查询一条记录并且总是通过。第二个有时不。

category.js

var keystone = require('keystone'); 
var async = require('async'); 
exports = module.exports = function (req, res) { 

    var view = new keystone.View(req, res); 
    var locals = res.locals; 

    // Init locals 
    locals.section = 'category'; 
    locals.filters = { 
     category: req.params.category 
    }; 
    locals.data = { 
     sections: [], 
     category: {} 
    }; 

    // Load current category 
    view.on('init', function (next) { 

     var q = keystone.list('Category').model.findOne({ 
      key: locals.filters.category 
     }); 

     q.exec(function (err, result) { 
      locals.data.category = result; 
      locals.section = locals.data.category.name.toLowerCase(); 
      next(err); 
     }); 
    }); 


    // Load sections 
    view.on('init', function (next) { 
     var q = keystone.list('Section').model.find().where('category').in([locals.data.category]).sort('sortOrder').exec(function(err, results) { 

      if (err || !results.length) { 
       return next(err); 
      } 

      async.each(results, function(section, next) { 
       keystone.list('Article').model.find().where('section').in([section.id]).sort('sortOrder').exec(function(err, articles){ 
        var s = section; 
        if (articles.length) { 
         s.articles = articles; 
         locals.data.sections.push(s); 
        } else { 
         locals.data.sections.push(s); 
        } 
       });    

      }, function(err) { 
       console.log(locals.data.sections); 
      }); 

      next(err); 
     }); 
    }); 

    view.render('category'); 
}; 

在我看来,我应该总是得到这样过去了:

sections: { _id: 574b909b43ff68163ed86bf2, publicTitle: 'Title 1', key: 'name-1', sortOrder: 3, name: 'Name 1', __v: 0, category: 574b8960947f45f034ac89b4, text: '', image: {} } 

category: { _id: 574b8960947f45f034ac89b4, key: 'blabla', sortOrder: 1, name: 'Blabla', __v: 0, image: {} } 

但是60%的时候,我得到这个:

sections: 

category: { _id: 574b8960947f45f034ac89b4, key: 'johndoe', sortOrder: 1, name: 'JohnDoe', __v: 0, image: {} } 

奇怪的是,如果我去另一个类别,其中有条更多条款和30条条款,我在90%的时间内获得了部分内容,但仍然错过了10%。这在开发和生产中都存在。

有些“下一个()”发射得太早?我看不到我弄乱了什么。

+0

尝试在1个init中嵌套查询。 – Darren

+0

我从一开始就试过,出于某种原因,如果我这样做,它会使页面永久加载。另外有两个inits似乎没有问题,这种方法被广泛用于keystone,甚至在样板中。 –

+0

我还被建议尝试下一步(犯错);在异步回调中,它也会导致永久加载。 –

回答

0

好的。在对async进行了一些阅读后,我设法使用async.parallel来正常工作。

var keystone = require('keystone'); 
var async = require('async'); 
exports = module.exports = function (req, res) { 

    var view = new keystone.View(req, res); 
    var locals = res.locals; 

    // Init locals 
    locals.section = 'category'; 
    locals.filters = { 
     category: req.params.category 
    }; 
    locals.data = { 
     sections: [], 
     category: {} 
    }; 

    // Load current category 
    view.on('init', function (next) { 

     var q = keystone.list('Category').model.findOne({ 
      key: locals.filters.category 
     }); 

     q.exec(function (err, result) { 
      locals.data.category = result; 
      locals.section = locals.data.category.name.toLowerCase(); 
      getChildrenRecords(locals.data.category, next); 
     }); 
    }); 


    function getChildrenRecords(category, next){ 

     var q = keystone.list('Section').model.find().where('category').in([category]).sort('sortOrder').exec(function(err, results) { 

      if (err || !results.length) { 
       return next(err); 
      } 

      callItems = []; 

      for(var i = 0; i < results.length; i++) { 
       var data = results[i]; 
       callItems.push(makeCallbackFunction(data)); 
      } 

      function makeCallbackFunction(section) { 
       return function (callback) { 

        keystone.list('Article').model.find().where('section').in([section.id]).sort('sortOrder').exec(function(err, articles){ 

         if (err) return callback(err); 

         if (articles.length) { section.articles = articles; } 
          locals.data.sections.push(section); 
          callback(); 
        }); 
       }; 
      } 

      async.parallel(callItems, function(err, results){ 
       next(err); 
      }); 
     }); 
    } 


    // Render the view 
    view.render('category'); 
};