--TEST--
Test V8::executeString() : Object with magic functions
--SKIPIF--
<?php require_once(dirname(__FILE__) . '/skipif.inc'); ?>
--FILE--
<?php

class Foo {
	var $bar = 'foobar';
	var $nullprop = null;

	function Foo() {
		echo "called constructor: ";
		var_dump(func_get_args());
	}

	function MyOwnFunc() {
		echo "called MyOwnFunc\n";
	}

	function __Get($name) {
		echo "Called __get(): ";
		var_dump($name);
		return null;
	}
	function __Set($name, $args) {
		echo "Called __set(): ";
		var_dump($name, $args);
	}
	function __Isset($name) {
		echo "Called __isset(): ";
		var_dump($name);
		return true;
	}
	function __unSet($name) {
		echo "Called __unset(): ";
		var_dump($name);
	}
	function __call($name, $args) {
		echo "Called __call(): ";
		var_dump($name, $args);
		return "call";
	}
	function __Invoke($name, $arg1, $arg2) {
		echo "Called __invoke(): ";
		var_dump(func_get_args());
		return 'foobar';
	}
	function __toString() {
		echo "Called __tostring: ";
		return $this->bar;
	}
}

class Bar {
	function foo($arg1, $arg2, $arg3) {
		echo "Called foo(): ";
		var_dump(func_get_args());
		return "test";
	}
}


$blaa = new V8Js();
$blaa->obj = $obj = new Foo;

try {
  echo "__invoke() [PHP]\n";
  var_dump($obj('arg1','arg2','arg3'));
  echo "__invoke() [JS]\n";
  $blaa->executeString("var_dump(PHP.obj('arg1','arg2','arg3'));", "invoke_test1 #1.js");
  echo "------------\n";

  echo " __invoke() with new [PHP]\n";
  $myobj = new $obj('arg1','arg2','arg3'); $myobj->myownfunc();
  echo " __invoke() with new [JS]\n";
  $blaa->executeString("myobj = new PHP.obj('arg1','arg2','arg3'); myobj.myownfunc();", "invoke_test2 #2.js");
  echo "------------\n";

  echo " __tostring() [PHP]\n";
  echo $obj; echo "\n";
  echo " __tostring() [JS]\n";
  $blaa->executeString('print(PHP.obj + "\n");', "tostring_test #3.js");
  echo "------------\n";

  echo " __isset() not called with existing property [PHP]\n";
  if (isset($obj->bar)) { echo "bar exists\n"; }
  echo " __isset() not called with existing property [JS]\n";
  $blaa->executeString('if ("bar" in PHP.obj) print("bar exists\n");', "isset_test1 #4.js");
  echo "------------\n";

  echo " __isset() with non-existing property [PHP]\n";
  if (!isset($obj->foobar)) { echo "foobar does not exist\n"; } else { echo "We called __isset and it said yes!\n"; }
  echo " __isset() with non-existing property [JS]\n";
  $blaa->executeString('if (!("foobar" in PHP.obj)) print("foobar does not exist\n"); else print("We called __isset and it said yes!\n");', "isset_test2 #5.js");
  echo "------------\n";

  echo " in works like isset [PHP]\n";
  echo "nullprop is ", (isset($obj->nullprop) ? "" : "not "), "set\n";
  echo " in works like isset [JS]\n";
  $blaa->executeString('print("nullprop is ", ("nullprop" in PHP.obj) ? "" : "not ", "set\n");', "isset_test3 #6.js");
  echo "------------\n";

  echo " __get() not called with existing property [PHP]\n";
  var_dump($obj->bar);
  echo " __get() not called with existing property [JS]\n";
  $blaa->executeString('var_dump(PHP.obj.bar);', "get_test1 #7.js");
  echo "------------\n";

  echo " __get() with non-existing property [PHP]\n";
  var_dump($obj->fooish);
  echo " __get() with non-existing property [JS]\n";
  $blaa->executeString('var_dump(PHP.obj.fooish);', "get_test2 #8.js");
  echo "------------\n";

  echo " __unset() with non-existing property [PHP]\n";
  unset($obj->foobar);
  echo " __unset() with non-existing property [JS]\n";
  $blaa->executeString('delete PHP.obj.foobar;', "unset_test1 #9.js");
  echo "------------\n";

  echo " __unset() with existing property [PHP]\n";
  $obj2 = new Foo; unset($obj2->bar);
  echo " __unset() with existing property [JS]\n";
  $blaa->obj2 = new Foo;
  $blaa->executeString('delete PHP.obj2.bar;', "unset_test2 #10.js");
  echo " fetching the unset property [PHP]\n";
  var_dump($obj2->bar);
  echo " fetching the unset property [JS]\n";
  $blaa->executeString('var_dump(PHP.obj2.bar);', "unset_test3 #11.js");
  echo "------------\n";

  echo " __call() [PHP]\n";
  var_dump($obj->fooish(1,2,3));
  echo " __call() [JS]\n";
  # note that 'PHP.obj.fooish(1,2,3)' won't work in JS, we need to use the
  # '__call' pseudo-method.
  $blaa->executeString('var_dump(PHP.obj.__call("fooish", [1,2,3]));', "call_test1 #12.js");
  echo "------------\n";

  # the __call pseudo-method should work in JS even if the PHP class doesn't
  # define an explicit __call magic function.  This makes it always safe to
  # use __call() if you want to be sure that any __call() handlers are invoked
  # (bypassing __get handlers, as is it done in PHP)
  $blaa->obj3 = $obj3 = new Bar;
  echo " __call() w/o handler [PHP]\n";
  var_dump($obj3->foo(1,2,3));
  echo " __call() w/o handler [JS]\n";
  $blaa->executeString('var_dump(PHP.obj3.__call("foo", [1,2,3]));', "call_test2 #13.js");
  echo "------------\n";

  # The Bar object should inherit toString() and hasOwnProperty() methods
  # from Object
  echo " __toString in Bar [PHP]\n";
  var_dump(method_exists( $obj3, '__toString' ));
  echo " toString in Bar [PHP]\n";
  var_dump(method_exists( $obj3, 'toString' ));
  echo " hasOwnProperty in Bar [PHP]\n";
  var_dump(method_exists( $obj3, 'hasOwnProperty' ));
  echo " __toString in Bar [JS]\n";
  $blaa->executeString('var_dump("__toString" in PHP.obj3 && typeof PHP.obj3.__toString == "function");', "inherit_test1 #14.js");
  # use '$toString' if you actually wanted to check for a PHP property
  # named 'toString' in Bar (instead of the inherited JavaScript property)
  echo " toString in Bar [JS]\n";
  $blaa->executeString('var_dump("toString" in PHP.obj3 && typeof PHP.obj3.toString == "function");', "inherit_test1 #15.js");
  # use '$hasOwnProperty' if you actually wanted to check for a PHP property
  # named 'hasOwnProperty' in Bar (instead of the inherited JavaScript property)
  echo " hasOwnProperty in Bar [JS]\n";
  $blaa->executeString('var_dump("hasOwnProperty" in PHP.obj3 && typeof PHP.obj3.hasOwnProperty == "function");', "inherit_test1 #16.js");
  echo "------------\n";

} catch (V8JsScriptException $e) {
  echo $e->getMessage(), "\n";
}
?>
===EOF===
--EXPECT--
called constructor: array(0) {
}
__invoke() [PHP]
Called __invoke(): array(3) {
  [0]=>
  string(4) "arg1"
  [1]=>
  string(4) "arg2"
  [2]=>
  string(4) "arg3"
}
string(6) "foobar"
__invoke() [JS]
Called __invoke(): array(3) {
  [0]=>
  string(4) "arg1"
  [1]=>
  string(4) "arg2"
  [2]=>
  string(4) "arg3"
}
string(6) "foobar"
------------
 __invoke() with new [PHP]
called constructor: array(3) {
  [0]=>
  string(4) "arg1"
  [1]=>
  string(4) "arg2"
  [2]=>
  string(4) "arg3"
}
called MyOwnFunc
 __invoke() with new [JS]
called constructor: array(3) {
  [0]=>
  string(4) "arg1"
  [1]=>
  string(4) "arg2"
  [2]=>
  string(4) "arg3"
}
called MyOwnFunc
------------
 __tostring() [PHP]
Called __tostring: foobar
 __tostring() [JS]
Called __get(): string(7) "valueOf"
Called __tostring: foobar
------------
 __isset() not called with existing property [PHP]
bar exists
 __isset() not called with existing property [JS]
bar exists
------------
 __isset() with non-existing property [PHP]
Called __isset(): string(6) "foobar"
We called __isset and it said yes!
 __isset() with non-existing property [JS]
Called __isset(): string(6) "foobar"
We called __isset and it said yes!
------------
 in works like isset [PHP]
nullprop is not set
 in works like isset [JS]
nullprop is not set
------------
 __get() not called with existing property [PHP]
string(6) "foobar"
 __get() not called with existing property [JS]
string(6) "foobar"
------------
 __get() with non-existing property [PHP]
Called __get(): string(6) "fooish"
NULL
 __get() with non-existing property [JS]
Called __get(): string(6) "fooish"
NULL
------------
 __unset() with non-existing property [PHP]
Called __unset(): string(6) "foobar"
 __unset() with non-existing property [JS]
Called __unset(): string(6) "foobar"
------------
 __unset() with existing property [PHP]
called constructor: array(0) {
}
 __unset() with existing property [JS]
called constructor: array(0) {
}
 fetching the unset property [PHP]
Called __get(): string(3) "bar"
NULL
 fetching the unset property [JS]
Called __get(): string(3) "bar"
NULL
------------
 __call() [PHP]
Called __call(): string(6) "fooish"
array(3) {
  [0]=>
  int(1)
  [1]=>
  int(2)
  [2]=>
  int(3)
}
string(4) "call"
 __call() [JS]
Called __call(): string(6) "fooish"
array(3) {
  [0]=>
  int(1)
  [1]=>
  int(2)
  [2]=>
  int(3)
}
string(4) "call"
------------
 __call() w/o handler [PHP]
Called foo(): array(3) {
  [0]=>
  int(1)
  [1]=>
  int(2)
  [2]=>
  int(3)
}
string(4) "test"
 __call() w/o handler [JS]
Called foo(): array(3) {
  [0]=>
  int(1)
  [1]=>
  int(2)
  [2]=>
  int(3)
}
string(4) "test"
------------
 __toString in Bar [PHP]
bool(false)
 toString in Bar [PHP]
bool(false)
 hasOwnProperty in Bar [PHP]
bool(false)
 __toString in Bar [JS]
bool(false)
 toString in Bar [JS]
bool(true)
 hasOwnProperty in Bar [JS]
bool(true)
------------
===EOF===