mirror of https://github.com/pulumi/pulumi.git
4147 lines
96 KiB
TypeScript
4147 lines
96 KiB
TypeScript
// Copyright 2016-2018, Pulumi Corporation.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
// tslint:disable
|
|
|
|
import * as assert from "assert";
|
|
import { EOL } from "os";
|
|
import { runtime } from "../../index";
|
|
import * as resource from "../../resource";
|
|
import { Output, output } from "../../resource";
|
|
import { assertAsyncThrows, asyncTest } from "../util";
|
|
|
|
interface ClosureCase {
|
|
pre?: () => void; // an optional function to run before this case.
|
|
title: string; // a title banner for the test case.
|
|
func: Function; // the function whose body and closure to serialize.
|
|
expectText: string | undefined; // optionally also validate the serialization to JavaScript text.
|
|
error?: string; // error message we expect to be thrown if we are unable to serialize closure.
|
|
afters?: ClosureCase[]; // an optional list of test cases to run afterwards.
|
|
}
|
|
|
|
export const exportedValue = 42;
|
|
|
|
// This group of tests ensure that we serialize closures properly.
|
|
describe("closure", () => {
|
|
const cases: ClosureCase[] = [];
|
|
|
|
cases.push({
|
|
title: "Empty function closure",
|
|
func: function () { },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function () { };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
cases.push({
|
|
title: "Empty named function",
|
|
func: function f() { },
|
|
expectText: `exports.handler = __f;
|
|
|
|
function __f() {
|
|
return (function() {
|
|
with({ f: __f }) {
|
|
|
|
return function /*f*/() { };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
cases.push({
|
|
title: "Named function with self-reference",
|
|
func: function f() { f(); },
|
|
expectText: `exports.handler = __f;
|
|
|
|
function __f() {
|
|
return (function() {
|
|
with({ f: __f }) {
|
|
|
|
return function /*f*/() { f(); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
cases.push({
|
|
title: "Function closure with this capture",
|
|
func: function () { console.log(this); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function () { console.log(this); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
cases.push({
|
|
title: "Function closure with this and arguments capture",
|
|
func: function () { console.log(this + arguments); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function () { console.log(this + arguments); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
cases.push({
|
|
title: "Empty arrow closure",
|
|
func: () => { },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return () => { };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
cases.push({
|
|
title: "Arrow closure with this capture",
|
|
func: () => { console.log(this); },
|
|
expectText: undefined,
|
|
error:
|
|
`Error serializing function 'func': closure.spec.js(0,0)
|
|
|
|
function 'func': closure.spec.js(0,0): which could not be serialized because
|
|
arrow function captured 'this'. Assign 'this' to another name outside function and capture that.
|
|
|
|
Function code:
|
|
() => { console.log(this); }
|
|
`,
|
|
});
|
|
|
|
const awaiterCode =
|
|
`
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function (thisArg, _arguments, P, generator) {
|
|
return new (P || (P = Promise))(function (resolve, reject) {
|
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
|
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
});
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`;
|
|
|
|
cases.push({
|
|
title: "Async lambda that does not capture this",
|
|
func: async () => { },
|
|
expectText: `exports.handler = __f0;
|
|
${awaiterCode}
|
|
function __f0() {
|
|
return (function() {
|
|
with({ __awaiter: __f1 }) {
|
|
|
|
return () => __awaiter(this, void 0, void 0, function* () { });
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
cases.push({
|
|
title: "Async lambda that does capture this",
|
|
func: async () => { console.log(this); },
|
|
expectText: undefined,
|
|
error: `Error serializing function 'func': closure.spec.js(0,0)
|
|
|
|
function 'func': closure.spec.js(0,0): which could not be serialized because
|
|
arrow function captured 'this'. Assign 'this' to another name outside function and capture that.
|
|
|
|
Function code:
|
|
() => __awaiter(this, void 0, void 0, function* () { console.log(this); })
|
|
`,
|
|
});
|
|
|
|
cases.push({
|
|
title: "Async function that does not capture this",
|
|
func: async function() { },
|
|
expectText: `exports.handler = __f0;
|
|
${awaiterCode}
|
|
function __f0() {
|
|
return (function() {
|
|
with({ __awaiter: __f1 }) {
|
|
|
|
return function () {
|
|
return __awaiter(this, void 0, void 0, function* () { });
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
cases.push({
|
|
title: "Async function that does capture this",
|
|
func: async function () { console.log(this); },
|
|
expectText: `exports.handler = __f0;
|
|
${awaiterCode}
|
|
function __f0() {
|
|
return (function() {
|
|
with({ __awaiter: __f1 }) {
|
|
|
|
return function () {
|
|
return __awaiter(this, void 0, void 0, function* () { console.log(this); });
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
cases.push({
|
|
title: "Arrow closure with this and arguments capture",
|
|
func: (function() { return () => { console.log(this + arguments); } }).apply(this, [0, 1]),
|
|
expectText: undefined,
|
|
error: `Error serializing function '<anonymous>': closure.spec.js(0,0)
|
|
|
|
function '<anonymous>': closure.spec.js(0,0): which could not be serialized because
|
|
arrow function captured 'this'. Assign 'this' to another name outside function and capture that.
|
|
|
|
Function code:
|
|
() => { console.log(this + arguments); }
|
|
`,
|
|
});
|
|
|
|
cases.push({
|
|
title: "Arrow closure with this capture inside function closure",
|
|
func: function () { () => { console.log(this); } },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function () { () => { console.log(this); }; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
cases.push({
|
|
title: "Arrow closure with this and arguments capture inside function closure",
|
|
func: function () { () => { console.log(this + arguments); } },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function () { () => { console.log(this + arguments); }; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
{
|
|
class Task {
|
|
run: any;
|
|
constructor() {
|
|
this.run = async function() { };
|
|
}
|
|
}
|
|
|
|
const task = new Task();
|
|
|
|
cases.push({
|
|
title: "Invocation of async function that does not capture this #1",
|
|
func: async function() { await task.run(); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __task = {run: __f2};
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function (thisArg, _arguments, P, generator) {
|
|
return new (P || (P = Promise))(function (resolve, reject) {
|
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
|
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
});
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ __awaiter: __f1 }) {
|
|
|
|
return function () {
|
|
return __awaiter(this, void 0, void 0, function* () { });
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ __awaiter: __f1, task: __task }) {
|
|
|
|
return function () {
|
|
return __awaiter(this, void 0, void 0, function* () { yield task.run(); });
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class Task {
|
|
run: any;
|
|
constructor() {
|
|
this.run = async function() { console.log(this); };
|
|
}
|
|
}
|
|
|
|
const task = new Task();
|
|
|
|
cases.push({
|
|
title: "Invocation of async function that does capture this #1",
|
|
func: async function() { await task.run(); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __task_proto = {};
|
|
Object.defineProperty(__task_proto, "constructor", { configurable: true, writable: true, value: __f2 });
|
|
var __task = Object.create(__task_proto);
|
|
__task.run = __f3;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function (thisArg, _arguments, P, generator) {
|
|
return new (P || (P = Promise))(function (resolve, reject) {
|
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
|
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
});
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ __awaiter: __f1 }) {
|
|
|
|
return function /*constructor*/() {
|
|
this.run = function () {
|
|
return __awaiter(this, void 0, void 0, function* () { console.log(this); });
|
|
};
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ __awaiter: __f1 }) {
|
|
|
|
return function () {
|
|
return __awaiter(this, void 0, void 0, function* () { console.log(this); });
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ __awaiter: __f1, task: __task }) {
|
|
|
|
return function () {
|
|
return __awaiter(this, void 0, void 0, function* () { yield task.run(); });
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class Task {
|
|
run: any;
|
|
constructor() {
|
|
this.run = async () => { };
|
|
}
|
|
}
|
|
|
|
const task = new Task();
|
|
|
|
cases.push({
|
|
title: "Invocation of async lambda that does not capture this #1",
|
|
func: async function() { await task.run(); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __task = {run: __f2};
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function (thisArg, _arguments, P, generator) {
|
|
return new (P || (P = Promise))(function (resolve, reject) {
|
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
|
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
});
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ __awaiter: __f1 }) {
|
|
|
|
return () => __awaiter(this, void 0, void 0, function* () { });
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ __awaiter: __f1, task: __task }) {
|
|
|
|
return function () {
|
|
return __awaiter(this, void 0, void 0, function* () { yield task.run(); });
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class Task {
|
|
run: any;
|
|
constructor() {
|
|
this.run = async () => { console.log(this); };
|
|
}
|
|
}
|
|
|
|
const task = new Task();
|
|
|
|
cases.push({
|
|
title: "Invocation of async lambda that capture this #1",
|
|
func: async function() { await task.run(); },
|
|
expectText: undefined,
|
|
error: `Error serializing function 'func': closure.spec.js(0,0)
|
|
|
|
function 'func': closure.spec.js(0,0): captured
|
|
variable 'task' which indirectly referenced
|
|
function '<anonymous>': closure.spec.js(0,0): which could not be serialized because
|
|
arrow function captured 'this'. Assign 'this' to another name outside function and capture that.
|
|
|
|
Function code:
|
|
() => __awaiter(this, void 0, void 0, function* () { console.log(this); })
|
|
`,
|
|
});
|
|
}
|
|
|
|
cases.push({
|
|
title: "Empty function closure w/ args",
|
|
func: function (x: any, y: any, z: any) { },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function (x, y, z) { };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
cases.push({
|
|
title: "Empty arrow closure w/ args",
|
|
func: (x: any, y: any, z: any) => { },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return (x, y, z) => { };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
// Serialize captures.
|
|
cases.push({
|
|
title: "Doesn't serialize global captures",
|
|
func: () => { console.log("Just a global object reference"); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return () => { console.log("Just a global object reference"); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
{
|
|
const wcap = "foo";
|
|
const xcap = 97;
|
|
const ycap = [ true, -1, "yup" ];
|
|
const zcap = {
|
|
a: "a",
|
|
b: false,
|
|
c: [ 0 ],
|
|
};
|
|
cases.push({
|
|
title: "Serializes basic captures",
|
|
func: () => { console.log(wcap + `${xcap}` + ycap.length + eval(zcap.a + zcap.b + zcap.c)); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __ycap = [true, -1, "yup"];
|
|
var __zcap = {};
|
|
__zcap.a = "a";
|
|
__zcap.b = false;
|
|
var __zcap_c = [0];
|
|
__zcap.c = __zcap_c;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ wcap: "foo", xcap: 97, ycap: __ycap, zcap: __zcap }) {
|
|
|
|
return () => { console.log(wcap + \`\${xcap}\` + ycap.length + eval(zcap.a + zcap.b + zcap.c)); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
{
|
|
let nocap1 = 1, nocap2 = 2, nocap3 = 3, nocap4 = 4, nocap5 = 5, nocap6 = 6, nocap7 = 7;
|
|
let nocap8 = 8, nocap9 = 9, nocap10 = 10;
|
|
let cap1 = 100, cap2 = 200, cap3 = 300, cap4 = 400, cap5 = 500, cap6 = 600, cap7 = 700;
|
|
let cap8 = 800;
|
|
|
|
const functext = `(nocap1, nocap2) => {
|
|
let zz = nocap1 + nocap2; // not a capture: args
|
|
let yy = nocap3; // not a capture: var later on
|
|
if (zz) {
|
|
zz += cap1; // true capture
|
|
let cap1 = 9; // because let is properly scoped
|
|
zz += nocap4; // not a capture
|
|
var nocap4 = 7; // because var is function scoped
|
|
zz += cap2; // true capture
|
|
const cap2 = 33;
|
|
var nocap3 = 8; // block the above capture
|
|
}
|
|
let f1 = (nocap5) => {
|
|
yy += nocap5; // not a capture: args
|
|
cap3++; // capture
|
|
};
|
|
let f2 = (function (nocap6) {
|
|
zz += nocap6; // not a capture: args
|
|
if (cap4) { // capture
|
|
yy = 0;
|
|
}
|
|
});
|
|
let www = nocap7(); // not a capture; it is defined below
|
|
if (true) {
|
|
function nocap7() {
|
|
}
|
|
}
|
|
let [{t: [nocap8]},,nocap9 = "hello",...nocap10] = [{t: [true]},null,undefined,1,2];
|
|
let vvv = [nocap8, nocap9, nocap10]; // not a capture; declarations from destructuring
|
|
let aaa = { // captures in property and method declarations
|
|
[cap5]: cap6,
|
|
[cap7]() {
|
|
cap8
|
|
}
|
|
}
|
|
}`;
|
|
cases.push({
|
|
title: "Doesn't serialize non-free variables (but retains frees)",
|
|
func: eval(functext),
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ cap1: 100, cap2: 200, cap3: 300, cap4: 400, cap5: 500, cap6: 600, cap7: 700, cap8: 800 }) {
|
|
|
|
return (nocap1, nocap2) => {
|
|
let zz = nocap1 + nocap2; // not a capture: args
|
|
let yy = nocap3; // not a capture: var later on
|
|
if (zz) {
|
|
zz += cap1; // true capture
|
|
let cap1 = 9; // because let is properly scoped
|
|
zz += nocap4; // not a capture
|
|
var nocap4 = 7; // because var is function scoped
|
|
zz += cap2; // true capture
|
|
const cap2 = 33;
|
|
var nocap3 = 8; // block the above capture
|
|
}
|
|
let f1 = (nocap5) => {
|
|
yy += nocap5; // not a capture: args
|
|
cap3++; // capture
|
|
};
|
|
let f2 = (function (nocap6) {
|
|
zz += nocap6; // not a capture: args
|
|
if (cap4) { // capture
|
|
yy = 0;
|
|
}
|
|
});
|
|
let www = nocap7(); // not a capture; it is defined below
|
|
if (true) {
|
|
function nocap7() {
|
|
}
|
|
}
|
|
let [{t: [nocap8]},,nocap9 = "hello",...nocap10] = [{t: [true]},null,undefined,1,2];
|
|
let vvv = [nocap8, nocap9, nocap10]; // not a capture; declarations from destructuring
|
|
let aaa = { // captures in property and method declarations
|
|
[cap5]: cap6,
|
|
[cap7]() {
|
|
cap8
|
|
}
|
|
}
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
{
|
|
let nocap1 = 1;
|
|
let cap1 = 100;
|
|
|
|
cases.push({
|
|
title: "Complex capturing cases #1",
|
|
func: () => {
|
|
// cap1 is captured here.
|
|
// nocap1 introduces a new variable that shadows the outer one.
|
|
let [nocap1 = cap1] = [];
|
|
console.log(nocap1);
|
|
},
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ cap1: 100 }) {
|
|
|
|
return () => {
|
|
// cap1 is captured here.
|
|
// nocap1 introduces a new variable that shadows the outer one.
|
|
let [nocap1 = cap1] = [];
|
|
console.log(nocap1);
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
{
|
|
let nocap1 = 1;
|
|
let cap1 = 100;
|
|
|
|
cases.push({
|
|
title: "Complex capturing cases #2",
|
|
func: () => {
|
|
// cap1 is captured here.
|
|
// nocap1 introduces a new variable that shadows the outer one.
|
|
let {nocap1 = cap1} = {};
|
|
console.log(nocap1);
|
|
},
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ cap1: 100 }) {
|
|
|
|
return () => {
|
|
// cap1 is captured here.
|
|
// nocap1 introduces a new variable that shadows the outer one.
|
|
let { nocap1 = cap1 } = {};
|
|
console.log(nocap1);
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
{
|
|
let nocap1 = 1;
|
|
let cap1 = 100;
|
|
|
|
cases.push({
|
|
title: "Complex capturing cases #3",
|
|
func: () => {
|
|
// cap1 is captured here.
|
|
// nocap1 introduces a new variable that shadows the outer one.
|
|
let {x: nocap1 = cap1} = {};
|
|
console.log(nocap1);
|
|
},
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ cap1: 100 }) {
|
|
|
|
return () => {
|
|
// cap1 is captured here.
|
|
// nocap1 introduces a new variable that shadows the outer one.
|
|
let { x: nocap1 = cap1 } = {};
|
|
console.log(nocap1);
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
cases.push({
|
|
title: "Don't capture built-ins",
|
|
func: () => { let x: any = eval("undefined + null + NaN + Infinity + __filename"); require("os"); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return () => { let x = eval("undefined + null + NaN + Infinity + __filename"); require("os"); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
{
|
|
const os = require("os");
|
|
|
|
cases.push({
|
|
title: "Fail to capture built-in modules due to native functions",
|
|
func: () => os,
|
|
expectText: undefined,
|
|
error:
|
|
`Error serializing '() => os': closure.spec.js(0,0)
|
|
|
|
'() => os': closure.spec.js(0,0): captured
|
|
module 'os' which indirectly referenced
|
|
(...)
|
|
it was a native code function.
|
|
|
|
Function code:
|
|
function (...)() { [native code] }
|
|
|
|
Capturing modules can sometimes cause problems.
|
|
Consider using import('os') or require('os') inside '() => os': closure.spec.js(0,0)`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const os = require("os");
|
|
|
|
cases.push({
|
|
title: "Wrapped lambda function",
|
|
func: (a: any,
|
|
b: any,
|
|
c: any) => {
|
|
const v =
|
|
os;
|
|
return { v };
|
|
},
|
|
expectText: undefined,
|
|
error: `Error serializing '(a, b, c) => { const v = os; ...': closure.spec.js(0,0)
|
|
|
|
'(a, b, c) => { const v = os; ...': closure.spec.js(0,0): captured
|
|
module 'os' which indirectly referenced
|
|
(...)
|
|
it was a native code function.
|
|
|
|
Function code:
|
|
function (...)() { [native code] }
|
|
|
|
Capturing modules can sometimes cause problems.
|
|
Consider using import('os') or require('os') inside '(a, b, c) => { const v = os; ...': closure.spec.js(0,0)`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const os = require("os");
|
|
function wrap(handler: Function) {
|
|
return () => handler;
|
|
}
|
|
|
|
const func = wrap(() => os);
|
|
|
|
cases.push({
|
|
title: "Fail to capture module through indirect function references",
|
|
func: func,
|
|
expectText: undefined,
|
|
error:
|
|
`Error serializing '() => handler': closure.spec.js(0,0)
|
|
|
|
'() => handler': closure.spec.js(0,0): captured
|
|
'handler', a function defined at
|
|
'() => os': closure.spec.js(0,0): which captured
|
|
module 'os' which indirectly referenced
|
|
(...)
|
|
it was a native code function.
|
|
|
|
Function code:
|
|
function (...)() { [native code] }
|
|
|
|
Capturing modules can sometimes cause problems.
|
|
Consider using import('os') or require('os') inside '() => os': closure.spec.js(0,0)`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const util = require("../util");
|
|
cases.push({
|
|
title: "Fail to capture user-defined modules due to native functions",
|
|
func: () => util,
|
|
expectText: undefined,
|
|
error:
|
|
`Error serializing '() => util': closure.spec.js(0,0)
|
|
|
|
'() => util': closure.spec.js(0,0): captured
|
|
module './bin/tests/util.js' which indirectly referenced
|
|
function 'assertAsyncThrows': util.js(0,0): which captured
|
|
module 'assert' which indirectly referenced
|
|
(...)
|
|
it was a native code function.
|
|
|
|
Function code:
|
|
function (...)() { [native code] }
|
|
|
|
Capturing modules can sometimes cause problems.
|
|
Consider using import('./bin/tests/util.js') or require('./bin/tests/util.js') inside '() => util': closure.spec.js(0,0)`,
|
|
});
|
|
}
|
|
|
|
cases.push({
|
|
title: "Don't capture catch variables",
|
|
func: () => { try { } catch (err) { console.log(err); } },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return () => { try { }
|
|
catch (err) {
|
|
console.log(err);
|
|
} };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
// Recursive function serialization.
|
|
{
|
|
const fff = "fff!";
|
|
const ggg = "ggg!";
|
|
const xcap = {
|
|
fff: function () { console.log(fff); },
|
|
ggg: () => { console.log(ggg); },
|
|
zzz: {
|
|
a: [ (a1: any, a2: any) => { console.log(a1 + a2); } ],
|
|
},
|
|
};
|
|
const func = () => {
|
|
xcap.fff();
|
|
xcap.ggg();
|
|
xcap.zzz.a[0]("x", "y");
|
|
};
|
|
|
|
cases.push({
|
|
title: "Serializes recursive function captures",
|
|
func: func,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __xcap = {};
|
|
__xcap.fff = __f1;
|
|
__xcap.ggg = __f2;
|
|
var __xcap_zzz = {};
|
|
var __xcap_zzz_a = [__f3];
|
|
__xcap_zzz.a = __xcap_zzz_a;
|
|
__xcap.zzz = __xcap_zzz;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ fff: "fff!" }) {
|
|
|
|
return function () { console.log(fff); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ ggg: "ggg!" }) {
|
|
|
|
return () => { console.log(ggg); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return (a1, a2) => { console.log(a1 + a2); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ xcap: __xcap }) {
|
|
|
|
return () => {
|
|
xcap.fff();
|
|
xcap.ggg();
|
|
xcap.zzz.a[0]("x", "y");
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class CapCap {
|
|
constructor() {
|
|
(<any>this).x = 42;
|
|
(<any>this).f = () => { console.log((<any>this).x); };
|
|
}
|
|
}
|
|
|
|
const cap: any = new CapCap();
|
|
|
|
cases.push({
|
|
title: "Serializes `this` capturing arrow functions",
|
|
func: cap.f,
|
|
expectText: undefined,
|
|
error: `Error serializing function '<anonymous>': closure.spec.js(0,0)
|
|
|
|
function '<anonymous>': closure.spec.js(0,0): which could not be serialized because
|
|
arrow function captured 'this'. Assign 'this' to another name outside function and capture that.
|
|
|
|
Function code:
|
|
() => { console.log(this.x); }
|
|
`,
|
|
});
|
|
}
|
|
|
|
cases.push({
|
|
title: "Don't serialize `this` in function expressions",
|
|
func: function() { return this; },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function () { return this; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
|
|
{
|
|
const mutable: any = {};
|
|
cases.push({
|
|
title: "Serialize mutable objects by value at the time of capture (pre-mutation)",
|
|
func: function() { return mutable; },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __mutable = {};
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ mutable: __mutable }) {
|
|
|
|
return function () { return mutable; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
afters: [{
|
|
pre: () => { mutable.timesTheyAreAChangin = true; },
|
|
title: "Serialize mutable objects by value at the time of capture (post-mutation)",
|
|
func: function() { return mutable; },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __mutable = {timesTheyAreAChangin: true};
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ mutable: __mutable }) {
|
|
|
|
return function () { return mutable; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
}],
|
|
});
|
|
}
|
|
|
|
{
|
|
const v = { d: output(4) };
|
|
cases.push({
|
|
title: "Output capture",
|
|
func: function () { console.log(v); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __v = {};
|
|
var __v_d_proto = {};
|
|
__f1.prototype = __v_d_proto;
|
|
Object.defineProperty(__v_d_proto, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
Object.defineProperty(__v_d_proto, "apply", { configurable: true, writable: true, value: __f2 });
|
|
Object.defineProperty(__v_d_proto, "get", { configurable: true, writable: true, value: __f3 });
|
|
var __v_d = Object.create(__v_d_proto);
|
|
__v_d.value = 4;
|
|
__v.d = __v_d;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/(value) {
|
|
this.value = value;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*apply*/(func) {
|
|
throw new Error("'apply' is not allowed from inside a cloud-callback. Use 'get' to retrieve the value of this Output directly.");
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*get*/() {
|
|
return this.value;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ v: __v }) {
|
|
|
|
return function () { console.log(v); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const v = {
|
|
d1: output(4),
|
|
d2: output("str"),
|
|
d3: output(undefined),
|
|
d4: output({ a: 1, b: true }),
|
|
};
|
|
cases.push({
|
|
title: "Multiple output capture",
|
|
func: function () { console.log(v); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __v = {};
|
|
var __v_d1_proto = {};
|
|
__f1.prototype = __v_d1_proto;
|
|
Object.defineProperty(__v_d1_proto, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
Object.defineProperty(__v_d1_proto, "apply", { configurable: true, writable: true, value: __f2 });
|
|
Object.defineProperty(__v_d1_proto, "get", { configurable: true, writable: true, value: __f3 });
|
|
var __v_d1 = Object.create(__v_d1_proto);
|
|
__v_d1.value = 4;
|
|
__v.d1 = __v_d1;
|
|
var __v_d2 = Object.create(__v_d1_proto);
|
|
__v_d2.value = "str";
|
|
__v.d2 = __v_d2;
|
|
var __v_d3 = Object.create(__v_d1_proto);
|
|
__v_d3.value = undefined;
|
|
__v.d3 = __v_d3;
|
|
var __v_d4 = Object.create(__v_d1_proto);
|
|
var __v_d4_value = {a: 1, b: true};
|
|
__v_d4.value = __v_d4_value;
|
|
__v.d4 = __v_d4;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/(value) {
|
|
this.value = value;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*apply*/(func) {
|
|
throw new Error("'apply' is not allowed from inside a cloud-callback. Use 'get' to retrieve the value of this Output directly.");
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*get*/() {
|
|
return this.value;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ v: __v }) {
|
|
|
|
return function () { console.log(v); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const v = {
|
|
d1: output(4),
|
|
d2: <any>undefined,
|
|
};
|
|
|
|
v.d2 = output({ a: 1, b: v });
|
|
|
|
cases.push({
|
|
title: "Recursive output capture",
|
|
func: function () { console.log(v); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __v = {};
|
|
var __v_d1_proto = {};
|
|
__f1.prototype = __v_d1_proto;
|
|
Object.defineProperty(__v_d1_proto, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
Object.defineProperty(__v_d1_proto, "apply", { configurable: true, writable: true, value: __f2 });
|
|
Object.defineProperty(__v_d1_proto, "get", { configurable: true, writable: true, value: __f3 });
|
|
var __v_d1 = Object.create(__v_d1_proto);
|
|
__v_d1.value = 4;
|
|
__v.d1 = __v_d1;
|
|
var __v_d2 = Object.create(__v_d1_proto);
|
|
var __v_d2_value = {};
|
|
__v_d2_value.a = 1;
|
|
__v_d2_value.b = __v;
|
|
__v_d2.value = __v_d2_value;
|
|
__v.d2 = __v_d2;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/(value) {
|
|
this.value = value;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*apply*/(func) {
|
|
throw new Error("'apply' is not allowed from inside a cloud-callback. Use 'get' to retrieve the value of this Output directly.");
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*get*/() {
|
|
return this.value;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ v: __v }) {
|
|
|
|
return function () { console.log(v); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const obj = { method1() { return this.method2(); }, method2: () => { return; } };
|
|
|
|
cases.push({
|
|
title: "Capture object with methods",
|
|
func: function () { console.log(obj); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __obj = {method1: __f1, method2: __f2};
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*method1*/() { return this.method2(); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return () => { return; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ obj: __obj }) {
|
|
|
|
return function () { console.log(obj); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
cases.push({
|
|
title: "Undeclared variable in typeof",
|
|
// @ts-ignore
|
|
func: function () { const x = typeof a; },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ a: undefined }) {
|
|
|
|
return function () { const x = typeof a; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const a = 0;
|
|
cases.push({
|
|
title: "Declared variable in typeof",
|
|
// @ts-ignore
|
|
func: function () { const x = typeof a; },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ a: 0 }) {
|
|
|
|
return function () { const x = typeof a; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const array: any[] = [];
|
|
const obj = { 80: "foo", arr: array };
|
|
array.push(obj);
|
|
|
|
cases.push({
|
|
title: "Capture numeric property",
|
|
func: function () { return array; },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __array = [];
|
|
var __array_0 = {};
|
|
__array_0["80"] = "foo";
|
|
__array_0.arr = __array;
|
|
__array[0] = __array_0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ array: __array }) {
|
|
|
|
return function () { return array; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const outer: any = { o: 1 };
|
|
const array = [outer];
|
|
outer.b = array;
|
|
const C = (function () {
|
|
function C() {}
|
|
C.prototype.m = function () { return this.n() + outer; };
|
|
C.prototype.n = function () { return array; };
|
|
(<any>C).m = function () { return this.n(); };
|
|
return C;
|
|
}());
|
|
|
|
cases.push({
|
|
title: "Serialize es5-style class",
|
|
func: () => C,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __C_prototype = {};
|
|
Object.defineProperty(__C_prototype, "constructor", { configurable: true, writable: true, value: __C });
|
|
var __outer = {};
|
|
__outer.o = 1;
|
|
var __outer_b = [];
|
|
__outer_b[0] = __outer;
|
|
__outer.b = __outer_b;
|
|
__C_prototype.m = __f1;
|
|
__C_prototype.n = __f2;
|
|
__C.prototype = __C_prototype;
|
|
__C.m = __f3;
|
|
|
|
function __C() {
|
|
return (function() {
|
|
with({ C: __C }) {
|
|
|
|
return function /*C*/() { };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ outer: __outer }) {
|
|
|
|
return function () { return this.n() + outer; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ array: __outer_b }) {
|
|
|
|
return function () { return array; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function () { return this.n(); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ C: __C }) {
|
|
|
|
return () => C;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const outer: any = { o: 1 };
|
|
const array = [outer];
|
|
outer.b = array;
|
|
class C {
|
|
public static s() { return array; }
|
|
public m() { return this.n(); }
|
|
public n() { return outer; }
|
|
}
|
|
cases.push({
|
|
title: "Serialize class",
|
|
func: () => C,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __f1_prototype = {};
|
|
Object.defineProperty(__f1_prototype, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
Object.defineProperty(__f1_prototype, "m", { configurable: true, writable: true, value: __f2 });
|
|
var __outer = {};
|
|
__outer.o = 1;
|
|
var __outer_b = [];
|
|
__outer_b[0] = __outer;
|
|
__outer.b = __outer_b;
|
|
Object.defineProperty(__f1_prototype, "n", { configurable: true, writable: true, value: __f3 });
|
|
__f1.prototype = __f1_prototype;
|
|
__f1.s = __f4;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/() { };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*m*/() { return this.n(); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ outer: __outer }) {
|
|
|
|
return function /*n*/() { return outer; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f4() {
|
|
return (function() {
|
|
with({ array: __outer_b }) {
|
|
|
|
return function /*s*/() { return array; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ C: __f1 }) {
|
|
|
|
return () => C;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class C {
|
|
private x: number;
|
|
public static s() { return 0; }
|
|
constructor() {
|
|
this.x = 1;
|
|
}
|
|
public m() { return this.n(); }
|
|
public n() { return 1; }
|
|
}
|
|
cases.push({
|
|
title: "Serialize class with constructor and field",
|
|
func: () => C,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __f1_prototype = {};
|
|
Object.defineProperty(__f1_prototype, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
Object.defineProperty(__f1_prototype, "m", { configurable: true, writable: true, value: __f2 });
|
|
Object.defineProperty(__f1_prototype, "n", { configurable: true, writable: true, value: __f3 });
|
|
__f1.prototype = __f1_prototype;
|
|
__f1.s = __f4;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/() {
|
|
this.x = 1;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*m*/() { return this.n(); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*n*/() { return 1; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f4() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*s*/() { return 0; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ C: __f1 }) {
|
|
|
|
return () => C;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class C {
|
|
private x: number;
|
|
public static s() { return 0; }
|
|
constructor() {
|
|
this.x = 1;
|
|
}
|
|
public m() { return this.n(); }
|
|
public n() { return 1; }
|
|
}
|
|
cases.push({
|
|
title: "Serialize constructed class",
|
|
func: () => new C(),
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __f1_prototype = {};
|
|
Object.defineProperty(__f1_prototype, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
Object.defineProperty(__f1_prototype, "m", { configurable: true, writable: true, value: __f2 });
|
|
Object.defineProperty(__f1_prototype, "n", { configurable: true, writable: true, value: __f3 });
|
|
__f1.prototype = __f1_prototype;
|
|
__f1.s = __f4;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/() {
|
|
this.x = 1;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*m*/() { return this.n(); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*n*/() { return 1; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f4() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*s*/() { return 0; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ C: __f1 }) {
|
|
|
|
return () => new C();
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class C {
|
|
public m() { return this.n(); }
|
|
public n() { return 0; }
|
|
}
|
|
cases.push({
|
|
title: "Serialize instance class methods",
|
|
func: new C().m,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*m*/() { return this.n(); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class C {
|
|
public delete() { return 0; }
|
|
}
|
|
cases.push({
|
|
title: "Serialize method with reserved name",
|
|
func: new C().delete,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*delete*/() { return 0; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
|
|
{
|
|
class C {
|
|
public static m() { return this.n(); }
|
|
public static n() { return 0; }
|
|
}
|
|
cases.push({
|
|
title: "Serialize static class methods",
|
|
func: C.m,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*m*/() { return this.n(); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const D = (function () {
|
|
function D() {
|
|
;
|
|
}
|
|
(<any>D).m = function () { return this.n(); };
|
|
(<any>D).n = function () { return 0; };
|
|
return D;
|
|
}());
|
|
cases.push({
|
|
title: "Serialize static class methods (es5 class style)",
|
|
func: (<any>D).m,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function () { return this.n(); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const array: any[] = [1];
|
|
array.push(array);
|
|
|
|
cases.push({
|
|
title: "Cyclic object #1",
|
|
func: () => array,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __array = [];
|
|
__array[0] = 1;
|
|
__array[1] = __array;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ array: __array }) {
|
|
|
|
return () => array;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const obj: any = {a: 1};
|
|
obj.b = obj;
|
|
|
|
cases.push({
|
|
title: "Cyclic object #2",
|
|
func: () => obj,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __obj = {};
|
|
__obj.a = 1;
|
|
__obj.b = __obj;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ obj: __obj }) {
|
|
|
|
return () => obj;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const obj: any = {a: []};
|
|
obj.a.push(obj);
|
|
obj.b = obj.a;
|
|
|
|
cases.push({
|
|
title: "Cyclic object #3",
|
|
func: () => obj,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __obj = {};
|
|
var __obj_a = [];
|
|
__obj_a[0] = __obj;
|
|
__obj.a = __obj_a;
|
|
__obj.b = __obj_a;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ obj: __obj }) {
|
|
|
|
return () => obj;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const obj: any = {a: []};
|
|
obj.a.push(obj);
|
|
obj.b = obj.a;
|
|
const obj2 = [obj, obj];
|
|
|
|
cases.push({
|
|
title: "Cyclic object #4",
|
|
func: () => obj2,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __obj2 = [];
|
|
var __obj2_0 = {};
|
|
var __obj2_0_a = [];
|
|
__obj2_0_a[0] = __obj2_0;
|
|
__obj2_0.a = __obj2_0_a;
|
|
__obj2_0.b = __obj2_0_a;
|
|
__obj2[0] = __obj2_0;
|
|
__obj2[1] = __obj2_0;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ obj2: __obj2 }) {
|
|
|
|
return () => obj2;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const obj: any = {a: 1};
|
|
|
|
function f1() { return obj; }
|
|
function f2() { console.log(obj); }
|
|
|
|
cases.push({
|
|
title: "Object captured across multiple functions",
|
|
func: () => { f1(); obj.a = 2; f2(); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __obj = {a: 1};
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ obj: __obj, f1: __f1 }) {
|
|
|
|
return function /*f1*/() { return obj; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ obj: __obj, f2: __f2 }) {
|
|
|
|
return function /*f2*/() { console.log(obj); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ f1: __f1, obj: __obj, f2: __f2 }) {
|
|
|
|
return () => { f1(); obj.a = 2; f2(); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const v = {};
|
|
Object.defineProperty(v, "key", {
|
|
configurable: true,
|
|
value: 1,
|
|
});
|
|
cases.push({
|
|
title: "Complex property descriptor #1",
|
|
func: () => v,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __v = {};
|
|
Object.defineProperty(__v, "key", { configurable: true, value: 1 });
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ v: __v }) {
|
|
|
|
return () => v;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const v = {};
|
|
Object.defineProperty(v, "key", {
|
|
writable: true,
|
|
enumerable: true,
|
|
value: 1,
|
|
});
|
|
cases.push({
|
|
title: "Complex property descriptor #2",
|
|
func: () => v,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __v = {};
|
|
Object.defineProperty(__v, "key", { enumerable: true, writable: true, value: 1 });
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ v: __v }) {
|
|
|
|
return () => v;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const v = [1, 2, 3];
|
|
delete v[1];
|
|
|
|
cases.push({
|
|
title: "Test array #1",
|
|
func: () => v,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __v = [];
|
|
__v[0] = 1;
|
|
__v[2] = 3;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ v: __v }) {
|
|
|
|
return () => v;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const v = [1, 2, 3];
|
|
delete v[1];
|
|
(<any>v).foo = "";
|
|
|
|
cases.push({
|
|
title: "Test array #2",
|
|
func: () => v,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __v = [];
|
|
__v[0] = 1;
|
|
__v[2] = 3;
|
|
__v.foo = "";
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ v: __v }) {
|
|
|
|
return () => v;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const v = () => { return 1; };
|
|
(<any>v).foo = "bar";
|
|
|
|
cases.push({
|
|
title: "Test function with property",
|
|
func: v,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
__f0.foo = "bar";
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return () => { return 1; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const x = Object.create(null);
|
|
const v = () => { return x; };
|
|
|
|
cases.push({
|
|
title: "Test null prototype",
|
|
func: v,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __x = Object.create(null);
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ x: __x }) {
|
|
|
|
return () => { return x; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const x = Object.create(Number.prototype);
|
|
const v = () => { return x; };
|
|
|
|
cases.push({
|
|
title: "Test non-default object prototype",
|
|
func: v,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __x = Object.create(global.Number.prototype);
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ x: __x }) {
|
|
|
|
return () => { return x; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const x = Object.create({ x() { return v; } });
|
|
const v = () => { return x; };
|
|
|
|
cases.push({
|
|
title: "Test recursive prototype object prototype",
|
|
func: v,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __x_proto = {x: __f1};
|
|
var __x = Object.create(__x_proto);
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ v: __f0 }) {
|
|
|
|
return function /*x*/() { return v; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ x: __x }) {
|
|
|
|
return () => { return x; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const v = () => { return 0; };
|
|
Object.setPrototypeOf(v, () => { return 1; });
|
|
|
|
cases.push({
|
|
title: "Test non-default function prototype",
|
|
func: v,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
Object.setPrototypeOf(__f0, __f1);
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return () => { return 0; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return () => { return 1; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
function *f() { yield 1; }
|
|
|
|
cases.push({
|
|
title: "Test generator func",
|
|
func: f,
|
|
expectText: `exports.handler = __f;
|
|
|
|
var __f_prototype = Object.create(Object.getPrototypeOf((function*(){}).prototype));
|
|
__f.prototype = __f_prototype;
|
|
Object.setPrototypeOf(__f, Object.getPrototypeOf(function*(){}));
|
|
|
|
function __f() {
|
|
return (function() {
|
|
with({ f: __f }) {
|
|
|
|
return function* /*f*/() { yield 1; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const gf = (function *() { yield 1; });
|
|
|
|
cases.push({
|
|
title: "Test anonymous generator func",
|
|
func: gf,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __f0_prototype = Object.create(Object.getPrototypeOf((function*(){}).prototype));
|
|
__f0.prototype = __f0_prototype;
|
|
Object.setPrototypeOf(__f0, Object.getPrototypeOf(function*(){}));
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function* () { yield 1; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class C {
|
|
private _x: number;
|
|
|
|
constructor() {
|
|
this._x = 0;
|
|
}
|
|
|
|
get foo() {
|
|
return this._x;
|
|
}
|
|
|
|
set foo(v: number) {
|
|
this._x = v;
|
|
}
|
|
}
|
|
|
|
cases.push({
|
|
title: "Test getter/setter #1",
|
|
func: () => C,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __f1_prototype = {};
|
|
Object.defineProperty(__f1_prototype, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
Object.defineProperty(__f1_prototype, "foo", { configurable: true, get: __f2, set: __f3 });
|
|
__f1.prototype = __f1_prototype;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/() {
|
|
this._x = 0;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*foo*/() {
|
|
return this._x;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*foo*/(v) {
|
|
this._x = v;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ C: __f1 }) {
|
|
|
|
return () => C;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const methodName = "method name";
|
|
class C {
|
|
[methodName](a: number) {
|
|
return a;
|
|
}
|
|
}
|
|
|
|
cases.push({
|
|
title: "Test computed method name.",
|
|
func: () => C,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __f1_prototype = {};
|
|
Object.defineProperty(__f1_prototype, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
Object.defineProperty(__f1_prototype, "method name", { configurable: true, writable: true, value: __f2 });
|
|
__f1.prototype = __f1_prototype;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/() { };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function (a) {
|
|
return a;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ C: __f1 }) {
|
|
|
|
return () => C;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const sym = Symbol("test_symbol");
|
|
class C {
|
|
[sym](a: number) {
|
|
return a;
|
|
}
|
|
|
|
getSym() { return sym; }
|
|
}
|
|
|
|
cases.push({
|
|
title: "Test symbols #1",
|
|
func: () => C,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __f1_prototype = {};
|
|
Object.defineProperty(__f1_prototype, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
var __sym = Object.create(global.Symbol.prototype);
|
|
Object.defineProperty(__f1_prototype, "getSym", { configurable: true, writable: true, value: __f2 });
|
|
Object.defineProperty(__f1_prototype, __sym, { configurable: true, writable: true, value: __f3 });
|
|
__f1.prototype = __f1_prototype;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/() { };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ sym: __sym }) {
|
|
|
|
return function /*getSym*/() { return sym; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function (a) {
|
|
return a;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ C: __f1 }) {
|
|
|
|
return () => C;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class C {
|
|
[Symbol.iterator](a: number) {
|
|
return a;
|
|
}
|
|
}
|
|
|
|
cases.push({
|
|
title: "Test Symbol.iterator",
|
|
func: () => C,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __f1_prototype = {};
|
|
Object.defineProperty(__f1_prototype, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
Object.defineProperty(__f1_prototype, Symbol.iterator, { configurable: true, writable: true, value: __f2 });
|
|
__f1.prototype = __f1_prototype;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/() { };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function (a) {
|
|
return a;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ C: __f1 }) {
|
|
|
|
return () => C;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class D {
|
|
public n: number;
|
|
constructor(n: number) {
|
|
this.n = n;
|
|
console.log("DConstructor");
|
|
}
|
|
dMethod(x: number) { return x; }
|
|
dVirtual() { return 1; }
|
|
}
|
|
class C extends D {
|
|
constructor(n: number) {
|
|
super(n + 1);
|
|
console.log("CConstructor");
|
|
}
|
|
cMethod() {
|
|
return "" +
|
|
super.dMethod + super["dMethod"] +
|
|
super.dMethod(1) + super["dMethod"](2) +
|
|
super.dMethod(super.dMethod(3));
|
|
}
|
|
dVirtual() { return 3; }
|
|
}
|
|
|
|
cases.push({
|
|
title: "Test class extension",
|
|
func: () => C,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __f2_prototype = {};
|
|
Object.defineProperty(__f2_prototype, "constructor", { configurable: true, writable: true, value: __f2 });
|
|
Object.defineProperty(__f2_prototype, "dMethod", { configurable: true, writable: true, value: __f3 });
|
|
Object.defineProperty(__f2_prototype, "dVirtual", { configurable: true, writable: true, value: __f4 });
|
|
__f2.prototype = __f2_prototype;
|
|
var __f1_prototype = Object.create(__f2_prototype);
|
|
Object.defineProperty(__f1_prototype, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
Object.defineProperty(__f1_prototype, "cMethod", { configurable: true, writable: true, value: __f5 });
|
|
Object.defineProperty(__f1_prototype, "dVirtual", { configurable: true, writable: true, value: __f6 });
|
|
__f1.prototype = __f1_prototype;
|
|
Object.setPrototypeOf(__f1, __f2);
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/(n) {
|
|
this.n = n;
|
|
console.log("DConstructor");
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*dMethod*/(x) { return x; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f4() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*dVirtual*/() { return 1; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ __super: __f2 }) {
|
|
|
|
return function /*constructor*/(n) {
|
|
__super.call(this, n + 1);
|
|
console.log("CConstructor");
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f5() {
|
|
return (function() {
|
|
with({ __super: __f2 }) {
|
|
|
|
return function /*cMethod*/() {
|
|
return "" +
|
|
__super.prototype.dMethod + __super.prototype["dMethod"] +
|
|
__super.prototype.dMethod.call(this, 1) + __super.prototype["dMethod"].call(this, 2) +
|
|
__super.prototype.dMethod.call(this, __super.prototype.dMethod.call(this, 3));
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f6() {
|
|
return (function() {
|
|
with({ __super: __f2 }) {
|
|
|
|
return function /*dVirtual*/() { return 3; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ C: __f1 }) {
|
|
|
|
return () => C;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class A {
|
|
public n: number;
|
|
constructor(n: number) {
|
|
this.n = n;
|
|
console.log("AConstruction");
|
|
}
|
|
method(x: number) { return x; }
|
|
}
|
|
class B extends A {
|
|
constructor(n: number) {
|
|
super(n + 1);
|
|
console.log("BConstructor");
|
|
}
|
|
method(n: number) { return 1 + super.method(n + 1); }
|
|
}
|
|
class C extends B {
|
|
constructor(n: number) {
|
|
super(n * 2);
|
|
console.log("CConstructor");
|
|
}
|
|
method(n: number) { return 2 * super.method(n * 2); }
|
|
}
|
|
|
|
cases.push({
|
|
title: "Three level inheritance",
|
|
func: () => C,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __f3_prototype = {};
|
|
Object.defineProperty(__f3_prototype, "constructor", { configurable: true, writable: true, value: __f3 });
|
|
Object.defineProperty(__f3_prototype, "method", { configurable: true, writable: true, value: __f4 });
|
|
__f3.prototype = __f3_prototype;
|
|
var __f2_prototype = Object.create(__f3_prototype);
|
|
Object.defineProperty(__f2_prototype, "constructor", { configurable: true, writable: true, value: __f2 });
|
|
Object.defineProperty(__f2_prototype, "method", { configurable: true, writable: true, value: __f5 });
|
|
__f2.prototype = __f2_prototype;
|
|
Object.setPrototypeOf(__f2, __f3);
|
|
var __f1_prototype = Object.create(__f2_prototype);
|
|
Object.defineProperty(__f1_prototype, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
Object.defineProperty(__f1_prototype, "method", { configurable: true, writable: true, value: __f6 });
|
|
__f1.prototype = __f1_prototype;
|
|
Object.setPrototypeOf(__f1, __f2);
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/(n) {
|
|
this.n = n;
|
|
console.log("AConstruction");
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f4() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*method*/(x) { return x; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ __super: __f3 }) {
|
|
|
|
return function /*constructor*/(n) {
|
|
__super.call(this, n + 1);
|
|
console.log("BConstructor");
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f5() {
|
|
return (function() {
|
|
with({ __super: __f3 }) {
|
|
|
|
return function /*method*/(n) { return 1 + __super.prototype.method.call(this, n + 1); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ __super: __f2 }) {
|
|
|
|
return function /*constructor*/(n) {
|
|
__super.call(this, n * 2);
|
|
console.log("CConstructor");
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f6() {
|
|
return (function() {
|
|
with({ __super: __f2 }) {
|
|
|
|
return function /*method*/(n) { return 2 * __super.prototype.method.call(this, n * 2); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ C: __f1 }) {
|
|
|
|
return () => C;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`});
|
|
}
|
|
|
|
{
|
|
const sym = Symbol();
|
|
|
|
class A {
|
|
public n: number;
|
|
constructor(n: number) {
|
|
this.n = n;
|
|
console.log("AConstruction");
|
|
}
|
|
public [sym](x: number) { return x; }
|
|
}
|
|
class B extends A {
|
|
constructor(n: number) {
|
|
super(n + 1);
|
|
console.log("BConstructor");
|
|
}
|
|
// @ts-ignore
|
|
public [sym](n: number) { return 1 + super[sym](n + 1); }
|
|
}
|
|
class C extends B {
|
|
constructor(n: number) {
|
|
super(n * 2);
|
|
console.log("CConstructor");
|
|
}
|
|
// @ts-ignore
|
|
public [sym](n: number) { return 2 * super[sym](n * 2); }
|
|
}
|
|
|
|
cases.push({
|
|
title: "Three level inheritance with symbols",
|
|
func: () => C,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __f3_prototype = {};
|
|
Object.defineProperty(__f3_prototype, "constructor", { configurable: true, writable: true, value: __f3 });
|
|
var __f3_prototype_sym = Object.create(global.Symbol.prototype);
|
|
Object.defineProperty(__f3_prototype, __f3_prototype_sym, { configurable: true, writable: true, value: __f4 });
|
|
__f3.prototype = __f3_prototype;
|
|
var __f2_prototype = Object.create(__f3_prototype);
|
|
Object.defineProperty(__f2_prototype, "constructor", { configurable: true, writable: true, value: __f2 });
|
|
Object.defineProperty(__f2_prototype, __f3_prototype_sym, { configurable: true, writable: true, value: __f5 });
|
|
__f2.prototype = __f2_prototype;
|
|
Object.setPrototypeOf(__f2, __f3);
|
|
var __f1_prototype = Object.create(__f2_prototype);
|
|
Object.defineProperty(__f1_prototype, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
Object.defineProperty(__f1_prototype, __f3_prototype_sym, { configurable: true, writable: true, value: __f6 });
|
|
__f1.prototype = __f1_prototype;
|
|
Object.setPrototypeOf(__f1, __f2);
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/(n) {
|
|
this.n = n;
|
|
console.log("AConstruction");
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f4() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function (x) { return x; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ __super: __f3 }) {
|
|
|
|
return function /*constructor*/(n) {
|
|
__super.call(this, n + 1);
|
|
console.log("BConstructor");
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f5() {
|
|
return (function() {
|
|
with({ sym: __f3_prototype_sym, __super: __f3 }) {
|
|
|
|
return function /*__computed*/(n) { return 1 + __super.prototype[sym].call(this, n + 1); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ __super: __f2 }) {
|
|
|
|
return function /*constructor*/(n) {
|
|
__super.call(this, n * 2);
|
|
console.log("CConstructor");
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f6() {
|
|
return (function() {
|
|
with({ sym: __f3_prototype_sym, __super: __f2 }) {
|
|
|
|
return function /*__computed*/(n) { return 2 * __super.prototype[sym].call(this, n * 2); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ C: __f1 }) {
|
|
|
|
return () => C;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`});
|
|
}
|
|
|
|
{
|
|
const sym = Symbol();
|
|
|
|
class A {
|
|
public n: number;
|
|
static method(x: number) { return x; }
|
|
static [sym](x: number) { return x * x; }
|
|
constructor(n: number) {
|
|
this.n = n;
|
|
console.log("AConstruction");
|
|
}
|
|
}
|
|
class B extends A {
|
|
static method(n: number) { return 1 + super.method(n + 1); }
|
|
// @ts-ignore
|
|
static [sym](x: number) { return x * super[sym](x + 1); }
|
|
constructor(n: number) {
|
|
super(n + 1);
|
|
console.log("BConstructor");
|
|
}
|
|
}
|
|
|
|
cases.push({
|
|
title: "Two level static inheritance",
|
|
func: () => B,
|
|
expectText: `exports.handler = __f0;
|
|
|
|
__f2.method = __f3;
|
|
var __f2_sym = Object.create(global.Symbol.prototype);
|
|
__f2[__f2_sym] = __f4;
|
|
__f1.method = __f5;
|
|
__f1[__f2_sym] = __f6;
|
|
Object.setPrototypeOf(__f1, __f2);
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/(n) {
|
|
this.n = n;
|
|
console.log("AConstruction");
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*method*/(x) { return x; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f4() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function (x) { return x * x; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ __super: __f2 }) {
|
|
|
|
return function /*constructor*/(n) {
|
|
__super.call(this, n + 1);
|
|
console.log("BConstructor");
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f5() {
|
|
return (function() {
|
|
with({ __super: __f2 }) {
|
|
|
|
return function /*method*/(n) { return 1 + __super.method.call(this, n + 1); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f6() {
|
|
return (function() {
|
|
with({ sym: __f2_sym, __super: __f2 }) {
|
|
|
|
return function /*__computed*/(x) { return x * __super[sym].call(this, x + 1); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ B: __f1 }) {
|
|
|
|
return () => B;
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, b: 2 };
|
|
|
|
cases.push({
|
|
title: "Capture subset of properties #1",
|
|
func: function () { console.log(o.a); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __o = {a: 1};
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ o: __o }) {
|
|
|
|
return function () { console.log(o.a); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, b: 2, c: 3 };
|
|
|
|
cases.push({
|
|
title: "Capture subset of properties #2",
|
|
func: function () { console.log(o.b + o.c); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __o = {b: 2, c: 3};
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ o: __o }) {
|
|
|
|
return function () { console.log(o.b + o.c); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, b: 2, c: 3 };
|
|
|
|
cases.push({
|
|
title: "Capture all if object is used as is.",
|
|
func: function () { console.log(o); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __o = {a: 1, b: 2, c: 3};
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ o: __o }) {
|
|
|
|
return function () { console.log(o); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, b: 2, c() { return this; } };
|
|
|
|
cases.push({
|
|
title: "Capture all if object property is invoked, and it uses this.",
|
|
func: function () { console.log(o.c()); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __o = {a: 1, b: 2, c: __f1};
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*c*/() { return this; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ o: __o }) {
|
|
|
|
return function () { console.log(o.c()); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, b: 2, c() { const v = () => this; } };
|
|
|
|
cases.push({
|
|
title: "Capture all if object property is invoked, and it uses this in nested arrow function.",
|
|
func: function () { console.log(o.c()); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __o = {a: 1, b: 2, c: __f1};
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*c*/() { const v = () => this; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ o: __o }) {
|
|
|
|
return function () { console.log(o.c()); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, b: 2, c() { const v = function () { return this; }; } };
|
|
|
|
cases.push({
|
|
title: "Capture one if object property is invoked, but it uses this in nested function.",
|
|
func: function () { console.log(o.c()); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __o = {c: __f1};
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*c*/() { const v = function () { return this; }; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ o: __o }) {
|
|
|
|
return function () { console.log(o.c()); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, b: 2, c() { return this; } };
|
|
|
|
cases.push({
|
|
title: "Capture one if object property is captured, uses this, but is not invoked",
|
|
func: function () { console.log(o.c); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __o = {c: __f1};
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*c*/() { return this; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ o: __o }) {
|
|
|
|
return function () { console.log(o.c); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, b: 2, c() { return 0; } };
|
|
|
|
cases.push({
|
|
title: "Capture one if object property is invoked, and it does not use this.",
|
|
func: function () { console.log(o.c()); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __o = {c: __f1};
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*c*/() { return 0; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ o: __o }) {
|
|
|
|
return function () { console.log(o.c()); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, b: { c() { return this; } } };
|
|
|
|
cases.push({
|
|
title: "Capture subset if sub object property is invoked.",
|
|
func: function () { console.log(o.b.c()); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __o = {};
|
|
var __o_b = {c: __f1};
|
|
__o.b = __o_b;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*c*/() { return this; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ o: __o }) {
|
|
|
|
return function () { console.log(o.b.c()); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, get b() { return this; } };
|
|
|
|
cases.push({
|
|
title: "Capture all if getter and getter uses this.",
|
|
func: function () { console.log(o.b); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __o = {};
|
|
__o.a = 1;
|
|
Object.defineProperty(__o, "b", { configurable: true, enumerable: true, get: __f1 });
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*b*/() { return this; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ o: __o }) {
|
|
|
|
return function () { console.log(o.b); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, get b() { return 0; } };
|
|
|
|
cases.push({
|
|
title: "Capture one if getter and getter does not use this.",
|
|
func: function () { console.log(o.b); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __o = {};
|
|
Object.defineProperty(__o, "b", { configurable: true, enumerable: true, get: __f1 });
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*b*/() { return 0; };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ o: __o }) {
|
|
|
|
return function () { console.log(o.b); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, b: 1, c: 2 };
|
|
function f1() {
|
|
console.log(o.a);
|
|
f2();
|
|
}
|
|
|
|
function f2() {
|
|
console.log(o.c);
|
|
}
|
|
|
|
cases.push({
|
|
title: "Capture multi props from different contexts #1",
|
|
func: f1,
|
|
expectText: `exports.handler = __f1;
|
|
|
|
var __o = {a: 1, c: 2};
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ o: __o, f2: __f2 }) {
|
|
|
|
return function /*f2*/() {
|
|
console.log(o.c);
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ o: __o, f2: __f2, f1: __f1 }) {
|
|
|
|
return function /*f1*/() {
|
|
console.log(o.a);
|
|
f2();
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1 };
|
|
function f1() {
|
|
// @ts-ignore
|
|
console.log(o.c);
|
|
}
|
|
|
|
cases.push({
|
|
title: "Do not capture non-existent prop",
|
|
func: f1,
|
|
expectText: `exports.handler = __f1;
|
|
|
|
var __o = {};
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ o: __o, f1: __f1 }) {
|
|
|
|
return function /*f1*/() {
|
|
// @ts-ignore
|
|
console.log(o.c);
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, b: 1, c: 2 };
|
|
function f1() {
|
|
console.log(o.a);
|
|
f2();
|
|
}
|
|
|
|
function f2() {
|
|
console.log(o);
|
|
}
|
|
|
|
cases.push({
|
|
title: "Capture all props from different contexts #1",
|
|
func: f1,
|
|
expectText: `exports.handler = __f1;
|
|
|
|
var __o = {a: 1, b: 1, c: 2};
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ o: __o, f2: __f2 }) {
|
|
|
|
return function /*f2*/() {
|
|
console.log(o);
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ o: __o, f2: __f2, f1: __f1 }) {
|
|
|
|
return function /*f1*/() {
|
|
console.log(o.a);
|
|
f2();
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, b: 1, c: 2 };
|
|
function f1() {
|
|
console.log(o);
|
|
f2();
|
|
}
|
|
|
|
function f2() {
|
|
console.log(o.a);
|
|
}
|
|
|
|
cases.push({
|
|
title: "Capture all props from different contexts #2",
|
|
func: f1,
|
|
expectText: `exports.handler = __f1;
|
|
|
|
var __o = {a: 1, b: 1, c: 2};
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ o: __o, f2: __f2 }) {
|
|
|
|
return function /*f2*/() {
|
|
console.log(o.a);
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ o: __o, f2: __f2, f1: __f1 }) {
|
|
|
|
return function /*f1*/() {
|
|
console.log(o);
|
|
f2();
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class C {
|
|
a: number;
|
|
b: number;
|
|
|
|
constructor() {
|
|
this.a = 1;
|
|
this.b = 2;
|
|
}
|
|
|
|
m() { console.log(this); }
|
|
}
|
|
const o = new C();
|
|
|
|
cases.push({
|
|
title: "Capture all props if prototype is and uses this #1",
|
|
func: function () { o.m(); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __o_proto = {};
|
|
__f1.prototype = __o_proto;
|
|
Object.defineProperty(__o_proto, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
Object.defineProperty(__o_proto, "m", { configurable: true, writable: true, value: __f2 });
|
|
var __o = Object.create(__o_proto);
|
|
__o.a = 1;
|
|
__o.b = 2;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/() {
|
|
this.a = 1;
|
|
this.b = 2;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*m*/() { console.log(this); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ o: __o }) {
|
|
|
|
return function () { o.m(); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class C {
|
|
a: number;
|
|
b: number;
|
|
|
|
constructor() {
|
|
this.a = 1;
|
|
this.b = 2;
|
|
}
|
|
|
|
m() { }
|
|
}
|
|
const o = new C();
|
|
|
|
cases.push({
|
|
title: "Capture no props if prototype is used but does not use this #1",
|
|
func: function () { o.m(); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __o = {};
|
|
Object.defineProperty(__o, "m", { configurable: true, writable: true, value: __f1 });
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*m*/() { };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ o: __o }) {
|
|
|
|
return function () { o.m(); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
class C {
|
|
a: number;
|
|
|
|
constructor() {
|
|
this.a = 1;
|
|
}
|
|
|
|
m() { (<any>this).n(); }
|
|
}
|
|
|
|
class D extends C {
|
|
b: number;
|
|
constructor() {
|
|
super();
|
|
this.b = 2;
|
|
}
|
|
n() {}
|
|
}
|
|
const o = new D();
|
|
|
|
cases.push({
|
|
title: "Capture all props if prototype is accessed #2",
|
|
func: function () { o.m(); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __o_proto_proto = {};
|
|
__f1.prototype = __o_proto_proto;
|
|
Object.defineProperty(__o_proto_proto, "constructor", { configurable: true, writable: true, value: __f1 });
|
|
Object.defineProperty(__o_proto_proto, "m", { configurable: true, writable: true, value: __f2 });
|
|
var __o_proto = Object.create(__o_proto_proto);
|
|
__f3.prototype = __o_proto;
|
|
Object.setPrototypeOf(__f3, __f1);
|
|
Object.defineProperty(__o_proto, "constructor", { configurable: true, writable: true, value: __f3 });
|
|
Object.defineProperty(__o_proto, "n", { configurable: true, writable: true, value: __f4 });
|
|
var __o = Object.create(__o_proto);
|
|
__o.a = 1;
|
|
__o.b = 2;
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*constructor*/() {
|
|
this.a = 1;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f2() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function /*m*/() { this.n(); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ __super: __f1 }) {
|
|
|
|
return function /*constructor*/() {
|
|
__super.call(this);
|
|
this.b = 2;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f4() {
|
|
return (function() {
|
|
with({ __super: __f1 }) {
|
|
|
|
return function /*n*/() { };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ o: __o }) {
|
|
|
|
return function () { o.m(); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const table1: any = { primaryKey: 1, insert: () => { }, scan: () => { } };
|
|
|
|
async function testScanReturnsAllValues() {
|
|
await table1.insert({[table1.primaryKey.get()]: "val1", value1: 1, value2: "1"});
|
|
await table1.insert({[table1.primaryKey.get()]: "val2", value1: 2, value2: "2"});
|
|
|
|
const values = null;
|
|
// @ts-ignore
|
|
const value1 = values.find(v => v[table1.primaryKey.get()] === "val1");
|
|
// @ts-ignore
|
|
const value2 = values.find(v => v[table1.primaryKey.get()] === "val2");
|
|
}
|
|
|
|
cases.push({
|
|
title: "Cloud table function",
|
|
func: testScanReturnsAllValues,
|
|
expectText: `exports.handler = __testScanReturnsAllValues;
|
|
|
|
var __table1 = {insert: __f1, primaryKey: 1};
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function (thisArg, _arguments, P, generator) {
|
|
return new (P || (P = Promise))(function (resolve, reject) {
|
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
|
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
});
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return () => { };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __testScanReturnsAllValues() {
|
|
return (function() {
|
|
with({ __awaiter: __f0, table1: __table1, testScanReturnsAllValues: __testScanReturnsAllValues }) {
|
|
|
|
return function /*testScanReturnsAllValues*/() {
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
yield table1.insert({ [table1.primaryKey.get()]: "val1", value1: 1, value2: "1" });
|
|
yield table1.insert({ [table1.primaryKey.get()]: "val2", value1: 2, value2: "2" });
|
|
const values = null;
|
|
// @ts-ignore
|
|
const value1 = values.find(v => v[table1.primaryKey.get()] === "val1");
|
|
// @ts-ignore
|
|
const value2 = values.find(v => v[table1.primaryKey.get()] === "val2");
|
|
});
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const o = { a: 1, b: { x: 1, doNotCapture: true }, c: 2 };
|
|
function f1() {
|
|
console.log(o);
|
|
}
|
|
|
|
cases.push({
|
|
title: "Do not capture #1",
|
|
func: f1,
|
|
expectText: `exports.handler = __f1;
|
|
|
|
var __o = {a: 1, b: undefined, c: 2};
|
|
|
|
function __f1() {
|
|
return (function() {
|
|
with({ o: __o, f1: __f1 }) {
|
|
|
|
return function /*f1*/() {
|
|
console.log(o);
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const lambda1 = () => console.log(1);
|
|
const lambda2 = () => console.log(1);
|
|
|
|
function f3() {
|
|
return (lambda1(), lambda2());
|
|
}
|
|
|
|
cases.push({
|
|
title: "Merge simple functions",
|
|
func: f3,
|
|
expectText: `exports.handler = __f3;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return () => console.log(1);
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ lambda1: __f0, lambda2: __f0, f3: __f3 }) {
|
|
|
|
return function /*f3*/() {
|
|
return (lambda1(), lambda2());
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
const awaiter1 = function (thisArg: any, _arguments: any, P: any, generator: any) {
|
|
return new (P || (P = Promise))(function (resolve: any, reject: any) {
|
|
function fulfilled(value: any) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
function rejected(value: any) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
function step(result: any) { result.done ? resolve(result.value) : new P(function (resolve1: any) { resolve1(result.value); }).then(fulfilled, rejected); }
|
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
});
|
|
};
|
|
const awaiter2 = function (thisArg: any, _arguments: any, P: any, generator: any) {
|
|
return new (P || (P = Promise))(function (resolve: any, reject: any) {
|
|
function fulfilled(value: any) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
function rejected(value: any) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
function step(result: any) { result.done ? resolve(result.value) : new P(function (resolve1: any) { resolve1(result.value); }).then(fulfilled, rejected); }
|
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
});
|
|
};
|
|
|
|
function f3() {
|
|
const v1 = awaiter1, v2 = awaiter2;
|
|
}
|
|
|
|
cases.push({
|
|
title: "Share __awaiter functions",
|
|
func: f3,
|
|
expectText: `exports.handler = __f3;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ }) {
|
|
|
|
return function (thisArg, _arguments, P, generator) {
|
|
return new (P || (P = Promise))(function (resolve, reject) {
|
|
function fulfilled(value) { try {
|
|
step(generator.next(value));
|
|
}
|
|
catch (e) {
|
|
reject(e);
|
|
} }
|
|
function rejected(value) { try {
|
|
step(generator["throw"](value));
|
|
}
|
|
catch (e) {
|
|
reject(e);
|
|
} }
|
|
function step(result) { result.done ? resolve(result.value) : new P(function (resolve1) { resolve1(result.value); }).then(fulfilled, rejected); }
|
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
});
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
|
|
function __f3() {
|
|
return (function() {
|
|
with({ awaiter1: __f0, awaiter2: __f0, f3: __f3 }) {
|
|
|
|
return function /*f3*/() {
|
|
const v1 = awaiter1, v2 = awaiter2;
|
|
};
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
cases.push({
|
|
title: "Capture of exported variable #1",
|
|
func: function () { console.log(exportedValue); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __exports = {exportedValue: 42};
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ exports: __exports }) {
|
|
|
|
return function () { console.log(exports.exportedValue); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
cases.push({
|
|
title: "Capture of exported variable #2",
|
|
func: function () { console.log(exports.exportedValue); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __exports = {exportedValue: 42};
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ exports: __exports }) {
|
|
|
|
return function () { console.log(exports.exportedValue); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
{
|
|
cases.push({
|
|
title: "Capture of exported variable #3",
|
|
func: function () { console.log(module.exports.exportedValue); },
|
|
expectText: `exports.handler = __f0;
|
|
|
|
var __module = {};
|
|
var __module_exports = {};
|
|
Object.defineProperty(__module_exports, "__esModule", { value: true });
|
|
__module_exports.exportedValue = 42;
|
|
__module.exports = __module_exports;
|
|
|
|
function __f0() {
|
|
return (function() {
|
|
with({ module: __module }) {
|
|
|
|
return function () { console.log(module.exports.exportedValue); };
|
|
|
|
}
|
|
}).apply(undefined, undefined).apply(this, arguments);
|
|
}
|
|
`,
|
|
});
|
|
}
|
|
|
|
// Run a bunch of direct checks on async js functions if we're in node 8 or above.
|
|
// We can't do this inline as node6 doesn't understand 'async functions'. And we
|
|
// can't do this in TS as TS will convert the async-function to be a normal non-async
|
|
// function.
|
|
const version = Number(process.version.match(/^v(\d+)\.\d+/)![1]);
|
|
if (version >= 8) {
|
|
const jsCases = require("./jsClosureCases");
|
|
cases.push(...jsCases.cases);
|
|
}
|
|
|
|
// Make a callback to keep running tests.
|
|
let remaining = cases;
|
|
while (true) {
|
|
const test = remaining.shift();
|
|
if (!test) {
|
|
return;
|
|
}
|
|
|
|
// if (test.title !== "Invocation of async function that does not capture this.") {
|
|
// continue;
|
|
// }
|
|
|
|
it(test.title, asyncTest(async () => {
|
|
// Run pre-actions.
|
|
if (test.pre) {
|
|
test.pre();
|
|
}
|
|
|
|
// Invoke the test case.
|
|
if (test.expectText) {
|
|
const text = await runtime.serializeFunctionAsync(test.func);
|
|
assert.equal(text, test.expectText);
|
|
}
|
|
else {
|
|
const message = await assertAsyncThrows(async () => {
|
|
await runtime.serializeFunctionAsync(test.func);
|
|
});
|
|
|
|
// replace real locations with (0,0) so that our test baselines do not need to
|
|
// updated any time this file changes.
|
|
const regex = /\([0-9]+,[0-9]+\)/g;
|
|
const withoutLocations = message.replace(regex, "(0,0)");
|
|
if (test.error) {
|
|
compareErrorText(test.error, withoutLocations);
|
|
}
|
|
}
|
|
}));
|
|
|
|
// Schedule any additional tests.
|
|
if (test.afters) {
|
|
remaining = test.afters.concat(remaining);
|
|
}
|
|
}
|
|
});
|
|
|
|
/**
|
|
* compareErrorText compares an "expected" error string and an "actual" error string
|
|
* and issues an error if they do not match.
|
|
*
|
|
* This function accepts two repetition operators to make writing tests easier against
|
|
* error messages that are dependent on the environment:
|
|
*
|
|
* * (...) alone on a single line causes the matcher to accept zero or more lines
|
|
* between the repetition and the next line.
|
|
* * (...) within in the context of a line causes the matcher to accept zero or more characters
|
|
* between the repetition and the next character.
|
|
*
|
|
* This is useful when testing error messages that you get when capturing bulit-in modules,
|
|
* because the specific error message differs between Node versions.
|
|
* @param expected The expected error message string, potentially containing repetitions
|
|
* @param actual The actual error message string
|
|
*/
|
|
function compareErrorText(expected: string, actual: string) {
|
|
const wildcard = "(...)";
|
|
|
|
if (!expected.includes(wildcard)) {
|
|
// We get a nice diff view if we diff the entire string, so do that
|
|
// if we didn't get a wildcard.
|
|
assert.equal(actual, expected);
|
|
return;
|
|
}
|
|
|
|
const expectedLines = expected.split(EOL);
|
|
const actualLines = actual.split(EOL);
|
|
let actualIndex = 0;
|
|
for (let expectedIndex = 0; expectedIndex < expectedLines.length; expectedIndex++) {
|
|
const expectedLine = expectedLines[expectedIndex].trim();
|
|
if (expectedLine === wildcard) {
|
|
if (expectedIndex + 1 === expectedLines.length) {
|
|
return;
|
|
}
|
|
|
|
const nextLine = expectedLines[++expectedIndex].trim();
|
|
while (true) {
|
|
const actualLine = actualLines[actualIndex++].trim();
|
|
if (actualLine === nextLine) {
|
|
break;
|
|
}
|
|
|
|
if (actualIndex === actualLines.length) {
|
|
assert.fail(`repetition failed to find match: expected terminator ${nextLine}, received ${actual}`);
|
|
}
|
|
}
|
|
} else if (expectedLine.includes(wildcard)) {
|
|
const line = actualLines[actualIndex++].trim();
|
|
const index = expectedLine.indexOf(wildcard);
|
|
const indexAfter = index + wildcard.length;
|
|
assert.equal(line.substring(0, index), expectedLine.substring(0, index));
|
|
|
|
let repetitionIndex = index;
|
|
for (; repetitionIndex < line.length; repetitionIndex++) {
|
|
if (line[repetitionIndex] === expectedLine[indexAfter]) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
assert.equal(line.substring(repetitionIndex), expectedLine.substring(indexAfter));
|
|
} else {
|
|
assert.equal(actualLines[actualIndex++].trim(), expectedLine);
|
|
}
|
|
}
|
|
}
|