最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - Node.js Error Handling -- how to deal with undefined values causing errors - Stack Overflow

programmeradmin1浏览0评论

Take this URL for instance: .xml.aspx?names=Khan

Using xml2js node.js module you may parse that XML, although it does not look pretty:

var CharacterID = response.eveapi.result[0].rowset[0].row[0].$.characterID;

The app crashed after 2 weeks of running, all because rowset[0] was undefined. Prior to that it crashed because eveapi was not defined. Seriously, does my if-else has to be like this just to prevent server from crashing due to stupid undefined object errors?

 if (!response.eveapi || 
     !response.eveapi.result[0] || 
     !response.eveapi.result[0].rowset[0] || 
     !response.eveapi.result[0].rowset[0].row[0]) {
            return res.send(500, "Error");

Besides the obvious if (err) return res.send(500, "Error"); error handling where applicable, what is the general practice for undefined errors?

Take this URL for instance: https://api.eveonline./eve/CharacterID.xml.aspx?names=Khan

Using xml2js node.js module you may parse that XML, although it does not look pretty:

var CharacterID = response.eveapi.result[0].rowset[0].row[0].$.characterID;

The app crashed after 2 weeks of running, all because rowset[0] was undefined. Prior to that it crashed because eveapi was not defined. Seriously, does my if-else has to be like this just to prevent server from crashing due to stupid undefined object errors?

 if (!response.eveapi || 
     !response.eveapi.result[0] || 
     !response.eveapi.result[0].rowset[0] || 
     !response.eveapi.result[0].rowset[0].row[0]) {
            return res.send(500, "Error");

Besides the obvious if (err) return res.send(500, "Error"); error handling where applicable, what is the general practice for undefined errors?

Share Improve this question edited Jun 28, 2013 at 6:23 Paul 27.5k13 gold badges89 silver badges126 bronze badges asked Jun 28, 2013 at 6:08 Sahat YalkabovSahat Yalkabov 33.7k44 gold badges115 silver badges176 bronze badges 4
  • 1 If you learn what the response is, you shouldn't have this problem. You could always create a function that accepts any number of arguments that looks through an object, basically doing what your if statement is, so you don't have to rewrite that long if statement – Ian Commented Jun 28, 2013 at 6:27
  • @Ian since [] can be used to dereference either objects or arrays, that's certainly possible. I think most people just aren't that used to variable argument functions, so if Twilight wants to see that answer maybe you could write an example? – Paul Commented Jun 28, 2013 at 6:36
  • 1 @Paul I don't know if it's a good example, or how universally it can be used, but here's what I got: jsfiddle/dqUng . Hopefully it helps the OP in some way – Ian Commented Jun 28, 2013 at 6:46
  • Yeah that's what I thought you were doing. Still needs to be called within a try/catch to branch off and return error responses. – Paul Commented Jun 28, 2013 at 6:54
Add a ment  | 

3 Answers 3

Reset to default 4

I wrote a library for this kind of thing, called dotty (https://github./deoxxa/dotty).

In your case, you could do this:

var dotty = require("dotty");

var CharacterID = dotty.get(response, "eveapi.result.0.rowset.0.row.0.$.characterID");

In the case of the path not being resolvable, it'll just return undefined.

As you've discovered, undefined is not itself an error, but using undefined as an array/object is an error.

x = {'a': { 'b': { 'c': { 'd': [1,2,3,4,5]} } } } ;
try { j = x.a.b.c.e[3] } catch(e) { console.log(e); }

prints

[TypeError: Cannot read property '3' of undefined]

This suggests to me that try/catch can be used with your code to return an error code, and if desired, an error text (or just stick the error text in console.log, a database or local file).

In your case, that could look like:

var CharacterID; // can't define it yet

try {
  CharacterID = response.eveapi.result[0].rowset[0].row[0].$.characterID;
} catch(e) {
// send description on the line with error
  return res.send(500, "Error: NodeJS assigning CharacterID: "+e); 
// return res.send(500, "error");  use this one if you dont want to reveal reason for errors
}

// code here can assume CharacterID evaluated.  It might still be undefined, though.

Maybe this function helps?

function tryPath(obj, path) {
    path = path.split(/[.,]/);
    while (path.length && obj) {
        obj = obj[path.shift()];
    }
    return obj || null;
}

For your code you'd use:

if (tryPath(response,'eveapi.result.0.rows.0.row.0') === null) {
  return res.send(500, "Error");
}

jsFiddle example
jsFiddle same example, but as extension to Object.prototype

发布评论

评论列表(0)

  1. 暂无评论