Não é possível definir o cabeçalho no KOA ao usar o retorno de chamada
Pergunta
Recentemente trabalhei em um novo projeto que usava retornos de chamada javascript.E eu estava usando koa estrutura.Mas quando chamei esta rota:
function * getCubes(next) {
var that = this;
_OLAPSchemaProvider.LoadCubesJSon(function(result) {
that.body = JSON.stringify(result.toString());
});
}
Eu recebo este erro:
_http_outgoing.js:331
throw new Error('Can\'t set headers after they are sent.');
^
Error: Can't set headers after they are sent.
at ServerResponse.OutgoingMessage.setHeader (_http_outgoing.js:331:11)
at Object.module.exports.set (G:\NAP\node_modules\koa\lib\response.js:396:16)
at Object.length (G:\NAP\node_modules\koa\lib\response.js:178:10)
at Object.body (G:\NAP\node_modules\koa\lib\response.js:149:19)
at Object.body (G:\NAP\node_modules\koa\node_modules\delegates\index.js:91:31)
at G:\NAP\Server\OlapServer\index.js:42:19
at G:\NAP\Server\OlapServer\OLAPSchemaProvider.js:1599:9
at _LoadCubes.xmlaRequest.success (G:\NAP\Server\OlapServer\OLAPSchemaProvider.js:1107:13)
at Object.Xmla._requestSuccess (G:\NAP\node_modules\xmla4js\src\Xmla.js:2110:50)
at Object.ajaxOptions.complete (G:\NAP\node_modules\xmla4js\src\Xmla.js:2021:34)
Solução
O problema é que sua chamada assíncrona LoadCubesJSon()
demora um pouco para retornar, mas Koa não percebe e continua com o fluxo de controle.Isso é basicamente o que yield
é para.
Objetos "rendíveis" incluem promessas, geradores e conversões (entre outros).
Pessoalmente, prefiro criar manualmente uma promessa com o Biblioteca 'Q'.Mas você pode usar qualquer outra biblioteca promissora ou node-thunkify
para criar uma conversão.
Aqui está um exemplo curto, mas prático, com Q
:
var koa = require('koa');
var q = require('q');
var app = koa();
app.use(function *() {
// We manually create a promise first.
var deferred = q.defer();
// setTimeout simulates an async call.
// Inside the traditional callback we would then resolve the promise with the callback return value.
setTimeout(function () {
deferred.resolve('Hello World');
}, 1000);
// Meanwhile, we return the promise to yield for.
this.body = yield deferred.promise;
});
app.listen(3000);
Então seu código ficaria assim:
function * getCubes(next) {
var deferred = q.defer();
_OLAPSchemaProvider.LoadCubesJSon(function (result) {
var output = JSON.stringify(result.toString());
deferred.resolve(output);
});
this.body = yield deferred.promise;
}
Licenciado em: CC-BY-SA com atribuição
Não afiliado a StackOverflow