วิธีพิมพ์ stack trace ใน Node.js?


คำตอบ:


615

Errorวัตถุใด ๆมีstackสมาชิกที่ดักจับจุดที่มันถูกสร้างขึ้น

var stack = new Error().stack
console.log( stack )

หรือมากกว่านั้น:

console.trace("Here I am!")

1
หรือเพียงแค่sys.puts(new Error().stack)(หลังจากที่เพิ่มโมดูลระบบ)
sirhc

5
ณ ตอนนี้ sys จะถูกลดค่า 'util'มันจะถูกแทนที่ด้วย
Pindatjuh

12
+1 สำหรับการแสดงnew Error().stackซึ่งทำงานในกรณีที่คุณไม่ต้องการเกี่ยวข้องกับคอนโซล
Eugene Beresovsky

1
ข้อดีอย่างหนึ่งของtraceมันคือมันแสดงให้เห็นถึงบรรทัด / บริบทปัจจุบันเช่นกันซึ่งstackไม่ได้ ข้อมูลอยู่ในวัตถุข้อผิดพลาดหากคุณต้องการสร้างบรรทัดที่ฉันเดาด้วยตนเอง
studgeek

130
console.log (err.stack) และ console.trace () ไม่ให้ผลลัพธ์เดียวกัน ในขณะที่ err.stack ให้การติดตามสแต็กสำหรับวัตถุ err ด้วยตัวเอง (ทำงานตามที่เราทุกคนคิดยกเว้น) console.trace () จะพิมพ์ call stack ที่จุดที่ console.trace () ถูกเรียก ดังนั้นหากคุณพบข้อผิดพลาดบางอย่างที่ถูกโยนโดยเลเยอร์โค้ดที่ลึกกว่า console.trace () จะไม่ประกอบด้วยโค้ดเลเยอร์ที่ลึกกว่าในการติดตามสแต็กเนื่องจากรหัสนั้นไม่อยู่ในสแต็กอีกต่อไป อย่างไรก็ตาม console.log (err.stack) จะมีเลเยอร์ที่ลึกกว่าถ้ามันโยนวัตถุข้อผิดพลาด
d512

200

ขณะนี้มีฟังก์ชั่นเฉพาะสำหรับคอนโซลที่:

console.trace()

11
เพียงให้แน่ใจว่าจะฟังความคิดเห็นดังกล่าวข้างต้นconsole.trace()เกี่ยวกับ
Qix - MONICA ถูกยกเลิก

5
โดยค่าเริ่มต้นนี้จะแสดงเพียง 10 เฟรมคุณสามารถใช้อาร์กิวเมนต์บรรทัดคำสั่งเพื่อเพิ่มสิ่งนี้เช่น--stack_trace_limit=200
Michael

ถ้าคุณต้องการส่งออกไปยังไฟล์บันทึก
ยะ

ดูเหมือนจะไม่ทำงานกับคำสัญญาและ async / คอยใช่ไหม?
bluenote10

97

ตามคำตอบแล้วคุณสามารถใช้คำสั่งติดตาม :

console.trace("I am here");

อย่างไรก็ตามหากคุณมาที่คำถามนี้ค้นหาวิธีบันทึกการติดตามสแต็กของข้อยกเว้นคุณสามารถบันทึกวัตถุข้อยกเว้นได้

try {  
  // if something unexpected
  throw new Error("Something unexpected has occurred.");     

} catch (e) {
  console.error(e);
}

มันจะเข้าสู่ระบบ:

ข้อผิดพลาด: มีบางสิ่งที่ไม่คาดคิดเกิดขึ้น
    ที่ main (c: \ Users \ Me \ Documents \ MyApp \ app.js: 9: 15)
    ที่ Object (c: \ Users \ Me \ Documents \ MyApp \ app.js: 17: 1)
    ที่ Module._compile (module.js: 460: 26)
    ที่ Object.Module._extensions..js (module.js: 478: 10 )
    ที่ Module.load (module.js: 355: 32)
    ที่ Function.Module._load (module.js: 310: 12)
    ที่ Function.Module.runMain (โมดูล.js: 501: 10)
    เมื่อเริ่มต้น (node.js) : 129: 16)
    ที่ node.js: 814: 3


หากเวอร์ชัน Node.js ของคุณคือ <มากกว่า 6.0.0การบันทึกวัตถุข้อยกเว้นจะไม่เพียงพอ ในกรณีนี้มันจะพิมพ์เท่านั้น:

[ข้อผิดพลาด: มีบางอย่างไม่คาดคิดเกิดขึ้น]

สำหรับโหนดเวอร์ชัน <6 ให้ใช้console.error(e.stack)แทนconsole.error(e)เพื่อพิมพ์ข้อความแสดงข้อผิดพลาดพร้อมกับสแต็กเต็มรูปแบบเช่นเดียวกับรุ่นโหนดปัจจุบัน


หมายเหตุ:ถ้ายกเว้นจะถูกสร้างขึ้นเป็นสตริงเหมือนthrow "myException"มันเป็นไปไม่ได้ที่จะดึงกองติดตามและเข้าสู่ระบบe.stackอัตราผลตอบแทนไม่ได้กำหนดไม่ได้กำหนด

เพื่อความปลอดภัยคุณสามารถใช้

console.error(e.stack || e);

และมันจะใช้ได้กับ Node.js ทั้งเก่าและใหม่


จะไม่console.error(e)พิมพ์ทุกอย่างในeวัตถุe.stackหรือไม่
drmrbrewer

1
@drmrbrewer ขอบคุณที่ชี้ให้เห็น ดูเหมือนว่าพฤติกรรมมีการเปลี่ยนแปลงระหว่างโหนดเวอร์ชัน 4.x และ 7.x (อาจเป็นการเปลี่ยนแปลง V8) ฉันอัพเดตคำตอบแล้ว
Zanon

1
@drmrbrewer ยืนยันว่าพฤติกรรมนี้เปลี่ยนไปในเวอร์ชัน 6.0.0
Zanon

2
ขอโทษฉันเพิ่งค้นพบสิ่งที่สำคัญ ดูความคิดเห็นของฉันกับการโพสต์ที่เกี่ยวข้อง: stackoverflow.com/questions/42528677/... ดูเหมือนว่าการบันทึกข้อผิดพลาดด้วยตนเองนั้นจะแสดงเนื้อหาทั้งหมดของข้อผิดพลาด แต่การพยายามเชื่อมโยง (เช่นสตริง) กับข้อความอื่นจะทำให้ส่วนข้อความสั้น ๆ เท่านั้นที่จะใช้ ทุกอย่างเข้าท่ากับการรับรู้นั้นมากขึ้น
drmrbrewer

1
คุณประหยัดวันของฉัน)
อเล็กซ์

39

หากต้องการพิมพ์ stacktrace Errorin ในคอนโซลในวิธีที่อ่านได้มากขึ้น:

console.log(ex, ex.stack.split("\n"));

ตัวอย่างผลลัพธ์:

[Error] [ 'Error',
  '    at repl:1:7',
  '    at REPLServer.self.eval (repl.js:110:21)',
  '    at Interface.<anonymous> (repl.js:239:12)',
  '    at Interface.EventEmitter.emit (events.js:95:17)',
  '    at Interface._onLine (readline.js:202:10)',
  '    at Interface._line (readline.js:531:8)',
  '    at Interface._ttyWrite (readline.js:760:14)',
  '    at ReadStream.onkeypress (readline.js:99:10)',
  '    at ReadStream.EventEmitter.emit (events.js:98:17)',
  '    at emitKey (readline.js:1095:12)' ]

9

ด้วยโมดูล Node ที่พร้อมใช้งานเป็นไปได้ที่จะได้รับการติดตามสแต็กแบบเต็มความยาวออกจาก Node (แม้ว่าจะมีโทษเล็กน้อย): http://www.mattinsler.com/post/26396305882/announcing-longjohn-long-stack -traces สำหรับโหนด js


4
เชื่อมโยงตาย แต่อย่างหนึ่งอาจจะอ่านสำเนาที่บันทึกไว้ที่นี่หรือเช็คเอาท์ห้องสมุด
Codebling

5

ลองError.captureStackTrace (Object เป้าหมาย [constructorOpt])

const myObj = {};
function c() {
  // pass
}

function b() {
    Error.captureStackTrace(myObj)
    c()
} 

function a() {
    b()
}

a()

console.log(myObj.stack)

ฟังก์ชั่นaและถูกจับในกองข้อผิดพลาดและเก็บไว้ในbmyObj


2
หากคุณต้องการแสดงข้อผิดพลาดที่จะมีstackคุณสมบัติที่คุณจะต้องเรียกสิ่งนี้ว่าถ้าโหน> = Error.captureStackTrace(error)6:
cjbarth

โปรดทราบว่าหากคุณไม่ต้องการให้เฟรมที่ถูกเรียกError.captureStackTraceให้แสดงในการติดตามสแต็กคุณสามารถข้ามมันได้โดยการผ่านมันเป็นconstructorOptarg
Ullauri

3

สำหรับสิ่งที่ฉันรู้ว่าการพิมพ์การติดตามสแต็กสมบูรณ์ใน nodejs เป็นไปไม่ได้คุณสามารถพิมพ์การติดตามสแต็ก "บางส่วน" คุณไม่สามารถดูได้ว่าคุณมาจากที่ใดในรหัสที่ซึ่งข้อยกเว้นเกิดขึ้น นั่นคือสิ่งที่ Ryan Dahl อธิบายในวิดีโอ youtube นี้ http://youtu.be/jo_B4LTHi3I เวลาขั้นต่ำ 56:30 เพื่อความแม่นยำ หวังว่านี่จะช่วยได้


2
จริง แต่โมดูลใน @ Timboudreau คำตอบของ "แก้ไข" ที่
Bogdan D

3

@isaacsคำตอบถูกต้อง แต่ถ้าคุณต้องการสแต็คข้อผิดพลาดที่เฉพาะเจาะจงมากขึ้นหรือสะอาดกว่าคุณสามารถใช้ฟังก์ชั่นนี้:

function getCleanerStack() {
   var err = new Error();
   Error.captureStackTrace(err, getStack);

   return err.stack;
}

ฟังก์ชั่นนี้เป็นแรงบันดาลใจโดยตรงจากconsole.traceฟังก์ชั่นในNodeJS NodeJS

รหัสที่มา: รุ่นล่าสุดหรือรุ่นเก่า


1
ไม่ทำงานแสดงสแต็กจากบรรทัดปัจจุบันเท่านั้น (ไม่ใช่บรรทัดที่เกิดข้อผิดพลาดนี้) err.stackเป็นคำตอบที่ถูกต้องมากขึ้น
เอียนจง

2

หากคุณต้องการบันทึกการติดตามสแต็กของข้อผิดพลาดเท่านั้น (และไม่ใช่ข้อความแสดงข้อผิดพลาด) โหนด 6 ขึ้นไปจะรวมชื่อข้อผิดพลาดและข้อความภายในการติดตามสแต็กโดยอัตโนมัติซึ่งค่อนข้างน่ารำคาญหากคุณต้องการจัดการข้อผิดพลาดเอง

console.log(error.stack.replace(error.message, ''))

วิธีแก้ปัญหานี้จะบันทึกเฉพาะชื่อข้อผิดพลาดและการติดตามสแต็ก (เพื่อให้คุณสามารถจัดรูปแบบข้อความแสดงข้อผิดพลาดและแสดงวิธีที่คุณต้องการที่อื่นในรหัสของคุณ)

ตัวอย่างด้านบนจะพิมพ์เฉพาะชื่อข้อผิดพลาดตามด้วยการติดตามสแต็กตัวอย่างเช่น:

Error: 
    at /Users/cfisher/Git/squashed/execProcess.js:6:17
    at ChildProcess.exithandler (child_process.js:213:5)
    at emitTwo (events.js:106:13)
    at ChildProcess.emit (events.js:191:7)
    at maybeClose (internal/child_process.js:877:16)
    at Socket.<anonymous> (internal/child_process.js:334:11)
    at emitOne (events.js:96:13)
    at Socket.emit (events.js:188:7)
    at Pipe._handle.close [as _onclose] (net.js:498:12)

แทน:

Error: Error: Command failed: sh ./commands/getBranchCommitCount.sh HEAD
git: 'rev-lists' is not a git command. See 'git --help'.

Did you mean this?
        rev-list

    at /Users/cfisher/Git/squashed/execProcess.js:6:17
    at ChildProcess.exithandler (child_process.js:213:5)
    at emitTwo (events.js:106:13)
    at ChildProcess.emit (events.js:191:7)
    at maybeClose (internal/child_process.js:877:16)
    at Socket.<anonymous> (internal/child_process.js:334:11)
    at emitOne (events.js:96:13)
    at Socket.emit (events.js:188:7)
    at Pipe._handle.close [as _onclose] (net.js:498:12)

1

ในกรณีที่ใครบางคนยังคงมองหาสิ่งนี้เหมือนฉันแล้วก็มีโมดูลที่เราสามารถใช้เรียกว่า "ร่องรอยร่องรอย" เป็นที่นิยมจริงๆลิงก์ NPM

จากนั้นเดินตามรอย

  var stackTrace = require('stack-trace');
  .
  .
  .
  var trace = stackTrace.get();
  trace.map(function (item){ 
    console.log(new Date().toUTCString() + ' : ' +  item.toString() );  
  });

หรือเพียงพิมพ์ร่องรอย:

var stackTrace = require('stack-trace');
.
.
.
var trace = stackTrace.get();
trace.toString();

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.