410 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			410 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * PEAR_RunTest
 | |
|  *
 | |
|  * PHP versions 4 and 5
 | |
|  *
 | |
|  * LICENSE: This source file is subject to version 3.0 of the PHP license
 | |
|  * that is available through the world-wide-web at the following URI:
 | |
|  * http://www.php.net/license/3_0.txt.  If you did not receive a copy of
 | |
|  * the PHP License and are unable to obtain it through the web, please
 | |
|  * send a note to license@php.net so we can mail you a copy immediately.
 | |
|  *
 | |
|  * @category   pear
 | |
|  * @package    PEAR
 | |
|  * @author     Tomas V.V.Cox <cox@idecnet.com>
 | |
|  * @author     Greg Beaver <cellog@php.net>
 | |
|  * @copyright  1997-2006 The PHP Group
 | |
|  * @license    http://www.php.net/license/3_0.txt  PHP License 3.0
 | |
|  * @version    CVS: $Id: RunTest.php,v 1.20 2006/02/03 02:08:11 cellog Exp $
 | |
|  * @link       http://pear.php.net/package/PEAR
 | |
|  * @since      File available since Release 1.3.3
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * for error handling
 | |
|  */
 | |
| require_once 'PEAR.php';
 | |
| require_once 'PEAR/Config.php';
 | |
| 
 | |
| define('DETAILED', 1);
 | |
| putenv("PHP_PEAR_RUNTESTS=1");
 | |
| 
 | |
| /**
 | |
|  * Simplified version of PHP's test suite
 | |
|  *
 | |
|  * Try it with:
 | |
|  *
 | |
|  * $ php -r 'include "../PEAR/RunTest.php"; $t=new PEAR_RunTest; $o=$t->run("./pear_system.phpt");print_r($o);'
 | |
|  *
 | |
|  *
 | |
|  * @category   pear
 | |
|  * @package    PEAR
 | |
|  * @author     Tomas V.V.Cox <cox@idecnet.com>
 | |
|  * @author     Greg Beaver <cellog@php.net>
 | |
|  * @copyright  1997-2006 The PHP Group
 | |
|  * @license    http://www.php.net/license/3_0.txt  PHP License 3.0
 | |
|  * @version    Release: 1.4.11
 | |
|  * @link       http://pear.php.net/package/PEAR
 | |
|  * @since      Class available since Release 1.3.3
 | |
|  */
 | |
| class PEAR_RunTest
 | |
| {
 | |
|     var $_logger;
 | |
|     var $_options;
 | |
| 
 | |
|     /**
 | |
|      * An object that supports the PEAR_Common->log() signature, or null
 | |
|      * @param PEAR_Common|null
 | |
|      */
 | |
|     function PEAR_RunTest($logger = null, $options = array())
 | |
|     {
 | |
|         if (is_null($logger)) {
 | |
|             require_once 'PEAR/Common.php';
 | |
|             $logger = new PEAR_Common;
 | |
|         }
 | |
|         $this->_logger = $logger;
 | |
|         $this->_options = $options;
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     //  Run an individual test case.
 | |
|     //
 | |
| 
 | |
|     function run($file, $ini_settings = '')
 | |
|     {
 | |
|         $cwd = getcwd();
 | |
|         $conf = &PEAR_Config::singleton();
 | |
|         $php = $conf->get('php_bin');
 | |
|         if (isset($this->_options['phpunit'])) {
 | |
|             $cmd = "$php$ini_settings -f $file";
 | |
|             if (isset($this->_logger)) {
 | |
|                 $this->_logger->log(2, 'Running command "' . $cmd . '"');
 | |
|             }
 | |
|     
 | |
|             $savedir = getcwd(); // in case the test moves us around
 | |
|             chdir(dirname($file));
 | |
|             echo `$cmd`;
 | |
|             chdir($savedir);
 | |
|             return 'PASSED'; // we have no way of knowing this information so assume passing
 | |
|         }
 | |
|         //var_dump($php);exit;
 | |
|         global $log_format, $info_params, $ini_overwrites;
 | |
| 
 | |
|         $info_params = '';
 | |
|         $log_format = 'LEOD';
 | |
| 
 | |
|         // Load the sections of the test file.
 | |
|         $section_text = array(
 | |
|             'TEST'    => '(unnamed test)',
 | |
|             'SKIPIF'  => '',
 | |
|             'GET'     => '',
 | |
|             'ARGS'    => '',
 | |
|             'CLEAN'   => '',
 | |
|         );
 | |
| 
 | |
|         $file = realpath($file);
 | |
|         if (!is_file($file) || !$fp = fopen($file, "r")) {
 | |
|             return PEAR::raiseError("Cannot open test file: $file");
 | |
|         }
 | |
| 
 | |
|         $section = '';
 | |
|         while (!feof($fp)) {
 | |
|             $line = fgets($fp);
 | |
| 
 | |
|             // Match the beginning of a section.
 | |
|             if (ereg('^--([A-Z]+)--',$line,$r)) {
 | |
|                 $section = $r[1];
 | |
|                 $section_text[$section] = '';
 | |
|                 continue;
 | |
|             } elseif (empty($section)) {
 | |
|                 fclose($fp);
 | |
|                 return PEAR::raiseError("Invalid sections formats in test file: $file");
 | |
| 			}
 | |
| 
 | |
|             // Add to the section text.
 | |
|             $section_text[$section] .= $line;
 | |
|         }
 | |
|         fclose($fp);
 | |
| 
 | |
|         $shortname = str_replace($cwd . DIRECTORY_SEPARATOR, '', $file);
 | |
|         if (!isset($this->_options['simple'])) {
 | |
|             $tested = trim($section_text['TEST']) . "[$shortname]";
 | |
|         } else {
 | |
|             $tested = trim($section_text['TEST']) . ' ';
 | |
|         }
 | |
| 
 | |
|         $tmp = realpath(dirname($file));
 | |
|         $tmp_skipif = $tmp . uniqid('/phpt.');
 | |
|         $tmp_file   = ereg_replace('\.phpt$','.php',$file);
 | |
|         $tmp_post   = $tmp . uniqid('/phpt.');
 | |
| 
 | |
|         @unlink($tmp_skipif);
 | |
|         @unlink($tmp_file);
 | |
|         @unlink($tmp_post);
 | |
| 
 | |
|         // unlink old test results
 | |
|         @unlink(ereg_replace('\.phpt$','.diff',$file));
 | |
|         @unlink(ereg_replace('\.phpt$','.log',$file));
 | |
|         @unlink(ereg_replace('\.phpt$','.exp',$file));
 | |
|         @unlink(ereg_replace('\.phpt$','.out',$file));
 | |
| 
 | |
|         // Check if test should be skipped.
 | |
|         $info = '';
 | |
|         $warn = false;
 | |
|         if (array_key_exists('SKIPIF', $section_text)) {
 | |
|             if (trim($section_text['SKIPIF'])) {
 | |
|                 $this->save_text($tmp_skipif, $section_text['SKIPIF']);
 | |
|                 //$extra = substr(PHP_OS, 0, 3) !== "WIN" ?
 | |
|                 //    "unset REQUEST_METHOD;": "";
 | |
| 
 | |
|                 //$output = `$extra $php $info_params -f $tmp_skipif`;
 | |
|                 $output = `$php $info_params -f $tmp_skipif`;
 | |
|                 unlink($tmp_skipif);
 | |
|                 if (eregi("^skip", trim($output))) {
 | |
|                     $skipreason = "SKIP $tested";
 | |
|                     $reason = (eregi("^skip[[:space:]]*(.+)\$", trim($output))) ? eregi_replace("^skip[[:space:]]*(.+)\$", "\\1", trim($output)) : FALSE;
 | |
|                     if ($reason) {
 | |
|                         $skipreason .= " (reason: $reason)";
 | |
|                     }
 | |
|                     if (!isset($this->_options['quiet'])) {
 | |
|                         $this->_logger->log(0, $skipreason);
 | |
|                     }
 | |
|                     if (isset($old_php)) {
 | |
|                         $php = $old_php;
 | |
|                     }
 | |
|                     return 'SKIPPED';
 | |
|                 }
 | |
|                 if (eregi("^info", trim($output))) {
 | |
|                     $reason = (ereg("^info[[:space:]]*(.+)\$", trim($output))) ? ereg_replace("^info[[:space:]]*(.+)\$", "\\1", trim($output)) : FALSE;
 | |
|                     if ($reason) {
 | |
|                         $info = " (info: $reason)";
 | |
|                     }
 | |
|                 }
 | |
|                 if (eregi("^warn", trim($output))) {
 | |
|                     $reason = (ereg("^warn[[:space:]]*(.+)\$", trim($output))) ? ereg_replace("^warn[[:space:]]*(.+)\$", "\\1", trim($output)) : FALSE;
 | |
|                     if ($reason) {
 | |
|                         $warn = true; /* only if there is a reason */
 | |
|                         $info = " (warn: $reason)";
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // We've satisfied the preconditions - run the test!
 | |
|         $this->save_text($tmp_file,$section_text['FILE']);
 | |
| 
 | |
|         $args = $section_text['ARGS'] ? ' -- '.$section_text['ARGS'] : '';
 | |
| 
 | |
|         $cmd = "$php$ini_settings -f $tmp_file$args 2>&1";
 | |
|         if (isset($this->_logger)) {
 | |
|             $this->_logger->log(2, 'Running command "' . $cmd . '"');
 | |
|         }
 | |
| 
 | |
|         $savedir = getcwd(); // in case the test moves us around
 | |
|         if (isset($section_text['RETURNS'])) {
 | |
|             ob_start();
 | |
|             system($cmd, $return_value);
 | |
|             $out = ob_get_contents();
 | |
|             ob_end_clean();
 | |
|             @unlink($tmp_post);
 | |
|             $section_text['RETURNS'] = (int) trim($section_text['RETURNS']);
 | |
|             $returnfail = ($return_value != $section_text['RETURNS']);
 | |
|         } else {
 | |
|             $out = `$cmd`;
 | |
|             $returnfail = false;
 | |
|         }
 | |
|         chdir($savedir);
 | |
| 
 | |
|         if ($section_text['CLEAN']) {
 | |
|             // perform test cleanup
 | |
|             $this->save_text($clean = $tmp . uniqid('/phpt.'), $section_text['CLEAN']);
 | |
|             `$php $clean`;
 | |
|             @unlink($clean);
 | |
|         }
 | |
|         // Does the output match what is expected?
 | |
|         $output = trim($out);
 | |
|         $output = preg_replace('/\r\n/', "\n", $output);
 | |
| 
 | |
|         if (isset($section_text['EXPECTF']) || isset($section_text['EXPECTREGEX'])) {
 | |
|             if (isset($section_text['EXPECTF'])) {
 | |
|                 $wanted = trim($section_text['EXPECTF']);
 | |
|             } else {
 | |
|                 $wanted = trim($section_text['EXPECTREGEX']);
 | |
|             }
 | |
|             $wanted_re = preg_replace('/\r\n/',"\n",$wanted);
 | |
|             if (isset($section_text['EXPECTF'])) {
 | |
|                 $wanted_re = preg_quote($wanted_re, '/');
 | |
|                 // Stick to basics
 | |
|                 $wanted_re = str_replace("%s", ".+?", $wanted_re); //not greedy
 | |
|                 $wanted_re = str_replace("%i", "[+\-]?[0-9]+", $wanted_re);
 | |
|                 $wanted_re = str_replace("%d", "[0-9]+", $wanted_re);
 | |
|                 $wanted_re = str_replace("%x", "[0-9a-fA-F]+", $wanted_re);
 | |
|                 $wanted_re = str_replace("%f", "[+\-]?\.?[0-9]+\.?[0-9]*(E-?[0-9]+)?", $wanted_re);
 | |
|                 $wanted_re = str_replace("%c", ".", $wanted_re);
 | |
|                 // %f allows two points "-.0.0" but that is the best *simple* expression
 | |
|             }
 | |
|     /* DEBUG YOUR REGEX HERE
 | |
|             var_dump($wanted_re);
 | |
|             print(str_repeat('=', 80) . "\n");
 | |
|             var_dump($output);
 | |
|     */
 | |
|             if (!$returnfail && preg_match("/^$wanted_re\$/s", $output)) {
 | |
|                 @unlink($tmp_file);
 | |
|                 if (!isset($this->_options['quiet'])) {
 | |
|                     $this->_logger->log(0, "PASS $tested$info");
 | |
|                 }
 | |
|                 if (isset($old_php)) {
 | |
|                     $php = $old_php;
 | |
|                 }
 | |
|                 return 'PASSED';
 | |
|             }
 | |
| 
 | |
|         } else {
 | |
|             $wanted = trim($section_text['EXPECT']);
 | |
|             $wanted = preg_replace('/\r\n/',"\n",$wanted);
 | |
|         // compare and leave on success
 | |
|             $ok = (0 == strcmp($output,$wanted));
 | |
|             if (!$returnfail && $ok) {
 | |
|                 @unlink($tmp_file);
 | |
|                 if (!isset($this->_options['quiet'])) {
 | |
|                     $this->_logger->log(0, "PASS $tested$info");
 | |
|                 }
 | |
|                 if (isset($old_php)) {
 | |
|                     $php = $old_php;
 | |
|                 }
 | |
|                 return 'PASSED';
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Test failed so we need to report details.
 | |
|         if ($warn) {
 | |
|             $this->_logger->log(0, "WARN $tested$info");
 | |
|         } else {
 | |
|             $this->_logger->log(0, "FAIL $tested$info");
 | |
|         }
 | |
| 
 | |
|         if (isset($section_text['RETURNS'])) {
 | |
|             $GLOBALS['__PHP_FAILED_TESTS__'][] = array(
 | |
|                             'name' => $file,
 | |
|                             'test_name' => $tested,
 | |
|                             'output' => ereg_replace('\.phpt$','.log', $file),
 | |
|                             'diff'   => ereg_replace('\.phpt$','.diff', $file),
 | |
|                             'info'   => $info,
 | |
|                             'return' => $return_value
 | |
|                             );
 | |
|         } else {
 | |
|             $GLOBALS['__PHP_FAILED_TESTS__'][] = array(
 | |
|                             'name' => $file,
 | |
|                             'test_name' => $tested,
 | |
|                             'output' => ereg_replace('\.phpt$','.log', $file),
 | |
|                             'diff'   => ereg_replace('\.phpt$','.diff', $file),
 | |
|                             'info'   => $info,
 | |
|                             );
 | |
|         }
 | |
| 
 | |
|         // write .exp
 | |
|         if (strpos($log_format,'E') !== FALSE) {
 | |
|             $logname = ereg_replace('\.phpt$','.exp',$file);
 | |
|             if (!$log = fopen($logname,'w')) {
 | |
|                 return PEAR::raiseError("Cannot create test log - $logname");
 | |
|             }
 | |
|             fwrite($log,$wanted);
 | |
|             fclose($log);
 | |
|         }
 | |
| 
 | |
|         // write .out
 | |
|         if (strpos($log_format,'O') !== FALSE) {
 | |
|             $logname = ereg_replace('\.phpt$','.out',$file);
 | |
|             if (!$log = fopen($logname,'w')) {
 | |
|                 return PEAR::raiseError("Cannot create test log - $logname");
 | |
|             }
 | |
|             fwrite($log,$output);
 | |
|             fclose($log);
 | |
|         }
 | |
| 
 | |
|         // write .diff
 | |
|         if (strpos($log_format,'D') !== FALSE) {
 | |
|             $logname = ereg_replace('\.phpt$','.diff',$file);
 | |
|             if (!$log = fopen($logname,'w')) {
 | |
|                 return PEAR::raiseError("Cannot create test log - $logname");
 | |
|             }
 | |
|             fwrite($log, $this->generate_diff($wanted, $output,
 | |
|                 isset($section_text['RETURNS']) ? array(trim($section_text['RETURNS']),
 | |
|                     $return_value) : null));
 | |
|             fclose($log);
 | |
|         }
 | |
| 
 | |
|         // write .log
 | |
|         if (strpos($log_format,'L') !== FALSE) {
 | |
|             $logname = ereg_replace('\.phpt$','.log',$file);
 | |
|             if (!$log = fopen($logname,'w')) {
 | |
|                 return PEAR::raiseError("Cannot create test log - $logname");
 | |
|             }
 | |
|             fwrite($log,"
 | |
| ---- EXPECTED OUTPUT
 | |
| $wanted
 | |
| ---- ACTUAL OUTPUT
 | |
| $output
 | |
| ---- FAILED
 | |
| ");
 | |
|             if ($returnfail) {
 | |
|                 fwrite($log,"
 | |
| ---- EXPECTED RETURN
 | |
| $section_text[RETURNS]
 | |
| ---- ACTUAL RETURN
 | |
| $return_value
 | |
| ");
 | |
|             }
 | |
|             fclose($log);
 | |
|             //error_report($file,$logname,$tested);
 | |
|         }
 | |
| 
 | |
|         if (isset($old_php)) {
 | |
|             $php = $old_php;
 | |
|         }
 | |
| 
 | |
|         return $warn ? 'WARNED' : 'FAILED';
 | |
|     }
 | |
| 
 | |
|     function generate_diff($wanted, $output, $return_value)
 | |
|     {
 | |
|         $w = explode("\n", $wanted);
 | |
|         $o = explode("\n", $output);
 | |
|         $w1 = array_diff_assoc($w,$o);
 | |
|         $o1 = array_diff_assoc($o,$w);
 | |
|         $w2 = array();
 | |
|         $o2 = array();
 | |
|         foreach($w1 as $idx => $val) $w2[sprintf("%03d<",$idx)] = sprintf("%03d- ", $idx+1).$val;
 | |
|         foreach($o1 as $idx => $val) $o2[sprintf("%03d>",$idx)] = sprintf("%03d+ ", $idx+1).$val;
 | |
|         $diff = array_merge($w2, $o2);
 | |
|         ksort($diff);
 | |
|         if ($return_value) {
 | |
|             $extra = "##EXPECTED: $return_value[0]\r\n##RETURNED: $return_value[1]";
 | |
|         } else {
 | |
|             $extra = '';
 | |
|         }
 | |
|         return implode("\r\n", $diff) . $extra;
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     //  Write the given text to a temporary file, and return the filename.
 | |
|     //
 | |
| 
 | |
|     function save_text($filename, $text)
 | |
|     {
 | |
|         if (!$fp = fopen($filename, 'w')) {
 | |
|             return PEAR::raiseError("Cannot open file '" . $filename . "' (save_text)");
 | |
|         }
 | |
|         fwrite($fp,$text);
 | |
|         fclose($fp);
 | |
|     if (1 < DETAILED) echo "
 | |
| FILE $filename {{{
 | |
| $text
 | |
| }}}
 | |
| ";
 | |
|     }
 | |
| 
 | |
| }
 | |
| ?>
 |