123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323 |
- <?php
- /**
- * $Id: PHPCPDTask.php 905 2010-10-05 16:28:03Z mrook $
- *
- * 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.
- *
- * This software consists of voluntary contributions made by many individuals
- * and is licensed under the LGPL. For more information please see
- * <http://phing.info>.
- */
- require_once 'phing/Task.php';
- /**
- * Runs PHP Copy & Paste Detector. Checking PHP files for duplicated code.
- * Refactored original PhpCpdTask provided by
- * Timo Haberkern <timo.haberkern@fantastic-bits.de>
- *
- * @package phing.tasks.ext.phpcpd
- * @author Benjamin Schultz <bschultz@proqrent.de>
- * @version $Id: PHPCPDTask.php 905 2010-10-05 16:28:03Z mrook $
- */
- class PHPCPDTask extends Task
- {
- /**
- * A php source code filename or directory
- *
- * @var PhingFile
- */
- protected $_file = null;
- /**
- * All fileset objects assigned to this task
- *
- * @var array<FileSet>
- */
- protected $_filesets = array();
- /**
- * Minimum number of identical lines.
- *
- * @var integer
- */
- protected $_minLines = 5;
- /**
- * Minimum number of identical tokens.
- *
- * @var integer
- */
- protected $_minTokens = 70;
- /**
- * List of valid file extensions for analyzed files.
- *
- * @var array
- */
- protected $_allowedFileExtensions = array('php');
- /**
- * List of exclude directory patterns.
- *
- * @var array
- */
- protected $_ignorePatterns = array('.git', '.svn', 'CVS', '.bzr', '.hg');
- /**
- * The format for the report
- *
- * @var string
- */
- protected $_format = 'default';
- /**
- * Formatter elements.
- *
- * @var array<PHPCPDFormatterElement>
- */
- protected $_formatters = array();
- /**
- * Load the necessary environment for running PHPCPD.
- *
- * @throws BuildException - if the phpcpd classes can't be loaded.
- */
- public function init()
- {
- /**
- * Determine PHPCPD installation
- */
- @include_once 'PHPCPD/TextUI/Command.php';
- if (! class_exists('PHPCPD_TextUI_Command')) {
- throw new BuildException(
- 'PHPCPDTask depends on PHPCPD being installed '
- . 'and on include_path.',
- $this->getLocation()
- );
- }
- // Other dependencies that should only be loaded
- // when class is actually used
- require_once 'phing/tasks/ext/phpcpd/PHPCPDFormatterElement.php';
- require_once 'PHPCPD/Detector.php';
- }
- /**
- * Set the input source file or directory.
- *
- * @param PhingFile $file The input source file or directory.
- *
- * @return void
- */
- public function setFile(PhingFile $file)
- {
- $this->_file = $file;
- }
- /**
- * Nested creator, adds a set of files (nested fileset attribute).
- *
- * @return FileSet The created fileset object
- */
- public function createFileSet()
- {
- $num = array_push($this->_filesets, new FileSet());
- return $this->_filesets[$num-1];
- }
- /**
- * Sets the minimum rule priority.
- *
- * @param integer $minimumPriority Minimum rule priority.
- *
- * @return void
- */
- public function setMinimumPriority($minimumPriority)
- {
- $this->_minimumPriority = $minimumPriority;
- }
- /**
- * Sets the minimum number of identical lines (default: 5).
- *
- * @param integer $minLines Minimum number of identical lines
- *
- * @return void
- */
- public function setMinLines($minLines)
- {
- $this->_minLines = $minLines;
- }
- /**
- * Sets the minimum number of identical tokens (default: 70).
- *
- * @param integer $minTokens Minimum number of identical tokens
- */
- public function setMinTokens($minTokens)
- {
- $this->_minTokens = $minTokens;
- }
- /**
- * Sets a list of filename extensions for valid php source code files.
- *
- * @param string $fileExtensions List of valid file extensions.
- *
- * @return void
- */
- public function setAllowedFileExtensions($fileExtensions)
- {
- $this->_allowedFileExtensions = array();
- $token = ' ,;';
- $ext = strtok($fileExtensions, $token);
- while ($ext !== false) {
- $this->_allowedFileExtensions[] = $ext;
- $ext = strtok($token);
- }
- }
- /**
- * Sets a list of ignore patterns that is used to exclude directories from
- * the source analysis.
- *
- * @param string $ignorePatterns List of ignore patterns.
- *
- * @return void
- */
- public function setIgnorePatterns($ignorePatterns)
- {
- $this->_ignorePatterns = array();
- $token = ' ,;';
- $pattern = strtok($ignorePatterns, $token);
- while ($pattern !== false) {
- $this->_ignorePatterns[] = $pattern;
- $pattern = strtok($token);
- }
- }
- /**
- * Sets the output format
- *
- * @param string $format Format of the report
- */
- public function setFormat($format)
- {
- $this->_format = $format;
- }
- /**
- * Create object for nested formatter element.
- *
- * @return PHPCPDFormatterElement
- */
- public function createFormatter()
- {
- $num = array_push(
- $this->_formatters,
- new PHPCPDFormatterElement($this)
- );
- return $this->_formatters[$num-1];
- }
- /**
- * Executes PHPCPD against PhingFile or a FileSet
- *
- * @return void
- */
- public function main()
- {
- if (!isset($this->_file) and count($this->_filesets) == 0) {
- throw new BuildException(
- "Missing either a nested fileset or attribute 'file' set"
- );
- }
- if (count($this->_formatters) == 0) {
- // turn legacy format attribute into formatter
- $fmt = new PHPCPDFormatterElement($this);
- $fmt->setType($this->format);
- $fmt->setUseFile(false);
- $this->_formatters[] = $fmt;
- }
- $this->validateFormatters();
- $filesToParse = array();
- if ($this->_file instanceof PhingFile) {
- $filesToParse[] = $this->_file->getPath();
- } else {
- // append any files in filesets
- foreach ($this->_filesets as $fs) {
- $files = $fs->getDirectoryScanner($this->project)
- ->getIncludedFiles();
- foreach ($files as $filename) {
- $f = new PhingFile($fs->getDir($this->project), $filename);
- $filesToParse[] = $f->getAbsolutePath();
- }
- }
- }
- $this->log('Processing files...');
- $detector = new PHPCPD_Detector();
- $clones = $detector->copyPasteDetection(
- $filesToParse,
- $this->_minLines,
- $this->_minTokens
- );
- $this->log('Finished copy/paste detection');
- foreach ($this->_formatters as $fe) {
- $formatter = $fe->getFormatter();
- $formatter->processClones(
- $clones,
- $fe->getOutfile(),
- $this->project
- );
- }
- }
- /**
- * Validates the available formatters
- *
- * @throws BuildException
- * @return void
- */
- protected function validateFormatters()
- {
- foreach ($this->_formatters as $fe) {
- if ($fe->getType() == '') {
- throw new BuildException(
- "Formatter missing required 'type' attribute."
- );
- }
- if ($fe->getUsefile() && $fe->getOutfile() === null) {
- throw new BuildException(
- "Formatter requires 'outfile' attribute "
- . "when 'useFile' is true."
- );
- }
- }
- }
- }
|