2007-07-19 12:40:28 +02:00
|
|
|
<?php
|
2014-08-18 05:39:42 +02:00
|
|
|
|
2007-07-19 12:40:28 +02:00
|
|
|
/**
|
|
|
|
* Provides introspection information about the class tree.
|
2014-08-18 05:39:42 +02:00
|
|
|
*
|
2015-02-27 01:10:32 +01:00
|
|
|
* It's a cached wrapper around the built-in class functions. SilverStripe uses
|
|
|
|
* class introspection heavily and without the caching it creates an unfortunate
|
2014-08-18 05:39:42 +02:00
|
|
|
* performance hit.
|
2008-02-25 03:10:37 +01:00
|
|
|
*
|
2012-04-12 08:02:46 +02:00
|
|
|
* @package framework
|
2008-02-25 03:10:37 +01:00
|
|
|
* @subpackage core
|
2007-07-19 12:40:28 +02:00
|
|
|
*/
|
|
|
|
class ClassInfo {
|
2014-08-18 05:39:42 +02:00
|
|
|
|
2012-04-17 04:46:07 +02:00
|
|
|
/**
|
|
|
|
* Wrapper for classes getter.
|
2008-03-03 00:24:10 +01:00
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public static function allClasses() {
|
2012-04-17 04:46:07 +02:00
|
|
|
return SS_ClassLoader::instance()->getManifest()->getClasses();
|
2007-07-19 12:40:28 +02:00
|
|
|
}
|
2008-03-03 00:24:10 +01:00
|
|
|
|
|
|
|
/**
|
2016-01-04 09:15:17 +01:00
|
|
|
* Returns true if a class or interface name exists.
|
|
|
|
*
|
|
|
|
* @param string $class
|
|
|
|
* @return bool
|
2008-03-03 00:24:10 +01:00
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public static function exists($class) {
|
2016-01-04 09:15:17 +01:00
|
|
|
return class_exists($class, false) || interface_exists($class, false) || SS_ClassLoader::instance()->getItemPath($class);
|
2007-07-19 12:40:28 +02:00
|
|
|
}
|
2008-03-03 00:24:10 +01:00
|
|
|
|
2009-03-04 04:44:11 +01:00
|
|
|
/**
|
|
|
|
* Cache for {@link hasTable()}
|
|
|
|
*/
|
2013-06-21 00:32:08 +02:00
|
|
|
private static $_cache_all_tables = array();
|
2012-01-02 14:01:16 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var Array Cache for {@link ancestry()}.
|
|
|
|
*/
|
|
|
|
private static $_cache_ancestry = array();
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2008-03-03 00:24:10 +01:00
|
|
|
/**
|
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 04:06:31 +01:00
|
|
|
* @todo Move this to SS_Database or DB
|
2008-03-03 00:24:10 +01:00
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public static function hasTable($class) {
|
2013-06-21 00:32:08 +02:00
|
|
|
// Cache the list of all table names to reduce on DB traffic
|
|
|
|
if(empty(self::$_cache_all_tables) && DB::is_active()) {
|
|
|
|
self::$_cache_all_tables = DB::get_schema()->tableList();
|
2008-11-09 23:11:25 +01:00
|
|
|
}
|
2013-06-21 00:32:08 +02:00
|
|
|
return !empty(self::$_cache_all_tables[strtolower($class)]);
|
2007-07-19 12:40:28 +02:00
|
|
|
}
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public static function reset_db_cache() {
|
2010-04-12 04:03:16 +02:00
|
|
|
self::$_cache_all_tables = null;
|
2012-01-02 14:01:16 +01:00
|
|
|
self::$_cache_ancestry = array();
|
2010-04-12 04:03:16 +02:00
|
|
|
}
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2007-07-19 12:40:28 +02:00
|
|
|
/**
|
|
|
|
* Returns the manifest of all classes which are present in the database.
|
2014-08-15 08:53:05 +02:00
|
|
|
*
|
2010-10-19 03:31:56 +02:00
|
|
|
* @param string $class Class name to check enum values for ClassName field
|
2014-08-15 08:53:05 +02:00
|
|
|
* @param boolean $includeUnbacked Flag indicating whether or not to include
|
2013-06-21 00:32:08 +02:00
|
|
|
* types that don't exist as implemented classes. By default these are excluded.
|
|
|
|
* @return array List of subclasses
|
2007-07-19 12:40:28 +02:00
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public static function getValidSubClasses($class = 'SiteTree', $includeUnbacked = false) {
|
2015-08-07 04:20:01 +02:00
|
|
|
if(is_string($class) && !class_exists($class)) return null;
|
|
|
|
|
2015-02-27 01:10:32 +01:00
|
|
|
$class = self::class_name($class);
|
2013-06-21 00:32:08 +02:00
|
|
|
$classes = DB::get_schema()->enumValuesForField($class, 'ClassName');
|
2012-08-29 04:30:10 +02:00
|
|
|
if (!$includeUnbacked) $classes = array_filter($classes, array('ClassInfo', 'exists'));
|
|
|
|
return $classes;
|
2007-07-19 12:40:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2011-02-19 05:35:59 +01:00
|
|
|
* Returns an array of the current class and all its ancestors and children
|
2013-06-21 00:32:08 +02:00
|
|
|
* which require a DB table.
|
2014-08-15 08:53:05 +02:00
|
|
|
*
|
2011-02-19 05:35:59 +01:00
|
|
|
* @param string|object $class
|
2008-11-06 05:51:25 +01:00
|
|
|
* @todo Move this into data object
|
2008-04-14 06:49:08 +02:00
|
|
|
* @return array
|
2007-07-19 12:40:28 +02:00
|
|
|
*/
|
2011-02-19 05:35:59 +01:00
|
|
|
public static function dataClassesFor($class) {
|
2015-08-07 04:20:01 +02:00
|
|
|
if(is_string($class) && !class_exists($class)) return null;
|
|
|
|
|
2011-02-19 05:35:59 +01:00
|
|
|
$result = array();
|
|
|
|
|
2015-02-27 01:10:32 +01:00
|
|
|
$class = self::class_name($class);
|
2007-07-19 12:40:28 +02:00
|
|
|
|
2011-02-19 05:35:59 +01:00
|
|
|
$classes = array_merge(
|
|
|
|
self::ancestry($class),
|
2013-06-21 00:32:08 +02:00
|
|
|
self::subclassesFor($class)
|
|
|
|
);
|
2011-02-19 05:35:59 +01:00
|
|
|
|
|
|
|
foreach ($classes as $class) {
|
2013-06-21 00:32:08 +02:00
|
|
|
if (DataObject::has_own_table($class)) $result[$class] = $class;
|
2007-07-19 12:40:28 +02:00
|
|
|
}
|
2011-02-19 05:35:59 +01:00
|
|
|
|
|
|
|
return $result;
|
2007-07-19 12:40:28 +02:00
|
|
|
}
|
2011-03-24 10:14:38 +01:00
|
|
|
|
2007-07-19 12:40:28 +02:00
|
|
|
/**
|
2011-03-24 10:14:38 +01:00
|
|
|
* Returns the root class (the first to extend from DataObject) for the
|
|
|
|
* passed class.
|
|
|
|
*
|
|
|
|
* @param string|object $class
|
|
|
|
* @return string
|
2007-07-19 12:40:28 +02:00
|
|
|
*/
|
2011-03-24 10:14:38 +01:00
|
|
|
public static function baseDataClass($class) {
|
2015-08-07 04:20:01 +02:00
|
|
|
if(is_string($class) && !class_exists($class)) return null;
|
|
|
|
|
2015-02-27 01:10:32 +01:00
|
|
|
$class = self::class_name($class);
|
2011-03-24 10:14:38 +01:00
|
|
|
|
2011-10-29 05:56:38 +02:00
|
|
|
if (!is_subclass_of($class, 'DataObject')) {
|
2011-12-16 23:51:40 +01:00
|
|
|
throw new InvalidArgumentException("$class is not a subclass of DataObject");
|
2011-03-24 10:14:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
while ($next = get_parent_class($class)) {
|
|
|
|
if ($next == 'DataObject') {
|
|
|
|
return $class;
|
|
|
|
}
|
|
|
|
|
|
|
|
$class = $next;
|
2007-07-19 12:40:28 +02:00
|
|
|
}
|
|
|
|
}
|
2011-03-24 10:14:38 +01:00
|
|
|
|
2008-03-03 00:24:10 +01:00
|
|
|
/**
|
2008-04-14 06:49:08 +02:00
|
|
|
* Returns a list of classes that inherit from the given class.
|
2009-02-02 00:49:53 +01:00
|
|
|
* The resulting array includes the base class passed
|
|
|
|
* through the $class parameter as the first array value.
|
2014-08-15 08:53:05 +02:00
|
|
|
*
|
2009-02-02 00:49:53 +01:00
|
|
|
* Example usage:
|
2009-03-22 23:59:14 +01:00
|
|
|
* <code>
|
2009-02-02 00:49:53 +01:00
|
|
|
* ClassInfo::subclassesFor('BaseClass');
|
|
|
|
* array(
|
2015-02-27 01:10:32 +01:00
|
|
|
* 'BaseClass' => 'BaseClass',
|
2009-02-02 00:49:53 +01:00
|
|
|
* 'ChildClass' => 'ChildClass',
|
|
|
|
* 'GrandChildClass' => 'GrandChildClass'
|
|
|
|
* )
|
2009-03-22 23:59:14 +01:00
|
|
|
* </code>
|
2014-08-15 08:53:05 +02:00
|
|
|
*
|
2008-04-14 06:49:08 +02:00
|
|
|
* @param mixed $class string of the classname or instance of the class
|
2009-02-02 00:49:53 +01:00
|
|
|
* @return array Names of all subclasses as an associative array.
|
2008-03-03 00:24:10 +01:00
|
|
|
*/
|
2011-02-21 06:16:52 +01:00
|
|
|
public static function subclassesFor($class) {
|
2015-08-07 04:20:01 +02:00
|
|
|
if(is_string($class) && !class_exists($class)) return null;
|
|
|
|
|
2015-02-27 01:10:32 +01:00
|
|
|
//normalise class case
|
|
|
|
$className = self::class_name($class);
|
2011-03-22 10:47:55 +01:00
|
|
|
$descendants = SS_ClassLoader::instance()->getManifest()->getDescendantsOf($class);
|
2015-02-27 01:10:32 +01:00
|
|
|
$result = array($className => $className);
|
2009-02-02 00:49:53 +01:00
|
|
|
|
2011-03-22 10:47:55 +01:00
|
|
|
if ($descendants) {
|
|
|
|
return $result + ArrayLib::valuekey($descendants);
|
|
|
|
} else {
|
|
|
|
return $result;
|
2009-02-02 00:49:53 +01:00
|
|
|
}
|
2007-07-19 12:40:28 +02:00
|
|
|
}
|
2011-03-22 10:47:55 +01:00
|
|
|
|
2015-02-27 01:10:32 +01:00
|
|
|
/**
|
|
|
|
* Convert a class name in any case and return it as it was defined in PHP
|
|
|
|
*
|
|
|
|
* eg: self::class_name('dataobJEct'); //returns 'DataObject'
|
|
|
|
*
|
|
|
|
* @param string|object $nameOrObject The classname or object you want to normalise
|
|
|
|
*
|
|
|
|
* @return string The normalised class name
|
|
|
|
*/
|
|
|
|
public static function class_name($nameOrObject) {
|
|
|
|
if (is_object($nameOrObject)) {
|
|
|
|
return get_class($nameOrObject);
|
2015-08-04 15:05:15 +02:00
|
|
|
} elseif (!self::exists($nameOrObject)) {
|
|
|
|
Deprecation::notice(
|
|
|
|
'4.0',
|
|
|
|
"ClassInfo::class_name() passed a class that doesn't exist. Support for this will be removed in 4.0",
|
|
|
|
Deprecation::SCOPE_GLOBAL
|
|
|
|
);
|
|
|
|
return $nameOrObject;
|
2015-02-27 01:10:32 +01:00
|
|
|
}
|
2015-08-07 04:20:01 +02:00
|
|
|
|
2015-02-27 01:10:32 +01:00
|
|
|
$reflection = new ReflectionClass($nameOrObject);
|
|
|
|
return $reflection->getName();
|
|
|
|
}
|
|
|
|
|
2008-03-03 00:24:10 +01:00
|
|
|
/**
|
2011-02-19 04:18:26 +01:00
|
|
|
* Returns the passed class name along with all its parent class names in an
|
|
|
|
* array, sorted with the root class first.
|
|
|
|
*
|
|
|
|
* @param string $class
|
|
|
|
* @param bool $tablesOnly Only return classes that have a table in the db.
|
|
|
|
* @return array
|
2008-03-03 00:24:10 +01:00
|
|
|
*/
|
2011-02-19 04:18:26 +01:00
|
|
|
public static function ancestry($class, $tablesOnly = false) {
|
2015-08-07 04:20:01 +02:00
|
|
|
if(is_string($class) && !class_exists($class)) return null;
|
|
|
|
|
2015-02-27 01:10:32 +01:00
|
|
|
$class = self::class_name($class);
|
2012-01-02 14:01:16 +01:00
|
|
|
|
2015-02-27 01:10:32 +01:00
|
|
|
$lClass = strtolower($class);
|
2012-01-02 14:01:16 +01:00
|
|
|
|
2015-02-27 01:10:32 +01:00
|
|
|
$cacheKey = $lClass . '_' . (string)$tablesOnly;
|
2012-01-02 14:01:16 +01:00
|
|
|
$parent = $class;
|
|
|
|
if(!isset(self::$_cache_ancestry[$cacheKey])) {
|
|
|
|
$ancestry = array();
|
|
|
|
do {
|
|
|
|
if (!$tablesOnly || DataObject::has_own_table($parent)) {
|
|
|
|
$ancestry[$parent] = $parent;
|
|
|
|
}
|
|
|
|
} while ($parent = get_parent_class($parent));
|
2014-08-15 08:53:05 +02:00
|
|
|
self::$_cache_ancestry[$cacheKey] = array_reverse($ancestry);
|
2007-07-19 12:40:28 +02:00
|
|
|
}
|
2011-02-19 04:18:26 +01:00
|
|
|
|
2012-01-02 14:01:16 +01:00
|
|
|
return self::$_cache_ancestry[$cacheKey];
|
2007-07-19 12:40:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2008-02-25 03:10:37 +01:00
|
|
|
* @return array A self-keyed array of class names. Note that this is only available with Silverstripe
|
|
|
|
* classes and not built-in PHP classes.
|
2007-07-19 12:40:28 +02:00
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public static function implementorsOf($interfaceName) {
|
2011-03-22 10:47:55 +01:00
|
|
|
return SS_ClassLoader::instance()->getManifest()->getImplementorsOf($interfaceName);
|
2007-07-19 12:40:28 +02:00
|
|
|
}
|
2008-08-09 04:16:46 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns true if the given class implements the given interface
|
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public static function classImplements($className, $interfaceName) {
|
2015-02-27 01:10:32 +01:00
|
|
|
return in_array($className, self::implementorsOf($interfaceName));
|
2008-08-09 04:16:46 +02:00
|
|
|
}
|
2009-08-05 06:01:22 +02:00
|
|
|
|
2008-10-17 17:18:26 +02:00
|
|
|
/**
|
|
|
|
* Get all classes contained in a file.
|
|
|
|
* @uses ManifestBuilder
|
2014-08-15 08:53:05 +02:00
|
|
|
*
|
2009-07-01 00:08:59 +02:00
|
|
|
* @todo Doesn't return additional classes that only begin
|
|
|
|
* with the filename, and have additional naming separated through underscores.
|
2014-08-15 08:53:05 +02:00
|
|
|
*
|
2008-10-17 17:18:26 +02:00
|
|
|
* @param string $filePath Path to a PHP file (absolute or relative to webroot)
|
|
|
|
* @return array
|
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public static function classes_for_file($filePath) {
|
2011-03-22 10:47:55 +01:00
|
|
|
$absFilePath = Director::getAbsFile($filePath);
|
2008-10-17 17:18:26 +02:00
|
|
|
$matchedClasses = array();
|
2011-03-22 10:47:55 +01:00
|
|
|
$manifest = SS_ClassLoader::instance()->getManifest()->getClasses();
|
|
|
|
|
|
|
|
foreach($manifest as $class => $compareFilePath) {
|
2008-10-17 17:18:26 +02:00
|
|
|
if($absFilePath == $compareFilePath) $matchedClasses[] = $class;
|
|
|
|
}
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2008-10-17 17:18:26 +02:00
|
|
|
return $matchedClasses;
|
|
|
|
}
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2009-07-01 00:08:59 +02:00
|
|
|
/**
|
|
|
|
* Returns all classes contained in a certain folder.
|
|
|
|
*
|
|
|
|
* @todo Doesn't return additional classes that only begin
|
|
|
|
* with the filename, and have additional naming separated through underscores.
|
2014-08-15 08:53:05 +02:00
|
|
|
*
|
2009-07-01 00:08:59 +02:00
|
|
|
* @param string $folderPath Relative or absolute folder path
|
|
|
|
* @return array Array of class names
|
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public static function classes_for_folder($folderPath) {
|
2011-03-22 10:47:55 +01:00
|
|
|
$absFolderPath = Director::getAbsFile($folderPath);
|
2009-07-01 00:08:59 +02:00
|
|
|
$matchedClasses = array();
|
2011-03-22 10:47:55 +01:00
|
|
|
$manifest = SS_ClassLoader::instance()->getManifest()->getClasses();
|
|
|
|
|
|
|
|
foreach($manifest as $class => $compareFilePath) {
|
2009-07-01 00:08:59 +02:00
|
|
|
if(stripos($compareFilePath, $absFolderPath) === 0) $matchedClasses[] = $class;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $matchedClasses;
|
|
|
|
}
|
2012-08-22 23:29:13 +02:00
|
|
|
|
|
|
|
private static $method_from_cache = array();
|
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public static function has_method_from($class, $method, $compclass) {
|
2015-02-27 01:10:32 +01:00
|
|
|
$lClass = strtolower($class);
|
|
|
|
$lMethod = strtolower($method);
|
|
|
|
$lCompclass = strtolower($compclass);
|
|
|
|
if (!isset(self::$method_from_cache[$lClass])) self::$method_from_cache[$lClass] = array();
|
2012-08-22 23:29:13 +02:00
|
|
|
|
2015-02-27 01:10:32 +01:00
|
|
|
if (!array_key_exists($lMethod, self::$method_from_cache[$lClass])) {
|
|
|
|
self::$method_from_cache[$lClass][$lMethod] = false;
|
2012-08-22 23:29:13 +02:00
|
|
|
|
|
|
|
$classRef = new ReflectionClass($class);
|
|
|
|
|
|
|
|
if ($classRef->hasMethod($method)) {
|
|
|
|
$methodRef = $classRef->getMethod($method);
|
2015-02-27 01:10:32 +01:00
|
|
|
self::$method_from_cache[$lClass][$lMethod] = $methodRef->getDeclaringClass()->getName();
|
2012-08-22 23:29:13 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-27 01:10:32 +01:00
|
|
|
return strtolower(self::$method_from_cache[$lClass][$lMethod]) == $lCompclass;
|
2012-08-22 23:29:13 +02:00
|
|
|
}
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2014-08-18 05:39:42 +02:00
|
|
|
|
|
|
|
/**
|
2015-02-27 01:10:32 +01:00
|
|
|
* Returns the table name in the class hierarchy which contains a given
|
2014-08-18 05:39:42 +02:00
|
|
|
* field column for a {@link DataObject}. If the field does not exist, this
|
|
|
|
* will return null.
|
|
|
|
*
|
2016-11-09 23:13:02 +01:00
|
|
|
* Note: In 3.x and below this method may return 'DataObject'. From 4.0 onwards
|
|
|
|
* null will be returned if a field is not a member of the object.
|
|
|
|
*
|
2014-08-18 05:39:42 +02:00
|
|
|
* @param string $candidateClass
|
|
|
|
* @param string $fieldName
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public static function table_for_object_field($candidateClass, $fieldName) {
|
2015-09-17 03:39:42 +02:00
|
|
|
if(!$candidateClass
|
|
|
|
|| !$fieldName
|
|
|
|
|| !class_exists($candidateClass)
|
|
|
|
|| !is_subclass_of($candidateClass, 'DataObject')
|
|
|
|
) {
|
2014-08-18 05:39:42 +02:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2015-02-27 01:10:32 +01:00
|
|
|
//normalise class name
|
|
|
|
$candidateClass = self::class_name($candidateClass);
|
|
|
|
$exists = self::exists($candidateClass);
|
2014-08-18 05:39:42 +02:00
|
|
|
|
2015-10-30 04:26:14 +01:00
|
|
|
// Short circuit for fixed fields
|
|
|
|
$fixed = DataObject::config()->fixed_fields;
|
|
|
|
if($exists && isset($fixed[$fieldName])) {
|
|
|
|
return self::baseDataClass($candidateClass);
|
|
|
|
}
|
2015-02-27 01:10:32 +01:00
|
|
|
|
2015-10-30 04:26:14 +01:00
|
|
|
// Find regular field
|
|
|
|
while($candidateClass && $candidateClass != 'DataObject' && $exists) {
|
|
|
|
if( DataObject::has_own_table($candidateClass)
|
|
|
|
&& DataObject::has_own_table_database_field($candidateClass, $fieldName)
|
|
|
|
) {
|
|
|
|
break;
|
2014-08-18 05:39:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$candidateClass = get_parent_class($candidateClass);
|
2015-02-27 01:10:32 +01:00
|
|
|
$exists = $candidateClass && self::exists($candidateClass);
|
2014-08-18 05:39:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if(!$candidateClass || !$exists) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $candidateClass;
|
|
|
|
}
|
2007-07-19 12:40:28 +02:00
|
|
|
}
|
2012-02-12 21:22:11 +01:00
|
|
|
|