|
11 | 11 | use MariaStan\Schema\Table; |
12 | 12 | use MariaStan\Util\MariaDbErrorCodes; |
13 | 13 |
|
| 14 | +use function array_column; |
| 15 | +use function array_fill_keys; |
14 | 16 | use function array_map; |
15 | 17 | use function file_get_contents; |
16 | 18 | use function hash; |
17 | 19 | use function is_array; |
| 20 | +use function is_string; |
18 | 21 | use function serialize; |
19 | 22 | use function unserialize; |
20 | 23 |
|
@@ -76,10 +79,15 @@ public function __construct( |
76 | 79 | $this->schemaDump = $dump; |
77 | 80 | } |
78 | 81 |
|
| 82 | + public function getDefaultDatabase(): string |
| 83 | + { |
| 84 | + return $this->defaultDatabase; |
| 85 | + } |
| 86 | + |
79 | 87 | /** @throws DbReflectionException */ |
80 | | - public function findTableSchema(string $table): Table |
| 88 | + public function findTableSchema(string $table, ?string $database = null): Table |
81 | 89 | { |
82 | | - $tableDump = $this->schemaDump['databases'][$this->defaultDatabase]['tables'][$table] ?? []; |
| 90 | + $tableDump = $this->schemaDump['databases'][$database ?? $this->defaultDatabase]['tables'][$table] ?? []; |
83 | 91 |
|
84 | 92 | if (! is_array($tableDump)) { |
85 | 93 | throw new UnexpectedValueException( |
@@ -136,59 +144,67 @@ public function getHash(): string |
136 | 144 | return hash('xxh128', serialize($this->schemaDump)); |
137 | 145 | } |
138 | 146 |
|
139 | | - public static function dumpSchema(\mysqli $db, string $database): string |
| 147 | + /** @param string|array<string>|null $databases null = all */ |
| 148 | + public static function dumpSchema(\mysqli $db, string|array|null $databases): string |
140 | 149 | { |
| 150 | + if (is_string($databases)) { |
| 151 | + $databases = [$databases]; |
| 152 | + } elseif ($databases === null) { |
| 153 | + $result = $db->query('SELECT SCHEMA_NAME FROM information_schema.SCHEMATA'); |
| 154 | + $databases = array_column($result->fetch_all(), 0); |
| 155 | + } |
| 156 | + |
141 | 157 | $result = [ |
142 | 158 | '__version' => self::DUMP_VERSION, |
143 | | - 'databases' => [ |
144 | | - $database => [ |
145 | | - 'tables' => [], |
146 | | - 'views' => [], |
147 | | - ], |
148 | | - ], |
| 159 | + 'databases' => array_fill_keys($databases, [ |
| 160 | + 'tables' => [], |
| 161 | + 'views' => [], |
| 162 | + ]), |
149 | 163 | ]; |
150 | 164 |
|
151 | | - $stmt = $db->prepare(' |
152 | | - SELECT * FROM information_schema.COLUMNS |
153 | | - WHERE TABLE_SCHEMA = ? |
154 | | - ORDER BY TABLE_NAME, ORDINAL_POSITION |
155 | | - '); |
156 | | - $stmt->execute([$database]); |
157 | | - $columns = $stmt->get_result()->fetch_all(\MYSQLI_ASSOC); |
158 | | - |
159 | | - /** @var array<string, scalar|null> $col */ |
160 | | - foreach ($columns as $col) { |
161 | | - $result['databases'][$database]['tables'][$col['TABLE_NAME']]['columns'][] = $col; |
162 | | - } |
| 165 | + foreach ($databases as $database) { |
| 166 | + $stmt = $db->prepare(' |
| 167 | + SELECT * FROM information_schema.COLUMNS |
| 168 | + WHERE TABLE_SCHEMA = ? |
| 169 | + ORDER BY TABLE_NAME, ORDINAL_POSITION |
| 170 | + '); |
| 171 | + $stmt->execute([$database]); |
| 172 | + $columns = $stmt->get_result()->fetch_all(MYSQLI_ASSOC); |
| 173 | + |
| 174 | + /** @var array<string, scalar|null> $col */ |
| 175 | + foreach ($columns as $col) { |
| 176 | + $result['databases'][$database]['tables'][$col['TABLE_NAME']]['columns'][] = $col; |
| 177 | + } |
163 | 178 |
|
164 | | - $stmt = $db->prepare(' |
165 | | - SELECT * FROM information_schema.TABLE_CONSTRAINTS tc |
166 | | - JOIN information_schema.KEY_COLUMN_USAGE kcu |
167 | | - USING (CONSTRAINT_SCHEMA, CONSTRAINT_NAME, TABLE_SCHEMA, TABLE_NAME) |
168 | | - WHERE TABLE_SCHEMA = ? AND tc.CONSTRAINT_TYPE = "FOREIGN KEY" |
169 | | - /* This happens when there is a FK and UNIQUE with same name. */ |
170 | | - AND kcu.REFERENCED_TABLE_NAME IS NOT NULL |
171 | | - ORDER BY TABLE_NAME, CONSTRAINT_NAME, kcu.ORDINAL_POSITION |
172 | | - '); |
173 | | - $stmt->execute([$database]); |
174 | | - $foreignKeys = $stmt->get_result()->fetch_all(\MYSQLI_ASSOC); |
175 | | - |
176 | | - /** @var array<string, scalar|null> $fk */ |
177 | | - foreach ($foreignKeys as $fk) { |
178 | | - $result['databases'][$database]['tables'][$fk['TABLE_NAME']]['foreign_keys'][] = $fk; |
179 | | - } |
| 179 | + $stmt = $db->prepare(' |
| 180 | + SELECT * FROM information_schema.TABLE_CONSTRAINTS tc |
| 181 | + JOIN information_schema.KEY_COLUMN_USAGE kcu |
| 182 | + USING (CONSTRAINT_SCHEMA, CONSTRAINT_NAME, TABLE_SCHEMA, TABLE_NAME) |
| 183 | + WHERE TABLE_SCHEMA = ? AND tc.CONSTRAINT_TYPE = "FOREIGN KEY" |
| 184 | + /* This happens when there is a FK and UNIQUE with same name. */ |
| 185 | + AND kcu.REFERENCED_TABLE_NAME IS NOT NULL |
| 186 | + ORDER BY TABLE_NAME, CONSTRAINT_NAME, kcu.ORDINAL_POSITION |
| 187 | + '); |
| 188 | + $stmt->execute([$database]); |
| 189 | + $foreignKeys = $stmt->get_result()->fetch_all(MYSQLI_ASSOC); |
| 190 | + |
| 191 | + /** @var array<string, scalar|null> $fk */ |
| 192 | + foreach ($foreignKeys as $fk) { |
| 193 | + $result['databases'][$database]['tables'][$fk['TABLE_NAME']]['foreign_keys'][] = $fk; |
| 194 | + } |
180 | 195 |
|
181 | | - $stmt = $db->prepare(' |
182 | | - SELECT TABLE_SCHEMA, TABLE_NAME, VIEW_DEFINITION |
183 | | - FROM information_schema.VIEWS |
184 | | - WHERE TABLE_SCHEMA = ? |
185 | | - '); |
186 | | - $stmt->execute([$database]); |
187 | | - $views = $stmt->get_result()->fetch_all(MYSQLI_ASSOC); |
188 | | - |
189 | | - foreach ($views as $view) { |
190 | | - $result['databases'][$view['TABLE_SCHEMA']]['views'][$view['TABLE_NAME']] |
191 | | - = ['definition' => $view['VIEW_DEFINITION']]; |
| 196 | + $stmt = $db->prepare(' |
| 197 | + SELECT TABLE_SCHEMA, TABLE_NAME, VIEW_DEFINITION |
| 198 | + FROM information_schema.VIEWS |
| 199 | + WHERE TABLE_SCHEMA = ? |
| 200 | + '); |
| 201 | + $stmt->execute([$database]); |
| 202 | + $views = $stmt->get_result()->fetch_all(MYSQLI_ASSOC); |
| 203 | + |
| 204 | + foreach ($views as $view) { |
| 205 | + $result['databases'][$view['TABLE_SCHEMA']]['views'][$view['TABLE_NAME']] |
| 206 | + = ['definition' => $view['VIEW_DEFINITION']]; |
| 207 | + } |
192 | 208 | } |
193 | 209 |
|
194 | 210 | return serialize($result); |
|
0 commit comments