Created
November 11, 2016 19:23
-
-
Save jeffsrepoaccount/d7613eb3b97c977a483562abef38bf40 to your computer and use it in GitHub Desktop.
Allows phing tasks to iterate over a delimited list property and call a target for every item in the list, setting a property for each item as well as the current key index.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<?php | |
/* | |
* $Id$ | |
* | |
* 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'; | |
require_once 'phing/system/io/FileSystem.php'; | |
include_once 'phing/mappers/FileNameMapper.php'; | |
include_once 'phing/tasks/system/PhingTask.php'; | |
/** | |
* <foreachkey> task | |
* | |
* Task definition for the foreachkey task. This task takes a list with | |
* delimited values, and executes a target with set param. | |
* | |
* Usage: | |
* <foreachkey list="values" target="targ" param="name" delimiter="|" /> | |
* | |
* Attributes: | |
* list --> The list of values to process, with the delimiter character, | |
* indicated by the "delimiter" attribute, separating each value. | |
* target --> The target to call for each token, passing the token as the | |
* parameter with the name indicated by the "param" attribute. | |
* param --> The name of the parameter to pass the tokens in as to the | |
* target. | |
* delimiter --> The delimiter string that separates the values in the "list" | |
* parameter. The default is ",". | |
* key --> The name of the key parameter to pass the tokens in as to the | |
* target. | |
* | |
* @author Jason Hines <jason@greenhell.com> | |
* @author Hans Lellelid <hans@xmpl.org> | |
* @author Jeff Lambert <http://stackoverflow.com/users/697370/jeff-lambert> | |
* @version $Id$ | |
* @package phing.tasks.system | |
*/ | |
class ForeachkeyTask extends Task | |
{ | |
protected $taskname = 'foreachkey'; | |
/** Delimter-separated list of values to process. */ | |
private $list; | |
/** Name of parameter to pass to callee */ | |
private $param; | |
/** Name of absolute path parameter to pass to callee */ | |
private $absparam; | |
/** Delimiter that separates items in $list */ | |
private $delimiter = ','; | |
/** Key name to pass to callee */ | |
private $key; | |
/** | |
* PhingCallTask that will be invoked w/ calleeTarget. | |
* @var PhingCallTask | |
*/ | |
private $callee; | |
/** Array of filesets */ | |
private $filesets = array(); | |
/** Instance of mapper **/ | |
private $mapperElement; | |
/** | |
* Array of filelists | |
* @var array | |
*/ | |
private $filelists = array(); | |
/** | |
* Target to execute. | |
* @var string | |
*/ | |
private $calleeTarget; | |
/** | |
* Total number of files processed | |
* @var integer | |
*/ | |
private $total_files = 0; | |
/** | |
* Total number of directories processed | |
* @var integer | |
*/ | |
private $total_dirs = 0; | |
public function init() | |
{ | |
$this->callee = $this->project->createTask("phingcall"); | |
$this->callee->setOwningTarget($this->getOwningTarget()); | |
$this->callee->setTaskName($this->getTaskName()); | |
$this->callee->setLocation($this->getLocation()); | |
$this->callee->init(); | |
} | |
/** | |
* This method does the work. | |
* @throws BuildException | |
* @return void | |
*/ | |
public function main() | |
{ | |
if ($this->list === null && count($this->filesets) == 0 && count($this->filelists) == 0) { | |
throw new BuildException("Need either list, nested fileset or nested filelist to iterate through"); | |
} | |
if ($this->param === null) { | |
throw new BuildException("You must supply a property name to set on each iteration in param"); | |
} | |
if ($this->calleeTarget === null) { | |
throw new BuildException("You must supply a target to perform"); | |
} | |
$callee = $this->callee; | |
$callee->setTarget($this->calleeTarget); | |
$callee->setInheritAll(true); | |
$callee->setInheritRefs(true); | |
$mapper = null; | |
if ($this->mapperElement !== null) { | |
$mapper = $this->mapperElement->getImplementation(); | |
} | |
$propertyList = trim($this->project->getProperty($this->list)); | |
if ($propertyList) { | |
$arr = explode($this->delimiter, $propertyList); | |
$total_entries = 0; | |
foreach ($arr as $k => $value) { | |
$value = trim($value); | |
$premapped = ''; | |
if ($mapper !== null) { | |
$premapped = $value; | |
$value = $mapper->main($value); | |
if ($value === null) { | |
continue; | |
} | |
$value = array_shift($value); | |
} | |
$this->log( | |
"Setting param '$this->param' to value '$value'" . ($premapped ? " (mapped from '$premapped')" : ''), | |
Project::MSG_VERBOSE | |
); | |
$this->createProperty($this->key, $k); | |
$this->createProperty($this->param, $value); | |
$callee->main(); | |
$total_entries++; | |
} | |
} | |
// filelists | |
foreach ($this->filelists as $fl) { | |
$srcFiles = $fl->getFiles($this->project); | |
$this->process($callee, $fl->getDir($this->project), $srcFiles, array()); | |
} | |
// filesets | |
foreach ($this->filesets as $fs) { | |
$ds = $fs->getDirectoryScanner($this->project); | |
$srcFiles = $ds->getIncludedFiles(); | |
$srcDirs = $ds->getIncludedDirectories(); | |
$this->process($callee, $fs->getDir($this->project), $srcFiles, $srcDirs); | |
} | |
if ($this->list === null) { | |
$this->log( | |
"Processed {$this->total_dirs} directories and {$this->total_files} files", | |
Project::MSG_VERBOSE | |
); | |
} else { | |
$this->log( | |
"Processed $total_entries entr" . ($total_entries > 1 ? 'ies' : 'y') . " in list", | |
Project::MSG_VERBOSE | |
); | |
} | |
} | |
/** | |
* Processes a list of files & directories | |
* | |
* @param Task $callee | |
* @param PhingFile $fromDir | |
* @param array $srcFiles | |
* @param array $srcDirs | |
*/ | |
protected function process(Task $callee, PhingFile $fromDir, $srcFiles, $srcDirs) | |
{ | |
$mapper = null; | |
if ($this->mapperElement !== null) { | |
$mapper = $this->mapperElement->getImplementation(); | |
} | |
$filecount = count($srcFiles); | |
$this->total_files += $filecount; | |
for ($j = 0; $j < $filecount; $j++) { | |
$value = $srcFiles[$j]; | |
$premapped = ""; | |
if ($this->absparam) { | |
$this->createProperty($this->key, $j); | |
$this->createProperty( | |
$this->param, | |
$fromDir . FileSystem::getFileSystem()->getSeparator() . $value | |
); | |
} | |
if ($mapper !== null) { | |
$premapped = $value; | |
$value = $mapper->main($value); | |
if ($value === null) { | |
continue; | |
} | |
$value = array_shift($value); | |
} | |
if ($this->param) { | |
$this->log( | |
"Setting param '$this->param' to value '$value'" . ($premapped ? " (mapped from '$premapped')" : ''), | |
Project::MSG_VERBOSE | |
); | |
$this->createProperty($this->key, $j); | |
$this->createProperty($this->param, $value); | |
} | |
$callee->main(); | |
} | |
$dircount = count($srcDirs); | |
$this->total_dirs += $dircount; | |
for ($j = 0; $j < $dircount; $j++) { | |
$value = $srcDirs[$j]; | |
$premapped = ""; | |
if ($this->absparam) { | |
$this->createProperty($this->key, $j); | |
$this->createProperty( | |
$this->absparam, | |
$fromDir . FileSystem::getFileSystem()->getSeparator() . $value | |
); | |
} | |
if ($mapper !== null) { | |
$premapped = $value; | |
$value = $mapper->main($value); | |
if ($value === null) { | |
continue; | |
} | |
$value = array_shift($value); | |
} | |
if ($this->param) { | |
$this->log( | |
"Setting param '$this->param' to value '$value'" . ($premapped ? " (mapped from '$premapped')" : ''), | |
Project::MSG_VERBOSE | |
); | |
$this->createProperty($this->key, $j); | |
$this->createProperty($this->param, $value); | |
} | |
$callee->main(); | |
} | |
} | |
/** | |
* @param $key | |
*/ | |
public function setKey($key) | |
{ | |
$this->key = $key; | |
} | |
/** | |
* @param $list | |
*/ | |
public function setList($list) | |
{ | |
$this->list = (string) $list; | |
} | |
/** | |
* @param $target | |
*/ | |
public function setTarget($target) | |
{ | |
$this->calleeTarget = (string) $target; | |
} | |
/** | |
* @param $param | |
*/ | |
public function setParam($param) | |
{ | |
$this->param = (string) $param; | |
} | |
/** | |
* @param $absparam | |
*/ | |
public function setAbsparam($absparam) | |
{ | |
$this->absparam = (string) $absparam; | |
} | |
/** | |
* @param $delimiter | |
*/ | |
public function setDelimiter($delimiter) | |
{ | |
$this->delimiter = (string) $delimiter; | |
} | |
/** | |
* Nested adder, adds a set of files (nested fileset attribute). | |
* | |
* @param FileSet $fs | |
* @return void | |
*/ | |
public function addFileSet(FileSet $fs) | |
{ | |
$this->filesets[] = $fs; | |
} | |
/** | |
* Nested creator, creates one Mapper for this task | |
* | |
* @return object The created Mapper type object | |
* @throws BuildException | |
*/ | |
public function createMapper() | |
{ | |
if ($this->mapperElement !== null) { | |
throw new BuildException("Cannot define more than one mapper", $this->location); | |
} | |
$this->mapperElement = new Mapper($this->project); | |
return $this->mapperElement; | |
} | |
/** | |
* @return SonarProperty | |
*/ | |
public function createProperty($name = null, $value = null) | |
{ | |
if($name === null && $value === null) { | |
return $this->callee->createProperty(); | |
} | |
$prop = $this->callee->createProperty(); | |
$prop->setOverride(true); | |
$prop->setName($name); | |
$prop->setValue($value); | |
$prop->setEnvironment('env'); | |
return $prop; | |
} | |
/** | |
* Supports embedded <filelist> element. | |
* @return FileList | |
*/ | |
public function createFileList() | |
{ | |
$num = array_push($this->filelists, new FileList()); | |
return $this->filelists[$num - 1]; | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment