Passed
Pull Request — master (#2945)
by Dorian
13:10
created

Comparator::compareSchemas()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 5
Code Lines 2

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 3
CRAP Score 1

Importance

Changes 0
Metric Value
dl 0
loc 5
ccs 3
cts 3
cp 1
rs 9.4285
c 0
b 0
f 0
cc 1
eloc 2
nc 1
nop 2
crap 1
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], $toFkeys[$key2]);
287
                } else {
288 3
                    if (strtolower($constraint1->getName()) == strtolower($constraint2->getName())) {
289 2
                        $tableDifferences->changedForeignKeys[] = $constraint2;
290 2
                        $changes++;
291 11
                        unset($fromFkeys[$key1], $toFkeys[$key2]);
292
                    }
293
                }
294
            }
295
        }
296
297 123
        foreach ($fromFkeys as $constraint1) {
298 7
            $tableDifferences->removedForeignKeys[] = $constraint1;
299 7
            $changes++;
300
        }
301
302 123
        foreach ($toFkeys as $constraint2) {
303 2
            $tableDifferences->addedForeignKeys[] = $constraint2;
304 2
            $changes++;
305
        }
306
307 123
        return $changes ? $tableDifferences : false;
308
    }
309
310
    /**
311
     * Try to find columns that only changed their name, rename operations maybe cheaper than add/drop
312
     * however ambiguities between different possibilities should not lead to renaming at all.
313
     *
314
     * @param \Doctrine\DBAL\Schema\TableDiff $tableDifferences
315
     *
316
     * @return void
317
     */
318 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...
319
    {
320 123
        $renameCandidates = [];
321 123
        foreach ($tableDifferences->addedColumns as $addedColumnName => $addedColumn) {
322 29
            foreach ($tableDifferences->removedColumns as $removedColumn) {
323 22
                if (count($this->diffColumn($addedColumn, $removedColumn)) == 0) {
324 29
                    $renameCandidates[$addedColumn->getName()][] = [$removedColumn, $addedColumn, $addedColumnName];
325
                }
326
            }
327
        }
328
329 123
        foreach ($renameCandidates as $candidateColumns) {
330 22
            if (count($candidateColumns) == 1) {
331 21
                list($removedColumn, $addedColumn) = $candidateColumns[0];
332 21
                $removedColumnName = strtolower($removedColumn->getName());
333 21
                $addedColumnName = strtolower($addedColumn->getName());
334
335 21
                if ( ! isset($tableDifferences->renamedColumns[$removedColumnName])) {
336 21
                    $tableDifferences->renamedColumns[$removedColumnName] = $addedColumn;
337
                    unset(
338 21
                        $tableDifferences->addedColumns[$addedColumnName],
339 22
                        $tableDifferences->removedColumns[$removedColumnName]
340
                    );
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
                    unset(
381 2
                        $tableDifferences->addedIndexes[$addedIndexName],
382 3
                        $tableDifferences->removedIndexes[$removedIndexName]
383
                    );
384
                }
385
            }
386
        }
387 123
    }
388
389
    /**
390
     * @param \Doctrine\DBAL\Schema\ForeignKeyConstraint $key1
391
     * @param \Doctrine\DBAL\Schema\ForeignKeyConstraint $key2
392
     *
393
     * @return boolean
394
     */
395 8
    public function diffForeignKey(ForeignKeyConstraint $key1, ForeignKeyConstraint $key2)
396
    {
397 8
        if (array_map('strtolower', $key1->getUnquotedLocalColumns()) != array_map('strtolower', $key2->getUnquotedLocalColumns())) {
398 1
            return true;
399
        }
400
401 7
        if (array_map('strtolower', $key1->getUnquotedForeignColumns()) != array_map('strtolower', $key2->getUnquotedForeignColumns())) {
402
            return true;
403
        }
404
405 7
        if ($key1->getUnqualifiedForeignTableName() !== $key2->getUnqualifiedForeignTableName()) {
406 1
            return true;
407
        }
408
409 6
        if ($key1->onUpdate() != $key2->onUpdate()) {
410 1
            return true;
411
        }
412
413 5
        if ($key1->onDelete() != $key2->onDelete()) {
414
            return true;
415
        }
416
417 5
        return false;
418
    }
419
420
    /**
421
     * Returns the difference between the fields $field1 and $field2.
422
     *
423
     * If there are differences this method returns $field2, otherwise the
424
     * boolean false.
425
     *
426
     * @param \Doctrine\DBAL\Schema\Column $column1
427
     * @param \Doctrine\DBAL\Schema\Column $column2
428
     *
429
     * @return array
430
     */
431 140
    public function diffColumn(Column $column1, Column $column2)
432
    {
433 140
        $properties1 = $column1->toArray();
434 140
        $properties2 = $column2->toArray();
435
436 140
        $changedProperties = [];
437
438 140
        foreach (['type', 'notnull', 'unsigned', 'autoincrement'] as $property) {
439 140
            if ($properties1[$property] != $properties2[$property]) {
440 140
                $changedProperties[] = $property;
441
            }
442
        }
443
444
        // This is a very nasty hack to make comparator work with the legacy json_array type, which should be killed in v3
445 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

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