2008-05-30 11:40:08 +00:00
< ? php
/**
* The TestTask handles creating and updating test files .
*
* PHP versions 4 and 5
*
2008-10-30 17:30:26 +00:00
* CakePHP ( tm ) : Rapid Development Framework ( http :// www . cakephp . org )
2009-07-16 23:55:41 -04:00
* Copyright 2005 - 2009 , Cake Software Foundation , Inc . ( http :// www . cakefoundation . org )
2008-05-30 11:40:08 +00:00
*
* Licensed under The MIT License
* Redistributions of files must retain the above copyright notice .
*
* @ filesource
2009-07-16 23:55:41 -04:00
* @ copyright Copyright 2005 - 2009 , Cake Software Foundation , Inc . ( http :// www . cakefoundation . org )
2008-10-30 17:30:26 +00:00
* @ link http :// www . cakefoundation . org / projects / info / cakephp CakePHP ( tm ) Project
* @ package cake
* @ subpackage cake . cake . console . libs . tasks
2009-07-16 23:55:41 -04:00
* @ since CakePHP ( tm ) v 1.3
2008-10-30 17:30:26 +00:00
* @ license http :// www . opensource . org / licenses / mit - license . php The MIT License
2008-05-30 11:40:08 +00:00
*/
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Task class for creating and updating test files .
*
2008-10-30 17:30:26 +00:00
* @ package cake
* @ subpackage cake . cake . console . libs . tasks
2008-05-30 11:40:08 +00:00
*/
class TestTask extends Shell {
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Name of plugin
*
* @ var string
* @ access public
2008-09-24 13:49:37 +00:00
*/
2008-05-30 11:40:08 +00:00
var $plugin = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* path to TESTS directory
*
* @ var string
* @ access public
*/
var $path = TESTS ;
2009-05-24 20:12:17 -04:00
2009-05-24 23:50:21 -04:00
/**
* Tasks used .
*
* @ var array
**/
var $tasks = array ( 'Template' );
2009-07-24 21:18:37 +02:00
2009-05-24 20:12:17 -04:00
/**
* class types that methods can be generated for
*
* @ var array
**/
var $classTypes = array ( 'Model' , 'Controller' , 'Component' , 'Behavior' , 'Helper' );
2009-07-24 21:18:37 +02:00
2009-05-24 23:50:21 -04:00
/**
* Internal list of fixtures that have been added so far .
*
* @ var string
**/
var $_fixtures = array ();
2009-07-24 21:18:37 +02:00
2009-05-24 23:50:21 -04:00
/**
* Flag for interactive mode
*
* @ var boolean
**/
var $interactive = false ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Execution method always used for tasks
*
* @ access public
*/
function execute () {
if ( empty ( $this -> args )) {
$this -> __interactive ();
}
2008-09-24 13:49:37 +00:00
2008-05-30 11:40:08 +00:00
if ( count ( $this -> args ) == 1 ) {
$this -> __interactive ( $this -> args [ 0 ]);
}
if ( count ( $this -> args ) > 1 ) {
2009-05-24 23:50:21 -04:00
$type = Inflector :: underscore ( $this -> args [ 0 ]);
if ( $this -> bake ( $type , $this -> args [ 1 ])) {
2008-05-30 11:40:08 +00:00
$this -> out ( 'done' );
}
}
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Handles interactive baking
*
* @ access private
*/
2009-05-24 20:12:17 -04:00
function __interactive ( $type = null ) {
2009-05-24 23:50:21 -04:00
$this -> interactive = true ;
2008-05-30 11:40:08 +00:00
$this -> hr ();
2009-05-24 20:12:17 -04:00
$this -> out ( __ ( 'Bake Tests' , true ));
$this -> out ( sprintf ( __ ( " Path: %s " , true ), $this -> path ));
2008-05-30 11:40:08 +00:00
$this -> hr ();
2008-09-24 13:49:37 +00:00
2009-05-24 20:12:17 -04:00
$selection = null ;
if ( $type ) {
$type = Inflector :: camelize ( $type );
2009-05-24 23:50:21 -04:00
if ( ! in_array ( $type , $this -> classTypes )) {
unset ( $type );
2008-05-30 11:40:08 +00:00
}
}
2009-05-24 23:50:21 -04:00
if ( ! $type ) {
$type = $this -> getObjectType ();
2009-05-24 20:12:17 -04:00
}
2009-05-24 23:50:21 -04:00
$className = $this -> getClassName ( $type );
return $this -> bake ( $type , $className );
2008-05-30 11:40:08 +00:00
}
2009-07-24 21:18:37 +02:00
2009-05-24 20:12:17 -04:00
/**
2009-05-24 23:50:21 -04:00
* Completes final steps for generating data to create test case .
2009-05-24 20:12:17 -04:00
*
2009-05-24 23:50:21 -04:00
* @ param string $type Type of object to bake test case for ie . Model , Controller
2009-07-24 21:18:37 +02:00
* @ param string $className the 'cake name' for the class ie . Posts for the PostsController
2009-05-24 23:50:21 -04:00
* @ access public
*/
function bake ( $type , $className ) {
if ( $this -> typeCanDetectFixtures ( $type ) && $this -> isLoadableClass ( $type , $className )) {
$this -> out ( __ ( 'Bake is detecting possible fixtures..' , true ));
$testSubject =& $this -> buildTestSubject ( $type , $className );
$this -> generateFixtureList ( $testSubject );
} elseif ( $this -> interactive ) {
$this -> getUserFixtures ();
}
$fullClassName = $this -> getRealClassName ( $type , $className );
2009-05-24 20:12:17 -04:00
2009-05-24 23:50:21 -04:00
$methods = array ();
if ( class_exists ( $fullClassName )) {
$methods = $this -> getTestableMethods ( $fullClassName );
2009-05-24 20:12:17 -04:00
}
2009-05-25 23:48:41 -04:00
$mock = $this -> hasMockClass ( $type , $fullClassName );
2009-05-25 00:54:23 -04:00
$construction = $this -> generateConstructor ( $type , $fullClassName );
$plugin = null ;
if ( $this -> plugin ) {
$plugin = $this -> plugin . '.' ;
}
2009-05-24 23:50:21 -04:00
$this -> Template -> set ( 'fixtures' , $this -> _fixtures );
2009-05-25 00:54:23 -04:00
$this -> Template -> set ( 'plugin' , $plugin );
$this -> Template -> set ( compact ( 'className' , 'methods' , 'type' , 'fullClassName' , 'mock' , 'construction' ));
2009-07-01 00:50:38 -04:00
$out = $this -> Template -> generate ( 'classes' , 'test' );
2009-05-24 23:50:21 -04:00
2009-06-06 21:03:26 -04:00
$filename = $this -> testCaseFileName ( $type , $className );
$made = $this -> createFile ( $filename , $out );
2009-05-24 23:50:21 -04:00
if ( $made ) {
return $out ;
}
return false ;
2008-05-30 11:40:08 +00:00
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2009-05-24 23:50:21 -04:00
* Interact with the user and get their chosen type . Can exit the script .
2008-05-30 11:40:08 +00:00
*
2009-05-24 23:50:21 -04:00
* @ return string Users chosen type .
**/
function getObjectType () {
$this -> hr ();
$this -> out ( __ ( " Select an object type: " , true ));
$this -> hr ();
$keys = array ();
foreach ( $this -> classTypes as $key => $option ) {
$this -> out ( ++ $key . '. ' . $option );
$keys [] = $key ;
2008-05-30 11:40:08 +00:00
}
2009-05-24 23:50:21 -04:00
$keys [] = 'q' ;
$selection = $this -> in ( __ ( " Enter the type of object to bake a test for or (q)uit " , true ), $keys , 'q' );
if ( $selection == 'q' ) {
return $this -> _stop ();
}
return $this -> classTypes [ $selection - 1 ];
}
2009-07-24 21:18:37 +02:00
2009-05-24 23:50:21 -04:00
/**
* Get the user chosen Class name for the chosen type
*
* @ param string $objectType Type of object to list classes for i . e . Model , Controller .
* @ return string Class name the user chose .
**/
function getClassName ( $objectType ) {
$options = Configure :: listObjects ( strtolower ( $objectType ));
$this -> out ( sprintf ( __ ( 'Choose a %s class' , true ), $objectType ));
$keys = array ();
foreach ( $options as $key => $option ) {
$this -> out ( ++ $key . '. ' . $option );
$keys [] = $key ;
}
$selection = $this -> in ( __ ( 'Choose an existing class, or enter the name of a class that does not exist' , true ));
if ( isset ( $options [ $selection - 1 ])) {
return $options [ $selection - 1 ];
}
return $selection ;
}
2009-07-24 21:18:37 +02:00
2009-05-24 23:50:21 -04:00
/**
* Checks whether the chosen type can find its own fixtures .
* Currently only model , and controller are supported
*
* @ return boolean
**/
function typeCanDetectFixtures ( $type ) {
$type = strtolower ( $type );
return ( $type == 'controller' || $type == 'model' );
}
2009-07-24 21:18:37 +02:00
2009-05-24 23:50:21 -04:00
/**
* Check if a class with the given type is loaded or can be loaded .
*
* @ return boolean
**/
function isLoadableClass ( $type , $class ) {
return App :: import ( $type , $class );
}
2009-07-24 21:18:37 +02:00
2009-05-24 23:50:21 -04:00
/**
* Construct an instance of the class to be tested .
2009-05-25 00:54:23 -04:00
* So that fixtures can be detected
2009-05-24 23:50:21 -04:00
*
* @ return object
**/
function & buildTestSubject ( $type , $class ) {
2009-05-26 00:36:25 -04:00
ClassRegistry :: flush ();
2009-05-24 23:50:21 -04:00
App :: import ( $type , $class );
$class = $this -> getRealClassName ( $type , $class );
if ( strtolower ( $type ) == 'model' ) {
$instance =& ClassRegistry :: init ( $class );
} else {
$instance =& new $class ();
}
return $instance ;
}
2009-07-24 21:18:37 +02:00
2009-05-24 23:50:21 -04:00
/**
* Gets the real class name from the cake short form .
*
* @ return string Real classname
**/
function getRealClassName ( $type , $class ) {
if ( strtolower ( $type ) == 'model' ) {
return $class ;
}
return $class . $type ;
2008-05-30 11:40:08 +00:00
}
2009-07-24 21:18:37 +02:00
2009-05-23 23:48:25 -04:00
/**
* Get methods declared in the class given .
* No parent methods will be returned
*
* @ param string $className Name of class to look at .
* @ return array Array of method names .
**/
function getTestableMethods ( $className ) {
$classMethods = get_class_methods ( $className );
$parentMethods = get_class_methods ( get_parent_class ( $className ));
$thisMethods = array_diff ( $classMethods , $parentMethods );
$out = array ();
foreach ( $thisMethods as $method ) {
if ( substr ( $method , 0 , 1 ) != '_' ) {
$out [] = $method ;
}
}
return $out ;
}
2009-07-24 21:18:37 +02:00
2009-05-24 01:15:31 -04:00
/**
2009-05-24 01:30:04 -04:00
* Generate the list of fixtures that will be required to run this test based on
2009-05-24 01:15:31 -04:00
* loaded models .
*
* @ param object The object you want to generate fixtures for .
* @ return array Array of fixtures to be included in the test .
**/
function generateFixtureList ( & $subject ) {
$this -> _fixtures = array ();
if ( is_a ( $subject , 'Model' )) {
$this -> _processModel ( $subject );
} elseif ( is_a ( $subject , 'Controller' )) {
$this -> _processController ( $subject );
}
return array_values ( $this -> _fixtures );
}
2009-07-24 21:18:37 +02:00
2009-05-24 01:15:31 -04:00
/**
2009-05-24 01:30:04 -04:00
* Process a model recursively and pull out all the
2009-05-24 01:15:31 -04:00
* model names converting them to fixture names .
*
* @ return void
2009-05-24 23:50:21 -04:00
* @ access protected
2009-05-24 01:15:31 -04:00
**/
function _processModel ( & $subject ) {
$this -> _addFixture ( $subject -> name );
$associated = $subject -> getAssociated ();
foreach ( $associated as $alias => $type ) {
$className = $subject -> { $alias } -> name ;
if ( ! isset ( $this -> _fixtures [ $className ])) {
$this -> _processModel ( $subject -> { $alias });
}
if ( $type == 'hasAndBelongsToMany' ) {
$joinModel = Inflector :: classify ( $subject -> hasAndBelongsToMany [ $alias ][ 'joinTable' ]);
if ( ! isset ( $this -> _fixtures [ $joinModel ])) {
$this -> _processModel ( $subject -> { $joinModel });
}
}
}
}
2009-07-24 21:18:37 +02:00
2009-05-24 01:30:04 -04:00
/**
* Process all the models attached to a controller
* and generate a fixture list .
*
* @ return void
2009-05-24 23:50:21 -04:00
* @ access protected
2009-05-24 01:30:04 -04:00
**/
function _processController ( & $subject ) {
$subject -> constructClasses ();
$models = array ( Inflector :: classify ( $subject -> name ));
if ( ! empty ( $subject -> uses )) {
$models = $subject -> uses ;
}
foreach ( $models as $model ) {
$this -> _processModel ( $subject -> { $model });
}
}
2009-07-24 21:18:37 +02:00
2009-05-24 01:15:31 -04:00
/**
* Add classname to the fixture list .
* Sets the app . or plugin . plugin_name . prefix .
*
* @ return void
2009-05-24 23:50:21 -04:00
* @ access protected
2009-05-24 01:15:31 -04:00
**/
function _addFixture ( $name ) {
$parent = get_parent_class ( $name );
$prefix = 'app.' ;
if ( strtolower ( $parent ) != 'appmodel' && strtolower ( substr ( $parent , - 8 )) == 'appmodel' ) {
$pluginName = substr ( $parent , 0 , strlen ( $parent ) - 8 );
$prefix = 'plugin.' . Inflector :: underscore ( $pluginName ) . '.' ;
}
$fixture = $prefix . Inflector :: underscore ( $name );
$this -> _fixtures [ $name ] = $fixture ;
}
2009-07-24 21:18:37 +02:00
2009-05-24 23:50:21 -04:00
/**
* Interact with the user to get additional fixtures they want to use .
*
* @ return void
**/
function getUserFixtures () {
$proceed = $this -> in ( __ ( 'Bake could not detect fixtures, would you like to add some?' , true ), array ( 'y' , 'n' ), 'n' );
$fixtures = array ();
if ( strtolower ( $proceed ) == 'y' ) {
$fixtureList = $this -> in ( __ ( " Please provide a comma separated list of the fixtures names you'd like to use. \n Example: 'app.comment, app.post, plugin.forums.post' " , true ));
$fixtureListTrimmed = str_replace ( ' ' , '' , $fixtureList );
$fixtures = explode ( ',' , $fixtureListTrimmed );
}
$this -> _fixtures = array_merge ( $this -> _fixtures , $fixtures );
return $fixtures ;
}
2009-07-24 21:18:37 +02:00
2009-05-25 00:54:23 -04:00
/**
2009-05-25 23:48:41 -04:00
* Is a mock class required for this type of test ?
* Controllers require a mock class .
2009-05-25 00:54:23 -04:00
*
2009-05-25 23:48:41 -04:00
* @ return boolean
2009-05-25 00:54:23 -04:00
**/
2009-05-25 23:48:41 -04:00
function hasMockClass ( $type ) {
$type = strtolower ( $type );
return $type == 'controller' ;
2009-05-25 00:54:23 -04:00
}
2009-07-24 21:18:37 +02:00
2009-05-25 00:54:23 -04:00
/**
* Generate a constructor code snippet for the type and classname
*
* @ return string Constructor snippet for the thing you are building .
**/
function generateConstructor ( $type , $fullClassName ) {
$type = strtolower ( $type );
if ( $type == 'model' ) {
2009-05-25 01:02:59 -04:00
return " ClassRegistry::init(' $fullClassName '); \n " ;
2009-05-25 00:54:23 -04:00
}
if ( $type == 'controller' ) {
2009-05-25 01:02:59 -04:00
return " new Test $fullClassName (); \n \t \t \$ this-> { $fullClassName } ->constructClasses(); \n " ;
2009-05-25 00:54:23 -04:00
}
2009-05-25 01:02:59 -04:00
return " new $fullClassName () \n " ;
2009-05-25 00:54:23 -04:00
}
2009-07-24 21:18:37 +02:00
2009-06-06 21:03:26 -04:00
/**
* make the filename for the test case . resolve the suffixes for controllers
* and get the plugin path if needed .
*
* @ return string filename the test should be created on
**/
function testCaseFileName ( $type , $className ) {
$path = $this -> path ;
if ( isset ( $this -> plugin )) {
$path = $this -> _pluginPath ( $this -> plugin ) . 'tests' . DS ;
}
$path .= 'cases' . DS . Inflector :: tableize ( $type ) . DS ;
if ( strtolower ( $type ) == 'controller' ) {
$className = $this -> getRealClassName ( $type , $className );
}
return $path . Inflector :: underscore ( $className ) . '.test.php' ;
}
2008-05-30 11:40:08 +00:00
}
?>