2008-05-30 11:40:08 +00:00
< ? php
/**
* The Project Task handles creating the base application
*
*
2010-10-03 12:38:58 -04:00
* PHP 5
2008-05-30 11:40:08 +00:00
*
2009-11-06 17:46:59 +11:00
* CakePHP ( tm ) : Rapid Development Framework ( http :// cakephp . org )
2011-05-29 17:31:39 -04:00
* Copyright 2005 - 2011 , Cake Software Foundation , Inc . ( http :// cakefoundation . org )
2008-05-30 11:40:08 +00:00
*
* Licensed under The MIT License
* Redistributions of files must retain the above copyright notice .
*
2011-05-29 17:31:39 -04:00
* @ copyright Copyright 2005 - 2011 , Cake Software Foundation , Inc . ( http :// cakefoundation . org )
2009-11-06 17:00:11 +11:00
* @ link http :// cakephp . org CakePHP ( tm ) Project
2010-12-24 14:26:26 -05:00
* @ package cake . console . shells . tasks
2008-10-30 17:30:26 +00:00
* @ since CakePHP ( tm ) v 1.2
2009-11-06 17:51:51 +11:00
* @ license MIT License ( http :// www . opensource . org / licenses / mit - license . php )
2008-05-30 11:40:08 +00:00
*/
2010-12-07 19:50:56 -04:30
App :: uses ( 'File' , 'Utility' );
2011-04-17 18:03:43 +02:00
App :: uses ( 'Folder' , 'Utility' );
2011-03-07 01:25:05 -04:30
App :: uses ( 'String' , 'Utility' );
App :: uses ( 'Security' , 'Utility' );
2010-10-30 01:26:18 -02:00
2008-05-30 11:40:08 +00:00
/**
* Task class for creating new project apps and plugins
*
2010-12-24 14:26:26 -05:00
* @ package cake . console . shells . tasks
2008-05-30 11:40:08 +00:00
*/
class ProjectTask extends Shell {
2009-07-24 21:18:37 +02:00
2009-06-30 20:44:09 -04:00
/**
* configs path ( used in testing ) .
*
* @ var string
2009-11-14 23:19:25 +11:00
*/
2010-04-04 17:14:00 +10:00
public $configPath = null ;
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Checks that given project path does not already exist , and
* finds the app directory in it . Then it calls bake () with that information .
*
* @ param string $project Project path
*/
2010-10-16 01:02:15 -04:00
public function execute () {
$project = null ;
if ( isset ( $this -> args [ 0 ])) {
$project = $this -> args [ 0 ];
2008-05-30 11:40:08 +00:00
}
while ( ! $project ) {
2011-04-17 18:12:25 +02:00
$prompt = __d ( 'cake_console' , " What is the path to the project you want to bake? " );
$project = $this -> in ( $prompt , null , APP_PATH . 'myapp' );
2008-05-30 11:40:08 +00:00
}
2011-05-22 08:49:51 -04:00
if ( $project && ! Folder :: isAbsolute ( $project ) && isset ( $_SERVER [ 'PWD' ])) {
$project = $_SERVER [ 'PWD' ] . DS . $project ;
}
$response = false ;
while ( $response == false && is_dir ( $project ) === true && file_exists ( $project . 'Config' . 'core.php' )) {
$prompt = __d ( 'cake_console' , '<warning>A project already exists in this location:</warning> %s Overwrite?' , $project );
$response = $this -> in ( $prompt , array ( 'y' , 'n' ), 'n' );
if ( strtolower ( $response ) === 'n' ) {
$response = $project = false ;
2008-05-30 11:40:08 +00:00
}
}
2010-10-17 22:38:32 -04:00
$success = true ;
2008-10-23 00:10:44 +00:00
if ( $this -> bake ( $project )) {
2008-05-30 11:40:08 +00:00
$path = Folder :: slashTerm ( $project );
if ( $this -> createHome ( $path )) {
2011-03-19 18:32:35 +01:00
$this -> out ( __d ( 'cake_console' , ' * Welcome page created' ));
2008-05-30 11:40:08 +00:00
} else {
2011-03-19 18:32:35 +01:00
$this -> err ( __d ( 'cake_console' , 'The Welcome page was <error>NOT</error> created' ));
2010-10-17 22:38:32 -04:00
$success = false ;
2008-05-30 11:40:08 +00:00
}
2010-10-17 22:38:32 -04:00
if ( $this -> securitySalt ( $path ) === true ) {
2011-03-19 18:32:35 +01:00
$this -> out ( __d ( 'cake_console' , ' * Random hash key created for \'Security.salt\'' ));
2008-05-30 11:40:08 +00:00
} else {
2011-04-17 13:13:02 +02:00
$this -> err ( __d ( 'cake_console' , 'Unable to generate random hash for \'Security.salt\', you should change it in %s' , APP . 'Config' . DS . 'core.php' ));
2010-10-17 22:38:32 -04:00
$success = false ;
2008-05-30 11:40:08 +00:00
}
2010-10-17 22:38:32 -04:00
if ( $this -> securityCipherSeed ( $path ) === true ) {
2011-03-19 18:32:35 +01:00
$this -> out ( __d ( 'cake_console' , ' * Random seed created for \'Security.cipherSeed\'' ));
2010-01-25 11:14:18 -05:00
} else {
2011-04-17 13:13:02 +02:00
$this -> err ( __d ( 'cake_console' , 'Unable to generate random seed for \'Security.cipherSeed\', you should change it in %s' , APP . 'Config' . DS . 'core.php' ));
2010-10-17 22:38:32 -04:00
$success = false ;
2010-01-25 11:14:18 -05:00
}
2010-10-17 22:38:32 -04:00
if ( $this -> consolePath ( $path ) === true ) {
2011-04-11 21:43:40 -04:30
$this -> out ( __d ( 'cake_console' , ' * app/Console/cake.php path set.' ));
2010-10-17 22:38:32 -04:00
} else {
2011-04-11 21:43:40 -04:30
$this -> err ( __d ( 'cake_console' , 'Unable to set console path for app/Console.' ));
2010-10-17 22:38:32 -04:00
$success = false ;
}
2011-07-13 23:54:24 +02:00
$this -> out ( __d ( 'cake_console' , 'The value for CAKE_CORE_INCLUDE_PATH can be hardcoded set to %s in webroot/index.php' , CAKE_CORE_INCLUDE_PATH ));
$this -> out ( __d ( 'cake_console' , '<warning>If you hard code it, the project will possibly run only in your computer.</warning>' ));
$setConstants = $this -> in ( __d ( 'cake_console' , 'Do you want to set CAKE_CORE_INCLUDE_PATH in webroot/index.php?' ), array ( 'y' , 'n' ), 'n' );
if ( strtolower ( $setConstants ) === 'y' ) {
if ( $this -> corePath ( $path ) === true ) {
$this -> out ( __d ( 'cake_console' , ' * CAKE_CORE_INCLUDE_PATH set to %s in webroot/index.php' , CAKE_CORE_INCLUDE_PATH ));
$this -> out ( __d ( 'cake_console' , ' * CAKE_CORE_INCLUDE_PATH set to %s in webroot/test.php' , CAKE_CORE_INCLUDE_PATH ));
$this -> out ( __d ( 'cake_console' , ' * <warning>Remember to check these values after moving to production server</warning>' ));
} else {
$this -> err ( __d ( 'cake_console' , 'Unable to set CAKE_CORE_INCLUDE_PATH, you should change it in %s' , $path . 'webroot' . DS . 'index.php' ));
$success = false ;
}
} else {
$this -> out ( __d ( 'cake_console' , '<warning>Please make sure your cake core is accessible, if you have problems edit CAKE_CORE_INCLUDE_PATH in webroot/index.php</warning>' ));
}
2008-05-30 11:40:08 +00:00
$Folder = new Folder ( $path );
if ( ! $Folder -> chmod ( $path . 'tmp' , 0777 )) {
2011-03-19 18:32:35 +01:00
$this -> err ( __d ( 'cake_console' , 'Could not set permissions on %s' , $path . DS . 'tmp' ));
$this -> out ( __d ( 'cake_console' , 'chmod -R 0777 %s' , $path . DS . 'tmp' ));
2010-10-17 22:38:32 -04:00
$success = false ;
2008-05-30 11:40:08 +00:00
}
2010-10-17 22:38:32 -04:00
if ( $success ) {
2011-03-19 18:32:35 +01:00
$this -> out ( __d ( 'cake_console' , '<success>Project baked successfully!</success>' ));
2010-10-17 22:38:32 -04:00
} else {
2011-03-19 18:32:35 +01:00
$this -> out ( __d ( 'cake_console' , 'Project baked but with <warning>some issues.</warning>.' ));
2008-05-30 11:40:08 +00:00
}
2010-10-23 00:35:53 -04:00
return $path ;
2008-05-30 11:40:08 +00:00
}
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
* Looks for a skeleton template of a Cake application ,
* and if not found asks the user for a path . When there is a path
* this method will make a deep copy of the skeleton to the project directory .
* A default home page will be added , and the tmp file storage will be chmod ' ed to 0777.
*
* @ param string $path Project path
* @ param string $skel Path to copy from
* @ param string $skip array of directories to skip when copying
* @ access private
*/
2011-05-30 22:02:32 +02:00
public function bake ( $path , $skel = null , $skip = array ( 'empty' )) {
2011-04-16 19:22:53 +02:00
if ( ! $skel && ! empty ( $this -> params [ 'skel' ])) {
2008-05-30 11:40:08 +00:00
$skel = $this -> params [ 'skel' ];
}
while ( ! $skel ) {
2011-04-16 20:15:15 +02:00
$skel = $this -> in (
__d ( 'cake_console' , " What is the path to the directory layout you wish to copy? " ),
null ,
2011-05-30 23:02:39 -04:00
CAKE . 'Console' . DS . 'Templates' . DS . 'skel'
2011-04-16 20:15:15 +02:00
);
if ( ! $skel ) {
2011-03-19 18:32:35 +01:00
$this -> err ( __d ( 'cake_console' , 'The directory path you supplied was empty. Please try again.' ));
2008-05-30 11:40:08 +00:00
} else {
while ( is_dir ( $skel ) === false ) {
2011-04-16 20:15:15 +02:00
$skel = $this -> in (
__d ( 'cake_console' , 'Directory path does not exist please choose another:' ),
null ,
2011-05-30 23:02:39 -04:00
CAKE . 'Console' . DS . 'Templates' . DS . 'skel'
2011-04-16 20:15:15 +02:00
);
2008-05-30 11:40:08 +00:00
}
}
}
$app = basename ( $path );
2011-03-19 18:32:35 +01:00
$this -> out ( __d ( 'cake_console' , '<info>Skel Directory</info>: ' ) . $skel );
$this -> out ( __d ( 'cake_console' , '<info>Will be copied to</info>: ' ) . $path );
2008-05-30 11:40:08 +00:00
$this -> hr ();
2011-03-19 18:32:35 +01:00
$looksGood = $this -> in ( __d ( 'cake_console' , 'Look okay?' ), array ( 'y' , 'n' , 'q' ), 'y' );
2008-05-30 11:40:08 +00:00
2011-05-22 08:56:19 -04:00
switch ( strtolower ( $looksGood )) {
case 'y' :
$Folder = new Folder ( $skel );
if ( ! empty ( $this -> params [ 'empty' ])) {
$skip = array ();
}
2010-10-17 19:01:44 -04:00
2011-05-22 08:56:19 -04:00
if ( $Folder -> copy ( array ( 'to' => $path , 'skip' => $skip ))) {
$this -> hr ();
$this -> out ( __d ( 'cake_console' , '<success>Created:</success> %s in %s' , $app , $path ));
$this -> hr ();
} else {
$this -> err ( __d ( 'cake_console' , " <error>Could not create</error> '%s' properly. " , $app ));
return false ;
}
2008-05-30 11:40:08 +00:00
2011-05-22 08:56:19 -04:00
foreach ( $Folder -> messages () as $message ) {
$this -> out ( String :: wrap ( ' * ' . $message ), 1 , Shell :: VERBOSE );
}
2008-05-30 11:40:08 +00:00
2011-05-22 08:56:19 -04:00
return true ;
case 'n' :
unset ( $this -> args [ 0 ]);
$this -> execute ();
return false ;
case 'q' :
$this -> out ( __d ( 'cake_console' , '<error>Bake Aborted.</error>' ));
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
/**
* Writes a file with a default home page to the project .
*
* @ param string $dir Path to project
* @ return boolean Success
*/
2010-04-05 13:19:38 +10:00
public function createHome ( $dir ) {
2008-05-30 11:40:08 +00:00
$app = basename ( $dir );
2011-05-17 18:37:12 -04:00
$path = $dir . 'View' . DS . 'Pages' . DS ;
2011-05-30 23:02:39 -04:00
$source = CAKE . 'Console' . DS . 'Templates' . DS . 'default' . DS . 'views' . DS . 'home.ctp' ;
2009-07-26 01:52:05 -04:00
include ( $source );
2008-05-30 11:40:08 +00:00
return $this -> createFile ( $path . 'home.ctp' , $output );
}
2009-07-24 21:18:37 +02:00
2010-10-17 22:38:32 -04:00
/**
* Generates the correct path to the CakePHP libs that are generating the project
* and points app / console / cake . php to the right place
*
* @ param string $path Project path .
* @ return boolean success
*/
public function consolePath ( $path ) {
2011-04-11 21:43:40 -04:30
$File = new File ( $path . 'Console' . DS . 'cake.php' );
2010-10-17 22:38:32 -04:00
$contents = $File -> read ();
if ( preg_match ( '/(__CAKE_PATH__)/' , $contents , $match )) {
2011-04-17 12:35:21 +02:00
$path = CAKE . 'Console' . DS ;
2010-10-17 22:38:32 -04:00
$replacement = " ' " . str_replace ( DS , " ' . DIRECTORY_SEPARATOR . ' " , $path ) . " ' " ;
$result = str_replace ( $match [ 0 ], $replacement , $contents );
if ( $File -> write ( $result )) {
return true ;
}
return false ;
}
return false ;
}
2008-05-30 11:40:08 +00:00
/**
* Generates and writes 'Security.salt'
*
* @ param string $path Project path
* @ return boolean Success
*/
2010-04-05 13:19:38 +10:00
public function securitySalt ( $path ) {
2011-05-13 03:36:45 -04:30
$File = new File ( $path . 'Config' . DS . 'core.php' );
2008-05-30 11:40:08 +00:00
$contents = $File -> read ();
2010-10-17 19:07:57 -04:00
if ( preg_match ( '/([\s]*Configure::write\(\'Security.salt\',[\s\'A-z0-9]*\);)/' , $contents , $match )) {
2008-05-30 11:40:08 +00:00
$string = Security :: generateAuthKey ();
$result = str_replace ( $match [ 0 ], " \t " . 'Configure::write(\'Security.salt\', \'' . $string . '\');' , $contents );
if ( $File -> write ( $result )) {
return true ;
}
return false ;
}
2008-09-08 20:33:48 +00:00
return false ;
2008-05-30 11:40:08 +00:00
}
2009-07-24 21:18:37 +02:00
2010-10-17 22:38:32 -04:00
/**
* Generates and writes 'Security.cipherSeed'
*
* @ param string $path Project path
* @ return boolean Success
*/
public function securityCipherSeed ( $path ) {
2011-05-13 03:36:45 -04:30
$File = new File ( $path . 'Config' . DS . 'core.php' );
2010-10-17 22:38:32 -04:00
$contents = $File -> read ();
if ( preg_match ( '/([\s]*Configure::write\(\'Security.cipherSeed\',[\s\'A-z0-9]*\);)/' , $contents , $match )) {
if ( ! class_exists ( 'Security' )) {
2011-04-17 12:35:21 +02:00
require CAKE . 'Utility' . DS . 'security.php' ;
2010-10-17 22:38:32 -04:00
}
$string = substr ( bin2hex ( Security :: generateAuthKey ()), 0 , 30 );
$result = str_replace ( $match [ 0 ], " \t " . 'Configure::write(\'Security.cipherSeed\', \'' . $string . '\');' , $contents );
if ( $File -> write ( $result )) {
return true ;
2010-01-25 11:14:18 -05:00
}
return false ;
}
2010-10-17 22:38:32 -04:00
return false ;
}
2010-01-25 11:14:18 -05:00
2008-05-30 11:40:08 +00:00
/**
* Generates and writes CAKE_CORE_INCLUDE_PATH
*
* @ param string $path Project path
* @ return boolean Success
*/
2010-04-05 13:19:38 +10:00
public function corePath ( $path ) {
2008-05-30 11:40:08 +00:00
if ( dirname ( $path ) !== CAKE_CORE_INCLUDE_PATH ) {
2010-10-17 22:38:32 -04:00
$File = new File ( $path . 'webroot' . DS . 'index.php' );
2008-05-30 11:40:08 +00:00
$contents = $File -> read ();
2011-05-17 14:34:10 -04:00
if ( preg_match ( '/([\s]*define\(\'CAKE_CORE_INCLUDE_PATH\',[\s\'A-z0-9\.]*\);)/' , $contents , $match )) {
2009-11-09 20:10:24 -05:00
$root = strpos ( CAKE_CORE_INCLUDE_PATH , '/' ) === 0 ? " DS . ' " : " ' " ;
2011-03-27 00:25:20 +01:00
$result = str_replace ( $match [ 0 ], " \n \t \t define('CAKE_CORE_INCLUDE_PATH', " . $root . str_replace ( DS , " ' . DS . ' " , trim ( CAKE_CORE_INCLUDE_PATH , DS )) . " '); " , $contents );
2008-05-30 11:40:08 +00:00
if ( ! $File -> write ( $result )) {
return false ;
}
} else {
return false ;
}
2010-10-17 22:38:32 -04:00
$File = new File ( $path . 'webroot' . DS . 'test.php' );
2008-05-30 11:40:08 +00:00
$contents = $File -> read ();
2011-07-13 18:23:07 +02:00
if ( preg_match ( '/([\s]*define\(\'CAKE_CORE_INCLUDE_PATH\',[\s\'A-z0-9\.]*\);)/' , $contents , $match )) {
2011-03-27 00:25:20 +01:00
$result = str_replace ( $match [ 0 ], " \n \t \t define('CAKE_CORE_INCLUDE_PATH', " . $root . str_replace ( DS , " ' . DS . ' " , trim ( CAKE_CORE_INCLUDE_PATH , DS )) . " '); " , $contents );
2008-05-30 11:40:08 +00:00
if ( ! $File -> write ( $result )) {
return false ;
}
} else {
return false ;
}
return true ;
}
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2011-06-19 20:28:40 -04:00
* Enables Configure :: read ( 'Routing.prefixes' ) in / app / Config / core . php
2008-05-30 11:40:08 +00:00
*
* @ param string $name Name to use as admin routing
* @ return boolean Success
*/
2010-04-05 13:19:38 +10:00
public function cakeAdmin ( $name ) {
2011-04-17 13:13:02 +02:00
$path = ( empty ( $this -> configPath )) ? APP . 'Config' . DS : $this -> configPath ;
2010-10-17 22:38:32 -04:00
$File = new File ( $path . 'core.php' );
2008-05-30 11:40:08 +00:00
$contents = $File -> read ();
2011-02-23 11:26:43 -05:00
if ( preg_match ( '%(\s*[/]*Configure::write\(\'Routing.prefixes\',[\s\'a-z,\)\(]*\);)%' , $contents , $match )) {
$result = str_replace ( $match [ 0 ], " \n " . 'Configure::write(\'Routing.prefixes\', array(\'' . $name . '\'));' , $contents );
2008-05-30 11:40:08 +00:00
if ( $File -> write ( $result )) {
2009-10-07 00:46:13 -04:00
Configure :: write ( 'Routing.prefixes' , array ( $name ));
2008-05-30 11:40:08 +00:00
return true ;
} else {
return false ;
}
} else {
return false ;
}
}
2009-07-24 21:18:37 +02:00
2009-06-30 20:44:09 -04:00
/**
2009-12-08 23:52:02 -05:00
* Checks for Configure :: read ( 'Routing.prefixes' ) and forces user to input it if not enabled
2009-06-30 20:44:09 -04:00
*
* @ return string Admin route to use
*/
2010-04-05 13:19:38 +10:00
public function getPrefix () {
2009-06-30 20:44:09 -04:00
$admin = '' ;
2009-10-07 00:46:13 -04:00
$prefixes = Configure :: read ( 'Routing.prefixes' );
if ( ! empty ( $prefixes )) {
2010-03-07 17:14:13 -05:00
if ( count ( $prefixes ) == 1 ) {
return $prefixes [ 0 ] . '_' ;
}
2009-11-11 09:56:35 -05:00
if ( $this -> interactive ) {
$this -> out ();
2011-03-19 18:32:35 +01:00
$this -> out ( __d ( 'cake_console' , 'You have more than one routing prefix configured' ));
2009-11-11 09:56:35 -05:00
}
2009-10-07 00:46:13 -04:00
$options = array ();
foreach ( $prefixes as $i => $prefix ) {
$options [] = $i + 1 ;
2009-11-11 09:56:35 -05:00
if ( $this -> interactive ) {
$this -> out ( $i + 1 . '. ' . $prefix );
}
2009-10-07 00:46:13 -04:00
}
2011-03-19 18:32:35 +01:00
$selection = $this -> in ( __d ( 'cake_console' , 'Please choose a prefix to bake with.' ), $options , 1 );
2009-10-07 00:46:13 -04:00
return $prefixes [ $selection - 1 ] . '_' ;
2009-06-30 20:44:09 -04:00
}
2009-11-11 09:56:35 -05:00
if ( $this -> interactive ) {
$this -> hr ();
2011-06-19 20:28:40 -04:00
$this -> out ( __d ( 'cake_console' , 'You need to enable Configure::write(\'Routing.prefixes\',array(\'admin\')) in /app/Config/core.php to use prefix routing.' ));
2011-03-19 18:32:35 +01:00
$this -> out ( __d ( 'cake_console' , 'What would you like the prefix route to be?' ));
$this -> out ( __d ( 'cake_console' , 'Example: www.example.com/admin/controller' ));
2009-11-11 09:56:35 -05:00
while ( $admin == '' ) {
2011-03-19 18:32:35 +01:00
$admin = $this -> in ( __d ( 'cake_console' , 'Enter a routing prefix:' ), null , 'admin' );
2009-11-11 09:56:35 -05:00
}
if ( $this -> cakeAdmin ( $admin ) !== true ) {
2011-06-19 20:28:40 -04:00
$this -> out ( __d ( 'cake_console' , '<error>Unable to write to</error> /app/Config/core.php.' ));
$this -> out ( __d ( 'cake_console' , 'You need to enable Configure::write(\'Routing.prefixes\',array(\'admin\')) in /app/Config/core.php to use prefix routing.' ));
2009-11-11 09:56:35 -05:00
$this -> _stop ();
}
return $admin . '_' ;
2009-06-30 20:44:09 -04:00
}
2009-11-11 09:56:35 -05:00
return '' ;
2009-06-30 20:44:09 -04:00
}
2009-07-24 21:18:37 +02:00
2008-05-30 11:40:08 +00:00
/**
2010-10-12 22:55:19 -04:00
* get the option parser .
2008-05-30 11:40:08 +00:00
*
2010-10-12 22:55:19 -04:00
* @ return ConsoleOptionParser
2008-05-30 11:40:08 +00:00
*/
2010-10-12 22:55:19 -04:00
public function getOptionParser () {
$parser = parent :: getOptionParser ();
return $parser -> description (
2011-03-19 18:32:35 +01:00
__d ( 'cake_console' , 'Generate a new CakePHP project skeleton.' )
2010-10-12 22:55:19 -04:00
) -> addArgument ( 'name' , array (
2011-03-19 18:32:35 +01:00
'help' => __d ( 'cake_console' , 'Application directory to make, if it starts with "/" the path is absolute.' )
2010-10-17 19:01:44 -04:00
)) -> addOption ( 'empty' , array (
2011-03-19 18:32:35 +01:00
'help' => __d ( 'cake_console' , 'Create empty files in each of the directories. Good if you are using git' )
2010-10-17 22:38:32 -04:00
)) -> addOption ( 'skel' , array (
2011-05-30 23:02:39 -04:00
'default' => current ( App :: core ( 'Console' )) . 'Templates' . DS . 'skel' ,
'help' => __d ( 'cake_console' , 'The directory layout to use for the new application skeleton. Defaults to cake/Console/Templates/skel of CakePHP used to create the project.' )
2010-10-12 22:55:19 -04:00
));
2008-05-30 11:40:08 +00:00
}
2011-04-17 18:03:43 +02:00
}