Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

'caller' and 'arguments' are restricted function properties and cannot be accessed in this context

Tags:

I am trying to create a simple debugging function that simply shows the caller of a function, like this:

function xe() {
  console.log(xe.caller().name)
}

With this I would just be able to add xe() to a function and it will log the calls to the function– just a short, simple addition to help with debugging. Debugging sugar, so to speak.

Unfortunately I get the error from the subject-line:

TypeError: 'caller' and 'arguments' are restricted function properties and cannot be accessed in this context.

I am using Babel/ES6, which injects "use strict" at the top of every module. This may be the cause, but searching has yielded limited information on why the error is raised, and I would like to understand it better.

If strict mode is the problem I would prefer not to disable strict mode for the entire project– just for the module/function.

like image 783
Brian M. Hunt Avatar asked Aug 10 '15 13:08

Brian M. Hunt


2 Answers

It is the cause. From MDN:

in strict mode it's no longer possible to "walk" the JavaScript stack via commonly-implemented extensions to ECMAScript. In normal code with these extensions, when a function fun is in the middle of being called, fun.caller is the function that most recently called fun, and fun.arguments are the arguments for that invocation of fun. Both extensions are problematic for "secure" JavaScript because they allow "secured" code to access "privileged" functions and their (potentially unsecured) arguments. If fun is in strict mode, both fun.caller and fun.arguments are non-deletable properties which throw when set or retrieved:

If you're doing ES6, you can't in the general case disable strict mode. It's implicit during certain conditions such as when in an ES6 module.

If you're just debugging, I'd suggest just using a break point in a debugger and checking the stack frame, but I'm sure you know that already.

If you're just outputting debugging information you could also, I suppose just read the stack of an Error object:

console.log(new Error().stack);

You can globaly disable (but I realize this isn't what you want) use strict with babel Using either:

require("6to5").transform("code", { blacklist: ["useStrict"] });

or

$ 6to5 --blacklist useStrict

If you must strip it out on a module level, I suspect you'll have to do it yourself. Basic string replace perhaps?

Additionally, as has been pointed out in ES5. It should be xe.caller.name and not xe.caller().name or you would re-invoke the function.

like image 115
Kit Sunde Avatar answered Sep 25 '22 10:09

Kit Sunde


As per this documentation. The Function.caller() property returns the function that invoked the specified function. Simply you will get whole caller function when you use xe.caller. Again you are executing caller function. Here you are doing recursion and that is the reason it is not allowing in strict mode.

You can execute your sample function in browser console. you will get Maximum call stack size exceeded error.

like image 29
Laxmikant Dange Avatar answered Sep 25 '22 10:09

Laxmikant Dange