You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

CreateTableBuilder.java 9.3KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. /*
  2. * SonarQube
  3. * Copyright (C) 2009-2023 SonarSource SA
  4. * mailto:info AT sonarsource DOT com
  5. *
  6. * This program is free software; you can redistribute it and/or
  7. * modify it under the terms of the GNU Lesser General Public
  8. * License as published by the Free Software Foundation; either
  9. * version 3 of the License, or (at your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  14. * Lesser General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU Lesser General Public License
  17. * along with this program; if not, write to the Free Software Foundation,
  18. * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  19. */
  20. package org.sonar.server.platform.db.migration.sql;
  21. import com.google.common.collect.HashMultimap;
  22. import com.google.common.collect.Multimap;
  23. import java.util.ArrayList;
  24. import java.util.Arrays;
  25. import java.util.Collection;
  26. import java.util.Iterator;
  27. import java.util.List;
  28. import java.util.Locale;
  29. import java.util.stream.Stream;
  30. import javax.annotation.CheckForNull;
  31. import org.sonar.db.dialect.Dialect;
  32. import org.sonar.db.dialect.H2;
  33. import org.sonar.db.dialect.MsSql;
  34. import org.sonar.db.dialect.Oracle;
  35. import org.sonar.db.dialect.PostgreSql;
  36. import org.sonar.server.platform.db.migration.def.BigIntegerColumnDef;
  37. import org.sonar.server.platform.db.migration.def.ColumnDef;
  38. import org.sonar.server.platform.db.migration.def.IntegerColumnDef;
  39. import static com.google.common.base.Preconditions.checkArgument;
  40. import static com.google.common.base.Preconditions.checkState;
  41. import static java.lang.String.format;
  42. import static java.util.Objects.requireNonNull;
  43. import static java.util.stream.Stream.of;
  44. import static org.sonar.server.platform.db.migration.def.Validations.validateConstraintName;
  45. import static org.sonar.server.platform.db.migration.def.Validations.validateTableName;
  46. public class CreateTableBuilder {
  47. public static final String PRIMARY_KEY_PREFIX = "pk_";
  48. private final Dialect dialect;
  49. private final String tableName;
  50. private final List<ColumnDef> columnDefs = new ArrayList<>();
  51. private final List<ColumnDef> pkColumnDefs = new ArrayList<>(2);
  52. private final Multimap<ColumnDef, ColumnFlag> flagsByColumn = HashMultimap.create(1, 1);
  53. @CheckForNull
  54. private String pkConstraintName;
  55. public CreateTableBuilder(Dialect dialect, String tableName) {
  56. this.dialect = requireNonNull(dialect, "dialect can't be null");
  57. this.tableName = validateTableName(tableName);
  58. }
  59. public List<String> build() {
  60. checkState(!columnDefs.isEmpty() || !pkColumnDefs.isEmpty(), "at least one column must be specified");
  61. return Stream.concat(of(createTableStatement()), createOracleAutoIncrementStatements())
  62. .toList();
  63. }
  64. public CreateTableBuilder addColumn(ColumnDef columnDef) {
  65. columnDefs.add(requireNonNull(columnDef, "column def can't be null"));
  66. return this;
  67. }
  68. public CreateTableBuilder addPkColumn(ColumnDef columnDef, ColumnFlag... flags) {
  69. pkColumnDefs.add(requireNonNull(columnDef, "column def can't be null"));
  70. addFlags(columnDef, flags);
  71. return this;
  72. }
  73. private void addFlags(ColumnDef columnDef, ColumnFlag[] flags) {
  74. Arrays.stream(flags)
  75. .forEach(flag -> {
  76. requireNonNull(flag, "flag can't be null");
  77. if (flag == ColumnFlag.AUTO_INCREMENT) {
  78. validateColumnDefForAutoIncrement(columnDef);
  79. }
  80. flagsByColumn.put(columnDef, flag);
  81. });
  82. }
  83. private void validateColumnDefForAutoIncrement(ColumnDef columnDef) {
  84. checkArgument("id".equals(columnDef.getName()),
  85. "Auto increment column name must be id");
  86. checkArgument(columnDef instanceof BigIntegerColumnDef
  87. || columnDef instanceof IntegerColumnDef,
  88. "Auto increment column must either be BigInteger or Integer");
  89. checkArgument(!columnDef.isNullable(),
  90. "Auto increment column can't be nullable");
  91. checkState(pkColumnDefs.stream().noneMatch(this::isAutoIncrement),
  92. "There can't be more than one auto increment column");
  93. }
  94. public CreateTableBuilder withPkConstraintName(String pkConstraintName) {
  95. this.pkConstraintName = validateConstraintName(pkConstraintName);
  96. return this;
  97. }
  98. private String createTableStatement() {
  99. StringBuilder res = new StringBuilder("CREATE TABLE ");
  100. res.append(tableName);
  101. res.append(" (");
  102. appendPkColumns(res);
  103. appendColumns(res, dialect, columnDefs);
  104. appendPkConstraint(res);
  105. res.append(')');
  106. return res.toString();
  107. }
  108. private void appendPkColumns(StringBuilder res) {
  109. appendColumns(res, dialect, pkColumnDefs);
  110. if (!pkColumnDefs.isEmpty() && !columnDefs.isEmpty()) {
  111. res.append(',');
  112. }
  113. }
  114. private void appendColumns(StringBuilder res, Dialect dialect, List<ColumnDef> columnDefs) {
  115. if (columnDefs.isEmpty()) {
  116. return;
  117. }
  118. Iterator<ColumnDef> columnDefIterator = columnDefs.iterator();
  119. while (columnDefIterator.hasNext()) {
  120. ColumnDef columnDef = columnDefIterator.next();
  121. res.append(columnDef.getName());
  122. res.append(' ');
  123. appendDataType(res, dialect, columnDef);
  124. appendDefaultValue(res, columnDef);
  125. appendNullConstraint(res, columnDef);
  126. appendColumnFlags(res, dialect, columnDef);
  127. if (columnDefIterator.hasNext()) {
  128. res.append(',');
  129. }
  130. }
  131. }
  132. private void appendDataType(StringBuilder res, Dialect dialect, ColumnDef columnDef) {
  133. if (PostgreSql.ID.equals(dialect.getId()) && isAutoIncrement(columnDef)) {
  134. if (columnDef instanceof BigIntegerColumnDef) {
  135. res.append("BIGSERIAL");
  136. } else if (columnDef instanceof IntegerColumnDef) {
  137. res.append("SERIAL");
  138. } else {
  139. throw new IllegalStateException("Column with autoincrement is neither BigInteger nor Integer");
  140. }
  141. } else {
  142. res.append(columnDef.generateSqlType(dialect));
  143. }
  144. }
  145. private boolean isAutoIncrement(ColumnDef columnDef) {
  146. Collection<ColumnFlag> columnFlags = this.flagsByColumn.get(columnDef);
  147. return columnFlags != null && columnFlags.contains(ColumnFlag.AUTO_INCREMENT);
  148. }
  149. private static void appendNullConstraint(StringBuilder res, ColumnDef columnDef) {
  150. if (columnDef.isNullable()) {
  151. res.append(" NULL");
  152. } else {
  153. res.append(" NOT NULL");
  154. }
  155. }
  156. private void appendDefaultValue(StringBuilder sql, ColumnDef columnDef) {
  157. Object defaultValue = columnDef.getDefaultValue();
  158. if (defaultValue != null) {
  159. sql.append(" DEFAULT ");
  160. if (defaultValue instanceof String) {
  161. sql.append(format("'%s'", defaultValue));
  162. } else if (defaultValue instanceof Boolean) {
  163. sql.append((boolean) defaultValue ? dialect.getTrueSqlValue() : dialect.getFalseSqlValue());
  164. } else {
  165. sql.append(defaultValue);
  166. }
  167. }
  168. }
  169. private void appendColumnFlags(StringBuilder res, Dialect dialect, ColumnDef columnDef) {
  170. Collection<ColumnFlag> columnFlags = this.flagsByColumn.get(columnDef);
  171. if (columnFlags != null && columnFlags.contains(ColumnFlag.AUTO_INCREMENT)) {
  172. switch (dialect.getId()) {
  173. case Oracle.ID:
  174. // no auto increment on Oracle, must use a sequence
  175. break;
  176. case PostgreSql.ID:
  177. // no specific clause on PostgreSQL but a specific type
  178. break;
  179. case MsSql.ID:
  180. res.append(" IDENTITY (1,1)");
  181. break;
  182. case H2.ID:
  183. res.append(" AUTO_INCREMENT (1,1)");
  184. break;
  185. default:
  186. throw new IllegalArgumentException("Unsupported dialect id " + dialect.getId());
  187. }
  188. }
  189. }
  190. private void appendPkConstraint(StringBuilder res) {
  191. if (pkColumnDefs.isEmpty()) {
  192. return;
  193. }
  194. res.append(", ");
  195. res.append("CONSTRAINT ");
  196. appendPkConstraintName(res);
  197. res.append(" PRIMARY KEY ");
  198. res.append('(');
  199. appendColumnNames(res, pkColumnDefs);
  200. res.append(')');
  201. }
  202. private void appendPkConstraintName(StringBuilder res) {
  203. if (pkConstraintName == null) {
  204. res.append(PRIMARY_KEY_PREFIX).append(tableName);
  205. } else {
  206. res.append(pkConstraintName.toLowerCase(Locale.ENGLISH));
  207. }
  208. }
  209. private static void appendColumnNames(StringBuilder res, List<ColumnDef> columnDefs) {
  210. Iterator<ColumnDef> columnDefIterator = columnDefs.iterator();
  211. while (columnDefIterator.hasNext()) {
  212. res.append(columnDefIterator.next().getName());
  213. if (columnDefIterator.hasNext()) {
  214. res.append(',');
  215. }
  216. }
  217. }
  218. private Stream<String> createOracleAutoIncrementStatements() {
  219. if (!Oracle.ID.equals(dialect.getId())) {
  220. return Stream.empty();
  221. }
  222. return pkColumnDefs.stream()
  223. .filter(this::isAutoIncrement)
  224. .flatMap(columnDef -> of(createSequenceFor(tableName), createOracleTriggerForTable(tableName)));
  225. }
  226. private static String createSequenceFor(String tableName) {
  227. return "CREATE SEQUENCE " + tableName + "_seq START WITH 1 INCREMENT BY 1";
  228. }
  229. static String createOracleTriggerForTable(String tableName) {
  230. return "CREATE OR REPLACE TRIGGER " + tableName + "_idt" +
  231. " BEFORE INSERT ON " + tableName +
  232. " FOR EACH ROW" +
  233. " BEGIN" +
  234. " IF :new.id IS null THEN" +
  235. " SELECT " + tableName + "_seq.nextval INTO :new.id FROM dual;" +
  236. " END IF;" +
  237. " END;";
  238. }
  239. public enum ColumnFlag {
  240. AUTO_INCREMENT
  241. }
  242. }