2012-03-05 12:26:46 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This file is copied from the Symfony package.
|
|
|
|
*
|
|
|
|
* (c) Fabien Potencier <fabien@symfony.com>
|
|
|
|
*
|
|
|
|
* For the full copyright and license information, please view the LICENSE
|
|
|
|
* file that was distributed with this source code.
|
|
|
|
*
|
|
|
|
* @license MIT
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace Composer\Autoload;
|
2014-03-29 16:46:55 +00:00
|
|
|
|
2013-06-29 20:46:04 +00:00
|
|
|
use Symfony\Component\Finder\Finder;
|
2014-03-29 16:46:55 +00:00
|
|
|
use Composer\IO\IOInterface;
|
2012-03-05 12:26:46 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* ClassMapGenerator
|
|
|
|
*
|
|
|
|
* @author Gyula Sallai <salla016@gmail.com>
|
2014-03-29 16:46:55 +00:00
|
|
|
* @author Jordi Boggiano <j.boggiano@seld.be>
|
2012-03-05 12:26:46 +00:00
|
|
|
*/
|
|
|
|
class ClassMapGenerator
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Generate a class map file
|
|
|
|
*
|
2014-02-11 10:09:30 +00:00
|
|
|
* @param \Traversable $dirs Directories or a single path to search in
|
2014-06-10 14:02:44 +00:00
|
|
|
* @param string $file The name of the class map file
|
2012-03-05 12:26:46 +00:00
|
|
|
*/
|
2012-05-22 15:13:15 +00:00
|
|
|
public static function dump($dirs, $file)
|
2012-03-05 12:26:46 +00:00
|
|
|
{
|
|
|
|
$maps = array();
|
|
|
|
|
|
|
|
foreach ($dirs as $dir) {
|
|
|
|
$maps = array_merge($maps, static::createMap($dir));
|
|
|
|
}
|
|
|
|
|
|
|
|
file_put_contents($file, sprintf('<?php return %s;', var_export($maps, true)));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Iterate over all files in the given directory searching for classes
|
|
|
|
*
|
2014-02-11 10:09:30 +00:00
|
|
|
* @param \Iterator|string $path The path to search in or an iterator
|
2014-04-29 12:35:02 +00:00
|
|
|
* @param string $whitelist Regex that matches against the file path
|
|
|
|
* @param IOInterface $io IO object
|
|
|
|
* @param string $namespace Optional namespace prefix to filter by
|
2012-03-05 12:26:46 +00:00
|
|
|
*
|
|
|
|
* @return array A class map array
|
2012-11-11 17:31:17 +00:00
|
|
|
*
|
|
|
|
* @throws \RuntimeException When the path is neither an existing file nor directory
|
2012-03-05 12:26:46 +00:00
|
|
|
*/
|
2014-04-29 12:35:02 +00:00
|
|
|
public static function createMap($path, $whitelist = null, IOInterface $io = null, $namespace = null)
|
2012-03-05 12:26:46 +00:00
|
|
|
{
|
2012-11-11 17:31:17 +00:00
|
|
|
if (is_string($path)) {
|
|
|
|
if (is_file($path)) {
|
|
|
|
$path = array(new \SplFileInfo($path));
|
2013-01-05 19:01:58 +00:00
|
|
|
} elseif (is_dir($path)) {
|
2014-03-18 21:37:37 +00:00
|
|
|
$path = Finder::create()->files()->followLinks()->name('/\.(php|inc|hh)$/')->in($path);
|
2012-04-01 18:23:47 +00:00
|
|
|
} else {
|
2012-11-11 17:31:17 +00:00
|
|
|
throw new \RuntimeException(
|
|
|
|
'Could not scan for classes inside "'.$path.
|
|
|
|
'" which does not appear to be a file nor a folder'
|
|
|
|
);
|
2012-04-01 18:23:47 +00:00
|
|
|
}
|
2012-03-05 12:26:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
$map = array();
|
|
|
|
|
2012-11-11 17:31:17 +00:00
|
|
|
foreach ($path as $file) {
|
|
|
|
$filePath = $file->getRealPath();
|
2012-03-05 12:26:46 +00:00
|
|
|
|
2014-03-18 21:37:37 +00:00
|
|
|
if (!in_array(pathinfo($filePath, PATHINFO_EXTENSION), array('php', 'inc', 'hh'))) {
|
2012-03-05 12:26:46 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2012-11-11 17:31:17 +00:00
|
|
|
if ($whitelist && !preg_match($whitelist, strtr($filePath, '\\', '/'))) {
|
2012-06-20 11:05:03 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2012-11-11 17:31:17 +00:00
|
|
|
$classes = self::findClasses($filePath);
|
2012-03-05 12:26:46 +00:00
|
|
|
|
|
|
|
foreach ($classes as $class) {
|
2014-04-29 12:35:02 +00:00
|
|
|
// skip classes not within the given namespace prefix
|
|
|
|
if (null !== $namespace && 0 !== strpos($class, $namespace)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2014-03-29 16:46:55 +00:00
|
|
|
if (!isset($map[$class])) {
|
|
|
|
$map[$class] = $filePath;
|
2014-04-10 11:15:35 +00:00
|
|
|
} elseif ($io && $map[$class] !== $filePath && !preg_match('{/(test|fixture|example)s?/}i', strtr($map[$class].' '.$filePath, '\\', '/'))) {
|
2015-02-06 12:52:44 +00:00
|
|
|
$io->writeError(
|
2014-03-29 16:46:55 +00:00
|
|
|
'<warning>Warning: Ambiguous class resolution, "'.$class.'"'.
|
|
|
|
' was found in both "'.$map[$class].'" and "'.$filePath.'", the first will be used.</warning>'
|
2014-03-24 13:34:02 +00:00
|
|
|
);
|
2014-03-20 12:37:05 +00:00
|
|
|
}
|
2012-03-05 12:26:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $map;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Extract the classes in the given file
|
|
|
|
*
|
2013-06-13 11:28:24 +00:00
|
|
|
* @param string $path The file to check
|
2013-06-13 00:05:44 +00:00
|
|
|
* @throws \RuntimeException
|
2013-06-13 11:28:24 +00:00
|
|
|
* @return array The found classes
|
2012-03-05 12:26:46 +00:00
|
|
|
*/
|
2012-05-22 15:13:15 +00:00
|
|
|
private static function findClasses($path)
|
2012-03-05 12:26:46 +00:00
|
|
|
{
|
2015-05-05 18:18:24 +00:00
|
|
|
$extraTypes = PHP_VERSION_ID < 50400 ? '' : '|trait';
|
2015-03-09 17:02:10 +00:00
|
|
|
if (defined('HHVM_VERSION') && version_compare(HHVM_VERSION, '3.3', '>=')) {
|
2015-03-09 17:05:12 +00:00
|
|
|
$extraTypes .= '|enum';
|
2015-03-09 16:37:56 +00:00
|
|
|
}
|
2012-11-10 16:29:45 +00:00
|
|
|
|
2012-05-21 10:51:21 +00:00
|
|
|
try {
|
2014-07-19 22:07:58 +00:00
|
|
|
$contents = @php_strip_whitespace($path);
|
2014-07-20 17:07:31 +00:00
|
|
|
if (!$contents) {
|
|
|
|
if (!file_exists($path)) {
|
|
|
|
throw new \Exception('File does not exist');
|
|
|
|
}
|
|
|
|
if (!is_readable($path)) {
|
|
|
|
throw new \Exception('File is not readable');
|
|
|
|
}
|
|
|
|
}
|
2012-11-10 16:29:45 +00:00
|
|
|
} catch (\Exception $e) {
|
|
|
|
throw new \RuntimeException('Could not scan for classes inside '.$path.": \n".$e->getMessage(), 0, $e);
|
|
|
|
}
|
2012-11-11 17:30:31 +00:00
|
|
|
|
2012-11-12 09:46:14 +00:00
|
|
|
// return early if there is no chance of matching anything in this file
|
2015-03-09 17:05:12 +00:00
|
|
|
if (!preg_match('{\b(?:class|interface'.$extraTypes.')\s}i', $contents)) {
|
2012-11-12 09:46:14 +00:00
|
|
|
return array();
|
|
|
|
}
|
|
|
|
|
2012-11-11 17:30:31 +00:00
|
|
|
// strip heredocs/nowdocs
|
2014-04-28 13:19:38 +00:00
|
|
|
$contents = preg_replace('{<<<\s*(\'?)(\w+)\\1(?:\r\n|\n|\r)(?:.*?)(?:\r\n|\n|\r)\\2(?=\r\n|\n|\r|;)}s', 'null', $contents);
|
2012-11-11 17:30:31 +00:00
|
|
|
// strip strings
|
2015-06-16 10:22:23 +00:00
|
|
|
$contents = preg_replace('{"[^"\\\\]*+(\\\\.[^"\\\\]*+)*+"|\'[^\'\\\\]*+(\\\\.[^\'\\\\]*+)*+\'}s', 'null', $contents);
|
2012-11-12 11:26:03 +00:00
|
|
|
// strip leading non-php code if needed
|
|
|
|
if (substr($contents, 0, 2) !== '<?') {
|
2014-04-14 10:47:47 +00:00
|
|
|
$contents = preg_replace('{^.+?<\?}s', '<?', $contents, 1, $replacements);
|
|
|
|
if ($replacements === 0) {
|
|
|
|
return array();
|
|
|
|
}
|
2012-11-12 08:15:35 +00:00
|
|
|
}
|
2012-11-12 11:26:03 +00:00
|
|
|
// strip non-php blocks in the file
|
2012-12-13 16:39:17 +00:00
|
|
|
$contents = preg_replace('{\?>.+<\?}s', '?><?', $contents);
|
|
|
|
// strip trailing non-php code if needed
|
|
|
|
$pos = strrpos($contents, '?>');
|
|
|
|
if (false !== $pos && false === strpos(substr($contents, $pos), '<?')) {
|
|
|
|
$contents = substr($contents, 0, $pos);
|
|
|
|
}
|
2012-11-12 08:15:35 +00:00
|
|
|
|
2012-11-11 17:30:31 +00:00
|
|
|
preg_match_all('{
|
|
|
|
(?:
|
2015-06-16 10:22:23 +00:00
|
|
|
\b(?<![\$:>])(?P<type>class|interface'.$extraTypes.') \s++ (?P<name>[a-zA-Z_\x7f-\xff:][a-zA-Z0-9_\x7f-\xff:\-]*+)
|
|
|
|
| \b(?<![\$:>])(?P<ns>namespace) (?P<nsname>\s++[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*+(?:\s*+\\\\\s*+[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*+)*+)? \s*+ [\{;]
|
2012-11-11 17:30:31 +00:00
|
|
|
)
|
2012-11-25 13:07:06 +00:00
|
|
|
}ix', $contents, $matches);
|
2012-11-11 17:30:31 +00:00
|
|
|
|
2012-11-12 09:46:14 +00:00
|
|
|
$classes = array();
|
2012-11-11 17:30:31 +00:00
|
|
|
$namespace = '';
|
|
|
|
|
|
|
|
for ($i = 0, $len = count($matches['type']); $i < $len; $i++) {
|
|
|
|
if (!empty($matches['ns'][$i])) {
|
|
|
|
$namespace = str_replace(array(' ', "\t", "\r", "\n"), '', $matches['nsname'][$i]) . '\\';
|
|
|
|
} else {
|
2014-04-14 04:21:53 +00:00
|
|
|
$name = $matches['name'][$i];
|
|
|
|
if ($name[0] === ':') {
|
2014-07-19 22:07:58 +00:00
|
|
|
// This is an XHP class, https://github.com/facebook/xhp
|
|
|
|
$name = 'xhp'.substr(str_replace(array('-', ':'), array('_', '__'), $name), 1);
|
2015-04-15 00:21:03 +00:00
|
|
|
} elseif ($matches['type'][$i] === 'enum') {
|
2015-03-09 16:37:56 +00:00
|
|
|
// In Hack, something like:
|
|
|
|
// enum Foo: int { HERP = '123'; }
|
|
|
|
// The regex above captures the colon, which isn't part of
|
|
|
|
// the class name.
|
|
|
|
$name = rtrim($name, ':');
|
2014-04-14 04:21:53 +00:00
|
|
|
}
|
|
|
|
$classes[] = ltrim($namespace . $name, '\\');
|
2012-11-11 17:30:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $classes;
|
2012-03-05 12:26:46 +00:00
|
|
|
}
|
|
|
|
}
|