Skip to content

Commit a52e957

Browse files
committed
fix: Support ES2015+ function definitions in function#toStringTokens()
Additionally revert from problematic regex based implementation. Addresses #21
1 parent 3551cdd commit a52e957

File tree

3 files changed

+54
-13
lines changed

3 files changed

+54
-13
lines changed

function/#/to-string-tokens.js

+46-11
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,52 @@
11
"use strict";
22

3-
var validFunction = require("../valid-function");
3+
var isValue = require("../../object/is-value")
4+
, esniff = require("esniff")
5+
, validFunction = require("../valid-function");
46

5-
var re1 = /^\s*function[\0-')-\uffff]*\(([\0-(*-\uffff]*)\)\s*\{([\0-\uffff]*)\}\s*$/
6-
, re2 = /^\s*\(?([\0-'*-\uffff]*)\)?\s*=>\s*(\{?[\0-\uffff]*\}?)\s*$/;
7+
var classRe = /^\s*class[\s{/}]/;
78

89
module.exports = function () {
9-
var str = String(validFunction(this)), data = str.match(re1);
10-
if (!data) {
11-
data = str.match(re2);
12-
if (!data) throw new Error("Unrecognized string format");
13-
data[1] = data[1].trim();
14-
if (data[2][0] === "{") data[2] = data[2].trim().slice(1, -1);
15-
}
16-
return { args: data[1], body: data[2] };
10+
var str = String(validFunction(this));
11+
if (classRe.test(str)) throw new Error("Class methods are not supported");
12+
13+
var argsStartIndex
14+
, argsEndIndex
15+
, bodyStartIndex
16+
, bodyEndReverseIndex = -1
17+
, shouldTrimArgs = false;
18+
19+
esniff(str, function (emitter, accessor) {
20+
emitter.once("trigger:(", function () { argsStartIndex = accessor.index + 1; });
21+
emitter.once("trigger:=", function () {
22+
if (isValue(argsStartIndex)) return;
23+
argsStartIndex = 0;
24+
argsEndIndex = accessor.index;
25+
shouldTrimArgs = true;
26+
if (!accessor.skipCodePart("=>")) {
27+
throw new Error("Unexpected function string: " + str);
28+
}
29+
accessor.skipWhitespace();
30+
if (!accessor.skipCodePart("{")) bodyEndReverseIndex = Infinity;
31+
bodyStartIndex = accessor.index;
32+
});
33+
emitter.on("trigger:)", function () {
34+
if (accessor.scopeDepth) return;
35+
argsEndIndex = accessor.index;
36+
accessor.skipCodePart(")");
37+
accessor.skipWhitespace();
38+
if (accessor.skipCodePart("=>")) {
39+
accessor.skipWhitespace();
40+
if (!accessor.skipCodePart("{")) bodyEndReverseIndex = Infinity;
41+
} else if (!accessor.skipCodePart("{")) {
42+
throw new Error("Unexpected function string: " + str);
43+
}
44+
bodyStartIndex = accessor.index;
45+
accessor.stop();
46+
});
47+
});
48+
49+
var argsString = str.slice(argsStartIndex, argsEndIndex);
50+
if (shouldTrimArgs) argsString = argsString.trim();
51+
return { args: argsString, body: str.slice(bodyStartIndex, bodyEndReverseIndex) };
1752
};

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@
2525
"dependencies": {
2626
"es6-iterator": "^2.0.3",
2727
"es6-symbol": "^3.1.3",
28+
"esniff": "^2.0.1",
2829
"next-tick": "^1.1.0"
2930
},
3031
"devDependencies": {

test/function/#/to-string-tokens.js

+7-2
Original file line numberDiff line numberDiff line change
@@ -10,10 +10,10 @@ module.exports = function (t, a) {
1010
a.deep(t.call(function () {}), { args: "", body: "" });
1111
// eslint-disable-next-line no-unused-vars
1212
a.deep(t.call(function (raz) {}), { args: "raz", body: "" });
13-
a.deep(t.call(function () { Object(); }), { args: "", body: " Object(); " });
13+
a.deep(t.call(function () { return {}; }), { args: "", body: " return {}; " });
1414

1515
try {
16-
eval("(() => {})");
16+
eval("((foo = bar) => {})");
1717
} catch (e) {
1818
// Non ES2015 env
1919
return;
@@ -23,4 +23,9 @@ module.exports = function (t, a) {
2323
a.deep(t.call(eval("((elo) => foo)")), { args: "elo", body: "foo" });
2424
a.deep(t.call(eval("(elo => foo)")), { args: "elo", body: "foo" });
2525
a.deep(t.call(eval("((elo, bar) => foo())")), { args: "elo, bar", body: "foo()" });
26+
a.deep(t.call(eval("foo=>bar")), { args: "foo", body: "bar" });
27+
a.deep(t.call(eval("(foo = { marko: (elo) => { someFunct(); } }) => { body(); }")), {
28+
args: "foo = { marko: (elo) => { someFunct(); } }",
29+
body: " body(); "
30+
});
2631
};

0 commit comments

Comments
 (0)