萬盛學電腦網

 萬盛學電腦網 >> 腳本專題 >> javascript >> 記錄幾個node.js錯誤及解決方案

記錄幾個node.js錯誤及解決方案

 幾個node.js錯誤及解決辦法整理如下

node.js Error: EBADF, write

最近工作寫了個小項目,本以為能好好喝下茶,可是讓人想掀桌的報錯出現了。

fs.js:77
      throw err;  // Forgot a callback but don't know where? Use NODE_DEBUG=fs
            ^
Error: EBADF, write
    at Error (native)

WTF?! fs.js: 77? 這是要我去看內核源代碼?好在項目還不大,拆拆更健康。花了半天的時間之後大概是清楚了。

首先是 Error 的名字 EBADF 其意義是 bad file descriptor 錯誤的文件描述符。
而 Error: EBADF, write 表示往錯誤的文件描述符裡面寫數據了。

出現這個BUG的場景簡而言之,是有一個 .on('data') 事件拿到數據往 fd 裡面寫,這個時候某個操作拋了 error 我在處理error 的時候 close 了這個 fd,而另外一邊去還在觸發 data 事件想往這個(已經被我 close 的)fd裡面寫數據。如下:

// ...
 
var fd = fs.openSync(path, 'w');
 
test.on('data', function(data) {
    fs.write(fd, data);
});
 
test.on('end', function() {
    fs.close(fd);
});
 
// 在 end 之前 close 就會出現 Error: EBADF, write
setTimeout(function() {
    fs.close(fd);
}, 10);
 
// ...

解決方案:所以我們排查好出現 fs.close 關閉文件描述符的地方,確保 close 之後不會再有 read/write 。


Error: EBADF, close

另外附上在谷歌的過程中看到了另外一個類似的錯誤。這是當你為多種情況做 fs.close(fd); 的處理,然而不幸的是,多個情況被都觸發, fs.close(fd) 調用了多遍,同樣也會出現 EBADF 錯誤。這樣就能出現:

test.on('end', function() {
    fs.close(fd);
    fs.close(fd); // 多調用了一次就會出現
});

不友好的報錯


fs.js:77
      throw err;  // Forgot a callback but don't know where? Use NODE_DEBUG=fs
            ^
Error: EBADF, close
    at Error (native)

解決方案:依舊是排查 fs.close,只不過這次是要保證多種處理不會反復執行 fs.close ,或者你可以使用 try/catch 來無視它。

Error: EBADF, bad file descriptor

最後,當 fd 失效以後進行 read 操作的話,我還以為會出現 Error: EBADF, read 結果並沒有。以下是嘗試出現BUG的代碼:


// ...
 
fs.closeSync(fd);
fs.readSync(fd, new Buffer(1024), 0, 1024);
 
// ...

不過這個報錯會友好很多,有將其調用棧打出來。

fs.js:552
  var r = binding.read(fd, buffer, offset, length, position);
                  ^
Error: EBADF, bad file descriptor
    at Error (native)
    at Object.fs.readSync (fs.js:552:19)
    at command.<anonymous> (/Users/Lellansin/Documents/workspace/node/test-server/app/services/TestService.js:40:6)
    at command.emit (events.js:110:17)
    at ChildProcess.emit (events.js:129:20)
    at maybeClose (child_process.js:1015:16)
    at Socket.<anonymous> (child_process.js:1183:11)
    at Socket.emit (events.js:107:17)
    at Pipe.close (net.js:485:12)

解決方案:看錯誤棧去改代碼就好了。。


node.js Error: stdout maxBuffer exceeded

在使用 child_process 模塊中的 exec 、execFile、spawnSync、execFileSync、execSync 方法時需要注意其 options 參數中的 maxBuffer 項。

以上方法在執行時會在內存中建一個 buffer 來緩沖組合所有的輸出數據,而 maxBuffer 則是指定該 buffer 大小的地方。如果輸出超過指定的大小則會報 maxBuffer exceeded 的錯誤。

解決方案是執行的時候估計好大小,設置更大的 maxBuffer:

var exec = require('child_process').exec;
 
var child = exec('ls -lah', {
    encoding: 'utf8',
    timeout: 0,
    maxBuffer: 5000 * 1024, // 默認 200 * 1024
    killSignal: 'SIGTERM'
}, function(err, stdout, stderr) {
    console.log(stdout);
});

或者是用 spawn 的 .on('data') 事件觸發時,手動拼接數據到 .on('close') 事件觸發的時候獲得完整數據。


pomelo Cannot call method 'forwardMessage' of undefined

報錯信息:
[2014-09-10 14:32:45.315] [DEBUG] pomelo - [E:svnxjmhtrunksrcservergame-servernode_modulespomelolibcomponents
connector.js] [connector-server-1] handleMessage session id: 1, msg: {"id":10,"type":0,"compressRoute":0,"route":"user.u
serHandler.login","body":"{username:'alan_1', password:'123456', dev_id:'6984654'}"}
[2014-09-10 14:32:45.320] [ERROR] pomelo - [E:svnxjmhtrunksrcservergame-servernode_modulespomelolibserverserv
er.js] fail to forward message:TypeError: Cannot call method 'forwardMessage' of undefined
    at doForward (E:svnxjmhtrunksrcservergame-servernode_modulespomelolibserverserver.js:334:50)
    at dispatch (E:svnxjmhtrunksrcservergame-servernode_modulespomelolibserverserver.js:103:7)
    at next (E:svnxjmhtrunksrcservergame-servernode_modulespomelolibcommonservicefilterService.js:50:7)
    at Service.beforeFilter (E:svnxjmhtrunksrcservergame-servernode_modulespomelolibcommonservicefilterServi
ce.js:65:3)
    at beforeFilter (E:svnxjmhtrunksrcservergame-servernode_modulespomelolibserverserver.js:242:8)
    at pro.globalHandle (E:svnxjmhtrunksrcservergame-servernode_modulespomelolibserverserver.js:112:3)
    at Component.globalHandle (E:svnxjmhtrunksrcservergame-servernode_modulespomelolibcomponentsserver.js:74:
14)
    at handleMessage (E:svnxjmhtrunksrcservergame-servernode_modulespomelolibcomponentsconnector.js:295:15)
    at null.<anonymous> (E:svnxjmhtrunksrcservergame-servernode_modulespomelolibcomponentsconnector.js:239:5)
 
    at EventEmitter.emit (events.js:95:17)

Pomelo Club 上有些說法是前端服務器將消息轉發到其它服務器過程中出現問題,rpc失敗。或者可能是 handler 拼錯。
博主檢查發現在 servers.json 中普通 server 如果配置了 "frontend":true 就會出現這個問題. 嘗試把 frontend 去掉或者改成 false 就正常了.

copyright © 萬盛學電腦網 all rights reserved