Отслеживание исходной строки ошибки в Javascript eval

Я создаю что-то, что включает javascripts на лету асинхронно, и это работает, но я стараюсь улучшить обнаружение ошибок (так что все ошибки не просто появляются из некоторой строки рядом с вызовом AJAX, тянет их вниз.

Если я использую eval для оценки многострочного файла javascript, есть ли способ отслеживать, на какой строке происходит ошибка?

Сохраняя ссылки на переменные, которые мне нужны при включении, у меня нет проблем с определением, в каком файле происходят ошибки. Моя проблема заключается в определении, в какой строке возникает ошибка.

Пример:

try {
  eval("var valid_statement = 7; \n invalid_statement())))");
} catch(e) {
  var err = new Error();
  err.message = 'Error in Evald Script: ' + e.message;
  err.lineNumber = ???
  throw err;
}

Как я могу сказать, что ошибка произошла во второй строке? В частности, я заинтересован в том, чтобы сделать это в Firefox.

Я знаю, что объекты ошибок имеют e.stack в браузерах Mozilla, но вывод, похоже, не правильно учитывает новые строки.

Ответы

Ответ 1

  • Номер строки в объявленном script начинается с того, на котором включен eval.
  • Объект ошибки имеет номер строки строки, в которой он был создан.

Так что-то вроде...

try {
  eval('var valid_statement = 7; \n invalid_statement())))');
} catch(e) {
  var err = e.constructor('Error in Evaled Script: ' + e.message);
  // +3 because `err` has the line number of the `eval` line plus two.
  err.lineNumber = e.lineNumber - err.lineNumber + 3;
  throw err;
}

Ответ 2

Может быть, глобальный event-listener ошибки поймает исключение из eval и отобразит правильные номера строк?

window.addEventListener('error', function(e) {
    console.log(e.message
        , '\n', e.filename, ':', e.lineno, (e.colno ? ':' + e.colno : '')
        , e.error && e.error.stack ? '\n' : '', e.error ? e.error.stack : undefined
    );
}, false);

Ответ 3

Я не думаю, что вы можете сделать это с уверенностью. Однако вы можете сделать это вместо eval:

try {
    $("<script />").html(scriptSource).appendTo("head").remove();
} catch (e) {
    alert(e.lineNumber);
}