mirror of
				https://github.com/robertkrimen/otto
				synced 2025-10-19 19:55:30 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			255 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			255 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package otto
 | |
| 
 | |
| import (
 | |
| 	. "./terst"
 | |
| 	"testing"
 | |
| )
 | |
| 
 | |
| func TestFunction(t *testing.T) {
 | |
| 	Terst(t)
 | |
| 
 | |
| 	test := runTest()
 | |
| 	test(`
 | |
|         var abc = Object.getOwnPropertyDescriptor(Function, "prototype");
 | |
|         [   [ typeof Function.prototype, typeof Function.prototype.length, Function.prototype.length ],
 | |
|             [ abc.writable, abc.enumerable, abc.configurable ] ];
 | |
|     `, "function,number,0,false,false,false")
 | |
| }
 | |
| 
 | |
| func Test_argumentList2parameterList(t *testing.T) {
 | |
| 	Terst(t)
 | |
| 
 | |
| 	Is(argumentList2parameterList([]Value{toValue("abc, def"), toValue("ghi")}), []string{"abc", "def", "ghi"})
 | |
| }
 | |
| 
 | |
| func TestFunction_new(t *testing.T) {
 | |
| 	Terst(t)
 | |
| 
 | |
| 	test := runTest()
 | |
| 	test(`raise:
 | |
|         new Function({});
 | |
|     `, "SyntaxError: Unexpected token Object")
 | |
| 
 | |
| 	test(`
 | |
|         var abc = Function("def, ghi", "jkl", "return def+ghi+jkl");
 | |
|         [ typeof abc, abc instanceof Function, abc("ab", "ba", 1) ];
 | |
|     `, "function,true,abba1")
 | |
| 
 | |
| 	test(`raise:
 | |
|         var abc = {
 | |
|             toString: function() { throw 1; }
 | |
|         };
 | |
|         var def = {
 | |
|             toString: function() { throw 2; }
 | |
|         };
 | |
|         var ghi = new Function(abc, def);
 | |
|         ghi;
 | |
|     `, "1")
 | |
| 
 | |
| 	test(`raise:
 | |
|         var abc = {
 | |
|             toString: function() { return "z;x"; }
 | |
|         };
 | |
|         var def = "return this";
 | |
|         var ghi = new Function(abc, def);
 | |
|         ghi;
 | |
|     `, "SyntaxError: z;x")
 | |
| 
 | |
| 	test(`raise:
 | |
|         var abc;
 | |
|         var def = "return true";
 | |
|         var ghi = new Function(null, def);
 | |
|         ghi;
 | |
|     `, "SyntaxError: null")
 | |
| }
 | |
| 
 | |
| func TestFunction_apply(t *testing.T) {
 | |
| 	Terst(t)
 | |
| 
 | |
| 	test := runTest()
 | |
| 	test(`Function.prototype.apply.length`, "2")
 | |
| 	test(`String.prototype.substring.apply("abc", [1, 11])`, "bc")
 | |
| }
 | |
| 
 | |
| func TestFunction_call(t *testing.T) {
 | |
| 	Terst(t)
 | |
| 
 | |
| 	test := runTest()
 | |
| 	test(`Function.prototype.call.length`, "1")
 | |
| 	test(`String.prototype.substring.call("abc", 1, 11)`, "bc")
 | |
| }
 | |
| 
 | |
| func TestFunctionArguments(t *testing.T) {
 | |
| 	Terst(t)
 | |
| 
 | |
| 	test := runTest()
 | |
| 	// Should not be able to delete arguments
 | |
| 	test(`
 | |
|         function abc(def, arguments){
 | |
|             delete def;
 | |
|             return def;
 | |
|         }
 | |
|         abc(1);
 | |
|     `, "1")
 | |
| 
 | |
| 	// Again, should not be able to delete arguments
 | |
| 	test(`
 | |
|         function abc(def){
 | |
|             delete def;
 | |
|             return def;
 | |
|         }
 | |
|         abc(1);
 | |
|     `, "1")
 | |
| 
 | |
| 	// Test typeof of a function argument
 | |
| 	test(`
 | |
|         function abc(def, ghi, jkl){
 | |
|             return typeof jkl
 | |
|         }
 | |
|         abc("1st", "2nd", "3rd", "4th", "5th");
 | |
|     `, "string")
 | |
| 
 | |
| 	test(`
 | |
|         function abc(def, ghi, jkl){
 | |
|             arguments[0] = 3.14;
 | |
|             arguments[1] = 'Nothing happens';
 | |
|             arguments[2] = 42;
 | |
|             if (3.14 === def && 'Nothing happens' === ghi && 42 === jkl)
 | |
|                 return true;
 | |
|         }
 | |
|         abc(-1, 4.2, 314);
 | |
|     `, "true")
 | |
| }
 | |
| 
 | |
| func TestFunctionDeclarationInFunction(t *testing.T) {
 | |
| 	Terst(t)
 | |
| 
 | |
| 	// Function declarations happen AFTER parameter/argument declarations
 | |
| 	// That is, a function declared within a function will shadow/overwrite
 | |
| 	// declared parameters
 | |
| 	test := runTest()
 | |
| 	test(`
 | |
|         function abc(def){
 | |
|             return def;
 | |
|             function def(){
 | |
|                 return 1;
 | |
|             }
 | |
|         }
 | |
|         typeof abc();
 | |
|     `, "function")
 | |
| }
 | |
| 
 | |
| func TestArguments_defineOwnProperty(t *testing.T) {
 | |
| 	Terst(t)
 | |
| 
 | |
| 	test := runTest()
 | |
| 
 | |
| 	test(`
 | |
|         var abc;
 | |
|         var def = true;
 | |
|         var ghi = {};
 | |
|         (function (a, b, c) {
 | |
|             Object.defineProperty(arguments, "0", {
 | |
|                 value: 42,
 | |
|                 writable: false,
 | |
|                 enumerable: false,
 | |
|                 configurable: false
 | |
|             });
 | |
|             Object.defineProperty(arguments, "1", {
 | |
|                 value: 3.14,
 | |
|                 configurable: true,
 | |
|                 enumerable: true
 | |
|             });
 | |
|             abc = Object.getOwnPropertyDescriptor(arguments, "0");
 | |
|             for (var name in arguments) {
 | |
|                 ghi[name] = (ghi[name] || 0) + 1;
 | |
|                 if (name === "0") {
 | |
|                     def = false;
 | |
|                 }
 | |
|             }
 | |
|         }(0, 1, 2));
 | |
|         [ abc.value, abc.writable, abc.enumerable, abc.configurable, def, ghi["1"] ];
 | |
|     `, "42,false,false,false,true,1")
 | |
| }
 | |
| 
 | |
| func TestFunction_bind(t *testing.T) {
 | |
| 	Terst(t)
 | |
| 
 | |
| 	test := runTest()
 | |
| 	test(`
 | |
|         abc = function(){
 | |
|             return "abc";
 | |
|         };
 | |
|         def = abc.bind();
 | |
|         [ typeof def.prototype, typeof def.hasOwnProperty, def.hasOwnProperty("caller"), def.hasOwnProperty("arguments"), def() ];
 | |
|     `, "object,function,true,true,abc")
 | |
| 
 | |
| 	test(`
 | |
|         abc = function(){
 | |
|             return arguments[1];
 | |
|         };
 | |
|         def = abc.bind(undefined, "abc");
 | |
|         ghi = abc.bind(undefined, "abc", "ghi");
 | |
|         [ def(), def("def"), ghi("def") ];
 | |
|     `, ",def,ghi")
 | |
| 
 | |
| 	test(`
 | |
|         var abc = function () {};
 | |
|         var ghi;
 | |
|         try {
 | |
|             Object.defineProperty(Function.prototype, "xyzzy", {
 | |
|                 value: 1001,
 | |
|                 writable: true,
 | |
|                 enumerable: true,
 | |
|                 configurable: true
 | |
|             });
 | |
|             var def = abc.bind({});
 | |
|             ghi = !def.hasOwnProperty("xyzzy") && ghi.xyzzy === 1001;
 | |
|         } finally {
 | |
|             delete Function.prototype.xyzzy;
 | |
|         }
 | |
|         [ ghi ];
 | |
|     `, "true")
 | |
| 
 | |
| 	test(`
 | |
|         var abc = function (def, ghi) {};
 | |
|         var jkl = abc.bind({});
 | |
|         var mno = abc.bind({}, 1, 2);
 | |
|         [ jkl.length, mno.length ];
 | |
|     `, "2,0")
 | |
| 
 | |
| 	test(`raise:
 | |
|         Math.bind();
 | |
|     `, "TypeError: undefined is not a function")
 | |
| 
 | |
| 	test(`
 | |
|         function construct(fn, arguments) {
 | |
|             var bound = Function.prototype.bind.apply(fn, [null].concat(arguments));
 | |
|             return new bound();
 | |
|         }
 | |
|         var abc = construct(Date, [1957, 4, 27]);
 | |
|         Object.prototype.toString.call(abc);
 | |
|     `, "[object Date]")
 | |
| 
 | |
| 	test(`
 | |
|         var fn = function (x, y, z) {
 | |
|             var result = {};
 | |
|             result.abc = x + y + z;
 | |
|             result.def = arguments[0] === "a" && arguments.length === 3;
 | |
|             return result;
 | |
|         };
 | |
|         var newFn = Function.prototype.bind.call(fn, {}, "a", "b", "c");
 | |
|         var result = new newFn();
 | |
|         [ result.hasOwnProperty("abc"), result.hasOwnProperty("def"), result.abc, result.def ];
 | |
|     `, "true,true,abc,true")
 | |
| }
 | |
| 
 | |
| func TestFunction_toString(t *testing.T) {
 | |
| 	Terst(t)
 | |
| 
 | |
| 	test := runTest()
 | |
| 	test(`raise:
 | |
|         Function.prototype.toString.call(undefined);
 | |
|     `, "TypeError")
 | |
| }
 | 
