--- /dev/null
+<?php
+
+namespace RedUNIT\Base;
+
+use RedUNIT\Base as Base;
+use RedBeanPHP\Facade as R;
+use RedBeanPHP\ModelHelper as ModelHelper;
+use RedBeanPHP\RedException as RedException;
+use RedBeanPHP\OODBBean as OODBBean;
+use RedBeanPHP\OODB as OODB;
+use RedBeanPHP\ToolBox as ToolBox;
+use RedBeanPHP\QueryWriter as QueryWriter;
+use RedBeanPHP\Adapter as Adapter;
+use RedBeanPHP\BeanHelper\SimpleFacadeBeanHelper as SimpleFacadeBeanHelper;
+
+/**
+ * FUSE
+ *
+ * Tests whether we can associate model logic on-the-fly
+ * by defining models extending from SimpleModel. Tests
+ * whether the calls to facade trigger the corresponding
+ * methods on the model.
+ *
+ * @file RedUNIT/Base/Fuse.php
+ * @desc Tests Fuse feature; coupling beans to models.
+ * @author Gabor de Mooij and the RedBeanPHP Community
+ * @license New BSD/GPLv2
+ *
+ * (c) G.J.G.T. (Gabor) de Mooij and the RedBeanPHP Community.
+ * This source file is subject to the New BSD/GPLv2 License that is bundled
+ * with this source code in the file license.txt.
+ */
+class Fuse extends Base
+{
+ /**
+ * Test whether we can override the getModelForBean() method
+ * of the BeanHelper and use a custom BeanHelper to attach a model
+ * based on type.
+ *
+ * @return void
+ */
+ public function testCustomBeanHelper()
+ {
+ $customBeanHelper = new \SoupBeanHelper( R::getToolbox() );
+ $oldBeanHelper = R::getRedBean()->getBeanHelper();
+ asrt( ( $oldBeanHelper instanceof SimpleFacadeBeanHelper ), TRUE );
+ R::getRedBean()->setBeanHelper( $customBeanHelper );
+ $meal = R::dispense( 'meal' );
+ asrt( ( $meal->box() instanceof \Model_Soup ), TRUE );
+ $cake = R::dispense( 'cake' );
+ asrt( is_null( $cake->box() ), TRUE );
+ $bean = R::dispense( 'coffee' );
+ asrt( ( $bean->box() instanceof \Model_Coffee ), TRUE );
+ $meal->setFlavour( 'tomato' );
+ asrt( $meal->getFlavour(), 'tomato' );
+ $meal->rating = 5;
+ R::store( $meal );
+ asrt( $meal->getFlavour(), 'tomato' );
+ $meal = $meal->unbox();
+ asrt( $meal->getFlavour(), 'tomato' );
+ $meal = R::findOne( 'meal' );
+ asrt( ( $meal->box() instanceof \Model_Soup ), TRUE );
+ asrt( $meal->getFlavour(), '' );
+ $meal->setFlavour( 'tomato' );
+ asrt( $meal->getFlavour(), 'tomato' );
+ $meal = $meal->unbox();
+ asrt( $meal->getFlavour(), 'tomato' );
+ R::getRedBean()->setBeanHelper( $oldBeanHelper );
+ }
+
+ /**
+ * Test FUSE hooks (i.e. open, update, update_after etc..)
+ *
+ * @return void
+ */
+ public function testHooks()
+ {
+ R::nuke();
+ $probe = R::dispense( 'probe' );
+ $probe->name = 'test';
+ asrt( $probe->getLogActionCount(), 1 );
+ asrt( $probe->getLogActionCount( 'dispense' ), 1 );
+ asrt( $probe->getLogActionCount( 'open' ), 0 );
+ asrt( $probe->getLogActionCount( 'update' ), 0 );
+ asrt( $probe->getLogActionCount( 'after_update' ), 0 );
+ asrt( $probe->getLogActionCount( 'delete' ), 0 );
+ asrt( $probe->getLogActionCount( 'after_delete' ), 0 );
+ asrt( ( $probe->getDataFromLog( 0, 'bean' ) === $probe ), TRUE );
+ R::store( $probe );
+ asrt( $probe->getLogActionCount(), 3 );
+ asrt( $probe->getLogActionCount( 'dispense' ), 1 );
+ asrt( $probe->getLogActionCount( 'open' ), 0 );
+ asrt( $probe->getLogActionCount( 'update' ), 1 );
+ asrt( $probe->getLogActionCount( 'after_update' ), 1 );
+ asrt( $probe->getLogActionCount( 'delete' ), 0 );
+ asrt( $probe->getLogActionCount( 'after_delete' ), 0 );
+ asrt( ( $probe->getDataFromLog( 2, 'bean' ) === $probe ), TRUE );
+ $probe = R::load( 'probe', $probe->id );
+ asrt( $probe->getLogActionCount(), 2 );
+ asrt( $probe->getLogActionCount( 'dispense' ), 1 );
+ asrt( $probe->getLogActionCount( 'open' ), 1 );
+ asrt( $probe->getLogActionCount( 'update' ), 0 );
+ asrt( $probe->getLogActionCount( 'after_update' ), 0 );
+ asrt( $probe->getLogActionCount( 'delete' ), 0 );
+ asrt( $probe->getLogActionCount( 'after_delete' ), 0 );
+ asrt( ( $probe->getDataFromLog( 0, 'bean' ) === $probe ), TRUE );
+ asrt( ( $probe->getDataFromLog( 1, 'id' ) === $probe->id ), TRUE );
+ $probe->clearLog();
+ R::trash( $probe );
+ asrt( $probe->getLogActionCount(), 2 );
+ asrt( $probe->getLogActionCount( 'dispense' ), 0 );
+ asrt( $probe->getLogActionCount( 'open' ), 0 );
+ asrt( $probe->getLogActionCount( 'update' ), 0 );
+ asrt( $probe->getLogActionCount( 'after_update' ), 0 );
+ asrt( $probe->getLogActionCount( 'delete' ), 1 );
+ asrt( $probe->getLogActionCount( 'after_delete' ), 1 );
+ asrt( ( $probe->getDataFromLog( 0, 'bean' ) === $probe ), TRUE );
+ asrt( ( $probe->getDataFromLog( 1, 'bean' ) === $probe ), TRUE );
+ //less 'normal scenarios'
+ $probe = R::dispense( 'probe' );
+ $probe->name = 'test';
+ asrt( $probe->getLogActionCount(), 1 );
+ asrt( $probe->getLogActionCount( 'dispense' ), 1 );
+ asrt( $probe->getLogActionCount( 'open' ), 0 );
+ asrt( $probe->getLogActionCount( 'update' ), 0 );
+ asrt( $probe->getLogActionCount( 'after_update' ), 0 );
+ asrt( $probe->getLogActionCount( 'delete' ), 0 );
+ asrt( $probe->getLogActionCount( 'after_delete' ), 0 );
+ asrt( ( $probe->getDataFromLog( 0, 'bean' ) === $probe ), TRUE );
+ R::store( $probe );
+ asrt( $probe->getLogActionCount(), 3 );
+ asrt( $probe->getLogActionCount( 'dispense' ), 1 );
+ asrt( $probe->getLogActionCount( 'open' ), 0 );
+ asrt( $probe->getLogActionCount( 'update' ), 1 );
+ asrt( $probe->getLogActionCount( 'after_update' ), 1 );
+ asrt( $probe->getLogActionCount( 'delete' ), 0 );
+ asrt( $probe->getLogActionCount( 'after_delete' ), 0 );
+ asrt( ( $probe->getDataFromLog( 2, 'bean' ) === $probe ), TRUE );
+ asrt( $probe->getMeta( 'tainted' ), FALSE );
+ asrt( $probe->getMeta( 'changed' ), FALSE );
+ R::store( $probe ); //not tainted, no FUSE save!
+ asrt( $probe->getLogActionCount(), 3 );
+ asrt( $probe->getLogActionCount( 'dispense' ), 1 );
+ asrt( $probe->getLogActionCount( 'open' ), 0 );
+ asrt( $probe->getLogActionCount( 'update' ), 1 );
+ asrt( $probe->getLogActionCount( 'after_update' ), 1 );
+ asrt( $probe->getLogActionCount( 'delete' ), 0 );
+ asrt( $probe->getLogActionCount( 'after_delete' ), 0 );
+ asrt( ( $probe->getDataFromLog( 2, 'bean' ) === $probe ), TRUE );
+ $probe->xownProbeList[] = R::dispense( 'probe' );
+ //tainted, not changed, triggers FUSE
+ asrt( $probe->getMeta( 'tainted' ), TRUE );
+ asrt( $probe->getMeta( 'changed' ), FALSE );
+ R::store( $probe );
+ asrt( $probe->getMeta( 'tainted' ), FALSE );
+ asrt( $probe->getMeta( 'changed' ), FALSE );
+ asrt( $probe->getLogActionCount(), 5 );
+ asrt( $probe->getLogActionCount( 'dispense' ), 1 );
+ asrt( $probe->getLogActionCount( 'open' ), 0 );
+ asrt( $probe->getLogActionCount( 'update' ), 2 );
+ asrt( $probe->getLogActionCount( 'after_update' ), 2 );
+ asrt( $probe->getLogActionCount( 'delete' ), 0 );
+ asrt( $probe->getLogActionCount( 'after_delete' ), 0 );
+ asrt( ( $probe->getDataFromLog( 2, 'bean' ) === $probe ), TRUE );
+ }
+
+ /**
+ * Tests the SimpleFacadeBeanHelper factory setter.
+ *
+ * @return void
+ */
+ public function testFactory()
+ {
+ SimpleFacadeBeanHelper::setFactoryFunction( function( $name ) {
+ $model = new $name();
+ $model->setNote( 'injected', 'dependency' );
+ return $model;
+ } );
+
+ $bean = R::dispense( 'band' )->box();
+
+ asrt( ( $bean instanceof \Model_Band ), TRUE );
+ asrt( ( $bean->getNote('injected') ), 'dependency' );
+
+ SimpleFacadeBeanHelper::setFactoryFunction( NULL );
+ }
+
+ /**
+ * Make sure that beans of type book_page can be fused with
+ * models like BookPage (beautified) as well as Book_Page (non-beautified).
+ */
+ public function testBeutificationOfLinkModel()
+ {
+ $page = R::dispense( 'page' );
+ $widget = R::dispense( 'widget' );
+ $page->sharedWidgetList[] = $widget;
+ R::store( $page );
+ $testReport = \Model_PageWidget::getTestReport();
+ asrt( $testReport, 'didSave' );
+
+ $page = R::dispense( 'page' );
+ $gadget = R::dispense( 'gadget' );
+ $page->sharedGadgetList[] = $gadget;
+ R::store( $page );
+ $testReport = \Model_Gadget_Page::getTestReport();
+ asrt( $testReport, 'didSave' );
+ }
+
+ /**
+ * Only theoretical.
+ *
+ * @return void
+ */
+ public function testTheoreticalBeautifications()
+ {
+ $bean = R::dispense('bean');
+ $bean->setMeta('type', 'a_b_c');
+ R::store($bean);
+ $testReport = \Model_A_B_C::getTestReport();
+ asrt( $testReport, 'didSave' );
+ }
+
+ /**
+ * Test extraction of toolbox.
+ *
+ * @return void
+ */
+ public function testGetExtractedToolBox()
+ {
+ $helper = new SimpleFacadeBeanHelper;
+
+ list( $redbean, $database, $writer, $toolbox ) = $helper->getExtractedToolbox();
+
+ asrt( ( $redbean instanceof OODB ), TRUE );
+ asrt( ( $database instanceof Adapter ), TRUE );
+ asrt( ( $writer instanceof QueryWriter ), TRUE );
+ asrt( ( $toolbox instanceof ToolBox ), TRUE );
+ }
+
+ /**
+ * Test FUSE and model formatting.
+ *
+ * @todo move tagging tests to tag tester.
+ *
+ * @return void
+ */
+ public function testFUSE()
+ {
+ $toolbox = R::getToolBox();
+ $adapter = $toolbox->getDatabaseAdapter();
+ $blog = R::dispense( 'blog' );
+ $blog->title = 'testing';
+ $blog->blog = 'tesing';
+ R::store( $blog );
+ $blogpost = R::load( "blog", 1 );
+ $post = R::dispense( "post" );
+ $post->message = "hello";
+ $blog->sharedPost[] = $post;
+ R::store($blog);
+ $a = R::getAll( "select * from blog " );
+ R::tag( $post, "lousy,smart" );
+ asrt( implode( ',', R::tag( $post ) ), "lousy,smart" );
+ R::tag( $post, "clever,smart" );
+ $tagz = implode( ',', R::tag( $post ) );
+ asrt( ( $tagz == "smart,clever" || $tagz == "clever,smart" ), TRUE );
+ R::tag( $blog, array( "smart", "interesting" ) );
+ asrt( implode( ',', R::tag( $blog ) ), "smart,interesting" );
+ try {
+ R::tag( $blog, array( "smart", "interesting", "lousy!" ) );
+ pass();
+ } catch ( RedException $e ) {
+ fail();
+ }
+ asrt( implode( ',', R::tag( $blog ) ), "smart,interesting,lousy!" );
+ asrt( implode( ",", R::tag( $blog ) ), "smart,interesting,lousy!" );
+ R::untag( $blog, array( "smart", "interesting" ) );
+ asrt( implode( ",", R::tag( $blog ) ), "lousy!" );
+ asrt( R::hasTag( $blog, array( "lousy!" ) ), TRUE );
+ asrt( R::hasTag( $blog, array( "lousy!", "smart" ) ), TRUE );
+ asrt( R::hasTag( $blog, array( "lousy!", "smart" ), TRUE ), FALSE );
+ R::tag( $blog, FALSE );
+ asrt( count( R::tag( $blog ) ), 0 );
+ R::tag( $blog, array( "funny", "comic" ) );
+ asrt( count( R::tag( $blog ) ), 2 );
+ R::addTags( $blog, array( "halloween" ) );
+ asrt( count( R::tag( $blog ) ), 3 );
+ asrt( R::hasTag( $blog, array( "funny", "commic", "halloween" ), TRUE ), FALSE );
+ R::unTag( $blog, array( "funny" ) );
+ R::addTags( $blog, "horror" );
+ asrt( count( R::tag( $blog ) ), 3 );
+ asrt( R::hasTag( $blog, array( "horror", "commic", "halloween" ), TRUE ), FALSE );
+ // No double tags
+ R::addTags( $blog, "horror" );
+ asrt( R::hasTag( $blog, array( "horror", "commic", "halloween" ), TRUE ), FALSE );
+ asrt( count( R::tag( $blog ) ), 3 );
+ }
+
+ /**
+ * Test error handling options FUSE.
+ */
+ public function testModelErrorHandling()
+ {
+ $test = R::dispense( 'feed' );
+ $test->nonExistantMethod();
+ pass();
+ $old = R::setErrorHandlingFUSE( OODBBean::C_ERR_LOG );
+ asrt( is_array( $old ), TRUE );
+ asrt( count( $old ), 2 );
+ asrt( $old[0], FALSE );
+ asrt( $old[1], NULL);
+ $test->nonExistantMethod(); //we cant really test this... :(
+ pass();
+ $old = R::setErrorHandlingFUSE( OODBBean::C_ERR_NOTICE );
+ asrt( is_array( $old ), TRUE );
+ asrt( count( $old ), 2 );
+ asrt( $old[0], OODBBean::C_ERR_LOG );
+ asrt( $old[1], NULL);
+ set_error_handler(function($error, $str) {
+ asrt( $str, 'FUSE: method does not exist in model: nonExistantMethod' );
+ }, E_USER_NOTICE);
+ $test->nonExistantMethod();
+ restore_error_handler();
+ $old = OODBBean::setErrorHandlingFUSE( OODBBean::C_ERR_WARN );
+ asrt( is_array( $old ), TRUE );
+ asrt( count( $old ), 2 );
+ asrt( $old[0], OODBBean::C_ERR_NOTICE );
+ asrt( $old[1], NULL);
+ set_error_handler(function($error, $str) {
+ asrt( $str, 'FUSE: method does not exist in model: nonExistantMethod' );
+ }, E_USER_WARNING);
+ $test->nonExistantMethod();
+ restore_error_handler();
+ $old = OODBBean::setErrorHandlingFUSE( OODBBean::C_ERR_FATAL );
+ asrt( is_array( $old ), TRUE );
+ asrt( count( $old ), 2 );
+ asrt( $old[0], OODBBean::C_ERR_WARN );
+ asrt( $old[1], NULL);
+ set_error_handler(function($error, $str) {
+ asrt( $str, 'FUSE: method does not exist in model: nonExistantMethod' );
+ }, E_USER_ERROR);
+ $test->nonExistantMethod();
+ restore_error_handler();
+ $old = OODBBean::setErrorHandlingFUSE( OODBBean::C_ERR_EXCEPTION );
+ asrt( is_array( $old ), TRUE );
+ asrt( count( $old ), 2 );
+ asrt( $old[0], OODBBean::C_ERR_FATAL );
+ asrt( $old[1], NULL);
+ try {
+ $test->nonExistantMethod();
+ fail();
+ } catch (\Exception $e) {
+ pass();
+ }
+ global $test_bean;
+ $test_bean = $test;
+ global $has_executed_error_func_fuse;
+ $has_executed_error_func_fuse = FALSE;
+ $old = OODBBean::setErrorHandlingFUSE( OODBBean::C_ERR_FUNC, function( $info ){
+ global $has_executed_error_func_fuse;
+ global $test_bean;
+ $has_executed_error_func_fuse = TRUE;
+ asrt( is_array( $info ), TRUE );
+ asrt( $info['method'], 'nonExistantMethod' );
+ asrt( json_encode( $info['bean']->export() ), json_encode( $test_bean->export() ) );
+ asrt( $info['message'], 'FUSE: method does not exist in model: nonExistantMethod' );
+ } );
+ asrt( is_array( $old ), TRUE );
+ asrt( count( $old ), 2 );
+ asrt( $old[0], OODBBean::C_ERR_EXCEPTION );
+ asrt( $old[1], NULL);
+ $test->nonExistantMethod();
+ asrt( $has_executed_error_func_fuse, TRUE );
+ $old = OODBBean::setErrorHandlingFUSE( OODBBean::C_ERR_IGNORE );
+ asrt( is_array( $old ), TRUE );
+ asrt( count( $old ), 2 );
+ asrt( $old[0], OODBBean::C_ERR_FUNC );
+ asrt( is_callable( $old[1] ), TRUE );
+ $old = OODBBean::setErrorHandlingFUSE( OODBBean::C_ERR_IGNORE );
+ asrt( is_array( $old ), TRUE );
+ asrt( count( $old ), 2 );
+ asrt( $old[0], OODBBean::C_ERR_IGNORE );
+ asrt( $old[1], NULL);
+ try {
+ OODBBean::setErrorHandlingFUSE( 900 );
+ fail();
+ } catch (\Exception $e) {
+ pass();
+ asrt( $e->getMessage(), 'Invalid error mode selected' );
+ }
+ try {
+ OODBBean::setErrorHandlingFUSE( OODBBean::C_ERR_FUNC, 'hello' );
+ fail();
+ } catch (\Exception $e) {
+ pass();
+ asrt( $e->getMessage(), 'Invalid error handler' );
+ }
+ OODBBean::setErrorHandlingFUSE( OODBBean::C_ERR_EXCEPTION );
+ //make sure ignore FUSE events
+ $test = R::dispense('feed');
+ R::store( $test );
+ $test = $test->fresh();
+ R::trash( $test );
+ pass();
+ OODBBean::setErrorHandlingFUSE( OODBBean::C_ERR_IGNORE );
+ }
+}