SQL.php 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  1. <?php
  2. namespace catcher\generate\factory;
  3. use catcher\exceptions\FailedException;
  4. use think\facade\Db;
  5. class SQL extends Factory
  6. {
  7. protected $index = '';
  8. public function done($params)
  9. {
  10. Db::execute($this->createSQL($params));
  11. // 判断表是否创建成功
  12. if (!$this->hasTableExists($params['table'])) {
  13. throw new FailedException(sprintf('create table [%s] failed', $params['table']));
  14. }
  15. return $params['table'];
  16. }
  17. /**
  18. * create table sql
  19. *
  20. * @time 2020年04月28日
  21. * @param $params
  22. * @return string
  23. */
  24. protected function createSQL($params)
  25. {
  26. if (!$params['table'] ?? false) {
  27. throw new FailedException('table name has lost~');
  28. }
  29. if ($this->hasTableExists($params['table'])) {
  30. throw new FailedException(sprintf('table [%s] has existed', $params['table']));
  31. }
  32. $extra = $params['extra'];
  33. // 主键
  34. $createSql = $this->primaryKey($extra['primary_key']);
  35. // 字段
  36. $ifHaveNotFields = true;
  37. foreach ($params['sql'] as $sql) {
  38. if (!$sql['field'] || !$sql['type']) {
  39. continue;
  40. }
  41. $ifHaveNotFields = false;
  42. $createSql .= $this->parseSQL($sql);
  43. }
  44. // 如果没有设置数据库字段
  45. if ($ifHaveNotFields) {
  46. throw new FailedException('Do you have set mysql fields?');
  47. }
  48. // 创建人
  49. if ($extra['creator_id'] ?? false) {
  50. $createSql .= $this->parseCreatorId();
  51. }
  52. // 创建时间
  53. if ($extra['created_at'] ?? false) {
  54. $createSql .= $this->parseCreatedAt();
  55. }
  56. // 软删除
  57. if ($extra['soft_delete'] ?? false) {
  58. $createSql .= $this->parseDeletedAt();
  59. }
  60. // 索引
  61. if ($this->index) {
  62. $createSql .= $this->index;
  63. }
  64. $createSql = rtrim($createSql, ',' . PHP_EOL);
  65. // 创建表 SQL
  66. return $this->createTable($params['table'], $createSql, $extra['engine'], 'utf8mb4', $extra['comment']);
  67. }
  68. /**
  69. * parse sql
  70. *
  71. * @time 2020年04月27日
  72. * @param $sql
  73. * @return string
  74. */
  75. protected function parseSQL($sql)
  76. {
  77. // 解析索引
  78. if ($sql['index']) {
  79. $this->parseIndex($sql['index'], $sql['field']);
  80. }
  81. // 字段
  82. $_sql[] = sprintf('`%s`', $sql['field']);
  83. // 类型
  84. if($sql['type'] =='double'){
  85. $_sql[] = $sql['type'] . ($sql['length'] ? sprintf('(%s,6)', $sql['length']) : '');
  86. }elseif($sql['type'] =='float'){
  87. $_sql[] = $sql['type'] . ($sql['length'] ? sprintf('(%s,4)', $sql['length']) : '');
  88. }else{
  89. $_sql[] = $sql['type'] . ($sql['length'] ? sprintf('(%s)', $sql['length']) : '');
  90. }
  91. if ($sql['unsigned']) {
  92. $_sql[] = 'unsigned';
  93. }
  94. // 默认值
  95. $default = trim(trim($sql['default'], '\''));
  96. if (!$sql['nullable']) {
  97. $_sql[] = 'not null';
  98. if ($default == '' || $default === '') {
  99. if (!$this->doNotNeedDefaultValueType($sql['type'])) {
  100. $_sql[] = ' default \'\'';
  101. }
  102. } else {
  103. if (strpos('int', $sql['type']) === false) {
  104. $_sql[] = ' default ' . (int)$default ;
  105. } else {
  106. $_sql[] = ' default ' . $default;
  107. }
  108. }
  109. }
  110. // 字段注释
  111. $_sql[] = $sql['comment'] ? sprintf('comment \'%s\'', $sql['comment']) : '';
  112. return implode(' ', $_sql) . ','. PHP_EOL;
  113. }
  114. /**
  115. * parse primary key
  116. *
  117. * @time 2020年04月27日
  118. * @param $id
  119. * @return string
  120. */
  121. protected function primaryKey($id)
  122. {
  123. return sprintf('`%s`', $id) . ' int unsigned not null auto_increment primary key,'. PHP_EOL;
  124. }
  125. /**
  126. * parse created_at & updated_at
  127. *
  128. * @time 2020年04月27日
  129. * @return string
  130. */
  131. protected function parseCreatedAt()
  132. {
  133. return sprintf('`created_at` int unsigned not null default 0 comment \'%s\',', '创建时间') . PHP_EOL .
  134. sprintf('`updated_at` int unsigned not null default 0 comment \'%s\',', '更新时间') . PHP_EOL;
  135. }
  136. /**
  137. * parse deleted_at
  138. *
  139. * @time 2020年04月27日
  140. * @return string
  141. */
  142. protected function parseDeletedAt()
  143. {
  144. return sprintf('`deleted_at` int unsigned not null default 0 comment \'%s\',', '软删除') . PHP_EOL;
  145. }
  146. /**
  147. * parse creator id
  148. *
  149. * @time 2020年07月01日
  150. * @return string
  151. */
  152. protected function parseCreatorId()
  153. {
  154. return sprintf('`creator_id` int unsigned not null default 0 comment \'%s\',', '创建人ID') . PHP_EOL;
  155. }
  156. /**
  157. * created table
  158. *
  159. * @time 2020年04月27日
  160. * @param $table
  161. * @param $sql
  162. * @param string $engine
  163. * @param string $charset
  164. * @param string $comment
  165. * @return string
  166. */
  167. protected function createTable($table, $sql, $engine='InnoDB', $charset = 'utf8mb4', $comment = '')
  168. {
  169. return sprintf('create table `%s`(' . PHP_EOL.
  170. '%s)'.PHP_EOL .
  171. 'engine=%s default charset=%s comment=\'%s\'', $table, $sql, $engine, $charset, $comment);
  172. }
  173. /**
  174. * parse index
  175. *
  176. * @time 2020年04月27日
  177. * @param $index
  178. * @param $field
  179. * @return void
  180. */
  181. protected function parseIndex($index, $field)
  182. {
  183. if ($index == 'unique') {
  184. $this->index .= "unique index unique_$field($field)," . PHP_EOL;
  185. } elseif ($index == 'index') {
  186. $this->index .= "index($field),". PHP_EOL;
  187. } elseif ($index == 'fulltext') {
  188. $this->index .= "fulltext key fulltext_$field($field)," . PHP_EOL;
  189. } elseif ($index == 'spatial') {
  190. $this->index .= "spatial index spatial_$field($field),". PHP_EOL;
  191. }
  192. }
  193. /**
  194. * 不需要默认值
  195. *
  196. * @param string $type
  197. * @time 2020年10月23日
  198. * @return bool
  199. */
  200. protected function doNotNeedDefaultValueType(string $type)
  201. {
  202. return in_array($type, [
  203. 'blob', 'text', 'geometry', 'json',
  204. 'tinytext', 'mediumtext', 'longtext',
  205. 'tinyblob', 'mediumblob', 'longblob'
  206. ]);
  207. }
  208. }