ChangeColumn.php 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. <?php
  2. namespace Illuminate\Database\Schema\Grammars;
  3. use RuntimeException;
  4. use Doctrine\DBAL\Types\Type;
  5. use Illuminate\Support\Fluent;
  6. use Doctrine\DBAL\Schema\Table;
  7. use Illuminate\Database\Connection;
  8. use Doctrine\DBAL\Schema\Comparator;
  9. use Illuminate\Database\Schema\Blueprint;
  10. use Doctrine\DBAL\Schema\AbstractSchemaManager as SchemaManager;
  11. class ChangeColumn
  12. {
  13. /**
  14. * Compile a change column command into a series of SQL statements.
  15. *
  16. * @param \Illuminate\Database\Schema\Grammars\Grammar $grammar
  17. * @param \Illuminate\Database\Schema\Blueprint $blueprint
  18. * @param \Illuminate\Support\Fluent $command
  19. * @param \Illuminate\Database\Connection $connection
  20. * @return array
  21. *
  22. * @throws \RuntimeException
  23. */
  24. public static function compile($grammar, Blueprint $blueprint, Fluent $command, Connection $connection)
  25. {
  26. if (! $connection->isDoctrineAvailable()) {
  27. throw new RuntimeException(sprintf(
  28. 'Changing columns for table "%s" requires Doctrine DBAL; install "doctrine/dbal".',
  29. $blueprint->getTable()
  30. ));
  31. }
  32. $tableDiff = static::getChangedDiff(
  33. $grammar, $blueprint, $schema = $connection->getDoctrineSchemaManager()
  34. );
  35. if ($tableDiff !== false) {
  36. return (array) $schema->getDatabasePlatform()->getAlterTableSQL($tableDiff);
  37. }
  38. return [];
  39. }
  40. /**
  41. * Get the Doctrine table difference for the given changes.
  42. *
  43. * @param \Illuminate\Database\Schema\Grammars\Grammar $grammar
  44. * @param \Illuminate\Database\Schema\Blueprint $blueprint
  45. * @param \Doctrine\DBAL\Schema\AbstractSchemaManager $schema
  46. * @return \Doctrine\DBAL\Schema\TableDiff|bool
  47. */
  48. protected static function getChangedDiff($grammar, Blueprint $blueprint, SchemaManager $schema)
  49. {
  50. $current = $schema->listTableDetails($grammar->getTablePrefix().$blueprint->getTable());
  51. return (new Comparator)->diffTable(
  52. $current, static::getTableWithColumnChanges($blueprint, $current)
  53. );
  54. }
  55. /**
  56. * Get a copy of the given Doctrine table after making the column changes.
  57. *
  58. * @param \Illuminate\Database\Schema\Blueprint $blueprint
  59. * @param \Doctrine\DBAL\Schema\Table $table
  60. * @return \Doctrine\DBAL\Schema\Table
  61. */
  62. protected static function getTableWithColumnChanges(Blueprint $blueprint, Table $table)
  63. {
  64. $table = clone $table;
  65. foreach ($blueprint->getChangedColumns() as $fluent) {
  66. $column = static::getDoctrineColumn($table, $fluent);
  67. // Here we will spin through each fluent column definition and map it to the proper
  68. // Doctrine column definitions - which is necessary because Laravel and Doctrine
  69. // use some different terminology for various column attributes on the tables.
  70. foreach ($fluent->getAttributes() as $key => $value) {
  71. if (! is_null($option = static::mapFluentOptionToDoctrine($key))) {
  72. if (method_exists($column, $method = 'set'.ucfirst($option))) {
  73. $column->{$method}(static::mapFluentValueToDoctrine($option, $value));
  74. continue;
  75. }
  76. $column->setCustomSchemaOption($option, static::mapFluentValueToDoctrine($option, $value));
  77. }
  78. }
  79. }
  80. return $table;
  81. }
  82. /**
  83. * Get the Doctrine column instance for a column change.
  84. *
  85. * @param \Doctrine\DBAL\Schema\Table $table
  86. * @param \Illuminate\Support\Fluent $fluent
  87. * @return \Doctrine\DBAL\Schema\Column
  88. */
  89. protected static function getDoctrineColumn(Table $table, Fluent $fluent)
  90. {
  91. return $table->changeColumn(
  92. $fluent['name'], static::getDoctrineColumnChangeOptions($fluent)
  93. )->getColumn($fluent['name']);
  94. }
  95. /**
  96. * Get the Doctrine column change options.
  97. *
  98. * @param \Illuminate\Support\Fluent $fluent
  99. * @return array
  100. */
  101. protected static function getDoctrineColumnChangeOptions(Fluent $fluent)
  102. {
  103. $options = ['type' => static::getDoctrineColumnType($fluent['type'])];
  104. if (in_array($fluent['type'], ['text', 'mediumText', 'longText'])) {
  105. $options['length'] = static::calculateDoctrineTextLength($fluent['type']);
  106. }
  107. if (in_array($fluent['type'], ['json', 'binary'])) {
  108. $options['customSchemaOptions'] = [
  109. 'collation' => '',
  110. ];
  111. }
  112. return $options;
  113. }
  114. /**
  115. * Get the doctrine column type.
  116. *
  117. * @param string $type
  118. * @return \Doctrine\DBAL\Types\Type
  119. */
  120. protected static function getDoctrineColumnType($type)
  121. {
  122. $type = strtolower($type);
  123. switch ($type) {
  124. case 'biginteger':
  125. $type = 'bigint';
  126. break;
  127. case 'smallinteger':
  128. $type = 'smallint';
  129. break;
  130. case 'mediumtext':
  131. case 'longtext':
  132. $type = 'text';
  133. break;
  134. case 'binary':
  135. $type = 'blob';
  136. break;
  137. }
  138. return Type::getType($type);
  139. }
  140. /**
  141. * Calculate the proper column length to force the Doctrine text type.
  142. *
  143. * @param string $type
  144. * @return int
  145. */
  146. protected static function calculateDoctrineTextLength($type)
  147. {
  148. switch ($type) {
  149. case 'mediumText':
  150. return 65535 + 1;
  151. case 'longText':
  152. return 16777215 + 1;
  153. default:
  154. return 255 + 1;
  155. }
  156. }
  157. /**
  158. * Get the matching Doctrine option for a given Fluent attribute name.
  159. *
  160. * @param string $attribute
  161. * @return string|null
  162. */
  163. protected static function mapFluentOptionToDoctrine($attribute)
  164. {
  165. switch ($attribute) {
  166. case 'type':
  167. case 'name':
  168. return;
  169. case 'nullable':
  170. return 'notnull';
  171. case 'total':
  172. return 'precision';
  173. case 'places':
  174. return 'scale';
  175. default:
  176. return $attribute;
  177. }
  178. }
  179. /**
  180. * Get the matching Doctrine value for a given Fluent attribute.
  181. *
  182. * @param string $option
  183. * @param mixed $value
  184. * @return mixed
  185. */
  186. protected static function mapFluentValueToDoctrine($option, $value)
  187. {
  188. return $option === 'notnull' ? ! $value : $value;
  189. }
  190. }