mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 12:05:37 +00:00
Merge pull request #6930 from dhensby/pulls/4/db-schema-indexes
Cleaning up DB index definition
This commit is contained in:
commit
9da2b5c4ab
@ -29,14 +29,14 @@ descriptor. There are several supported notations:
|
|||||||
|
|
||||||
class MyObject extends DataObject {
|
class MyObject extends DataObject {
|
||||||
|
|
||||||
private static $indexes = array(
|
private static $indexes = [
|
||||||
'<column-name>' => true,
|
'<column-name>' => true,
|
||||||
'<index-name>' => 'unique("<column-name>")'
|
'<index-name>' => [
|
||||||
'<index-name>' => array(
|
|
||||||
'type' => '<type>',
|
'type' => '<type>',
|
||||||
'value' => '"<column-name>"'
|
'columns' => ['<column-name>', '<other-column-name>'],
|
||||||
),
|
],
|
||||||
);
|
'<index-name>' => ['<column-name>', '<other-column-name>'],
|
||||||
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
The `<column-name>` is used to put a standard non-unique index on the column specified. For complex or large tables
|
The `<column-name>` is used to put a standard non-unique index on the column specified. For complex or large tables
|
||||||
@ -57,17 +57,14 @@ support the following:
|
|||||||
|
|
||||||
class MyTestObject extends DataObject {
|
class MyTestObject extends DataObject {
|
||||||
|
|
||||||
private static $db = array(
|
private static $db = [
|
||||||
'MyField' => 'Varchar',
|
'MyField' => 'Varchar',
|
||||||
'MyOtherField' => 'Varchar',
|
'MyOtherField' => 'Varchar',
|
||||||
);
|
];
|
||||||
|
|
||||||
private static $indexes = array(
|
private static $indexes = [
|
||||||
'MyIndexName' => array(
|
'MyIndexName' => ['MyField', 'MyOtherField'],
|
||||||
'type' => 'index',
|
];
|
||||||
'value' => '"MyField","MyOtherField"'
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
## Complex/Composite Indexes
|
## Complex/Composite Indexes
|
||||||
|
@ -57,8 +57,7 @@ Example DataObject:
|
|||||||
private static $indexes = array(
|
private static $indexes = array(
|
||||||
'SearchFields' => array(
|
'SearchFields' => array(
|
||||||
'type' => 'fulltext',
|
'type' => 'fulltext',
|
||||||
'name' => 'SearchFields',
|
'columns' => ['Title', 'Content'],
|
||||||
'value' => '"Title", "Content"',
|
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -405,8 +405,8 @@ abstract class DBSchemaManager
|
|||||||
|
|
||||||
// Create custom indexes
|
// Create custom indexes
|
||||||
if ($indexSchema) {
|
if ($indexSchema) {
|
||||||
foreach ($indexSchema as $indexName => $indexDetails) {
|
foreach ($indexSchema as $indexName => $indexSpec) {
|
||||||
$this->requireIndex($table, $indexName, $indexDetails);
|
$this->requireIndex($table, $indexName, $indexSpec);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -449,7 +449,6 @@ abstract class DBSchemaManager
|
|||||||
$newTable = !isset($this->tableList[strtolower($table)]);
|
$newTable = !isset($this->tableList[strtolower($table)]);
|
||||||
|
|
||||||
// Force spec into standard array format
|
// Force spec into standard array format
|
||||||
$spec = $this->parseIndexSpec($index, $spec);
|
|
||||||
$specString = $this->convertIndexSpec($spec);
|
$specString = $this->convertIndexSpec($spec);
|
||||||
|
|
||||||
// Check existing index
|
// Check existing index
|
||||||
@ -544,40 +543,6 @@ abstract class DBSchemaManager
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Converts an array or string index spec into a universally useful array
|
|
||||||
*
|
|
||||||
* @see convertIndexSpec() for approximate inverse
|
|
||||||
* @param string $name Index name
|
|
||||||
* @param string|array $spec
|
|
||||||
* @return array The resulting spec array with the required fields name, type, and value
|
|
||||||
*/
|
|
||||||
protected function parseIndexSpec($name, $spec)
|
|
||||||
{
|
|
||||||
// Support $indexes = array('ColumnName' => true) for quick indexes
|
|
||||||
if ($spec === true) {
|
|
||||||
return array(
|
|
||||||
'name' => $name,
|
|
||||||
'value' => $this->quoteColumnSpecString($name),
|
|
||||||
'type' => 'index'
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Do minimal cleanup on any already parsed spec
|
|
||||||
if (is_array($spec)) {
|
|
||||||
$spec['value'] = $this->quoteColumnSpecString($spec['value']);
|
|
||||||
$spec['type'] = empty($spec['type']) ? 'index' : trim($spec['type']);
|
|
||||||
return $spec;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Nicely formatted spec!
|
|
||||||
return array(
|
|
||||||
'name' => $name,
|
|
||||||
'value' => $this->quoteColumnSpecString($spec),
|
|
||||||
'type' => $this->determineIndexType($spec)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This takes the index spec which has been provided by a class (ie static $indexes = blah blah)
|
* This takes the index spec which has been provided by a class (ie static $indexes = blah blah)
|
||||||
* and turns it into a proper string.
|
* and turns it into a proper string.
|
||||||
@ -593,12 +558,12 @@ abstract class DBSchemaManager
|
|||||||
protected function convertIndexSpec($indexSpec)
|
protected function convertIndexSpec($indexSpec)
|
||||||
{
|
{
|
||||||
// Return already converted spec
|
// Return already converted spec
|
||||||
if (!is_array($indexSpec)) {
|
if (!is_array($indexSpec) || !array_key_exists('type', $indexSpec) || !array_key_exists('columns', $indexSpec) || !is_array($indexSpec['columns'])) {
|
||||||
return $indexSpec;
|
throw new \InvalidArgumentException(sprintf('argument to convertIndexSpec must be correct indexSpec, %s given', var_export($indexSpec, true)));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Combine elements into standard string format
|
// Combine elements into standard string format
|
||||||
return "{$indexSpec['type']} ({$indexSpec['value']})";
|
return sprintf('%s (%s)', $indexSpec['type'], $this->implodeColumnList($indexSpec['columns']));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -39,6 +39,10 @@ class MySQLSchemaManager extends DBSchemaManager
|
|||||||
}
|
}
|
||||||
if ($indexes) {
|
if ($indexes) {
|
||||||
foreach ($indexes as $k => $v) {
|
foreach ($indexes as $k => $v) {
|
||||||
|
// force MyISAM if we have a fulltext index
|
||||||
|
if ($v['type'] === 'fulltext') {
|
||||||
|
$addOptions = 'ENGINE=MyISAM';
|
||||||
|
}
|
||||||
$indexSchemas .= $this->getIndexSqlDefinition($k, $v) . ",\n";
|
$indexSchemas .= $this->getIndexSqlDefinition($k, $v) . ",\n";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -301,19 +305,24 @@ class MySQLSchemaManager extends DBSchemaManager
|
|||||||
*/
|
*/
|
||||||
protected function getIndexSqlDefinition($indexName, $indexSpec)
|
protected function getIndexSqlDefinition($indexName, $indexSpec)
|
||||||
{
|
{
|
||||||
$indexSpec = $this->parseIndexSpec($indexName, $indexSpec);
|
|
||||||
if ($indexSpec['type'] == 'using') {
|
if ($indexSpec['type'] == 'using') {
|
||||||
return "index \"$indexName\" using ({$indexSpec['value']})";
|
return sprintf('index "%s" using (%s)', $indexName, $this->implodeColumnList($indexSpec['columns']));
|
||||||
} else {
|
} else {
|
||||||
return "{$indexSpec['type']} \"$indexName\" ({$indexSpec['value']})";
|
return sprintf('%s "%s" (%s)', $indexSpec['type'], $indexName, $this->implodeColumnList($indexSpec['columns']));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function alterIndex($tableName, $indexName, $indexSpec)
|
public function alterIndex($tableName, $indexName, $indexSpec)
|
||||||
{
|
{
|
||||||
$indexSpec = $this->parseIndexSpec($indexName, $indexSpec);
|
$indexSpec = $this->parseIndexSpec($indexName, $indexSpec);
|
||||||
$this->query("ALTER TABLE \"$tableName\" DROP INDEX \"$indexName\"");
|
$this->query(sprintf('ALTER TABLE "%s" DROP INDEX "%s"', $tableName, $indexName));
|
||||||
$this->query("ALTER TABLE \"$tableName\" ADD {$indexSpec['type']} \"$indexName\" {$indexSpec['value']}");
|
$this->query(sprintf(
|
||||||
|
'ALTER TABLE "%s" ADD %s "%s" %s',
|
||||||
|
$tableName,
|
||||||
|
$indexSpec['type'],
|
||||||
|
$indexName,
|
||||||
|
$this->implodeColumnList($indexSpec['columns'])
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function indexKey($table, $index, $spec)
|
protected function indexKey($table, $index, $spec)
|
||||||
@ -347,11 +356,11 @@ class MySQLSchemaManager extends DBSchemaManager
|
|||||||
if ($groupedIndexes) {
|
if ($groupedIndexes) {
|
||||||
foreach ($groupedIndexes as $index => $details) {
|
foreach ($groupedIndexes as $index => $details) {
|
||||||
ksort($details['fields']);
|
ksort($details['fields']);
|
||||||
$indexList[$index] = $this->parseIndexSpec($index, array(
|
$indexList[$index] = array(
|
||||||
'name' => $index,
|
'name' => $index,
|
||||||
'value' => $this->implodeColumnList($details['fields']),
|
'columns' => $details['fields'],
|
||||||
'type' => $details['type']
|
'type' => $details['type'],
|
||||||
));
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3052,38 +3052,6 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
|
|
||||||
//-------------------------------------------------------------------------------------------//
|
//-------------------------------------------------------------------------------------------//
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the database indexes on this table.
|
|
||||||
* This array is indexed by the name of the field with the index, and
|
|
||||||
* the value is the type of index.
|
|
||||||
*/
|
|
||||||
public function databaseIndexes()
|
|
||||||
{
|
|
||||||
$has_one = $this->uninherited('has_one');
|
|
||||||
$classIndexes = $this->uninherited('indexes');
|
|
||||||
//$fileIndexes = $this->uninherited('fileIndexes', true);
|
|
||||||
|
|
||||||
$indexes = array();
|
|
||||||
|
|
||||||
if ($has_one) {
|
|
||||||
foreach ($has_one as $relationshipName => $fieldType) {
|
|
||||||
$indexes[$relationshipName . 'ID'] = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($classIndexes) {
|
|
||||||
foreach ($classIndexes as $indexName => $indexType) {
|
|
||||||
$indexes[$indexName] = $indexType;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (get_parent_class($this) == self::class) {
|
|
||||||
$indexes['ClassName'] = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $indexes;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check the database schema and update it as necessary.
|
* Check the database schema and update it as necessary.
|
||||||
*
|
*
|
||||||
@ -3093,12 +3061,11 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
{
|
{
|
||||||
// Only build the table if we've actually got fields
|
// Only build the table if we've actually got fields
|
||||||
$schema = static::getSchema();
|
$schema = static::getSchema();
|
||||||
$fields = $schema->databaseFields(static::class, false);
|
|
||||||
$table = $schema->tableName(static::class);
|
$table = $schema->tableName(static::class);
|
||||||
|
$fields = $schema->databaseFields(static::class, false);
|
||||||
|
$indexes = $schema->databaseIndexes(static::class, false);
|
||||||
$extensions = self::database_extensions(static::class);
|
$extensions = self::database_extensions(static::class);
|
||||||
|
|
||||||
$indexes = $this->databaseIndexes();
|
|
||||||
|
|
||||||
if (empty($table)) {
|
if (empty($table)) {
|
||||||
throw new LogicException(
|
throw new LogicException(
|
||||||
"Class " . static::class . " not loaded by manifest, or no database table configured"
|
"Class " . static::class . " not loaded by manifest, or no database table configured"
|
||||||
@ -3144,10 +3111,18 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Build index list
|
// Build index list
|
||||||
$manymanyIndexes = array(
|
$manymanyIndexes = [
|
||||||
$parentField => true,
|
$parentField => [
|
||||||
$childField => true,
|
'type' => 'index',
|
||||||
);
|
'name' => $parentField,
|
||||||
|
'columns' => [$parentField],
|
||||||
|
],
|
||||||
|
$childField => [
|
||||||
|
'type' => 'index',
|
||||||
|
'name' =>$childField,
|
||||||
|
'columns' => [$childField],
|
||||||
|
],
|
||||||
|
];
|
||||||
DB::require_table($tableOrClass, $manymanyFields, $manymanyIndexes, true, null, $extensions);
|
DB::require_table($tableOrClass, $manymanyFields, $manymanyIndexes, true, null, $extensions);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -36,6 +36,13 @@ class DataObjectSchema
|
|||||||
*/
|
*/
|
||||||
protected $databaseFields = [];
|
protected $databaseFields = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cache of database indexes
|
||||||
|
*
|
||||||
|
* @var array
|
||||||
|
*/
|
||||||
|
protected $databaseIndexes = [];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Cache of composite database field
|
* Cache of composite database field
|
||||||
*
|
*
|
||||||
@ -57,6 +64,7 @@ class DataObjectSchema
|
|||||||
{
|
{
|
||||||
$this->tableNames = [];
|
$this->tableNames = [];
|
||||||
$this->databaseFields = [];
|
$this->databaseFields = [];
|
||||||
|
$this->databaseIndexes = [];
|
||||||
$this->compositeFields = [];
|
$this->compositeFields = [];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -333,6 +341,26 @@ class DataObjectSchema
|
|||||||
return isset($fields[$field]) ? $fields[$field] : null;
|
return isset($fields[$field]) ? $fields[$field] : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $class
|
||||||
|
* @param bool $aggregated
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function databaseIndexes($class, $aggregated = true)
|
||||||
|
{
|
||||||
|
$class = ClassInfo::class_name($class);
|
||||||
|
if ($class === DataObject::class) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
$this->cacheDatabaseIndexes($class);
|
||||||
|
$indexes = $this->databaseIndexes[$class];
|
||||||
|
if (!$aggregated) {
|
||||||
|
return $indexes;
|
||||||
|
}
|
||||||
|
return array_merge($indexes, $this->databaseIndexes(get_parent_class($class)));
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if the given class has a table
|
* Check if the given class has a table
|
||||||
*
|
*
|
||||||
@ -456,6 +484,70 @@ class DataObjectSchema
|
|||||||
$this->compositeFields[$class] = $compositeFields;
|
$this->compositeFields[$class] = $compositeFields;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cache all indexes for the given class.
|
||||||
|
* Will do nothing if already cached
|
||||||
|
*
|
||||||
|
* @param $class
|
||||||
|
*/
|
||||||
|
protected function cacheDatabaseIndexes($class)
|
||||||
|
{
|
||||||
|
if (array_key_exists($class, $this->databaseIndexes)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
$indexes = [];
|
||||||
|
|
||||||
|
// look for indexable field types
|
||||||
|
foreach ($this->databaseFields($class, false) as $field => $type) {
|
||||||
|
if ($type === 'ForeignKey' || $type === 'DBClassName') {
|
||||||
|
$indexes[$field] = [
|
||||||
|
'type' => 'index',
|
||||||
|
'columns' => [$field],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// look for custom indexes declared on the class
|
||||||
|
$classIndexes = Config::inst()->get($class, 'indexes', Config::UNINHERITED) ?: [];
|
||||||
|
foreach ($classIndexes as $indexName => $indexSpec) {
|
||||||
|
if (array_key_exists($indexName, $indexes)) {
|
||||||
|
throw new InvalidArgumentException(sprintf(
|
||||||
|
'Index named "%s" already exists on class %s',
|
||||||
|
$indexName,
|
||||||
|
$class
|
||||||
|
));
|
||||||
|
}
|
||||||
|
if (is_array($indexSpec)) {
|
||||||
|
if (!ArrayLib::is_associative($indexSpec)) {
|
||||||
|
$indexSpec = [
|
||||||
|
'columns' => $indexSpec,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
if (!isset($indexSpec['type'])) {
|
||||||
|
$indexSpec['type'] = 'index';
|
||||||
|
}
|
||||||
|
if (!isset($indexSpec['columns'])) {
|
||||||
|
$indexSpec['columns'] = [$indexName];
|
||||||
|
} elseif (!is_array($indexSpec['columns'])) {
|
||||||
|
throw new InvalidArgumentException(sprintf(
|
||||||
|
'Index %s on %s is not valid. columns should be an array %s given',
|
||||||
|
var_export($indexName, true),
|
||||||
|
var_export($class, true),
|
||||||
|
var_export($indexSpec['columns'], true)
|
||||||
|
));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$indexSpec = [
|
||||||
|
'type' => 'index',
|
||||||
|
'columns' => [$indexName],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
$indexes[$indexName] = $indexSpec;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->databaseIndexes[$class] = $indexes;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the table name in the class hierarchy which contains a given
|
* Returns the table name in the class hierarchy which contains a given
|
||||||
* field column for a {@link DataObject}. If the field does not exist, this
|
* field column for a {@link DataObject}. If the field does not exist, this
|
||||||
|
@ -2,9 +2,9 @@
|
|||||||
|
|
||||||
namespace SilverStripe\ORM\Filters;
|
namespace SilverStripe\ORM\Filters;
|
||||||
|
|
||||||
|
use SilverStripe\Core\Convert;
|
||||||
use SilverStripe\ORM\DataQuery;
|
use SilverStripe\ORM\DataQuery;
|
||||||
use SilverStripe\ORM\DataObject;
|
use SilverStripe\ORM\DataObject;
|
||||||
use SilverStripe\Core\Config\Config;
|
|
||||||
use Exception;
|
use Exception;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -20,9 +20,11 @@ use Exception;
|
|||||||
* database table, using the {$indexes} hash in your DataObject subclass:
|
* database table, using the {$indexes} hash in your DataObject subclass:
|
||||||
*
|
*
|
||||||
* <code>
|
* <code>
|
||||||
* private static $indexes = array(
|
* private static $indexes = [
|
||||||
* 'SearchFields' => 'fulltext(Name, Title, Description)'
|
* 'SearchFields' => [
|
||||||
* );
|
* 'type' => 'fulltext',
|
||||||
|
* 'columns' => ['Name', 'Title', 'Description'],
|
||||||
|
* ];
|
||||||
* </code>
|
* </code>
|
||||||
*
|
*
|
||||||
* @todo Add support for databases besides MySQL
|
* @todo Add support for databases besides MySQL
|
||||||
@ -63,28 +65,22 @@ class FulltextFilter extends SearchFilter
|
|||||||
*/
|
*/
|
||||||
public function getDbName()
|
public function getDbName()
|
||||||
{
|
{
|
||||||
$indexes = Config::inst()->get($this->model, "indexes");
|
$indexes = DataObject::getSchema()->databaseIndexes($this->model);
|
||||||
if (is_array($indexes) && array_key_exists($this->getName(), $indexes)) {
|
if (array_key_exists($this->getName(), $indexes)) {
|
||||||
$index = $indexes[$this->getName()];
|
$index = $indexes[$this->getName()];
|
||||||
if (is_array($index) && array_key_exists("value", $index)) {
|
|
||||||
return $this->prepareColumns($index['value']);
|
|
||||||
} else {
|
} else {
|
||||||
// Parse a fulltext string (eg. fulltext ("ColumnA", "ColumnB")) to figure out which columns
|
return parent::getDbName();
|
||||||
// we need to search.
|
}
|
||||||
if (preg_match('/^fulltext\s+\((.+)\)$/i', $index, $matches)) {
|
if (is_array($index) && array_key_exists('columns', $index)) {
|
||||||
return $this->prepareColumns($matches[1]);
|
return $this->prepareColumns($index['columns']);
|
||||||
} else {
|
} else {
|
||||||
throw new Exception(sprintf(
|
throw new Exception(sprintf(
|
||||||
"Invalid fulltext index format for '%s' on '%s'",
|
"Invalid fulltext index format for '%s' on '%s'",
|
||||||
$this->getName(),
|
var_export($this->getName(), true),
|
||||||
$this->model
|
var_export($this->model, true)
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return parent::getDbName();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adds table identifier to the every column.
|
* Adds table identifier to the every column.
|
||||||
@ -95,11 +91,10 @@ class FulltextFilter extends SearchFilter
|
|||||||
*/
|
*/
|
||||||
protected function prepareColumns($columns)
|
protected function prepareColumns($columns)
|
||||||
{
|
{
|
||||||
$cols = preg_split('/"?\s*,\s*"?/', trim($columns, '(") '));
|
$table = DataObject::getSchema()->tableForField($this->model, current($columns));
|
||||||
$table = DataObject::getSchema()->tableForField($this->model, current($cols));
|
$columns = array_map(function ($column) use ($table) {
|
||||||
$cols = array_map(function ($col) use ($table) {
|
return Convert::symbol2sql("$table.$column");
|
||||||
return sprintf('"%s"."%s"', $table, $col);
|
}, $columns);
|
||||||
}, $cols);
|
return implode(',', $columns);
|
||||||
return implode(',', $cols);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -56,8 +56,8 @@ class FulltextSearchable extends DataExtension
|
|||||||
public static function enable($searchableClasses = [SiteTree::class, File::class])
|
public static function enable($searchableClasses = [SiteTree::class, File::class])
|
||||||
{
|
{
|
||||||
$defaultColumns = array(
|
$defaultColumns = array(
|
||||||
SiteTree::class => '"Title","MenuTitle","Content","MetaDescription"',
|
SiteTree::class => ['Title','MenuTitle','Content','MetaDescription'],
|
||||||
File::class => '"Name","Title"'
|
File::class => ['Name','Title'],
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!is_array($searchableClasses)) {
|
if (!is_array($searchableClasses)) {
|
||||||
@ -69,12 +69,7 @@ class FulltextSearchable extends DataExtension
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (isset($defaultColumns[$class])) {
|
if (isset($defaultColumns[$class])) {
|
||||||
Config::modify()->set(
|
$class::add_extension(sprintf('%s(%s)', static::class, "'" . implode("','", $defaultColumns[$class]) . "''"));
|
||||||
$class,
|
|
||||||
'create_table_options',
|
|
||||||
array(MySQLSchemaManager::ID => 'ENGINE=MyISAM')
|
|
||||||
);
|
|
||||||
$class::add_extension(__CLASS__."('{$defaultColumns[$class]}')");
|
|
||||||
} else {
|
} else {
|
||||||
throw new Exception(
|
throw new Exception(
|
||||||
"FulltextSearchable::enable() I don't know the default search columns for class '$class'"
|
"FulltextSearchable::enable() I don't know the default search columns for class '$class'"
|
||||||
@ -94,9 +89,12 @@ class FulltextSearchable extends DataExtension
|
|||||||
public function __construct($searchFields = array())
|
public function __construct($searchFields = array())
|
||||||
{
|
{
|
||||||
if (is_array($searchFields)) {
|
if (is_array($searchFields)) {
|
||||||
$this->searchFields = '"'.implode('","', $searchFields).'"';
|
|
||||||
} else {
|
|
||||||
$this->searchFields = $searchFields;
|
$this->searchFields = $searchFields;
|
||||||
|
} else {
|
||||||
|
$this->searchFields = explode(',', $searchFields);
|
||||||
|
foreach ($this->searchFields as &$field) {
|
||||||
|
$field = trim($field);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -107,7 +105,7 @@ class FulltextSearchable extends DataExtension
|
|||||||
'SearchFields' => array(
|
'SearchFields' => array(
|
||||||
'type' => 'fulltext',
|
'type' => 'fulltext',
|
||||||
'name' => 'SearchFields',
|
'name' => 'SearchFields',
|
||||||
'value' => $args[0]
|
'columns' => $args,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
@ -175,11 +175,11 @@ class DataObjectSchemaGenerationTest extends SapphireTest
|
|||||||
// Update the SearchFields index here
|
// Update the SearchFields index here
|
||||||
TestIndexObject::config()->update(
|
TestIndexObject::config()->update(
|
||||||
'indexes',
|
'indexes',
|
||||||
array(
|
[
|
||||||
'SearchFields' => array(
|
'SearchFields' => [
|
||||||
'value' => 'Title'
|
'columns' => ['Title'],
|
||||||
)
|
],
|
||||||
)
|
]
|
||||||
);
|
);
|
||||||
|
|
||||||
// Verify that the above index change triggered a schema update
|
// Verify that the above index change triggered a schema update
|
||||||
|
@ -7,29 +7,35 @@ use SilverStripe\Dev\TestOnly;
|
|||||||
class TestIndexObject extends TestObject implements TestOnly
|
class TestIndexObject extends TestObject implements TestOnly
|
||||||
{
|
{
|
||||||
private static $table_name = 'DataObjectSchemaGenerationTest_IndexDO';
|
private static $table_name = 'DataObjectSchemaGenerationTest_IndexDO';
|
||||||
private static $db = array(
|
private static $db = [
|
||||||
'Title' => 'Varchar(255)',
|
'Title' => 'Varchar(255)',
|
||||||
'Content' => 'Text'
|
'Content' => 'Text',
|
||||||
);
|
];
|
||||||
|
|
||||||
private static $indexes = array(
|
private static $indexes = [
|
||||||
'NameIndex' => 'unique ("Title")',
|
'NameIndex' => [
|
||||||
'SearchFields' => array(
|
'type' => 'unique',
|
||||||
|
'columns' => ['Title'],
|
||||||
|
],
|
||||||
|
'SearchFields' => [
|
||||||
'type' => 'fulltext',
|
'type' => 'fulltext',
|
||||||
'name' => 'SearchFields',
|
'name' => 'SearchFields',
|
||||||
'value' => '"Title","Content"'
|
'columns' => ['Title', 'Content'],
|
||||||
)
|
],
|
||||||
);
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @config
|
* @config
|
||||||
*/
|
*/
|
||||||
private static $indexes_alt = array(
|
private static $indexes_alt = [
|
||||||
'NameIndex' => array(
|
'NameIndex' => [
|
||||||
'type' => 'unique',
|
'type' => 'unique',
|
||||||
'name' => 'NameIndex',
|
'name' => 'NameIndex',
|
||||||
'value' => '"Title"'
|
'columns' => ['Title'],
|
||||||
),
|
],
|
||||||
'SearchFields' => 'fulltext ("Title","Content")'
|
'SearchFields' => [
|
||||||
);
|
'type' => 'fulltext',
|
||||||
|
'columns' => ['Title', 'Content'],
|
||||||
|
],
|
||||||
|
];
|
||||||
}
|
}
|
||||||
|
@ -6,6 +6,7 @@ use SilverStripe\Core\ClassInfo;
|
|||||||
use SilverStripe\ORM\DataObject;
|
use SilverStripe\ORM\DataObject;
|
||||||
use SilverStripe\Dev\SapphireTest;
|
use SilverStripe\Dev\SapphireTest;
|
||||||
use SilverStripe\ORM\DataObjectSchema;
|
use SilverStripe\ORM\DataObjectSchema;
|
||||||
|
use SilverStripe\ORM\Tests\DataObjectSchemaTest\AllIndexes;
|
||||||
use SilverStripe\ORM\Tests\DataObjectSchemaTest\BaseClass;
|
use SilverStripe\ORM\Tests\DataObjectSchemaTest\BaseClass;
|
||||||
use SilverStripe\ORM\Tests\DataObjectSchemaTest\BaseDataClass;
|
use SilverStripe\ORM\Tests\DataObjectSchemaTest\BaseDataClass;
|
||||||
use SilverStripe\ORM\Tests\DataObjectSchemaTest\ChildClass;
|
use SilverStripe\ORM\Tests\DataObjectSchemaTest\ChildClass;
|
||||||
@ -23,7 +24,7 @@ use SilverStripe\ORM\Tests\DataObjectSchemaTest\WithRelation;
|
|||||||
*/
|
*/
|
||||||
class DataObjectSchemaTest extends SapphireTest
|
class DataObjectSchemaTest extends SapphireTest
|
||||||
{
|
{
|
||||||
protected static $extra_dataobjects = array(
|
protected static $extra_dataobjects = [
|
||||||
// Classes in base namespace
|
// Classes in base namespace
|
||||||
BaseClass::class,
|
BaseClass::class,
|
||||||
BaseDataClass::class,
|
BaseDataClass::class,
|
||||||
@ -33,8 +34,9 @@ class DataObjectSchemaTest extends SapphireTest
|
|||||||
NoFields::class,
|
NoFields::class,
|
||||||
WithCustomTable::class,
|
WithCustomTable::class,
|
||||||
WithRelation::class,
|
WithRelation::class,
|
||||||
DefaultTableName::class
|
DefaultTableName::class,
|
||||||
);
|
AllIndexes::class,
|
||||||
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test table name generation
|
* Test table name generation
|
||||||
@ -234,4 +236,39 @@ class DataObjectSchemaTest extends SapphireTest
|
|||||||
$this->setExpectedException('InvalidArgumentException');
|
$this->setExpectedException('InvalidArgumentException');
|
||||||
$schema->baseDataClass(DataObject::class);
|
$schema->baseDataClass(DataObject::class);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testDatabaseIndexes()
|
||||||
|
{
|
||||||
|
$indexes = DataObject::getSchema()->databaseIndexes(AllIndexes::class);
|
||||||
|
$this->assertCount(5, $indexes);
|
||||||
|
$this->assertArrayHasKey('ClassName', $indexes);
|
||||||
|
$this->assertEquals([
|
||||||
|
'type' => 'index',
|
||||||
|
'columns' => ['ClassName'],
|
||||||
|
], $indexes['ClassName']);
|
||||||
|
|
||||||
|
$this->assertArrayHasKey('Content', $indexes);
|
||||||
|
$this->assertEquals([
|
||||||
|
'type' => 'index',
|
||||||
|
'columns' => ['Content'],
|
||||||
|
], $indexes['Content']);
|
||||||
|
|
||||||
|
$this->assertArrayHasKey('IndexCols', $indexes);
|
||||||
|
$this->assertEquals([
|
||||||
|
'type' => 'index',
|
||||||
|
'columns' => ['Title', 'Content'],
|
||||||
|
], $indexes['IndexCols']);
|
||||||
|
|
||||||
|
$this->assertArrayHasKey('IndexUnique', $indexes);
|
||||||
|
$this->assertEquals([
|
||||||
|
'type' => 'unique',
|
||||||
|
'columns' => ['Number'],
|
||||||
|
], $indexes['IndexUnique']);
|
||||||
|
|
||||||
|
$this->assertArrayHasKey('IndexNormal', $indexes);
|
||||||
|
$this->assertEquals([
|
||||||
|
'type' => 'index',
|
||||||
|
'columns' => ['Title'],
|
||||||
|
], $indexes['IndexNormal']);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
28
tests/php/ORM/DataObjectSchemaTest/AllIndexes.php
Normal file
28
tests/php/ORM/DataObjectSchemaTest/AllIndexes.php
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
<?php
|
||||||
|
namespace SilverStripe\ORM\Tests\DataObjectSchemaTest;
|
||||||
|
|
||||||
|
use SilverStripe\Dev\TestOnly;
|
||||||
|
use SilverStripe\ORM\DataObject;
|
||||||
|
|
||||||
|
class AllIndexes extends DataObject implements TestOnly
|
||||||
|
{
|
||||||
|
private static $table_name = 'DataObjectSchemaTest_AllIndexes';
|
||||||
|
|
||||||
|
private static $db = [
|
||||||
|
'Title' => 'Varchar',
|
||||||
|
'Content' => 'Varchar',
|
||||||
|
'Number' => 'Int',
|
||||||
|
];
|
||||||
|
|
||||||
|
private static $indexes = [
|
||||||
|
'Content' => true,
|
||||||
|
'IndexCols' => ['Title', 'Content'],
|
||||||
|
'IndexUnique' => [
|
||||||
|
'type' => 'unique',
|
||||||
|
'columns' => ['Number'],
|
||||||
|
],
|
||||||
|
'IndexNormal' => [
|
||||||
|
'columns' => ['Title'],
|
||||||
|
],
|
||||||
|
];
|
||||||
|
}
|
@ -11,22 +11,28 @@ class TestObject extends DataObject implements TestOnly
|
|||||||
|
|
||||||
private static $table_name = 'FulltextFilterTest_DataObject';
|
private static $table_name = 'FulltextFilterTest_DataObject';
|
||||||
|
|
||||||
private static $db = array(
|
private static $db = [
|
||||||
"ColumnA" => "Varchar(255)",
|
'ColumnA' => 'Varchar(255)',
|
||||||
"ColumnB" => "HTMLText",
|
'ColumnB' => 'HTMLText',
|
||||||
"ColumnC" => "Varchar(255)",
|
'ColumnC' => 'Varchar(255)',
|
||||||
"ColumnD" => "HTMLText",
|
'ColumnD' => 'HTMLText',
|
||||||
"ColumnE" => 'Varchar(255)'
|
'ColumnE' => 'Varchar(255)',
|
||||||
);
|
];
|
||||||
|
|
||||||
private static $indexes = array(
|
private static $indexes = array(
|
||||||
'SearchFields' => array(
|
'SearchFields' => [
|
||||||
'type' => 'fulltext',
|
'type' => 'fulltext',
|
||||||
'name' => 'SearchFields',
|
'name' => 'SearchFields',
|
||||||
'value' => '"ColumnA", "ColumnB"',
|
'columns' => ['ColumnA', 'ColumnB'],
|
||||||
),
|
],
|
||||||
'OtherSearchFields' => 'fulltext ("ColumnC", "ColumnD")',
|
'OtherSearchFields' => [
|
||||||
'SingleIndex' => 'fulltext ("ColumnE")'
|
'type' => 'fulltext',
|
||||||
|
'columns' => ['ColumnC', 'ColumnD'],
|
||||||
|
],
|
||||||
|
'SingleIndex' => [
|
||||||
|
'type' => 'fulltext',
|
||||||
|
'columns' => ['ColumnE'],
|
||||||
|
],
|
||||||
);
|
);
|
||||||
|
|
||||||
private static $create_table_options = array(
|
private static $create_table_options = array(
|
||||||
|
@ -6,6 +6,7 @@ use SilverStripe\Assets\File;
|
|||||||
use SilverStripe\ORM\Connect\MySQLSchemaManager;
|
use SilverStripe\ORM\Connect\MySQLSchemaManager;
|
||||||
use SilverStripe\Core\Config\Config;
|
use SilverStripe\Core\Config\Config;
|
||||||
use SilverStripe\Dev\SapphireTest;
|
use SilverStripe\Dev\SapphireTest;
|
||||||
|
use SilverStripe\ORM\DataObject;
|
||||||
use SilverStripe\ORM\Search\FulltextSearchable;
|
use SilverStripe\ORM\Search\FulltextSearchable;
|
||||||
|
|
||||||
class FulltextSearchableTest extends SapphireTest
|
class FulltextSearchableTest extends SapphireTest
|
||||||
@ -49,4 +50,13 @@ class FulltextSearchableTest extends SapphireTest
|
|||||||
File::remove_extension(FulltextSearchable::class);
|
File::remove_extension(FulltextSearchable::class);
|
||||||
$this->assertFalse(File::has_extension(FulltextSearchable::class));
|
$this->assertFalse(File::has_extension(FulltextSearchable::class));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testIndexesAdded()
|
||||||
|
{
|
||||||
|
$indexes = DataObject::getSchema()->databaseIndexes(File::class);
|
||||||
|
$this->assertArrayHasKey('SearchFields', $indexes);
|
||||||
|
$this->assertCount(2, $indexes['SearchFields']['columns']);
|
||||||
|
$this->assertContains('Name', $indexes['SearchFields']['columns']);
|
||||||
|
$this->assertContains('Title', $indexes['SearchFields']['columns']);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user