Passed
Pull Request — master (#2945)
by Dorian
20:40 queued 09:56
created

Comparator   F

Complexity

Total Complexity 118

Size/Duplication

Total Lines 513
Duplicated Lines 11.11 %

Test Coverage

Coverage 98.21%

Importance

Changes 0
Metric Value
wmc 118
dl 57
loc 513
rs 1.5789
c 0
b 0
f 0
ccs 220
cts 224
cp 0.9821

11 Methods

Rating   Name   Duplication   Size   Complexity  
A diffSequence() 0 11 3
F compare() 0 98 25
A isAutoIncrementSequenceInSchema() 0 9 3
A compareSchemas() 0 5 1
C detectIndexRenamings() 28 28 7
F diffColumn() 0 80 32
F diffTable() 0 106 25
B diffForeignKey() 0 23 6
B isALegacyJsonComparison() 0 8 6
A diffIndex() 0 7 3
B detectColumnRenamings() 21 21 7

How to fix   Duplicated Code    Complexity   

Duplicated Code

Duplicate code is one of the most pungent code smells. A rule that is often used is to re-structure code once it is duplicated in three or more places.

Common duplication problems, and corresponding solutions are:

Complex Class

 Tip:   Before tackling complexity, make sure that you eliminate any duplication first. This often can reduce the size of classes significantly.

Complex classes like Comparator often do a lot of different things. To break such a class down, we need to identify a cohesive component within that class. A common approach to find such a component is to look for fields/methods that share the same prefixes, or suffixes.

Once you have determined the fields that belong together, you can apply the Extract Class refactoring. If the component makes sense as a sub-class, Extract Subclass is also a candidate, and is often faster.

While breaking up the class, it is a good idea to analyze how other classes use Comparator, and based on these observations, apply Extract Interface, too.

1
<?php
2
/*
3
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
4
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
5
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
6
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
7
 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
8
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
9
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
10
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
11
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
12
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
13
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
14
 *
15
 * This software consists of voluntary contributions made by many individuals
16
 * and is licensed under the MIT license. For more information, see
17
 * <http://www.doctrine-project.org>.
18
 */
19
20
namespace Doctrine\DBAL\Schema;
21
22
use Doctrine\DBAL\Types;
23
24
/**
25
 * Compares two Schemas and return an instance of SchemaDiff.
26
 *
27
 * @link   www.doctrine-project.org
28
 * @since  2.0
29
 * @author Benjamin Eberlei <[email protected]>
30
 */
31
class Comparator
32
{
33
    /**
34
     * @param \Doctrine\DBAL\Schema\Schema $fromSchema
35
     * @param \Doctrine\DBAL\Schema\Schema $toSchema
36
     *
37
     * @return \Doctrine\DBAL\Schema\SchemaDiff
38
     */
39 17
    public static function compareSchemas(Schema $fromSchema, Schema $toSchema)
40
    {
41 17
        $c = new self();
42
43 17
        return $c->compare($fromSchema, $toSchema);
44
    }
45
46
    /**
47
     * Returns a SchemaDiff object containing the differences between the schemas $fromSchema and $toSchema.
48
     *
49
     * The returned differences are returned in such a way that they contain the
50
     * operations to change the schema stored in $fromSchema to the schema that is
51
     * stored in $toSchema.
52
     *
53
     * @param \Doctrine\DBAL\Schema\Schema $fromSchema
54
     * @param \Doctrine\DBAL\Schema\Schema $toSchema
55
     *
56
     * @return \Doctrine\DBAL\Schema\SchemaDiff
57
     */
58 27
    public function compare(Schema $fromSchema, Schema $toSchema)
59
    {
60 27
        $diff = new SchemaDiff();
61 27
        $diff->fromSchema = $fromSchema;
62
63 27
        $foreignKeysToTable = [];
64
65 27
        foreach ($toSchema->getNamespaces() as $namespace) {
66 2
            if ( ! $fromSchema->hasNamespace($namespace)) {
67 2
                $diff->newNamespaces[$namespace] = $namespace;
68
            }
69
        }
70
71 27
        foreach ($fromSchema->getNamespaces() as $namespace) {
72 2
            if ( ! $toSchema->hasNamespace($namespace)) {
73 2
                $diff->removedNamespaces[$namespace] = $namespace;
74
            }
75
        }
76
77 27
        foreach ($toSchema->getTables() as $table) {
78 21
            $tableName = $table->getShortestName($toSchema->getName());
79 21
            if ( ! $fromSchema->hasTable($tableName)) {
80 5
                $diff->newTables[$tableName] = $toSchema->getTable($tableName);
81
            } else {
82 19
                $tableDifferences = $this->diffTable($fromSchema->getTable($tableName), $toSchema->getTable($tableName));
83 19
                if ($tableDifferences !== false) {
84 21
                    $diff->changedTables[$tableName] = $tableDifferences;
85
                }
86
            }
87
        }
88
89
        /* Check if there are tables removed */
90 27
        foreach ($fromSchema->getTables() as $table) {
91 20
            $tableName = $table->getShortestName($fromSchema->getName());
92
93 20
            $table = $fromSchema->getTable($tableName);
94 20
            if ( ! $toSchema->hasTable($tableName)) {
95 5
                $diff->removedTables[$tableName] = $table;
96
            }
97
98
            // also remember all foreign keys that point to a specific table
99 20
            foreach ($table->getForeignKeys() as $foreignKey) {
100 2
                $foreignTable = strtolower($foreignKey->getForeignTableName());
101 2
                if (!isset($foreignKeysToTable[$foreignTable])) {
102 2
                    $foreignKeysToTable[$foreignTable] = [];
103
                }
104 20
                $foreignKeysToTable[$foreignTable][] = $foreignKey;
105
            }
106
        }
107
108 27
        foreach ($diff->removedTables as $tableName => $table) {
109 5
            if (isset($foreignKeysToTable[$tableName])) {
110 2
                $diff->orphanedForeignKeys = array_merge($diff->orphanedForeignKeys, $foreignKeysToTable[$tableName]);
111
112
                // deleting duplicated foreign keys present on both on the orphanedForeignKey
113
                // and the removedForeignKeys from changedTables
114 2
                foreach ($foreignKeysToTable[$tableName] as $foreignKey) {
115
                    // strtolower the table name to make if compatible with getShortestName
116 2
                    $localTableName = strtolower($foreignKey->getLocalTableName());
117 2
                    if (isset($diff->changedTables[$localTableName])) {
118 2
                        foreach ($diff->changedTables[$localTableName]->removedForeignKeys as $key => $removedForeignKey) {
119
                            // We check if the key is from the removed table if not we skip.
120 2
                            if ($tableName !== strtolower($removedForeignKey->getForeignTableName())) {
121 1
                                continue;
122
                            }
123 5
                            unset($diff->changedTables[$localTableName]->removedForeignKeys[$key]);
124
                        }
125
                    }
126
                }
127
            }
128
        }
129
130 27
        foreach ($toSchema->getSequences() as $sequence) {
131 4
            $sequenceName = $sequence->getShortestName($toSchema->getName());
132 4
            if ( ! $fromSchema->hasSequence($sequenceName)) {
133 3
                if ( ! $this->isAutoIncrementSequenceInSchema($fromSchema, $sequence)) {
134 3
                    $diff->newSequences[] = $sequence;
135
                }
136
            } else {
137 2
                if ($this->diffSequence($sequence, $fromSchema->getSequence($sequenceName))) {
138 4
                    $diff->changedSequences[] = $toSchema->getSequence($sequenceName);
139
                }
140
            }
141
        }
142
143 27
        foreach ($fromSchema->getSequences() as $sequence) {
144 4
            if ($this->isAutoIncrementSequenceInSchema($toSchema, $sequence)) {
145 1
                continue;
146
            }
147
148 3
            $sequenceName = $sequence->getShortestName($fromSchema->getName());
149
150 3
            if ( ! $toSchema->hasSequence($sequenceName)) {
151 3
                $diff->removedSequences[] = $sequence;
152
            }
153
        }
154
155 27
        return $diff;
156
    }
157
158
    /**
159
     * @param \Doctrine\DBAL\Schema\Schema   $schema
160
     * @param \Doctrine\DBAL\Schema\Sequence $sequence
161
     *
162
     * @return boolean
163
     */
164 6
    private function isAutoIncrementSequenceInSchema($schema, $sequence)
165
    {
166 6
        foreach ($schema->getTables() as $table) {
167 2
            if ($sequence->isAutoIncrementsFor($table)) {
168 2
                return true;
169
            }
170
        }
171
172 4
        return false;
173
    }
174
175
    /**
176
     * @param \Doctrine\DBAL\Schema\Sequence $sequence1
177
     * @param \Doctrine\DBAL\Schema\Sequence $sequence2
178
     *
179
     * @return boolean
180
     */
181 3
    public function diffSequence(Sequence $sequence1, Sequence $sequence2)
182
    {
183 3
        if ($sequence1->getAllocationSize() != $sequence2->getAllocationSize()) {
184 2
            return true;
185
        }
186
187 2
        if ($sequence1->getInitialValue() != $sequence2->getInitialValue()) {
188 1
            return true;
189
        }
190
191 1
        return false;
192
    }
193
194
    /**
195
     * Returns the difference between the tables $table1 and $table2.
196
     *
197
     * If there are no differences this method returns the boolean false.
198
     *
199
     * @param \Doctrine\DBAL\Schema\Table $table1
200
     * @param \Doctrine\DBAL\Schema\Table $table2
201
     *
202
     * @return boolean|\Doctrine\DBAL\Schema\TableDiff
203
     */
204 123
    public function diffTable(Table $table1, Table $table2)
205
    {
206 123
        $changes = 0;
207 123
        $tableDifferences = new TableDiff($table1->getName());
208 123
        $tableDifferences->fromTable = $table1;
209
210 123
        $table1Columns = $table1->getColumns();
211 123
        $table2Columns = $table2->getColumns();
212
213
        /* See if all the fields in table 1 exist in table 2 */
214 123
        foreach ($table2Columns as $columnName => $column) {
215 118
            if ( !$table1->hasColumn($columnName)) {
216 29
                $tableDifferences->addedColumns[$columnName] = $column;
217 118
                $changes++;
218
            }
219
        }
220
        /* See if there are any removed fields in table 2 */
221 123
        foreach ($table1Columns as $columnName => $column) {
222
            // See if column is removed in table 2.
223 118
            if ( ! $table2->hasColumn($columnName)) {
224 28
                $tableDifferences->removedColumns[$columnName] = $column;
225 28
                $changes++;
226 28
                continue;
227
            }
228
229
            // See if column has changed properties in table 2.
230 98
            $changedProperties = $this->diffColumn($column, $table2->getColumn($columnName));
231
232 98
            if ( ! empty($changedProperties)) {
233 42
                $columnDiff = new ColumnDiff($column->getName(), $table2->getColumn($columnName), $changedProperties);
234 42
                $columnDiff->fromColumn = $column;
235 42
                $tableDifferences->changedColumns[$column->getName()] = $columnDiff;
236 98
                $changes++;
237
            }
238
        }
239
240 123
        $this->detectColumnRenamings($tableDifferences);
241
242 123
        $table1Indexes = $table1->getIndexes();
243 123
        $table2Indexes = $table2->getIndexes();
244
245
        /* See if all the indexes in table 1 exist in table 2 */
246 123
        foreach ($table2Indexes as $indexName => $index) {
247 50
            if (($index->isPrimary() && $table1->hasPrimaryKey()) || $table1->hasIndex($indexName)) {
248 34
                continue;
249
            }
250
251 16
            $tableDifferences->addedIndexes[$indexName] = $index;
252 16
            $changes++;
253
        }
254
        /* See if there are any removed indexes in table 2 */
255 123
        foreach ($table1Indexes as $indexName => $index) {
256
            // See if index is removed in table 2.
257 47
            if (($index->isPrimary() && ! $table2->hasPrimaryKey()) ||
258 47
                ! $index->isPrimary() && ! $table2->hasIndex($indexName)
259
            ) {
260 17
                $tableDifferences->removedIndexes[$indexName] = $index;
261 17
                $changes++;
262 17
                continue;
263
            }
264
265
            // See if index has changed in table 2.
266 34
            $table2Index = $index->isPrimary() ? $table2->getPrimaryKey() : $table2->getIndex($indexName);
267
268 34
            if ($this->diffIndex($index, $table2Index)) {
0 ignored issues
show
Bug introduced by
It seems like $table2Index can also be of type null; however, parameter $index2 of Doctrine\DBAL\Schema\Comparator::diffIndex() does only seem to accept Doctrine\DBAL\Schema\Index, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

268
            if ($this->diffIndex($index, /** @scrutinizer ignore-type */ $table2Index)) {
Loading history...
269 15
                $tableDifferences->changedIndexes[$indexName] = $table2Index;
270 34
                $changes++;
271
            }
272
        }
273
274 123
        $this->detectIndexRenamings($tableDifferences);
275
276 123
        $fromFkeys = $table1->getForeignKeys();
277 123
        $toFkeys = $table2->getForeignKeys();
278
279 123
        foreach ($fromFkeys as $key1 => $constraint1) {
280 11
            foreach ($toFkeys as $key2 => $constraint2) {
281 5
                if ($this->diffForeignKey($constraint1, $constraint2) === false) {
282 2
                    if (strcasecmp($constraint1->getName(), $constraint2->getName()) !== 0) {
283 2
                        $tableDifferences->renamedForeignKeys[$constraint1->getName()] = $constraint2;
284 2
                        $changes++;
285
                    }
286 2
                    unset($fromFkeys[$key1]);
287 2
                    unset($toFkeys[$key2]);
288
                } else {
289 3
                    if (strtolower($constraint1->getName()) == strtolower($constraint2->getName())) {
290 2
                        $tableDifferences->changedForeignKeys[] = $constraint2;
291 2
                        $changes++;
292 2
                        unset($fromFkeys[$key1]);
293 11
                        unset($toFkeys[$key2]);
294
                    }
295
                }
296
            }
297
        }
298
299 123
        foreach ($fromFkeys as $constraint1) {
300 7
            $tableDifferences->removedForeignKeys[] = $constraint1;
301 7
            $changes++;
302
        }
303
304 123
        foreach ($toFkeys as $constraint2) {
305 2
            $tableDifferences->addedForeignKeys[] = $constraint2;
306 2
            $changes++;
307
        }
308
309 123
        return $changes ? $tableDifferences : false;
310
    }
311
312
    /**
313
     * Try to find columns that only changed their name, rename operations maybe cheaper than add/drop
314
     * however ambiguities between different possibilities should not lead to renaming at all.
315
     *
316
     * @param \Doctrine\DBAL\Schema\TableDiff $tableDifferences
317
     *
318
     * @return void
319
     */
320 123 View Code Duplication
    private function detectColumnRenamings(TableDiff $tableDifferences)
0 ignored issues
show
Duplication introduced by
This method seems to be duplicated in your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
321
    {
322 123
        $renameCandidates = [];
323 123
        foreach ($tableDifferences->addedColumns as $addedColumnName => $addedColumn) {
324 29
            foreach ($tableDifferences->removedColumns as $removedColumn) {
325 22
                if (count($this->diffColumn($addedColumn, $removedColumn)) == 0) {
326 29
                    $renameCandidates[$addedColumn->getName()][] = [$removedColumn, $addedColumn, $addedColumnName];
327
                }
328
            }
329
        }
330
331 123
        foreach ($renameCandidates as $candidateColumns) {
332 22
            if (count($candidateColumns) == 1) {
333 21
                list($removedColumn, $addedColumn) = $candidateColumns[0];
334 21
                $removedColumnName = strtolower($removedColumn->getName());
335 21
                $addedColumnName = strtolower($addedColumn->getName());
336
337 21
                if ( ! isset($tableDifferences->renamedColumns[$removedColumnName])) {
338 21
                    $tableDifferences->renamedColumns[$removedColumnName] = $addedColumn;
339 21
                    unset($tableDifferences->addedColumns[$addedColumnName]);
340 22
                    unset($tableDifferences->removedColumns[$removedColumnName]);
341
                }
342
            }
343
        }
344 123
    }
345
346
    /**
347
     * Try to find indexes that only changed their name, rename operations maybe cheaper than add/drop
348
     * however ambiguities between different possibilities should not lead to renaming at all.
349
     *
350
     * @param \Doctrine\DBAL\Schema\TableDiff $tableDifferences
351
     *
352
     * @return void
353
     */
354 123 View Code Duplication
    private function detectIndexRenamings(TableDiff $tableDifferences)
0 ignored issues
show
Duplication introduced by
This method seems to be duplicated in your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
355
    {
356 123
        $renameCandidates = [];
357
358
        // Gather possible rename candidates by comparing each added and removed index based on semantics.
359 123
        foreach ($tableDifferences->addedIndexes as $addedIndexName => $addedIndex) {
360 16
            foreach ($tableDifferences->removedIndexes as $removedIndex) {
361 7
                if (! $this->diffIndex($addedIndex, $removedIndex)) {
362 16
                    $renameCandidates[$addedIndex->getName()][] = [$removedIndex, $addedIndex, $addedIndexName];
363
                }
364
            }
365
        }
366
367 123
        foreach ($renameCandidates as $candidateIndexes) {
368
            // If the current rename candidate contains exactly one semantically equal index,
369
            // we can safely rename it.
370
            // Otherwise it is unclear if a rename action is really intended,
371
            // therefore we let those ambiguous indexes be added/dropped.
372 3
            if (count($candidateIndexes) === 1) {
373 2
                list($removedIndex, $addedIndex) = $candidateIndexes[0];
374
375 2
                $removedIndexName = strtolower($removedIndex->getName());
376 2
                $addedIndexName = strtolower($addedIndex->getName());
377
378 2
                if (! isset($tableDifferences->renamedIndexes[$removedIndexName])) {
379 2
                    $tableDifferences->renamedIndexes[$removedIndexName] = $addedIndex;
380 2
                    unset($tableDifferences->addedIndexes[$addedIndexName]);
381 3
                    unset($tableDifferences->removedIndexes[$removedIndexName]);
382
                }
383
            }
384
        }
385 123
    }
386
387
    /**
388
     * @param \Doctrine\DBAL\Schema\ForeignKeyConstraint $key1
389
     * @param \Doctrine\DBAL\Schema\ForeignKeyConstraint $key2
390
     *
391
     * @return boolean
392
     */
393 8
    public function diffForeignKey(ForeignKeyConstraint $key1, ForeignKeyConstraint $key2)
394
    {
395 8
        if (array_map('strtolower', $key1->getUnquotedLocalColumns()) != array_map('strtolower', $key2->getUnquotedLocalColumns())) {
396 1
            return true;
397
        }
398
399 7
        if (array_map('strtolower', $key1->getUnquotedForeignColumns()) != array_map('strtolower', $key2->getUnquotedForeignColumns())) {
400
            return true;
401
        }
402
403 7
        if ($key1->getUnqualifiedForeignTableName() !== $key2->getUnqualifiedForeignTableName()) {
404 1
            return true;
405
        }
406
407 6
        if ($key1->onUpdate() != $key2->onUpdate()) {
408 1
            return true;
409
        }
410
411 5
        if ($key1->onDelete() != $key2->onDelete()) {
412
            return true;
413
        }
414
415 5
        return false;
416
    }
417
418
    /**
419
     * Returns the difference between the fields $field1 and $field2.
420
     *
421
     * If there are differences this method returns $field2, otherwise the
422
     * boolean false.
423
     *
424
     * @param \Doctrine\DBAL\Schema\Column $column1
425
     * @param \Doctrine\DBAL\Schema\Column $column2
426
     *
427
     * @return array
428
     */
429 140
    public function diffColumn(Column $column1, Column $column2)
430
    {
431 140
        $properties1 = $column1->toArray();
432 140
        $properties2 = $column2->toArray();
433
434 140
        $changedProperties = [];
435
436 140
        foreach (['type', 'notnull', 'unsigned', 'autoincrement'] as $property) {
437 140
            if ($properties1[$property] != $properties2[$property]) {
438 140
                $changedProperties[] = $property;
439
            }
440
        }
441
442
        // This is a very nasty hack to make comparator work with the legacy json_array type, which should be killed in v3
443 140
        if ($this->isALegacyJsonComparison($properties1['type'], $properties2['type'])) {
0 ignored issues
show
Deprecated Code introduced by
The function Doctrine\DBAL\Schema\Com...ALegacyJsonComparison() has been deprecated. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-deprecated  annotation

443
        if (/** @scrutinizer ignore-deprecated */ $this->isALegacyJsonComparison($properties1['type'], $properties2['type'])) {
Loading history...
444 1
            array_shift($changedProperties);
445
446 1
            $changedProperties[] = 'comment';
447
        }
448
449 140
        if ($properties1['default'] != $properties2['default'] ||
450
            // Null values need to be checked additionally as they tell whether to create or drop a default value.
451
            // null != 0, null != false, null != '' etc. This affects platform's table alteration SQL generation.
452 137
            (null === $properties1['default'] && null !== $properties2['default']) ||
453 140
            (null === $properties2['default'] && null !== $properties1['default'])
454
        ) {
455 4
            $changedProperties[] = 'default';
456
        }
457
458 140
        if (($properties1['type'] instanceof Types\StringType && ! $properties1['type'] instanceof Types\GuidType) ||
459 140
            $properties1['type'] instanceof Types\BinaryType
460
        ) {
461
            // check if value of length is set at all, default value assumed otherwise.
462 28
            $length1 = $properties1['length'] ?: 255;
463 28
            $length2 = $properties2['length'] ?: 255;
464 28
            if ($length1 != $length2) {
465 13
                $changedProperties[] = 'length';
466
            }
467
468 28
            if ($properties1['fixed'] != $properties2['fixed']) {
469 28
                $changedProperties[] = 'fixed';
470
            }
471 123
        } elseif ($properties1['type'] instanceof Types\DecimalType) {
472 2
            if (($properties1['precision'] ?: 10) != ($properties2['precision'] ?: 10)) {
473
                $changedProperties[] = 'precision';
474
            }
475 2
            if ($properties1['scale'] != $properties2['scale']) {
476
                $changedProperties[] = 'scale';
477
            }
478
        }
479
480
        // A null value and an empty string are actually equal for a comment so they should not trigger a change.
481 140
        if ($properties1['comment'] !== $properties2['comment'] &&
482 140
            ! (null === $properties1['comment'] && '' === $properties2['comment']) &&
483 140
            ! (null === $properties2['comment'] && '' === $properties1['comment'])
484
        ) {
485 45
            $changedProperties[] = 'comment';
486
        }
487
488 140
        $customOptions1 = $column1->getCustomSchemaOptions();
489 140
        $customOptions2 = $column2->getCustomSchemaOptions();
490
491 140
        foreach (array_merge(array_keys($customOptions1), array_keys($customOptions2)) as $key) {
492 2
            if ( ! array_key_exists($key, $properties1) || ! array_key_exists($key, $properties2)) {
493 1
                $changedProperties[] = $key;
494 2
            } elseif ($properties1[$key] !== $properties2[$key]) {
495 2
                $changedProperties[] = $key;
496
            }
497
        }
498
499 140
        $platformOptions1 = $column1->getPlatformOptions();
500 140
        $platformOptions2 = $column2->getPlatformOptions();
501
502 140
        foreach (array_keys(array_intersect_key($platformOptions1, $platformOptions2)) as $key) {
503 2
            if ($properties1[$key] !== $properties2[$key]) {
504 2
                $changedProperties[] = $key;
505
            }
506
        }
507
508 140
        return array_unique($changedProperties);
509
    }
510
511
    /**
512
     * TODO: kill with fire on v3.0
513
     *
514
     * @deprecated
515
     */
516 140
    private function isALegacyJsonComparison(Types\Type $one, Types\Type $other) : bool
517
    {
518 140
        if ( ! $one instanceof Types\JsonType || ! $other instanceof Types\JsonType) {
519 138
            return false;
520
        }
521
522 2
        return ( ! $one instanceof Types\JsonArrayType && $other instanceof Types\JsonArrayType)
523 2
            || ( ! $other instanceof Types\JsonArrayType && $one instanceof Types\JsonArrayType);
524
    }
525
526
    /**
527
     * Finds the difference between the indexes $index1 and $index2.
528
     *
529
     * Compares $index1 with $index2 and returns $index2 if there are any
530
     * differences or false in case there are no differences.
531
     *
532
     * @param \Doctrine\DBAL\Schema\Index $index1
533
     * @param \Doctrine\DBAL\Schema\Index $index2
534
     *
535
     * @return boolean
536
     */
537 41
    public function diffIndex(Index $index1, Index $index2)
538
    {
539 41
        if ($index1->isFullfilledBy($index2) && $index2->isFullfilledBy($index1)) {
540 22
            return false;
541
        }
542
543 19
        return true;
544
    }
545
}
546