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.

SQLDialectSQLite.java 5.3KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. /*
  2. * Copyright 2014 James Moger.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package com.iciql;
  17. import java.sql.Date;
  18. import java.sql.ResultSet;
  19. import java.sql.SQLException;
  20. import java.sql.Time;
  21. import java.sql.Timestamp;
  22. import com.iciql.Iciql.DataTypeAdapter;
  23. import com.iciql.TableDefinition.FieldDefinition;
  24. import com.iciql.TableDefinition.IndexDefinition;
  25. import com.iciql.util.IciqlLogger;
  26. import com.iciql.util.StatementBuilder;
  27. /**
  28. * SQLite database dialect.
  29. */
  30. public class SQLDialectSQLite extends SQLDialectDefault {
  31. @Override
  32. public boolean supportsSavePoints() {
  33. // SAVEPOINT support was added after the 3.8.7 release
  34. String [] chunks = productVersion.split("\\.");
  35. if (Integer.parseInt(chunks[0]) > 3) {
  36. return true;
  37. }
  38. float f = Float.parseFloat(chunks[1] + "." + chunks[2]);
  39. return (f > 8.7);
  40. }
  41. @Override
  42. protected <T> String prepareCreateTable(TableDefinition<T> def) {
  43. return "CREATE TABLE IF NOT EXISTS";
  44. }
  45. @Override
  46. protected <T> String prepareCreateView(TableDefinition<T> def) {
  47. return "CREATE VIEW IF NOT EXISTS";
  48. }
  49. @Override
  50. public String convertSqlType(String sqlType) {
  51. if (isIntegerType(sqlType)) {
  52. return "INTEGER";
  53. }
  54. return sqlType;
  55. }
  56. @Override
  57. protected boolean prepareColumnDefinition(StatementBuilder buff, String dataType,
  58. boolean isAutoIncrement, boolean isPrimaryKey) {
  59. String convertedType = convertSqlType(dataType);
  60. buff.append(convertedType);
  61. if (isPrimaryKey) {
  62. buff.append(" PRIMARY KEY");
  63. if (isAutoIncrement) {
  64. buff.append(" AUTOINCREMENT");
  65. }
  66. return true;
  67. }
  68. return false;
  69. }
  70. @Override
  71. public <T> void prepareDropView(SQLStatement stat, TableDefinition<T> def) {
  72. StatementBuilder buff = new StatementBuilder("DROP VIEW IF EXISTS "
  73. + prepareTableName(def.schemaName, def.tableName));
  74. stat.setSQL(buff.toString());
  75. return;
  76. }
  77. @Override
  78. public void prepareCreateIndex(SQLStatement stat, String schemaName, String tableName,
  79. IndexDefinition index) {
  80. StatementBuilder buff = new StatementBuilder();
  81. buff.append("CREATE ");
  82. switch (index.type) {
  83. case UNIQUE:
  84. buff.append("UNIQUE ");
  85. break;
  86. case UNIQUE_HASH:
  87. buff.append("UNIQUE ");
  88. break;
  89. default:
  90. IciqlLogger.warn("{0} does not support hash indexes", getClass().getSimpleName());
  91. }
  92. buff.append("INDEX IF NOT EXISTS ");
  93. buff.append(index.indexName);
  94. buff.append(" ON ");
  95. buff.append(tableName);
  96. buff.append("(");
  97. for (String col : index.columnNames) {
  98. buff.appendExceptFirst(", ");
  99. buff.append(prepareColumnName(col));
  100. }
  101. buff.append(") ");
  102. stat.setSQL(buff.toString().trim());
  103. }
  104. @Override
  105. public <T> void prepareMerge(SQLStatement stat, String schemaName, String tableName,
  106. TableDefinition<T> def, Object obj) {
  107. StatementBuilder buff = new StatementBuilder("INSERT OR REPLACE INTO ");
  108. buff.append(prepareTableName(schemaName, tableName)).append(" (");
  109. buff.resetCount();
  110. for (FieldDefinition field : def.fields) {
  111. buff.appendExceptFirst(", ");
  112. buff.append(field.columnName);
  113. }
  114. buff.append(") ");
  115. buff.resetCount();
  116. buff.append("VALUES (");
  117. for (FieldDefinition field : def.fields) {
  118. buff.appendExceptFirst(", ");
  119. buff.append('?');
  120. Object value = def.getValue(obj, field);
  121. Object parameter = serialize(value, field.typeAdapter);
  122. stat.addParameter(parameter);
  123. }
  124. buff.append(')');
  125. stat.setSQL(buff.toString());
  126. }
  127. @Override
  128. public Object deserialize(ResultSet rs, int columnIndex, Class<?> targetType, Class<? extends DataTypeAdapter<?>> typeAdapter) {
  129. try {
  130. return super.deserialize(rs, columnIndex, targetType, typeAdapter);
  131. } catch (IciqlException e) {
  132. if (typeAdapter == null && e.getMessage().startsWith("Can not convert")) {
  133. try {
  134. // give the SQLite JDBC driver an opportunity to deserialize DateTime objects
  135. if (Timestamp.class.equals(targetType)) {
  136. return rs.getTimestamp(columnIndex);
  137. } else if (Time.class.equals(targetType)) {
  138. return rs.getTime(columnIndex);
  139. } else if (Date.class.equals(targetType)) {
  140. return rs.getDate(columnIndex);
  141. } else if (java.util.Date.class.equals(targetType)) {
  142. Timestamp timestamp = rs.getTimestamp(columnIndex);
  143. return new java.util.Date(timestamp.getTime());
  144. }
  145. } catch (SQLException x) {
  146. throw new IciqlException(x, "Can not convert the value at column {0} to {1}",
  147. columnIndex, targetType.getName());
  148. }
  149. }
  150. // rethrow e
  151. throw e;
  152. }
  153. }
  154. @Override
  155. public String prepareStringParameter(Object o) {
  156. if (o instanceof Boolean) {
  157. // SQLite does not have an explicit BOOLEAN type
  158. Boolean bool = (Boolean) o;
  159. return bool ? "1" : "0";
  160. }
  161. return super.prepareStringParameter(o);
  162. }
  163. }