Problems with SyntaxError construction
A. Sundararajan
sundararajan.athijegannathan at oracle.com
Fri Feb 20 11:48:29 UTC 2015
Unlike TypeError or ReferenceError or any other execution error from
eval'ed code, SyntaxError is thrown "up front". It is an "early error"
as defined in "Clause 16" of ES 5.1 spec (https://es5.github.io/#x16).
If we set evaluated code's file name and line number of syntax error,
that would seem to imply that the execution reached that line and
resulted in error. But, that is not the case here - execution of eval'ed
code is yet to start. Error was detected during "compilation" and thrown
from the "eval line" itself. Also, if there are nested eval calls and
deepest one resulted in SyntaxError, we want to know which eval
attempted to eval code with the syntax error.
The "message" property of SyntaxError does contain the file and line
(like file:N) of the eval-ed code that caused syntax error.
-Sundar
On Friday 20 February 2015 03:11 PM, Tim Fox wrote:
> HI Sundar -
>
> On 20/02/15 08:09, A. Sundararajan wrote:
>> * SyntaxError constructor accepting extra arguments apart from
>> message is not as per ECMAScript standard. That one is
>> Firefox/Mozilla specific extension. We try to *add* properties,
>> functions for extension - try not to modify/extend standard defined
>> functions, constructors.
>>
>> * User can fileName, lineNumber and columnNumber of constructed
>> SyntaxError instance.
>>
>> * On the java side, the NashornException and ScriptException's file,
>> line, column use the actual throw point from the script source. While
>> this can read the values from SyntaxError or any Error for that
>> matter, there is a problem. fileName, lineNumber, columnNumber could
>> be set to any value in script. These have to be converted to String,
>> int, int respectively. Conversion means calling [[DefaultValue]] with
>> String, Number hint - which is calling out to script code. This
>> process can itself throw exception!! All this may destroy info. on
>> where exactly the underlying exception was thrown.
>>
>> * There is clear workaround. Script code can always read SyntaxError
>> instance's fileName, lineNumber etc. - which can be set to anything
>> post construction. Java code can get underling ECMA error instance
>> via NashornException.getEcmaError - which would return a
>> ScriptObjectMirror. Java code can call
>> ScriptObjectMirror.getMember("fileName") etc. to retrieve properties
>> set by user's script.
>
> I've tried what you suggest, it seems to work (ish), but I end up with
> seriously grungy code that relies on casting and non public APIs that
> (I guess) could be changed without warning in any release, so I don't
> think I'd be happy doing this in Vert.x. Here's an example:
>
> (Yikes!)
>
> https://gist.github.com/purplefox/003fae1e32fd44368ffa
>
> So.... rewinding a bit here. The only reason I find myself shaving
> this particular Yak is because syntax errors thrown from evals()
> aren't reported by Nashorn with the fileName and lineNumber of the
> _eval_, instead they are reported with the fileName/lineNumber of
> where the eval was executed.
>
> I mentioned this in a previous post, but here is the gist again that
> demonstrates the issue:
>
> https://gist.github.com/purplefox/7ba0d87d2d1670b871a4
>
> This means I have to manually catch SyntaxErrors in our require()
> implementation. Fill in the correct fileName, lineNumber etc and
> rethrow as new SyntaxError instances. But, as we discussed in this
> thread, that doesn't seem to work easily as I can't set fileName,
> lineNumber etc when constructing an instance in Nashorn, and if I set
> it afterwards those values aren't visible in Java (at least not in a
> portable way that doesn't rely on hacky reliance on non public APIs).
> If I don't do this, our users see a pretty useless error message when
> there's a syntax error in their module and it all leads to a very poor
> user experience.
>
> Imho, a better way to solve this would be for Nashorn to create syntax
> error instances when doing evals() that properly reflect the
> lineNumber and fileName of the _eval_ not of the place the eval was
> executed. This seems a far more intuitive way to do it and it would be
> consistent with the way other Error types are thrown from code in
> evals (e.g. TypeError, RangeError etc) which *do* reflect the correct
> line in the eval where the error occurred. Fundamentally a user is not
> going to care about the lineNumber where the eval was executed, they
> want to know where in the actual code the syntax error is - that's the
> useful information which is currently being obscured.
>
> So, any chance of changing/fixing the current behaviour? Doing that
> would mean I wouldn't have to little our code with piles of Yak hair
> trying to get the important information out of private classes, and I
> think it would lead to a better and more intuitive user experience :)
>
>
>
>
>
>>
>> Hope this helps,
>> -Sundar
>>
>> On Wednesday 18 February 2015 08:29 PM, Tim Fox wrote:
>>> Hi all,
>>>
>>> I've started a new thread for this, so as not to clutter the
>>> previous thread on exceptions.
>>>
>>> I've come across a possible issue with creating SyntaxError objects
>>> (and potentially other exceptions although I haven't tested them).
>>>
>>> https://gist.github.com/purplefox/dfe30a38098c1708e24b
>>>
>>> In the above gist I create a SyntaxError specifying message,
>>> fileName, and lineNumber.
>>>
>>> However fileName and lineNumber appear to be ignored, and instead
>>> the SyntaxError fileName and lineNumber are set to the fileName of
>>> the file where it was created and the lineNumber of the creation.
>>>
>>> I'm not sure if I'm doing something wrong, but according to
>>> https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/SyntaxError
>>> I should be able to create SyntaxError objects like this.
>>>
>>> There appears to be a further issue. If I then throw the syntax
>>> error from the JS script, and catch it in Java like this:
>>>
>>> try {
>>> engine.eval(script);
>>> } catch (ScriptException e) {
>>> System.out.println("message:" + e.getMessage());
>>> System.out.println("fileName:" + e.getFileName());
>>> System.out.println("lineNumber:" + e.getLineNumber());
>>> }
>>>
>>> Then the message, fileName and lineNumber always correspond to place
>>> where the object was created, not to the subsequently set values.
>>>
>>> In other words, I can't figure out a way of creating a SyntaxError
>>> object in JS with my own values of fileName and lineNumber and have
>>> those values available in Java when caught.
>>>
>>> Any ideas from the experts? :)
>>>
>>>
>>
>
More information about the nashorn-dev
mailing list