2012-10-10 22:49:47 +04:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* Copyright (c) 2012 Bart Visscher <bartv@thisnet.nl>
|
|
|
|
* This file is licensed under the Affero General Public License version 3 or
|
|
|
|
* later.
|
|
|
|
* See the COPYING-README file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
class OC_DB_MDB2SchemaReader {
|
|
|
|
static protected $DBNAME;
|
|
|
|
static protected $DBTABLEPREFIX;
|
2013-03-17 20:00:07 +04:00
|
|
|
static protected $platform;
|
2012-10-10 22:49:47 +04:00
|
|
|
|
2013-03-19 21:52:54 +04:00
|
|
|
/**
|
|
|
|
* @param $file
|
|
|
|
* @param $platform
|
|
|
|
* @return \Doctrine\DBAL\Schema\Schema
|
|
|
|
* @throws DomainException
|
|
|
|
*/
|
2013-03-17 20:00:07 +04:00
|
|
|
public static function loadSchemaFromFile($file, $platform) {
|
2012-10-10 22:49:47 +04:00
|
|
|
self::$DBNAME = OC_Config::getValue( "dbname", "owncloud" );
|
|
|
|
self::$DBTABLEPREFIX = OC_Config::getValue( "dbtableprefix", "oc_" );
|
2013-03-17 20:00:07 +04:00
|
|
|
self::$platform = $platform;
|
2012-10-10 22:49:47 +04:00
|
|
|
$schema = new \Doctrine\DBAL\Schema\Schema();
|
|
|
|
$xml = simplexml_load_file($file);
|
|
|
|
foreach($xml->children() as $child) {
|
|
|
|
switch($child->getName()) {
|
|
|
|
case 'name':
|
|
|
|
$name = (string)$child;
|
|
|
|
$name = str_replace( '*dbname*', self::$DBNAME, $name );
|
|
|
|
break;
|
|
|
|
case 'create':
|
|
|
|
case 'overwrite':
|
|
|
|
case 'charset':
|
|
|
|
break;
|
|
|
|
case 'table':
|
|
|
|
self::loadTable($schema, $child);
|
|
|
|
break;
|
|
|
|
default:
|
2013-03-17 17:15:36 +04:00
|
|
|
throw new DomainException('Unknown element: '.$child->getName());
|
2012-10-10 22:49:47 +04:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $schema;
|
|
|
|
}
|
|
|
|
|
2013-03-19 21:52:54 +04:00
|
|
|
/**
|
|
|
|
* @param\Doctrine\DBAL\Schema\Schema $schema
|
|
|
|
* @param $xml
|
|
|
|
* @throws DomainException
|
|
|
|
*/
|
2012-10-10 22:49:47 +04:00
|
|
|
private static function loadTable($schema, $xml) {
|
|
|
|
foreach($xml->children() as $child) {
|
|
|
|
switch($child->getName()) {
|
|
|
|
case 'name':
|
|
|
|
$name = (string)$child;
|
|
|
|
$name = str_replace( '*dbprefix*', self::$DBTABLEPREFIX, $name );
|
|
|
|
$table = $schema->createTable($name);
|
|
|
|
break;
|
|
|
|
case 'create':
|
|
|
|
case 'overwrite':
|
|
|
|
case 'charset':
|
|
|
|
break;
|
|
|
|
case 'declaration':
|
|
|
|
self::loadDeclaration($table, $child);
|
|
|
|
break;
|
|
|
|
default:
|
2013-03-17 17:15:36 +04:00
|
|
|
throw new DomainException('Unknown element: '.$child->getName());
|
2012-10-10 22:49:47 +04:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-19 21:52:54 +04:00
|
|
|
/**
|
|
|
|
* @param \Doctrine\DBAL\Schema\Table $table
|
|
|
|
* @param $xml
|
|
|
|
* @throws DomainException
|
|
|
|
*/
|
2012-10-10 22:49:47 +04:00
|
|
|
private static function loadDeclaration($table, $xml) {
|
|
|
|
foreach($xml->children() as $child) {
|
|
|
|
switch($child->getName()) {
|
|
|
|
case 'field':
|
|
|
|
self::loadField($table, $child);
|
|
|
|
break;
|
|
|
|
case 'index':
|
|
|
|
self::loadIndex($table, $child);
|
|
|
|
break;
|
|
|
|
default:
|
2013-03-17 17:15:36 +04:00
|
|
|
throw new DomainException('Unknown element: '.$child->getName());
|
2012-10-10 22:49:47 +04:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static function loadField($table, $xml) {
|
|
|
|
$options = array();
|
|
|
|
foreach($xml->children() as $child) {
|
|
|
|
switch($child->getName()) {
|
|
|
|
case 'name':
|
|
|
|
$name = (string)$child;
|
|
|
|
break;
|
|
|
|
case 'type':
|
|
|
|
$type = (string)$child;
|
|
|
|
switch($type) {
|
|
|
|
case 'text':
|
|
|
|
$type = 'string';
|
|
|
|
break;
|
|
|
|
case 'clob':
|
|
|
|
$type = 'text';
|
|
|
|
break;
|
|
|
|
case 'timestamp':
|
|
|
|
$type = 'datetime';
|
|
|
|
break;
|
|
|
|
// TODO
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'length':
|
|
|
|
$length = (string)$child;
|
|
|
|
$options['length'] = $length;
|
|
|
|
break;
|
|
|
|
case 'unsigned':
|
|
|
|
$unsigned = self::asBool($child);
|
|
|
|
$options['unsigned'] = $unsigned;
|
|
|
|
break;
|
|
|
|
case 'notnull':
|
|
|
|
$notnull = self::asBool($child);
|
|
|
|
$options['notnull'] = $notnull;
|
|
|
|
break;
|
|
|
|
case 'autoincrement':
|
|
|
|
$autoincrement = self::asBool($child);
|
|
|
|
$options['autoincrement'] = $autoincrement;
|
|
|
|
break;
|
|
|
|
case 'default':
|
|
|
|
$default = (string)$child;
|
|
|
|
$options['default'] = $default;
|
|
|
|
break;
|
|
|
|
default:
|
2013-03-17 17:15:36 +04:00
|
|
|
throw new DomainException('Unknown element: '.$child->getName());
|
2012-10-10 22:49:47 +04:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (isset($name) && isset($type)) {
|
|
|
|
if (empty($options['default'])) {
|
2013-03-17 18:25:41 +04:00
|
|
|
if (empty($options['notnull']) || !$options['notnull']) {
|
|
|
|
unset($options['default']);
|
2013-06-25 00:37:07 +04:00
|
|
|
$options['notnull'] = false;
|
2013-03-17 18:25:41 +04:00
|
|
|
}
|
2012-10-10 22:49:47 +04:00
|
|
|
if ($type == 'integer') {
|
2013-03-17 18:25:41 +04:00
|
|
|
$options['default'] = 0;
|
2012-10-10 22:49:47 +04:00
|
|
|
}
|
2013-03-17 18:25:41 +04:00
|
|
|
if (!empty($options['autoincrement']) && $options['autoincrement']) {
|
2012-10-10 22:49:47 +04:00
|
|
|
unset($options['default']);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ($type == 'integer') {
|
|
|
|
$length = $options['length'];
|
2013-03-17 20:34:35 +04:00
|
|
|
if ($length < 4) {
|
2012-10-10 22:49:47 +04:00
|
|
|
$type = 'smallint';
|
|
|
|
}
|
|
|
|
else if ($length > 4) {
|
|
|
|
$type = 'bigint';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$table->addColumn($name, $type, $options);
|
|
|
|
if (!empty($options['autoincrement'])
|
|
|
|
&& !empty($options['notnull'])) {
|
|
|
|
$table->setPrimaryKey(array($name));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static function loadIndex($table, $xml) {
|
|
|
|
$name = null;
|
|
|
|
$fields = array();
|
|
|
|
foreach($xml->children() as $child) {
|
|
|
|
switch($child->getName()) {
|
|
|
|
case 'name':
|
|
|
|
$name = (string)$child;
|
|
|
|
break;
|
|
|
|
case 'primary':
|
|
|
|
$primary = self::asBool($child);
|
|
|
|
break;
|
|
|
|
case 'unique':
|
|
|
|
$unique = self::asBool($child);
|
|
|
|
break;
|
|
|
|
case 'field':
|
|
|
|
foreach($child->children() as $field) {
|
|
|
|
switch($field->getName()) {
|
|
|
|
case 'name':
|
|
|
|
$field_name = (string)$field;
|
2013-03-17 20:00:07 +04:00
|
|
|
$keywords = self::$platform->getReservedKeywordsList();
|
|
|
|
if ($keywords->isKeyword($field_name)) {
|
|
|
|
$field_name = self::$platform->quoteIdentifier($field_name);
|
|
|
|
}
|
2012-10-10 22:49:47 +04:00
|
|
|
$fields[] = $field_name;
|
|
|
|
break;
|
|
|
|
case 'sorting':
|
|
|
|
break;
|
|
|
|
default:
|
2013-03-17 17:15:36 +04:00
|
|
|
throw new DomainException('Unknown element: '.$field->getName());
|
2012-10-10 22:49:47 +04:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
2013-03-17 17:15:36 +04:00
|
|
|
throw new DomainException('Unknown element: '.$child->getName());
|
2012-10-10 22:49:47 +04:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!empty($fields)) {
|
|
|
|
if (isset($primary) && $primary) {
|
|
|
|
$table->setPrimaryKey($fields, $name);
|
|
|
|
} else
|
|
|
|
if (isset($unique) && $unique) {
|
|
|
|
$table->addUniqueIndex($fields, $name);
|
|
|
|
} else {
|
|
|
|
$table->addIndex($fields, $name);
|
|
|
|
}
|
|
|
|
} else {
|
2013-03-17 17:15:36 +04:00
|
|
|
throw new DomainException('Empty index definition: '.$name.' options:'. print_r($fields, true));
|
2012-10-10 22:49:47 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static function asBool($xml) {
|
|
|
|
$result = (string)$xml;
|
|
|
|
if ($result == 'true') {
|
|
|
|
$result = true;
|
|
|
|
} else
|
|
|
|
if ($result == 'false') {
|
|
|
|
$result = false;
|
|
|
|
}
|
|
|
|
return (bool)$result;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|