. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * * * Neither the name of Sebastian Bergmann nor the names of his * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * * @category Testing * @package PHPUnit * @author Sebastian Bergmann * @copyright 2002-2009 Sebastian Bergmann * @license http://www.opensource.org/licenses/bsd-license.php BSD License * @link http://www.phpunit.de/ * @since File available since Release 3.1.4 */ require_once 'PHPUnit/Framework.php'; require_once 'PHPUnit/Util/Metrics/Project.php'; require_once 'PHPUnit/Util/Class.php'; require_once 'PHPUnit/Util/CodeCoverage.php'; require_once 'PHPUnit/Util/Filesystem.php'; require_once 'PHPUnit/Util/Filter.php'; PHPUnit_Util_Filter::addFileToFilter(__FILE__, 'PHPUNIT'); /** * * * @category Testing * @package PHPUnit * @author Sebastian Bergmann * @copyright 2002-2009 Sebastian Bergmann * @license http://www.opensource.org/licenses/bsd-license.php BSD License * @version Release: 3.3.17 * @link http://www.phpunit.de/ * @since Class available since Release 3.1.4 */ class PHPUnit_Util_Log_CodeCoverage_Database { /** * @var PDO */ protected $dbh; /** * Constructor. * * @param PDO $dbh * @throws PDOException */ public function __construct(PDO $dbh) { $this->dbh = $dbh; } /** * Stores code coverage information. * * @param PHPUnit_Framework_TestResult $result * @param integer $runId * @param integer $revision * @param string $commonPath */ public function storeCodeCoverage(PHPUnit_Framework_TestResult $result, $runId, $revision, $commonPath = '') { $codeCoverage = $result->getCodeCoverageInformation(FALSE); $summary = PHPUnit_Util_CodeCoverage::getSummary($codeCoverage); $files = array_keys($summary); $projectMetrics = new PHPUnit_Util_Metrics_Project($files, $summary); $storedClasses = array(); if (empty($commonPath)) { $commonPath = PHPUnit_Util_Filesystem::getCommonPath($files); } $this->dbh->beginTransaction(); foreach ($files as $file) { $filename = str_replace($commonPath, '', $file); $fileId = FALSE; $fileMetrics = $projectMetrics->getFile($file); $lines = $fileMetrics->getLines(); $hash = md5_file($file); $stmt = $this->dbh->prepare( 'SELECT code_file_id FROM code_file WHERE code_file_name = :filename AND revision = :revision;' ); $stmt->bindParam(':filename', $filename, PDO::PARAM_STR); $stmt->bindParam(':revision', $revision, PDO::PARAM_INT); $stmt->execute(); if ($stmt) { $fileId = (int)$stmt->fetchColumn(); } unset($stmt); if ($fileId == 0) { $stmt = $this->dbh->prepare( 'INSERT INTO code_file (code_file_name, code_file_md5, revision) VALUES(:filename, :hash, :revision);' ); $stmt->bindParam(':filename', $filename, PDO::PARAM_STR); $stmt->bindParam(':hash', $hash, PDO::PARAM_STR); $stmt->bindParam(':revision', $revision, PDO::PARAM_INT); $stmt->execute(); $fileId = $this->dbh->lastInsertId(); $stmt = $this->dbh->prepare( 'INSERT INTO code_class (code_file_id, code_class_name, code_class_start_line, code_class_end_line) VALUES(:fileId, :className, :startLine, :endLine);' ); foreach ($fileMetrics->getClasses() as $classMetrics) { $className = $classMetrics->getClass()->getName(); $classStartLine = $classMetrics->getClass()->getStartLine(); $classEndLine = $classMetrics->getClass()->getEndLine(); $stmt->bindParam(':fileId', $fileId, PDO::PARAM_INT); $stmt->bindParam(':className', $className, PDO::PARAM_STR); $stmt->bindParam(':startLine', $classStartLine, PDO::PARAM_INT); $stmt->bindParam(':endLine', $classEndLine, PDO::PARAM_INT); $stmt->execute(); $classId = $this->dbh->lastInsertId(); $storedClasses[$className] = $classId; $stmt2 = $this->dbh->prepare( 'INSERT INTO code_method (code_class_id, code_method_name, code_method_start_line, code_method_end_line) VALUES(:classId, :methodName, :startLine, :endLine);' ); foreach ($classMetrics->getMethods() as $methodMetrics) { $methodName = $methodMetrics->getMethod()->getName(); $methodStartLine = $methodMetrics->getMethod()->getStartLine(); $methodEndLine = $methodMetrics->getMethod()->getEndLine(); $stmt2->bindParam(':classId', $classId, PDO::PARAM_INT); $stmt2->bindParam(':methodName', $methodName, PDO::PARAM_STR); $stmt2->bindParam(':startLine', $methodStartLine, PDO::PARAM_INT); $stmt2->bindParam(':endLine', $methodEndLine, PDO::PARAM_INT); $stmt2->execute(); } unset($stmt2); } $stmt = $this->dbh->prepare( 'INSERT INTO code_line (code_file_id, code_line_number, code_line, code_line_covered) VALUES(:fileId, :lineNumber, :line, :covered);' ); $i = 1; foreach ($lines as $line) { $covered = 0; if (isset($summary[$file][$i])) { if (is_int($summary[$file][$i])) { $covered = $summary[$file][$i]; } else { $covered = 1; } } $stmt->bindParam(':fileId', $fileId, PDO::PARAM_INT); $stmt->bindParam(':lineNumber', $i, PDO::PARAM_INT); $stmt->bindParam(':line', $line, PDO::PARAM_STR); $stmt->bindParam(':covered', $covered, PDO::PARAM_INT); $stmt->execute(); $i++; } } $stmt = $this->dbh->prepare( 'INSERT INTO metrics_file (run_id, code_file_id, metrics_file_coverage, metrics_file_loc, metrics_file_cloc, metrics_file_ncloc, metrics_file_loc_executable, metrics_file_loc_executed) VALUES(:runId, :fileId, :coverage, :loc, :cloc, :ncloc, :locExecutable, :locExecuted);' ); $fileCoverage = $fileMetrics->getCoverage(); $fileLoc = $fileMetrics->getLoc(); $fileCloc = $fileMetrics->getCloc(); $fileNcloc = $fileMetrics->getNcloc(); $fileLocExecutable = $fileMetrics->getLocExecutable(); $fileLocExecuted = $fileMetrics->getLocExecuted(); $stmt->bindParam(':runId', $runId, PDO::PARAM_INT); $stmt->bindParam(':fileId', $fileId, PDO::PARAM_INT); $stmt->bindParam(':coverage', $fileCoverage); $stmt->bindParam(':loc', $fileLoc, PDO::PARAM_INT); $stmt->bindParam(':cloc', $fileCloc, PDO::PARAM_INT); $stmt->bindParam(':ncloc', $fileNcloc, PDO::PARAM_INT); $stmt->bindParam(':locExecutable', $fileLocExecutable, PDO::PARAM_INT); $stmt->bindParam(':locExecuted', $fileLocExecuted, PDO::PARAM_INT); $stmt->execute(); $stmtSelectFunctionId = $this->dbh->prepare( 'SELECT code_function_id FROM code_file, code_function WHERE code_function.code_file_id = code_file.code_file_id AND code_file.revision = :revision AND code_function.code_function_name = :functionName;' ); $stmtInsertFunction = $this->dbh->prepare( 'INSERT INTO metrics_function (run_id, code_function_id, metrics_function_coverage, metrics_function_loc, metrics_function_loc_executable, metrics_function_loc_executed, metrics_function_ccn, metrics_function_crap, metrics_function_npath) VALUES(:runId, :functionId, :coverage, :loc, :locExecutable, :locExecuted, :ccn, :crap, :npath);' ); $stmtSelectClassId = $this->dbh->prepare( 'SELECT code_class_id FROM code_file, code_class WHERE code_class.code_file_id = code_file.code_file_id AND code_file.revision = :revision AND code_class.code_class_name = :className;' ); $stmtInsertClass = $this->dbh->prepare( 'INSERT INTO metrics_class (run_id, code_class_id, metrics_class_coverage, metrics_class_loc, metrics_class_loc_executable, metrics_class_loc_executed, metrics_class_aif, metrics_class_ahf, metrics_class_cis, metrics_class_csz, metrics_class_dit, metrics_class_impl, metrics_class_mif, metrics_class_mhf, metrics_class_noc, metrics_class_pf, metrics_class_vars, metrics_class_varsnp, metrics_class_varsi, metrics_class_wmc, metrics_class_wmcnp, metrics_class_wmci) VALUES(:runId, :classId, :coverage, :loc, :locExecutable, :locExecuted, :aif, :ahf, :cis, :csz, :dit, :impl, :mif, :mhf, :noc, :pf, :vars, :varsnp, :varsi, :wmc, :wmcnp, :wmci);' ); $stmtSelectMethodId = $this->dbh->prepare( 'SELECT code_method_id FROM code_file, code_class, code_method WHERE code_class.code_file_id = code_file.code_file_id AND code_class.code_class_id = code_method.code_class_id AND code_file.revision = :revision AND code_class.code_class_name = :className AND code_method.code_method_name = :methodName;' ); $stmtInsertMethod = $this->dbh->prepare( 'INSERT INTO metrics_method (run_id, code_method_id, metrics_method_coverage, metrics_method_loc, metrics_method_loc_executable, metrics_method_loc_executed, metrics_method_ccn, metrics_method_crap, metrics_method_npath) VALUES(:runId, :methodId, :coverage, :loc, :locExecutable, :locExecuted, :ccn, :crap, :npath);' ); foreach ($fileMetrics->getFunctions() as $functionMetrics) { $functionName = $functionMetrics->getFunction()->getName(); $stmtSelectFunctionId->bindParam(':functionName', $functionName, PDO::PARAM_STR); $stmtSelectFunctionId->bindParam(':revision', $revision, PDO::PARAM_INT); $stmtSelectFunctionId->execute(); $functionId = (int)$stmtSelectFunctionId->fetchColumn(); $stmtSelectFunctionId->closeCursor(); $functionCoverage = $functionMetrics->getCoverage(); $functionLoc = $functionMetrics->getLoc(); $functionLocExecutable = $functionMetrics->getLocExecutable(); $functionLocExecuted = $functionMetrics->getLocExecuted(); $functionCcn = $functionMetrics->getCCN(); $functionCrap = $functionMetrics->getCrapIndex(); $functionNpath = $functionMetrics->getNPath(); $stmtInsertFunction->bindParam(':runId', $runId, PDO::PARAM_INT); $stmtInsertFunction->bindParam(':functionId', $functionId, PDO::PARAM_INT); $stmtInsertFunction->bindParam(':coverage', $functionCoverage); $stmtInsertFunction->bindParam(':loc', $functionLoc, PDO::PARAM_INT); $stmtInsertFunction->bindParam(':locExecutable', $functionLocExecutable, PDO::PARAM_INT); $stmtInsertFunction->bindParam(':locExecuted', $functionLocExecuted, PDO::PARAM_INT); $stmtInsertFunction->bindParam(':ccn', $functionCcn, PDO::PARAM_INT); $stmtInsertFunction->bindParam(':crap', $functionCrap); $stmtInsertFunction->bindParam(':npath', $functionNpath, PDO::PARAM_INT); $stmtInsertFunction->execute(); } foreach ($fileMetrics->getClasses() as $classMetrics) { $className = $classMetrics->getClass()->getName(); $stmtSelectClassId->bindParam(':className', $className, PDO::PARAM_STR); $stmtSelectClassId->bindParam(':revision', $revision, PDO::PARAM_INT); $stmtSelectClassId->execute(); $classId = (int)$stmtSelectClassId->fetchColumn(); $stmtSelectClassId->closeCursor(); $classCoverage = $classMetrics->getCoverage(); $classLoc = $classMetrics->getLoc(); $classLocExecutable = $classMetrics->getLocExecutable(); $classLocExecuted = $classMetrics->getLocExecuted(); $classAif = $classMetrics->getAIF(); $classAhf = $classMetrics->getAHF(); $classCis = $classMetrics->getCIS(); $classCsz = $classMetrics->getCSZ(); $classDit = $classMetrics->getDIT(); $classImpl = $classMetrics->getIMPL(); $classMif = $classMetrics->getMIF(); $classMhf = $classMetrics->getMHF(); $classNoc = $classMetrics->getNOC(); $classPf = $classMetrics->getPF(); $classVars = $classMetrics->getVARS(); $classVarsnp = $classMetrics->getVARSnp(); $classVarsi = $classMetrics->getVARSi(); $classWmc = $classMetrics->getWMC(); $classWmcnp = $classMetrics->getWMCnp(); $classWmci = $classMetrics->getWMCi(); $stmtInsertClass->bindParam(':runId', $runId, PDO::PARAM_INT); $stmtInsertClass->bindParam(':classId', $classId, PDO::PARAM_INT); $stmtInsertClass->bindParam(':coverage', $classCoverage); $stmtInsertClass->bindParam(':loc', $classLoc, PDO::PARAM_INT); $stmtInsertClass->bindParam(':locExecutable', $classLocExecutable, PDO::PARAM_INT); $stmtInsertClass->bindParam(':locExecuted', $classLocExecuted, PDO::PARAM_INT); $stmtInsertClass->bindParam(':aif', $classAif); $stmtInsertClass->bindParam(':ahf', $classAhf); $stmtInsertClass->bindParam(':cis', $classCis, PDO::PARAM_INT); $stmtInsertClass->bindParam(':csz', $classCsz, PDO::PARAM_INT); $stmtInsertClass->bindParam(':dit', $classDit, PDO::PARAM_INT); $stmtInsertClass->bindParam(':impl', $classImpl, PDO::PARAM_INT); $stmtInsertClass->bindParam(':mif', $classMif); $stmtInsertClass->bindParam(':mhf', $classMhf); $stmtInsertClass->bindParam(':noc', $classNoc, PDO::PARAM_INT); $stmtInsertClass->bindParam(':pf', $classPf); $stmtInsertClass->bindParam(':vars', $classVars, PDO::PARAM_INT); $stmtInsertClass->bindParam(':varsnp', $classVarsnp, PDO::PARAM_INT); $stmtInsertClass->bindParam(':varsi', $classVarsi, PDO::PARAM_INT); $stmtInsertClass->bindParam(':wmc', $classWmc, PDO::PARAM_INT); $stmtInsertClass->bindParam(':wmcnp', $classWmcnp, PDO::PARAM_INT); $stmtInsertClass->bindParam(':wmci', $classWmci, PDO::PARAM_INT); $stmtInsertClass->execute(); foreach ($classMetrics->getMethods() as $methodMetrics) { $methodName = $methodMetrics->getMethod()->getName(); $stmtSelectMethodId->bindParam(':className', $className, PDO::PARAM_STR); $stmtSelectMethodId->bindParam(':methodName', $methodName, PDO::PARAM_STR); $stmtSelectMethodId->bindParam(':revision', $revision, PDO::PARAM_INT); $stmtSelectMethodId->execute(); $methodId = (int)$stmtSelectMethodId->fetchColumn(); $stmtSelectMethodId->closeCursor(); $methodCoverage = $methodMetrics->getCoverage(); $methodLoc = $methodMetrics->getLoc(); $methodLocExecutable = $methodMetrics->getLocExecutable(); $methodLocExecuted = $methodMetrics->getLocExecuted(); $methodCcn = $methodMetrics->getCCN(); $methodCrap = $methodMetrics->getCrapIndex(); $methodNpath = $methodMetrics->getNPath(); $stmtInsertMethod->bindParam(':runId', $runId, PDO::PARAM_INT); $stmtInsertMethod->bindParam(':methodId', $methodId, PDO::PARAM_INT); $stmtInsertMethod->bindParam(':coverage', $methodCoverage); $stmtInsertMethod->bindParam(':loc', $methodLoc, PDO::PARAM_INT); $stmtInsertMethod->bindParam(':locExecutable', $methodLocExecutable, PDO::PARAM_INT); $stmtInsertMethod->bindParam(':locExecuted', $methodLocExecuted, PDO::PARAM_INT); $stmtInsertMethod->bindParam(':ccn', $methodCcn, PDO::PARAM_INT); $stmtInsertMethod->bindParam(':crap', $methodCrap); $stmtInsertMethod->bindParam(':npath', $methodNpath, PDO::PARAM_INT); $stmtInsertMethod->execute(); } } unset($stmtSelectFunctionId); unset($stmtInsertFunction); unset($stmtSelectClassId); unset($stmtInsertClass); unset($stmtSelectMethodId); unset($stmtInsertMethod); $stmt = $this->dbh->prepare( 'SELECT code_line_id, code_line_covered FROM code_line WHERE code_file_id = :fileId AND code_line_number = :lineNumber;' ); $stmt2 = $this->dbh->prepare( 'UPDATE code_line SET code_line_covered = :lineCovered WHERE code_line_id = :lineId;' ); $stmt3 = $this->dbh->prepare( 'INSERT INTO code_coverage (test_id, code_line_id) VALUES(:testId, :lineId);' ); for ($lineNumber = 1; $lineNumber <= $fileLoc; $lineNumber++) { $coveringTests = PHPUnit_Util_CodeCoverage::getCoveringTests( $codeCoverage, $file, $lineNumber ); if (is_array($coveringTests)) { $stmt->bindParam(':fileId', $fileId, PDO::PARAM_INT); $stmt->bindParam(':lineNumber', $lineNumber, PDO::PARAM_INT); $stmt->execute(); $codeLineId = (int)$stmt->fetchColumn(0); $oldCoverageFlag = (int)$stmt->fetchColumn(1); $newCoverageFlag = isset($summary[$file][$lineNumber]) ? 1 : 0; if (($oldCoverageFlag == 0 && $newCoverageFlag != 0) || ($oldCoverageFlag < 0 && $newCoverageFlag > 0)) { $stmt2->bindParam(':lineCovered', $newCoverageFlag, PDO::PARAM_INT); $stmt2->bindParam(':lineId', $codeLineId, PDO::PARAM_INT); $stmt2->execute(); } foreach ($coveringTests as $test) { $stmt3->bindParam(':testId', $test->__db_id, PDO::PARAM_INT); $stmt3->bindParam(':lineId', $codeLineId, PDO::PARAM_INT); $stmt3->execute(); } } } } unset($stmt); unset($stmt2); unset($stmt3); $stmt = $this->dbh->prepare( 'SELECT code_method.code_method_id FROM code_class, code_method WHERE code_class.code_class_id = code_method.code_class_id AND code_class.code_class_name = :className AND code_method.code_method_name = :methodName;' ); $stmt2 = $this->dbh->prepare( 'UPDATE test SET code_method_id = :methodId WHERE test_id = :testId;' ); foreach ($result->topTestSuite() as $test) { if ($test instanceof PHPUnit_Framework_TestCase) { $className = get_class($test); $methodName = $test->getName(); $stmt->bindParam(':className', $className, PDO::PARAM_STR); $stmt->bindParam(':methodName', $methodName, PDO::PARAM_STR); $stmt->execute(); $methodId = (int)$stmt->fetchColumn(); $stmt->closeCursor(); $stmt2->bindParam(':methodId', $methodId, PDO::PARAM_INT); $stmt2->bindParam(':testId', $test->__db_id, PDO::PARAM_INT); $stmt2->execute(); } } unset($stmt); unset($stmt2); $stmt = $this->dbh->prepare( 'INSERT INTO metrics_project (run_id, metrics_project_cls, metrics_project_clsa, metrics_project_clsc, metrics_project_roots, metrics_project_leafs, metrics_project_interfs, metrics_project_maxdit) VALUES(:runId, :cls, :clsa, :clsc, :roots, :leafs, :interfs, :maxdit);' ); $cls = $projectMetrics->getCLS(); $clsa = $projectMetrics->getCLSa(); $clsc = $projectMetrics->getCLSc(); $interfs = $projectMetrics->getInterfs(); $roots = $projectMetrics->getRoots(); $leafs = $projectMetrics->getLeafs(); $maxDit = $projectMetrics->getMaxDit(); $stmt->bindParam(':runId', $runId, PDO::PARAM_INT); $stmt->bindParam(':cls', $cls, PDO::PARAM_INT); $stmt->bindParam(':clsa', $clsa, PDO::PARAM_INT); $stmt->bindParam(':clsc', $clsc, PDO::PARAM_INT); $stmt->bindParam(':roots', $roots, PDO::PARAM_INT); $stmt->bindParam(':leafs', $leafs, PDO::PARAM_INT); $stmt->bindParam(':interfs', $interfs, PDO::PARAM_INT); $stmt->bindParam(':maxdit', $maxDit, PDO::PARAM_INT); $stmt->execute(); unset($stmt); $stmt = $this->dbh->prepare( 'UPDATE code_class SET code_class_parent_id = :parentClassId WHERE code_class_id = :classId;' ); $stmt2 = $this->dbh->prepare( 'SELECT code_class.code_class_id as code_class_id FROM code_class, code_file WHERE code_class.code_file_id = code_file.code_file_id AND code_file.revision = :revision AND code_class.code_class_name = :parentClassName;' ); foreach ($storedClasses as $className => $classId) { $class = new ReflectionClass($className); $parentClass = $class->getParentClass(); if ($parentClass !== FALSE) { $parentClassName = $parentClass->getName(); $parentClassId = 0; if (isset($storedClasses[$parentClassName])) { $parentClassId = $storedClasses[$parentClassName]; } else { $stmt2->bindParam(':parentClassName', $parentClassName, PDO::PARAM_STR); $stmt2->bindParam(':revision', $revision, PDO::PARAM_INT); $stmt2->execute(); $parentClassId = (int)$stmt2->fetchColumn(); $stmt2->closeCursor(); } if ($parentClassId > 0) { $stmt->bindParam(':classId', $classId, PDO::PARAM_INT); $stmt->bindParam(':parentClassId', $parentClassId, PDO::PARAM_INT); $stmt->execute(); } } } unset($stmt); unset($stmt2); $this->dbh->commit(); } } ?>