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.

SQLDialectDefault.java 20KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555
  1. /*
  2. * Copyright 2004-2011 H2 Group.
  3. * Copyright 2011 James Moger.
  4. * Copyright 2012 Frédéric Gaillard.
  5. *
  6. * Licensed under the Apache License, Version 2.0 (the "License");
  7. * you may not use this file except in compliance with the License.
  8. * You may obtain a copy of the License at
  9. *
  10. * http://www.apache.org/licenses/LICENSE-2.0
  11. *
  12. * Unless required by applicable law or agreed to in writing, software
  13. * distributed under the License is distributed on an "AS IS" BASIS,
  14. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  15. * See the License for the specific language governing permissions and
  16. * limitations under the License.
  17. */
  18. package com.iciql;
  19. import com.iciql.Iciql.ConstraintDeleteType;
  20. import com.iciql.Iciql.ConstraintUpdateType;
  21. import com.iciql.Iciql.DataTypeAdapter;
  22. import com.iciql.Iciql.Mode;
  23. import com.iciql.TableDefinition.ConstraintForeignKeyDefinition;
  24. import com.iciql.TableDefinition.ConstraintUniqueDefinition;
  25. import com.iciql.TableDefinition.FieldDefinition;
  26. import com.iciql.TableDefinition.IndexDefinition;
  27. import com.iciql.util.IciqlLogger;
  28. import com.iciql.util.StatementBuilder;
  29. import com.iciql.util.StringUtils;
  30. import com.iciql.util.Utils;
  31. import java.sql.Blob;
  32. import java.sql.Clob;
  33. import java.sql.Connection;
  34. import java.sql.DatabaseMetaData;
  35. import java.sql.ResultSet;
  36. import java.sql.SQLException;
  37. import java.text.MessageFormat;
  38. import java.text.SimpleDateFormat;
  39. import java.util.Map;
  40. import java.util.concurrent.ConcurrentHashMap;
  41. /**
  42. * Default implementation of an SQL dialect.
  43. */
  44. public class SQLDialectDefault implements SQLDialect {
  45. final String LITERAL = "'";
  46. protected float databaseVersion;
  47. protected int databaseMajorVersion;
  48. protected int databaseMinorVersion;
  49. protected String databaseName;
  50. protected String productVersion;
  51. protected Mode mode;
  52. protected Map<Class<? extends DataTypeAdapter<?>>, DataTypeAdapter<?>> typeAdapters;
  53. public SQLDialectDefault() {
  54. typeAdapters = new ConcurrentHashMap<Class<? extends DataTypeAdapter<?>>, DataTypeAdapter<?>>();
  55. }
  56. @Override
  57. public String toString() {
  58. return getClass().getName() + ": " + databaseName + " " + productVersion;
  59. }
  60. @Override
  61. public void configureDialect(Db db) {
  62. Connection conn = db.getConnection();
  63. DatabaseMetaData data = null;
  64. try {
  65. data = conn.getMetaData();
  66. databaseName = data.getDatabaseProductName();
  67. databaseMajorVersion = data.getDatabaseMajorVersion();
  68. databaseMinorVersion = data.getDatabaseMinorVersion();
  69. databaseVersion = Float.parseFloat(databaseMajorVersion + "."
  70. + databaseMinorVersion);
  71. productVersion = data.getDatabaseProductVersion();
  72. } catch (SQLException e) {
  73. throw new IciqlException(e, "failed to retrieve database metadata!");
  74. }
  75. mode = db.getMode();
  76. }
  77. @Override
  78. public boolean supportsSavePoints() {
  79. return true;
  80. }
  81. /**
  82. * Allows subclasses to change the type of a column for a CREATE statement.
  83. *
  84. * @param sqlType
  85. * @return the SQL type or a preferred alternative
  86. */
  87. @Override
  88. public String convertSqlType(String sqlType) {
  89. return sqlType;
  90. }
  91. @Override
  92. public Class<? extends java.util.Date> getDateTimeClass() {
  93. return java.util.Date.class;
  94. }
  95. @Override
  96. public String prepareTableName(String schemaName, String tableName) {
  97. if (StringUtils.isNullOrEmpty(schemaName)) {
  98. return tableName;
  99. }
  100. return schemaName + "." + tableName;
  101. }
  102. @Override
  103. public String prepareColumnName(String name) {
  104. return name;
  105. }
  106. @Override
  107. public String extractColumnName(String name) {
  108. return name.replace('\"', ' ').replace('\'', ' ').trim();
  109. }
  110. @Override
  111. public <T> void prepareDropTable(SQLStatement stat, TableDefinition<T> def) {
  112. StatementBuilder buff = new StatementBuilder("DROP TABLE IF EXISTS "
  113. + prepareTableName(def.schemaName, def.tableName));
  114. stat.setSQL(buff.toString());
  115. return;
  116. }
  117. protected <T> String prepareCreateTable(TableDefinition<T> def) {
  118. return "CREATE TABLE";
  119. }
  120. @Override
  121. public <T> void prepareCreateTable(SQLStatement stat, TableDefinition<T> def) {
  122. StatementBuilder buff = new StatementBuilder();
  123. buff.append(prepareCreateTable(def));
  124. buff.append(" ");
  125. buff.append(prepareTableName(def.schemaName, def.tableName)).append('(');
  126. boolean hasIdentityColumn = false;
  127. for (FieldDefinition field : def.fields) {
  128. buff.appendExceptFirst(", ");
  129. buff.append(prepareColumnName(field.columnName)).append(' ');
  130. String dataType = field.dataType;
  131. if (dataType.equals("VARCHAR")) {
  132. // check to see if we should use VARCHAR or CLOB
  133. if (field.length <= 0) {
  134. dataType = "CLOB";
  135. }
  136. buff.append(convertSqlType(dataType));
  137. if (field.length > 0) {
  138. buff.append('(').append(field.length).append(')');
  139. }
  140. } else if (dataType.equals("DECIMAL")) {
  141. // DECIMAL(precision,scale)
  142. buff.append(convertSqlType(dataType));
  143. if (field.length > 0) {
  144. buff.append('(').append(field.length);
  145. if (field.scale > 0) {
  146. buff.append(',').append(field.scale);
  147. }
  148. buff.append(')');
  149. }
  150. } else {
  151. // other
  152. hasIdentityColumn |= prepareColumnDefinition(buff, convertSqlType(dataType),
  153. field.isAutoIncrement, field.isPrimaryKey);
  154. }
  155. buff.append(prepareColumnConstraint(field.isAutoIncrement, field.isPrimaryKey, field.nullable, field.field.getType(),
  156. field.dataType, field.defaultValue));
  157. }
  158. // if table does not have identity column then specify primary key
  159. if (!hasIdentityColumn) {
  160. if (def.primaryKeyColumnNames != null && def.primaryKeyColumnNames.size() > 0) {
  161. buff.append(", PRIMARY KEY(");
  162. buff.resetCount();
  163. for (String n : def.primaryKeyColumnNames) {
  164. buff.appendExceptFirst(", ");
  165. buff.append(prepareColumnName(n));
  166. }
  167. buff.append(')');
  168. }
  169. }
  170. // create unique constraints
  171. if (def.constraintsUnique.size() > 0) {
  172. buff.append(", ");
  173. buff.resetCount();
  174. for (ConstraintUniqueDefinition constraint : def.constraintsUnique) {
  175. buff.append("CONSTRAINT ");
  176. buff.append(constraint.constraintName);
  177. buff.append(" UNIQUE ");
  178. buff.append(" (");
  179. for (String col : constraint.uniqueColumns) {
  180. buff.appendExceptFirst(", ");
  181. buff.append(prepareColumnName(col));
  182. }
  183. buff.append(") ");
  184. }
  185. }
  186. // create foreign key constraints
  187. if (def.constraintsForeignKey.size() > 0) {
  188. buff.append(", ");
  189. buff.resetCount();
  190. for (ConstraintForeignKeyDefinition constraint : def.constraintsForeignKey) {
  191. buff.appendExceptFirst(", ");
  192. buff.append(String.format("CONSTRAINT %s FOREIGN KEY(%s) REFERENCES %s(%s)",
  193. constraint.constraintName,
  194. constraint.foreignColumns.get(0),
  195. constraint.referenceTable,
  196. constraint.referenceColumns.get(0)));
  197. if (constraint.deleteType != ConstraintDeleteType.UNSET) {
  198. buff.append(" ON DELETE ");
  199. switch (constraint.deleteType) {
  200. case CASCADE:
  201. buff.append("CASCADE ");
  202. break;
  203. case RESTRICT:
  204. buff.append("RESTRICT ");
  205. break;
  206. case SET_NULL:
  207. buff.append("SET NULL ");
  208. break;
  209. case NO_ACTION:
  210. buff.append("NO ACTION ");
  211. break;
  212. case SET_DEFAULT:
  213. buff.append("SET DEFAULT ");
  214. break;
  215. }
  216. }
  217. if (constraint.updateType != ConstraintUpdateType.UNSET) {
  218. buff.append(" ON UPDATE ");
  219. switch (constraint.updateType) {
  220. case CASCADE:
  221. buff.append("CASCADE ");
  222. break;
  223. case RESTRICT:
  224. buff.append("RESTRICT ");
  225. break;
  226. case SET_NULL:
  227. buff.append("SET NULL ");
  228. break;
  229. case NO_ACTION:
  230. buff.append("NO ACTION ");
  231. break;
  232. case SET_DEFAULT:
  233. buff.append("SET DEFAULT ");
  234. break;
  235. }
  236. }
  237. switch (constraint.deferrabilityType) {
  238. case DEFERRABLE_INITIALLY_DEFERRED:
  239. buff.append("DEFERRABLE INITIALLY DEFERRED ");
  240. break;
  241. case DEFERRABLE_INITIALLY_IMMEDIATE:
  242. buff.append("DEFERRABLE INITIALLY IMMEDIATE ");
  243. break;
  244. case NOT_DEFERRABLE:
  245. buff.append("NOT DEFERRABLE ");
  246. break;
  247. case UNSET:
  248. break;
  249. }
  250. }
  251. }
  252. buff.append(')');
  253. stat.setSQL(buff.toString());
  254. }
  255. @Override
  256. public <T> void prepareDropView(SQLStatement stat, TableDefinition<T> def) {
  257. StatementBuilder buff = new StatementBuilder("DROP VIEW "
  258. + prepareTableName(def.schemaName, def.tableName));
  259. stat.setSQL(buff.toString());
  260. return;
  261. }
  262. protected <T> String prepareCreateView(TableDefinition<T> def) {
  263. return "CREATE VIEW";
  264. }
  265. @Override
  266. public <T> void prepareCreateView(SQLStatement stat, TableDefinition<T> def) {
  267. StatementBuilder buff = new StatementBuilder();
  268. buff.append(" FROM ");
  269. buff.append(prepareTableName(def.schemaName, def.viewTableName));
  270. StatementBuilder where = new StatementBuilder();
  271. for (FieldDefinition field : def.fields) {
  272. if (!StringUtils.isNullOrEmpty(field.constraint)) {
  273. where.appendExceptFirst(", ");
  274. String col = prepareColumnName(field.columnName);
  275. String constraint = field.constraint.replace("{0}", col).replace("this", col);
  276. where.append(constraint);
  277. }
  278. }
  279. if (where.length() > 0) {
  280. buff.append(" WHERE ");
  281. buff.append(where.toString());
  282. }
  283. prepareCreateView(stat, def, buff.toString());
  284. }
  285. @Override
  286. public <T> void prepareCreateView(SQLStatement stat, TableDefinition<T> def, String fromWhere) {
  287. StatementBuilder buff = new StatementBuilder();
  288. buff.append(prepareCreateView(def));
  289. buff.append(" ");
  290. buff.append(prepareTableName(def.schemaName, def.tableName));
  291. buff.append(" AS SELECT ");
  292. for (FieldDefinition field : def.fields) {
  293. buff.appendExceptFirst(", ");
  294. buff.append(prepareColumnName(field.columnName));
  295. }
  296. buff.append(fromWhere);
  297. stat.setSQL(buff.toString());
  298. }
  299. protected boolean isIntegerType(String dataType) {
  300. if ("INT".equals(dataType)) {
  301. return true;
  302. } else if ("INTEGER".equals(dataType)) {
  303. return true;
  304. } else if ("TINYINT".equals(dataType)) {
  305. return true;
  306. } else if ("SMALLINT".equals(dataType)) {
  307. return true;
  308. } else if ("MEDIUMINT".equals(dataType)) {
  309. return true;
  310. } else if ("BIGINT".equals(dataType)) {
  311. return true;
  312. }
  313. return false;
  314. }
  315. protected boolean prepareColumnDefinition(StatementBuilder buff, String dataType,
  316. boolean isAutoIncrement, boolean isPrimaryKey) {
  317. buff.append(dataType);
  318. if (isAutoIncrement) {
  319. buff.append(" AUTO_INCREMENT");
  320. }
  321. return false;
  322. }
  323. @Override
  324. public String prepareColumnConstraint(boolean isAutoIncrement, boolean isPrimaryKey, boolean nullable, Class<?> fieldType, String dataType, String defaultValue) {
  325. StringBuilder sb = new StringBuilder();
  326. if (!isAutoIncrement && !isPrimaryKey) {
  327. if (nullable && (defaultValue == null || ((dataType.equals("TIMESTAMP") || dataType.equals("DATETIME")) && StringUtils.isNullOrEmpty(defaultValue)))) {
  328. sb.append(" NULL");
  329. } else if (!StringUtils.isNullOrEmpty(defaultValue)) {
  330. if (ModelUtils.isProperlyFormattedDefaultValue(defaultValue)
  331. && ModelUtils.isValidDefaultValue(fieldType, defaultValue)) {
  332. sb.append(" DEFAULT ").append(defaultValue);
  333. }
  334. }
  335. }
  336. if (!nullable) {
  337. sb.append(" NOT NULL");
  338. }
  339. return sb.toString();
  340. }
  341. @Override
  342. public void prepareCreateIndex(SQLStatement stat, String schemaName, String tableName,
  343. IndexDefinition index) {
  344. StatementBuilder buff = new StatementBuilder();
  345. buff.append("CREATE ");
  346. switch (index.type) {
  347. case UNIQUE:
  348. buff.append("UNIQUE ");
  349. break;
  350. case UNIQUE_HASH:
  351. buff.append("UNIQUE ");
  352. break;
  353. default:
  354. IciqlLogger.warn("{0} does not support hash indexes", getClass().getSimpleName());
  355. }
  356. buff.append("INDEX ");
  357. buff.append(index.indexName);
  358. buff.append(" ON ");
  359. // FIXME maybe we can use schemaName ?
  360. // buff.append(prepareTableName(schemaName, tableName));
  361. buff.append(tableName);
  362. buff.append("(");
  363. for (String col : index.columnNames) {
  364. buff.appendExceptFirst(", ");
  365. buff.append(prepareColumnName(col));
  366. }
  367. buff.append(") ");
  368. stat.setSQL(buff.toString().trim());
  369. }
  370. /**
  371. * PostgreSQL and Derby do not support the SQL2003 MERGE syntax, but we can
  372. * use a trick to insert a row if it does not exist and call update() in
  373. * Db.merge() if the affected row count is 0.
  374. * <p>
  375. * Databases that do support a MERGE syntax should override this method.
  376. * <p>
  377. * http://stackoverflow.com/questions/407688
  378. */
  379. @Override
  380. public <T> void prepareMerge(SQLStatement stat, String schemaName, String tableName,
  381. TableDefinition<T> def, Object obj) {
  382. StatementBuilder buff = new StatementBuilder("INSERT INTO ");
  383. buff.append(prepareTableName(schemaName, tableName));
  384. buff.append(" (");
  385. buff.resetCount();
  386. for (FieldDefinition field : def.fields) {
  387. buff.appendExceptFirst(", ");
  388. buff.append(prepareColumnName(field.columnName));
  389. }
  390. buff.append(") (SELECT ");
  391. buff.resetCount();
  392. for (FieldDefinition field : def.fields) {
  393. buff.appendExceptFirst(", ");
  394. buff.append('?');
  395. Object value = def.getValue(obj, field);
  396. Object parameter = serialize(value, field.typeAdapter);
  397. stat.addParameter(parameter);
  398. }
  399. buff.append(" FROM ");
  400. buff.append(prepareTableName(schemaName, tableName));
  401. buff.append(" WHERE ");
  402. buff.resetCount();
  403. for (FieldDefinition field : def.fields) {
  404. if (field.isPrimaryKey) {
  405. buff.appendExceptFirst(" AND ");
  406. buff.append(MessageFormat.format("{0} = ?", prepareColumnName(field.columnName)));
  407. Object value = def.getValue(obj, field);
  408. Object parameter = serialize(value, field.typeAdapter);
  409. stat.addParameter(parameter);
  410. }
  411. }
  412. buff.append(" HAVING count(*)=0)");
  413. stat.setSQL(buff.toString());
  414. }
  415. @Override
  416. public void appendLimitOffset(SQLStatement stat, long limit, long offset) {
  417. if (limit > 0) {
  418. stat.appendSQL(" LIMIT " + limit);
  419. }
  420. if (offset > 0) {
  421. stat.appendSQL(" OFFSET " + offset);
  422. }
  423. }
  424. @Override
  425. public void registerAdapter(DataTypeAdapter<?> typeAdapter) {
  426. typeAdapters.put((Class<? extends DataTypeAdapter<?>>) typeAdapter.getClass(), typeAdapter);
  427. }
  428. @Override
  429. public DataTypeAdapter<?> getAdapter(Class<? extends DataTypeAdapter<?>> typeAdapter) {
  430. DataTypeAdapter<?> dta = typeAdapters.get(typeAdapter);
  431. if (dta == null) {
  432. dta = Utils.newObject(typeAdapter);
  433. typeAdapters.put(typeAdapter, dta);
  434. }
  435. dta.setMode(mode);
  436. return dta;
  437. }
  438. @SuppressWarnings("unchecked")
  439. @Override
  440. public <T> Object serialize(T value, Class<? extends DataTypeAdapter<?>> typeAdapter) {
  441. if (typeAdapter == null) {
  442. // pass-through
  443. return value;
  444. }
  445. DataTypeAdapter<T> dta = (DataTypeAdapter<T>) getAdapter(typeAdapter);
  446. return dta.serialize(value);
  447. }
  448. @Override
  449. public Object deserialize(ResultSet rs, int columnIndex, Class<?> targetType, Class<? extends DataTypeAdapter<?>> typeAdapter) {
  450. Object value = null;
  451. try {
  452. if (typeAdapter == null) {
  453. // standard object deserialization
  454. Object o = rs.getObject(columnIndex);
  455. if (o == null) {
  456. // no-op
  457. value = null;
  458. } else if (Clob.class.isAssignableFrom(o.getClass())) {
  459. value = Utils.convert(o, String.class);
  460. } else if (Blob.class.isAssignableFrom(o.getClass())) {
  461. value = Utils.convert(o, byte[].class);
  462. } else {
  463. value = Utils.convert(o, targetType);
  464. }
  465. } else {
  466. // custom object deserialization with a DataTypeAdapter
  467. DataTypeAdapter<?> dta = getAdapter(typeAdapter);
  468. Object object = rs.getObject(columnIndex);
  469. value = dta.deserialize(object);
  470. }
  471. } catch (SQLException e) {
  472. throw new IciqlException(e, "Can not convert the value at column {0} to {1}",
  473. columnIndex, targetType.getName());
  474. }
  475. return value;
  476. }
  477. @Override
  478. public String prepareStringParameter(Object o) {
  479. if (o instanceof String) {
  480. return LITERAL + o.toString().replace(LITERAL, "''") + LITERAL;
  481. } else if (o instanceof Character) {
  482. return LITERAL + o.toString() + LITERAL;
  483. } else if (o instanceof java.sql.Time) {
  484. return LITERAL + new SimpleDateFormat("HH:mm:ss").format(o) + LITERAL;
  485. } else if (o instanceof java.sql.Date) {
  486. return LITERAL + new SimpleDateFormat("yyyy-MM-dd").format(o) + LITERAL;
  487. } else if (o instanceof java.util.Date) {
  488. return LITERAL + new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(o) + LITERAL;
  489. }
  490. return o.toString();
  491. }
  492. @Override
  493. public <T, A> void prepareBitwiseAnd(SQLStatement stat, Query<T> query, A x, A y) {
  494. query.appendSQL(stat, null, x);
  495. stat.appendSQL(" & ");
  496. query.appendSQL(stat, x, y);
  497. }
  498. @Override
  499. public <T, A> void prepareBitwiseXor(SQLStatement stat, Query<T> query, A x, A y) {
  500. query.appendSQL(stat, null, x);
  501. stat.appendSQL(" ^ ");
  502. query.appendSQL(stat, x, y);
  503. }
  504. }