Page Menu
Home
Phabricator
Search
Configure Global Search
Log In
Files
F15418362
D10494.id.diff
No One
Temporary
Actions
View File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Award Token
Flag For Later
Size
21 KB
Referenced Files
None
Subscribers
None
D10494.id.diff
View Options
diff --git a/src/__phutil_library_map__.php b/src/__phutil_library_map__.php
--- a/src/__phutil_library_map__.php
+++ b/src/__phutil_library_map__.php
@@ -1328,8 +1328,11 @@
'PhabricatorConduitTokenController' => 'applications/conduit/controller/PhabricatorConduitTokenController.php',
'PhabricatorConfigAllController' => 'applications/config/controller/PhabricatorConfigAllController.php',
'PhabricatorConfigApplication' => 'applications/config/application/PhabricatorConfigApplication.php',
+ 'PhabricatorConfigColumnSchema' => 'applications/config/schema/PhabricatorConfigColumnSchema.php',
'PhabricatorConfigConfigPHIDType' => 'applications/config/phid/PhabricatorConfigConfigPHIDType.php',
'PhabricatorConfigController' => 'applications/config/controller/PhabricatorConfigController.php',
+ 'PhabricatorConfigDatabaseController' => 'applications/config/controller/PhabricatorConfigDatabaseController.php',
+ 'PhabricatorConfigDatabaseSchema' => 'applications/config/schema/PhabricatorConfigDatabaseSchema.php',
'PhabricatorConfigDatabaseSource' => 'infrastructure/env/PhabricatorConfigDatabaseSource.php',
'PhabricatorConfigDefaultSource' => 'infrastructure/env/PhabricatorConfigDefaultSource.php',
'PhabricatorConfigDictionarySource' => 'infrastructure/env/PhabricatorConfigDictionarySource.php',
@@ -1356,8 +1359,11 @@
'PhabricatorConfigOptionType' => 'applications/config/custom/PhabricatorConfigOptionType.php',
'PhabricatorConfigProxySource' => 'infrastructure/env/PhabricatorConfigProxySource.php',
'PhabricatorConfigResponse' => 'applications/config/response/PhabricatorConfigResponse.php',
+ 'PhabricatorConfigSchemaQuery' => 'applications/config/schema/PhabricatorConfigSchemaQuery.php',
+ 'PhabricatorConfigServerSchema' => 'applications/config/schema/PhabricatorConfigServerSchema.php',
'PhabricatorConfigSource' => 'infrastructure/env/PhabricatorConfigSource.php',
'PhabricatorConfigStackSource' => 'infrastructure/env/PhabricatorConfigStackSource.php',
+ 'PhabricatorConfigTableSchema' => 'applications/config/schema/PhabricatorConfigTableSchema.php',
'PhabricatorConfigTransaction' => 'applications/config/storage/PhabricatorConfigTransaction.php',
'PhabricatorConfigTransactionQuery' => 'applications/config/query/PhabricatorConfigTransactionQuery.php',
'PhabricatorConfigValidationException' => 'applications/config/exception/PhabricatorConfigValidationException.php',
@@ -4210,8 +4216,11 @@
'PhabricatorConduitTokenController' => 'PhabricatorConduitController',
'PhabricatorConfigAllController' => 'PhabricatorConfigController',
'PhabricatorConfigApplication' => 'PhabricatorApplication',
+ 'PhabricatorConfigColumnSchema' => 'Phobject',
'PhabricatorConfigConfigPHIDType' => 'PhabricatorPHIDType',
'PhabricatorConfigController' => 'PhabricatorController',
+ 'PhabricatorConfigDatabaseController' => 'PhabricatorConfigController',
+ 'PhabricatorConfigDatabaseSchema' => 'Phobject',
'PhabricatorConfigDatabaseSource' => 'PhabricatorConfigProxySource',
'PhabricatorConfigDefaultSource' => 'PhabricatorConfigProxySource',
'PhabricatorConfigDictionarySource' => 'PhabricatorConfigSource',
@@ -4242,7 +4251,10 @@
),
'PhabricatorConfigProxySource' => 'PhabricatorConfigSource',
'PhabricatorConfigResponse' => 'AphrontHTMLResponse',
+ 'PhabricatorConfigSchemaQuery' => 'Phobject',
+ 'PhabricatorConfigServerSchema' => 'Phobject',
'PhabricatorConfigStackSource' => 'PhabricatorConfigSource',
+ 'PhabricatorConfigTableSchema' => 'Phobject',
'PhabricatorConfigTransaction' => 'PhabricatorApplicationTransaction',
'PhabricatorConfigTransactionQuery' => 'PhabricatorApplicationTransactionQuery',
'PhabricatorConfigValidationException' => 'Exception',
diff --git a/src/applications/config/application/PhabricatorConfigApplication.php b/src/applications/config/application/PhabricatorConfigApplication.php
--- a/src/applications/config/application/PhabricatorConfigApplication.php
+++ b/src/applications/config/application/PhabricatorConfigApplication.php
@@ -42,6 +42,8 @@
'edit/(?P<key>[\w\.\-]+)/' => 'PhabricatorConfigEditController',
'group/(?P<key>[^/]+)/' => 'PhabricatorConfigGroupController',
'welcome/' => 'PhabricatorConfigWelcomeController',
+ 'database/(?:(?P<database>[^/]+)/(?:(?P<table>[^/]+)/)?)?'
+ => 'PhabricatorConfigDatabaseController',
'(?P<verb>ignore|unignore)/(?P<key>[^/]+)/'
=> 'PhabricatorConfigIgnoreController',
'issue/' => array(
diff --git a/src/applications/config/controller/PhabricatorConfigController.php b/src/applications/config/controller/PhabricatorConfigController.php
--- a/src/applications/config/controller/PhabricatorConfigController.php
+++ b/src/applications/config/controller/PhabricatorConfigController.php
@@ -15,6 +15,7 @@
$nav->addFilter('/', pht('Option Groups'));
$nav->addFilter('all/', pht('All Settings'));
$nav->addFilter('issue/', pht('Setup Issues'));
+ $nav->addFilter('database/', pht('Database Status'));
$nav->addFilter('welcome/', pht('Welcome Screen'));
return $nav;
diff --git a/src/applications/config/controller/PhabricatorConfigDatabaseController.php b/src/applications/config/controller/PhabricatorConfigDatabaseController.php
new file mode 100644
--- /dev/null
+++ b/src/applications/config/controller/PhabricatorConfigDatabaseController.php
@@ -0,0 +1,258 @@
+<?php
+
+final class PhabricatorConfigDatabaseController
+ extends PhabricatorConfigController {
+
+ private $database;
+ private $table;
+
+ public function willProcessRequest(array $data) {
+ $this->database = idx($data, 'database');
+ $this->table = idx($data, 'table');
+ }
+
+ public function processRequest() {
+ $request = $this->getRequest();
+ $viewer = $request->getUser();
+
+ $conf = PhabricatorEnv::newObjectFromConfig(
+ 'mysql.configuration-provider',
+ array($dao = null, 'w'));
+
+ $api = id(new PhabricatorStorageManagementAPI())
+ ->setUser($conf->getUser())
+ ->setHost($conf->getHost())
+ ->setPort($conf->getPort())
+ ->setNamespace(PhabricatorLiskDAO::getDefaultStorageNamespace())
+ ->setPassword($conf->getPassword());
+
+ $query = id(new PhabricatorConfigSchemaQuery())
+ ->setAPI($api);
+
+ $actual = $query->loadActualSchema();
+ $expect = $query->loadExpectedSchema();
+
+ if ($this->table) {
+ return $this->renderTable(
+ $actual,
+ $expect,
+ $this->database,
+ $this->table);
+ } else if ($this->database) {
+ return $this->renderDatabase(
+ $actual,
+ $expect,
+ $this->database);
+ } else {
+ return $this->renderServer(
+ $actual,
+ $expect);
+ }
+ }
+
+ private function buildResponse($title, $body) {
+ $nav = $this->buildSideNavView();
+ $nav->selectFilter('database/');
+
+ $crumbs = $this->buildApplicationCrumbs();
+ if ($this->database) {
+ $crumbs->addTextCrumb(
+ pht('Database Status'),
+ $this->getApplicationURI('database/'));
+ if ($this->table) {
+ $crumbs->addTextCrumb(
+ $this->database,
+ $this->getApplicationURI('database/'.$this->database.'/'));
+ $crumbs->addTextCrumb($this->table);
+ } else {
+ $crumbs->addTextCrumb($this->database);
+ }
+ } else {
+ $crumbs->addTextCrumb(pht('Database Status'));
+ }
+
+ $nav->setCrumbs($crumbs);
+ $nav->appendChild($body);
+
+ return $this->buildApplicationPage(
+ $nav,
+ array(
+ 'title' => $title,
+ ));
+ }
+
+
+ private function renderServer(
+ PhabricatorConfigServerSchema $schema,
+ PhabricatorConfigServerSchema $expect) {
+
+ $icon_ok = id(new PHUIIconView())
+ ->setIconFont('fa-check-circle green');
+
+ $icon_warn = id(new PHUIIconView())
+ ->setIconFont('fa-exclamation-circle yellow');
+
+ $rows = array();
+ foreach ($schema->getDatabases() as $database_name => $database) {
+
+ $expect_database = $expect->getDatabase($database_name);
+ if ($expect_database) {
+ $expect_set = $expect_database->getCharacterSet();
+ $expect_collation = $expect_database->getCollation();
+
+ if ($database->isSameSchema($expect_database)) {
+ $icon = $icon_ok;
+ } else {
+ $icon = $icon_warn;
+ }
+ } else {
+ $expect_set = null;
+ $expect_collation = null;
+ $icon = $icon_warn;
+ }
+
+ $actual_set = $database->getCharacterSet();
+ $actual_collation = $database->getCollation();
+
+
+
+ $rows[] = array(
+ $icon,
+ phutil_tag(
+ 'a',
+ array(
+ 'href' => $this->getApplicationURI(
+ '/database/'.$database_name.'/'),
+ ),
+ $database_name),
+ $actual_set,
+ $expect_set,
+ $actual_collation,
+ $expect_collation,
+ );
+ }
+
+ $table = id(new AphrontTableView($rows))
+ ->setHeaders(
+ array(
+ null,
+ pht('Database'),
+ pht('Charset'),
+ pht('Expected Charset'),
+ pht('Collation'),
+ pht('Expected Collation'),
+ ))
+ ->setColumnClasses(
+ array(
+ '',
+ 'wide pri',
+ null,
+ null,
+ ));
+
+ $title = pht('Database Status');
+
+ $box = id(new PHUIObjectBoxView())
+ ->setHeaderText($title)
+ ->appendChild($table);
+
+ return $this->buildResponse($title, $box);
+ }
+
+ private function renderDatabase(
+ PhabricatorConfigServerSchema $schema,
+ PhabricatorConfigServerSchema $expect,
+ $database_name) {
+
+ $database = $schema->getDatabase($database_name);
+ if (!$database) {
+ return new Aphront404Response();
+ }
+
+ $rows = array();
+ foreach ($database->getTables() as $table_name => $table) {
+ $rows[] = array(
+ phutil_tag(
+ 'a',
+ array(
+ 'href' => $this->getApplicationURI(
+ '/database/'.$database_name.'/'.$table_name.'/'),
+ ),
+ $table_name),
+ $table->getCollation(),
+ );
+ }
+
+ $table = id(new AphrontTableView($rows))
+ ->setHeaders(
+ array(
+ pht('Table'),
+ pht('Collation'),
+ ))
+ ->setColumnClasses(
+ array(
+ 'wide pri',
+ null,
+ ));
+
+ $title = pht('Database Status: %s', $database_name);
+
+ $box = id(new PHUIObjectBoxView())
+ ->setHeaderText($title)
+ ->appendChild($table);
+
+ return $this->buildResponse($title, $box);
+ }
+
+ private function renderTable(
+ PhabricatorConfigServerSchema $schema,
+ PhabricatorConfigServerSchema $expect,
+ $database_name,
+ $table_name) {
+
+ $database = $schema->getDatabase($database_name);
+ if (!$database) {
+ return new Aphront404Response();
+ }
+
+ $table = $database->getTable($table_name);
+ if (!$table) {
+ return new Aphront404Response();
+ }
+
+ $rows = array();
+ foreach ($table->getColumns() as $column_name => $column) {
+ $rows[] = array(
+ $column_name,
+ $column->getColumnType(),
+ $column->getCharacterSet(),
+ $column->getCollation(),
+ );
+ }
+
+ $table = id(new AphrontTableView($rows))
+ ->setHeaders(
+ array(
+ pht('Table'),
+ pht('Column Type'),
+ pht('Character Set'),
+ pht('Collation'),
+ ))
+ ->setColumnClasses(
+ array(
+ 'wide pri',
+ null,
+ null,
+ null
+ ));
+
+ $title = pht('Database Status: %s.%s', $database_name, $table_name);
+
+ $box = id(new PHUIObjectBoxView())
+ ->setHeaderText($title)
+ ->appendChild($table);
+
+ return $this->buildResponse($title, $box);
+ }
+
+}
diff --git a/src/applications/config/schema/PhabricatorConfigColumnSchema.php b/src/applications/config/schema/PhabricatorConfigColumnSchema.php
new file mode 100644
--- /dev/null
+++ b/src/applications/config/schema/PhabricatorConfigColumnSchema.php
@@ -0,0 +1,46 @@
+<?php
+
+final class PhabricatorConfigColumnSchema extends Phobject {
+
+ private $name;
+ private $characterSet;
+ private $collation;
+ private $columnType;
+
+ public function setColumnType($column_type) {
+ $this->columnType = $column_type;
+ return $this;
+ }
+
+ public function getColumnType() {
+ return $this->columnType;
+ }
+
+ public function setCollation($collation) {
+ $this->collation = $collation;
+ return $this;
+ }
+
+ public function getCollation() {
+ return $this->collation;
+ }
+
+ public function setCharacterSet($character_set) {
+ $this->characterSet = $character_set;
+ return $this;
+ }
+
+ public function getCharacterSet() {
+ return $this->characterSet;
+ }
+
+ public function setName($name) {
+ $this->name = $name;
+ return $this;
+ }
+
+ public function getName() {
+ return $this->name;
+ }
+
+}
diff --git a/src/applications/config/schema/PhabricatorConfigDatabaseSchema.php b/src/applications/config/schema/PhabricatorConfigDatabaseSchema.php
new file mode 100644
--- /dev/null
+++ b/src/applications/config/schema/PhabricatorConfigDatabaseSchema.php
@@ -0,0 +1,67 @@
+<?php
+
+final class PhabricatorConfigDatabaseSchema extends Phobject {
+
+ private $name;
+ private $characterSet;
+ private $collation;
+ private $tables = array();
+
+ public function addTable(PhabricatorConfigTableSchema $table) {
+ $key = $table->getName();
+ if (isset($this->tables[$key])) {
+ throw new Exception(
+ pht('Trying to add duplicate table "%s"!', $key));
+ }
+ $this->tables[$key] = $table;
+ return $this;
+ }
+
+ public function getTables() {
+ return $this->tables;
+ }
+
+ public function getTable($key) {
+ return idx($this->tables, $key);
+ }
+
+ public function isSameSchema(PhabricatorConfigDatabaseSchema $expect) {
+ return ($this->toDictionary() === $expect->toDictionary());
+ }
+
+ public function toDictionary() {
+ return array(
+ 'name' => $this->getName(),
+ 'characterSet' => $this->getCharacterSet(),
+ 'collation' => $this->getCollation(),
+ );
+ }
+
+ public function setCollation($collation) {
+ $this->collation = $collation;
+ return $this;
+ }
+
+ public function getCollation() {
+ return $this->collation;
+ }
+
+ public function setCharacterSet($character_set) {
+ $this->characterSet = $character_set;
+ return $this;
+ }
+
+ public function getCharacterSet() {
+ return $this->characterSet;
+ }
+
+ public function setName($name) {
+ $this->name = $name;
+ return $this;
+ }
+
+ public function getName() {
+ return $this->name;
+ }
+
+}
diff --git a/src/applications/config/schema/PhabricatorConfigSchemaQuery.php b/src/applications/config/schema/PhabricatorConfigSchemaQuery.php
new file mode 100644
--- /dev/null
+++ b/src/applications/config/schema/PhabricatorConfigSchemaQuery.php
@@ -0,0 +1,135 @@
+<?php
+
+final class PhabricatorConfigSchemaQuery extends Phobject {
+
+ private $api;
+
+ public function setAPI(PhabricatorStorageManagementAPI $api) {
+ $this->api = $api;
+ return $this;
+ }
+
+ protected function getAPI() {
+ if (!$this->api) {
+ throw new Exception(pht('Call setAPI() before issuing a query!'));
+ }
+ return $this->api;
+ }
+
+ protected function getConn() {
+ return $this->getAPI()->getConn(null);
+ }
+
+ private function getDatabaseNames() {
+ $api = $this->getAPI();
+ $patches = PhabricatorSQLPatchList::buildAllPatches();
+ return $api->getDatabaseList(
+ $patches,
+ $only_living = true);
+ }
+
+ public function loadActualSchema() {
+ $databases = $this->getDatabaseNames();
+
+ $conn = $this->getConn();
+ $tables = queryfx_all(
+ $conn,
+ 'SELECT TABLE_SCHEMA, TABLE_NAME, TABLE_COLLATION
+ FROM INFORMATION_SCHEMA.TABLES
+ WHERE TABLE_SCHEMA IN (%Ls)',
+ $databases);
+
+ $database_info = queryfx_all(
+ $conn,
+ 'SELECT SCHEMA_NAME, DEFAULT_CHARACTER_SET_NAME, DEFAULT_COLLATION_NAME
+ FROM INFORMATION_SCHEMA.SCHEMATA
+ WHERE SCHEMA_NAME IN (%Ls)',
+ $databases);
+ $database_info = ipull($database_info, null, 'SCHEMA_NAME');
+
+ $server_schema = new PhabricatorConfigServerSchema();
+
+ $tables = igroup($tables, 'TABLE_SCHEMA');
+ foreach ($tables as $database_name => $database_tables) {
+ $info = $database_info[$database_name];
+
+ $database_schema = id(new PhabricatorConfigDatabaseSchema())
+ ->setName($database_name)
+ ->setCharacterSet($info['DEFAULT_CHARACTER_SET_NAME'])
+ ->setCollation($info['DEFAULT_COLLATION_NAME']);
+
+ foreach ($database_tables as $table) {
+ $table_name = $table['TABLE_NAME'];
+
+ $table_schema = id(new PhabricatorConfigTableSchema())
+ ->setName($table_name)
+ ->setCollation($table['TABLE_COLLATION']);
+
+ $columns = queryfx_all(
+ $conn,
+ 'SELECT COLUMN_NAME, CHARACTER_SET_NAME, COLLATION_NAME, COLUMN_TYPE
+ FROM INFORMATION_SCHEMA.COLUMNS
+ WHERE TABLE_SCHEMA = %s AND TABLE_NAME = %s',
+ $database_name,
+ $table_name);
+
+ foreach ($columns as $column) {
+ $column_schema = id(new PhabricatorConfigColumnSchema())
+ ->setName($column['COLUMN_NAME'])
+ ->setCharacterSet($column['CHARACTER_SET_NAME'])
+ ->setCollation($column['COLLATION_NAME'])
+ ->setColumnType($column['COLUMN_TYPE']);
+
+ $table_schema->addColumn($column_schema);
+ }
+
+ $database_schema->addTable($table_schema);
+ }
+
+ $server_schema->addDatabase($database_schema);
+ }
+
+ return $server_schema;
+ }
+
+
+ public function loadExpectedSchema() {
+ $databases = $this->getDatabaseNames();
+
+ $api = $this->getAPI();
+
+ if ($api->isCharacterSetAvailable('utf8mb4')) {
+ // If utf8mb4 is available, we use it with the utf8mb4_unicode_ci
+ // collation. This is most correct, and will sort properly.
+
+ $utf8_charset = 'utf8mb4';
+ $utf8_collate = 'utf8mb4_unicode_ci';
+ } else {
+ // If utf8mb4 is not available, we use binary. This allows us to store
+ // 4-byte unicode characters. This has some tradeoffs:
+ //
+ // Unicode characters won't sort correctly. There's nothing we can do
+ // about this while still supporting 4-byte characters.
+ //
+ // It's possible that strings will be truncated in the middle of a
+ // character on insert. We encourage users to set STRICT_ALL_TABLES
+ // to prevent this.
+
+ $utf8_charset = 'binary';
+ $utf8_collate = 'binary';
+ }
+
+ $server_schema = new PhabricatorConfigServerSchema();
+ foreach ($databases as $database_name) {
+ $database_schema = id(new PhabricatorConfigDatabaseSchema())
+ ->setName($database_name)
+ ->setCharacterSet($utf8_charset)
+ ->setCollation($utf8_collate);
+
+ $server_schema->addDatabase($database_schema);
+ }
+
+ return $server_schema;
+ }
+
+}
diff --git a/src/applications/config/schema/PhabricatorConfigServerSchema.php b/src/applications/config/schema/PhabricatorConfigServerSchema.php
new file mode 100644
--- /dev/null
+++ b/src/applications/config/schema/PhabricatorConfigServerSchema.php
@@ -0,0 +1,25 @@
+<?php
+
+final class PhabricatorConfigServerSchema extends Phobject {
+
+ private $databases = array();
+
+ public function addDatabase(PhabricatorConfigDatabaseSchema $database) {
+ $key = $database->getName();
+ if (isset($this->databases[$key])) {
+ throw new Exception(
+ pht('Trying to add duplicate database "%s"!', $key));
+ }
+ $this->databases[$key] = $database;
+ return $this;
+ }
+
+ public function getDatabases() {
+ return $this->databases;
+ }
+
+ public function getDatabase($key) {
+ return idx($this->getDatabases(), $key);
+ }
+
+}
diff --git a/src/applications/config/schema/PhabricatorConfigTableSchema.php b/src/applications/config/schema/PhabricatorConfigTableSchema.php
new file mode 100644
--- /dev/null
+++ b/src/applications/config/schema/PhabricatorConfigTableSchema.php
@@ -0,0 +1,41 @@
+<?php
+
+final class PhabricatorConfigTableSchema extends Phobject {
+
+ private $name;
+ private $collation;
+ private $columns = array();
+
+ public function addColumn(PhabricatorConfigColumnSchema $column) {
+ $key = $column->getName();
+ if (isset($this->columns[$key])) {
+ throw new Exception(
+ pht('Trying to add duplicate column "%s"!', $key));
+ }
+ $this->columns[$key] = $column;
+ return $this;
+ }
+
+ public function getColumns() {
+ return $this->columns;
+ }
+
+ public function setCollation($collation) {
+ $this->collation = $collation;
+ return $this;
+ }
+
+ public function getCollation() {
+ return $this->collation;
+ }
+
+ public function setName($name) {
+ $this->name = $name;
+ return $this;
+ }
+
+ public function getName() {
+ return $this->name;
+ }
+
+}
diff --git a/src/infrastructure/storage/management/PhabricatorStorageManagementAPI.php b/src/infrastructure/storage/management/PhabricatorStorageManagementAPI.php
--- a/src/infrastructure/storage/management/PhabricatorStorageManagementAPI.php
+++ b/src/infrastructure/storage/management/PhabricatorStorageManagementAPI.php
@@ -196,4 +196,16 @@
require_once $script;
}
+ public function isCharacterSetAvailable($character_set) {
+ $conn = $this->getConn(null);
+
+ $result = queryfx_one(
+ $conn,
+ 'SELECT CHARACTER_SET_NAME FROM INFORMATION_SCHEMA.CHARACTER_SETS
+ WHERE CHARACTER_SET_NAME = %s',
+ $character_set);
+
+ return (bool)$result;
+ }
+
}
File Metadata
Details
Attached
Mime Type
text/plain
Expires
Fri, Mar 21, 10:51 PM (2 w, 1 d ago)
Storage Engine
blob
Storage Format
Encrypted (AES-256-CBC)
Storage Handle
7222475
Default Alt Text
D10494.id.diff (21 KB)
Attached To
Mode
D10494: Add a UI for reviewing database schemata
Attached
Detach File
Event Timeline
Log In to Comment