/sql-processor/src/main/java/org/sqlproc/engine/impl/SqlDefaultFeatures.java

http://github.com/hudec/sql-processor · Java · 336 lines · 116 code · 10 blank · 210 comment · 6 complexity · bb7687ebbfbe1f9e6b39003f9f1770e0 MD5 · raw file

  1. package org.sqlproc.engine.impl;
  2. import static org.sqlproc.engine.SqlFeature.CASSANDRA;
  3. import static org.sqlproc.engine.SqlFeature.DB2;
  4. import static org.sqlproc.engine.SqlFeature.H2;
  5. import static org.sqlproc.engine.SqlFeature.HSQLDB;
  6. import static org.sqlproc.engine.SqlFeature.IDSEL;
  7. import static org.sqlproc.engine.SqlFeature.IDSEL_Integer;
  8. import static org.sqlproc.engine.SqlFeature.IDSEL_Long;
  9. import static org.sqlproc.engine.SqlFeature.INFORMIX;
  10. import static org.sqlproc.engine.SqlFeature.JDBC;
  11. import static org.sqlproc.engine.SqlFeature.LIKE_STRING;
  12. import static org.sqlproc.engine.SqlFeature.LIMIT_FROM_TO;
  13. import static org.sqlproc.engine.SqlFeature.LIMIT_FROM_TO_ORDERED;
  14. import static org.sqlproc.engine.SqlFeature.LIMIT_TO;
  15. import static org.sqlproc.engine.SqlFeature.LIMIT_TO_ORDERED;
  16. import static org.sqlproc.engine.SqlFeature.LOG_SQL_COMMAND_FOR_EXCEPTION;
  17. import static org.sqlproc.engine.SqlFeature.METHODS_ENUM_IN;
  18. import static org.sqlproc.engine.SqlFeature.METHODS_ENUM_OUT;
  19. import static org.sqlproc.engine.SqlFeature.MSSQL;
  20. import static org.sqlproc.engine.SqlFeature.MYSQL;
  21. import static org.sqlproc.engine.SqlFeature.OPERATOR_ATTRIBUTE;
  22. import static org.sqlproc.engine.SqlFeature.OPERATOR_ATTRIBUTE_IN_MAP;
  23. import static org.sqlproc.engine.SqlFeature.ORACLE;
  24. import static org.sqlproc.engine.SqlFeature.POSTGRESQL;
  25. import static org.sqlproc.engine.SqlFeature.SEQ;
  26. import static org.sqlproc.engine.SqlFeature.SEQ_NAME;
  27. import static org.sqlproc.engine.SqlFeature.SURROUND_QUERY_LIKE_PARTIAL;
  28. import static org.sqlproc.engine.SqlFeature.SURROUND_QUERY_MIN_LEN;
  29. import static org.sqlproc.engine.SqlFeature.VERSION_COLUMN;
  30. import static org.sqlproc.engine.SqlFeature.WILDCARD_CHARACTER;
  31. import java.util.HashMap;
  32. import java.util.Map;
  33. import java.util.Map.Entry;
  34. import org.sqlproc.engine.SqlFeature;
  35. /**
  36. * The default optional features.
  37. *
  38. * <p>
  39. * For more info please see the <a href="https://github.com/hudec/sql-processor/wiki">Tutorials</a>.
  40. *
  41. * @author <a href="mailto:Vladimir.Hudec@gmail.com">Vladimir Hudec</a>
  42. */
  43. public class SqlDefaultFeatures {
  44. public static final Map<SqlFeature, Object> FEATURES = new HashMap<SqlFeature, Object>();
  45. public static final Map<String, Map<SqlFeature, Object>> FILTERED_FEATURES = new HashMap<String, Map<SqlFeature, Object>>();
  46. static {
  47. FILTERED_FEATURES.put(ORACLE.name(), new HashMap<SqlFeature, Object>());
  48. FILTERED_FEATURES.put(HSQLDB.name(), new HashMap<SqlFeature, Object>());
  49. FILTERED_FEATURES.put(H2.name(), new HashMap<SqlFeature, Object>());
  50. FILTERED_FEATURES.put(MYSQL.name(), new HashMap<SqlFeature, Object>());
  51. FILTERED_FEATURES.put(POSTGRESQL.name(), new HashMap<SqlFeature, Object>());
  52. FILTERED_FEATURES.put(INFORMIX.name(), new HashMap<SqlFeature, Object>());
  53. FILTERED_FEATURES.put(MSSQL.name(), new HashMap<SqlFeature, Object>());
  54. FILTERED_FEATURES.put(DB2.name(), new HashMap<SqlFeature, Object>());
  55. FILTERED_FEATURES.put(CASSANDRA.name(), new HashMap<SqlFeature, Object>());
  56. }
  57. static {
  58. /**
  59. * This is the default value related to the key <code>WILDCARD_CHARACTER</code> .
  60. */
  61. FEATURES.put(WILDCARD_CHARACTER, "%");
  62. /**
  63. * This is the default value related to the key <code>SURROUND_QUERY_LIKE_PARTIAL</code> .
  64. */
  65. FEATURES.put(SURROUND_QUERY_LIKE_PARTIAL, Boolean.TRUE);
  66. /**
  67. * This is the default value related to the key <code>SURROUND_QUERY_MIN_LEN</code>.
  68. */
  69. FEATURES.put(SURROUND_QUERY_MIN_LEN, 2);
  70. /**
  71. * This is the default value related to the key <code>LIKE_STRING</code>.
  72. */
  73. FEATURES.put(LIKE_STRING, "like");
  74. /**
  75. * This is the default value related to the key <code>METHODS_ENUM_IN</code>. It is list of the next methods:
  76. * <ul>
  77. * <li>getCode</li>
  78. * <li>getValue</li>
  79. * <li>getName</li>
  80. * <li>name</li>
  81. * </ul>
  82. */
  83. FEATURES.put(METHODS_ENUM_IN, new String[] { "getCode", "getValue", "getName", "name" });
  84. /**
  85. * This is the default value related to the key <code>METHODS_ENUM_OUT</code>. It is list of the next methods:
  86. * <ul>
  87. * <li>fromCode</li>
  88. * <li>fromValue</li>
  89. * <li>valueOf</li>
  90. * </ul>
  91. */
  92. FEATURES.put(METHODS_ENUM_OUT, new String[] { "fromCode", "fromValue", "valueOf" });
  93. /**
  94. * <code>HSQLDB_DEFAULT_LIMIT_FROM_TO</code> is the default value related to the key <code>LIMIT_FROM_TO</code>
  95. * in the case the filter value <code>HSQLDB</code> is used for the {@link SqlProcessorLoader} instance
  96. * creation.
  97. */
  98. FILTERED_FEATURES.get(HSQLDB.name()).put(LIMIT_FROM_TO, "$S offset $F limit $M");
  99. /**
  100. * <code>HSQLDB_DEFAULT_LIMIT_FROM_TO_ORDERED</code> is the default value related to the key
  101. * <code>LIMIT_FROM_TO_ORDERED</code> in the case the filter value <code>HSQLDB</code> is used for the
  102. * {@link SqlProcessorLoader} instance creation.
  103. */
  104. FILTERED_FEATURES.get(HSQLDB.name()).put(LIMIT_FROM_TO_ORDERED, "$S offset $F limit $M using index");
  105. /**
  106. * <code>HSQLDB_DEFAULT_LIMIT_TO</code> is the default value related to the key <code>LIMIT_TO</code> in the
  107. * case the filter value <code>HSQLDB</code> is used for the {@link SqlProcessorLoader} instance creation.
  108. */
  109. FILTERED_FEATURES.get(HSQLDB.name()).put(LIMIT_TO, "$S limit $M");
  110. /**
  111. * <code>HSQLDB_DEFAULT_LIMIT_TO_ORDERED</code> is the default value related to the key
  112. * <code>LIMIT_TO_ORDERED</code> in the case the filter value <code>HSQLDB</code> is used for the
  113. * {@link SqlProcessorLoader} instance creation.
  114. */
  115. FILTERED_FEATURES.get(HSQLDB.name()).put(LIMIT_TO_ORDERED, "$S limit $M using index");
  116. /**
  117. * <code>H2_DEFAULT_LIMIT_FROM_TO</code> is the default value related to the key <code>LIMIT_FROM_TO</code> in
  118. * the case the filter value <code>H2</code> is used for the {@link SqlProcessorLoader} instance creation.
  119. */
  120. FILTERED_FEATURES.get(H2.name()).put(LIMIT_FROM_TO, "$S limit $M offset $F");
  121. /**
  122. * <code>H2_DEFAULT_LIMIT_TO</code> is the default value related to the key <code>LIMIT_TO</code> in the case
  123. * the filter value <code>H2</code> is used for the {@link SqlProcessorLoader} instance creation.
  124. */
  125. FILTERED_FEATURES.get(H2.name()).put(LIMIT_TO, "$S limit $M");
  126. /**
  127. * <code>ORACLE_DEFAULT_LIMIT_FROM_TO</code> is the default value related to the key <code>LIMIT_FROM_TO</code>
  128. * in the case the filter value <code>ORACLE</code> is used for the {@link SqlProcessorLoader} instance
  129. * creation.
  130. */
  131. FILTERED_FEATURES.get(ORACLE.name()).put(LIMIT_FROM_TO,
  132. "select * from ( select row_.*, rownum rownum_ from ($S) row_ where rownum <= $m) where rownum_ > $F");
  133. /**
  134. * <code>ORACLE_DEFAULT_LIMIT_TO</code> is the default value related to the key <code>LIMIT_TO</code> in the
  135. * case the filter value <code>ORACLE</code> is used for the {@link SqlProcessorLoader} instance creation.
  136. */
  137. FILTERED_FEATURES.get(ORACLE.name()).put(LIMIT_TO, "select * from ($S) where rownum <= $m");
  138. /**
  139. * <code>MYSQL_DEFAULT_LIMIT_FROM_TO</code> is the default value related to the key <code>LIMIT_FROM_TO</code>
  140. * in the case the filter value <code>MYSQL</code> is used for the {@link SqlProcessorLoader} instance creation.
  141. */
  142. FILTERED_FEATURES.get(MYSQL.name()).put(LIMIT_FROM_TO, "$S limit $F, $M");
  143. /**
  144. * <code>MYSQL_DEFAULT_LIMIT_TO</code> is the default value related to the key <code>LIMIT_TO</code> in the case
  145. * the filter value <code>MYSQL</code> is used for the {@link SqlProcessorLoader} instance creation.
  146. */
  147. FILTERED_FEATURES.get(MYSQL.name()).put(LIMIT_TO, "$S limit $M");
  148. /**
  149. * <code>POSTGRESQL_DEFAULT_LIMIT_FROM_TO</code> is the default value related to the key
  150. * <code>LIMIT_FROM_TO</code> in the case the filter value <code>POSTGRESQL</code> is used for the
  151. * {@link SqlProcessorLoader} instance creation.
  152. */
  153. FILTERED_FEATURES.get(POSTGRESQL.name()).put(LIMIT_FROM_TO, "$S limit $M offset $F");
  154. /**
  155. * <code>POSTGRESQL_DEFAULT_LIMIT_TO</code> is the default value related to the key <code>LIMIT_TO</code> in the
  156. * case the filter value <code>POSTGRESQL</code> is used for the {@link SqlProcessorLoader} instance creation.
  157. */
  158. FILTERED_FEATURES.get(POSTGRESQL.name()).put(LIMIT_TO, "$S limit $M");
  159. /**
  160. * <code>INFORMIX_DEFAULT_LIMIT_FROM_TO</code> is the default value related to the key
  161. * <code>LIMIT_FROM_TO</code> in the case the filter value <code>INFORMIX</code> is used for the
  162. * {@link SqlProcessorLoader} instance creation.
  163. */
  164. FILTERED_FEATURES.get(INFORMIX.name()).put(LIMIT_FROM_TO, "select skip $F first $M $s");
  165. /**
  166. * <code>INFORMIX_DEFAULT_LIMIT_TO</code> is the default value related to the key <code>LIMIT_TO</code> in the
  167. * case the filter value <code>INFORMIX</code> is used for the {@link SqlProcessorLoader} instance creation.
  168. */
  169. FILTERED_FEATURES.get(INFORMIX.name()).put(LIMIT_TO, "select first $M $s");
  170. /**
  171. * Version for MS SQL Server 2012: <code>MSSQL_DEFAULT_LIMIT_FROM_TO</code> is the default value related to the
  172. * key <code>LIMIT_FROM_TO</code> in the case the filter value <code>MSSQL</code> is used for the
  173. * {@link SqlProcessorLoader} instance creation.
  174. */
  175. FILTERED_FEATURES.get(MSSQL.name()).put(LIMIT_FROM_TO, "$S OFFSET ($F) ROWS FETCH NEXT ($M) ROWS ONLY");
  176. /**
  177. * <code>MSSQL_DEFAULT_LIMIT_TO</code> is the default value related to the key <code>LIMIT_TO</code> in the case
  178. * the filter value <code>MSSQL</code> is used for the {@link SqlProcessorLoader} instance creation.
  179. */
  180. FILTERED_FEATURES.get(MSSQL.name()).put(LIMIT_TO, "select top ($M) $s");
  181. /**
  182. * <code>DB2_DEFAULT_LIMIT_FROM_TO</code> is the default value related to the key <code>LIMIT_FROM_TO</code> in
  183. * the case the filter value <code>DB2</code> is used for the {@link SqlProcessorLoader} instance creation.
  184. */
  185. FILTERED_FEATURES.get(DB2.name()).put(LIMIT_FROM_TO,
  186. "select * from (select row_.*, rownumber() over() rownum_ from ($S) row_) where rownum_ <= $m and rownum_ > $F");
  187. /**
  188. * <code>DB2_DEFAULT_LIMIT_TO</code> is the default value related to the key <code>LIMIT_TO</code> in the case
  189. * the filter value <code>DB2</code> is used for the {@link SqlProcessorLoader} instance creation.
  190. */
  191. FILTERED_FEATURES.get(DB2.name()).put(LIMIT_TO,
  192. "select * from (select row_.*, rownumber() over() rownum_ from ($S) row_) where rownum_ <= $m");
  193. /**
  194. * <code>HSQLDB_DEFAULT_SEQ</code> is the default value related to the key <code>SEQ</code> in the case the
  195. * filter value <code>HSQLDB</code> is used for the {@link SqlProcessorLoader} instance creation.
  196. */
  197. FILTERED_FEATURES.get(HSQLDB.name()).put(SEQ, "call next value for $n");
  198. /**
  199. * <code>H2_DEFAULT_SEQ</code> is the default value related to the key <code>SEQ</code> in the case the filter
  200. * value <code>H2</code> is used for the {@link SqlProcessorLoader} instance creation.
  201. */
  202. FILTERED_FEATURES.get(H2.name()).put(SEQ, "call next value for $n");
  203. /**
  204. * <code>MYSQL_DEFAULT_SEQ</code> is the default value related to the key <code>SEQ</code> in the case the
  205. * filter value <code>MYSQL</code> is used for the {@link SqlProcessorLoader} instance creation.
  206. */
  207. // TODO $t - table name
  208. // public static final String MYSQL_DEFAULT_SEQ,
  209. // "select auto_increment from information_schema.tables where table_name, '$t'");
  210. /**
  211. * <code>ORACLE_DEFAULT_SEQ</code> is the default value related to the key <code>SEQ</code> in the case the
  212. * filter value <code>ORACLE</code> is used for the {@link SqlProcessorLoader} instance creation.
  213. */
  214. FILTERED_FEATURES.get(ORACLE.name()).put(SEQ, "select $n.nextval from dual");
  215. /**
  216. * <code>POSTGRESQL_DEFAULT_SEQ</code> is the default value related to the key <code>SEQ</code> in the case the
  217. * filter value <code>POSTGRESQL</code> is used for the {@link SqlProcessorLoader} instance creation.
  218. */
  219. FILTERED_FEATURES.get(POSTGRESQL.name()).put(SEQ, "select nextval('$n')");
  220. /**
  221. * <code>INFORMIX_DEFAULT_SEQ</code> is the default value related to the key <code>SEQ</code> in the case the
  222. * filter value <code>INFORMIX</code> is used for the {@link SqlProcessorLoader} instance creation.
  223. */
  224. FILTERED_FEATURES.get(INFORMIX.name()).put(SEQ, "SELECT FIRST 1 $n.NEXTVAL FROM systables");
  225. /**
  226. * <code>DB2_DEFAULT_SEQ</code> is the default value related to the key <code>SEQ</code> in the case the filter
  227. * value <code>DB2</code> is used for the {@link SqlProcessorLoader} instance creation.
  228. */
  229. FILTERED_FEATURES.get(DB2.name()).put(SEQ, "values nextval for $n");
  230. /**
  231. * <code>DEFAULT_SEQ_NAME</code> is the default sequence name.
  232. */
  233. FEATURES.put(SEQ_NAME, "SQLPROC_SEQUENCE");
  234. /**
  235. * <code>HSQLDB_DEFAULT_IDSEL</code> is the default value related to the key <code>IDSEL</code> in the case the
  236. * filter value <code>HSQLDB</code> is used for the {@link SqlProcessorLoader} instance creation.
  237. */
  238. FILTERED_FEATURES.get(HSQLDB.name()).put(IDSEL, "call identity()");
  239. /**
  240. * <code>H2_DEFAULT_IDSEL</code> is the default value related to the key <code>IDSEL</code> in the case the
  241. * filter value <code>H2</code> is used for the {@link SqlProcessorLoader} instance creation.
  242. */
  243. FILTERED_FEATURES.get(H2.name()).put(IDSEL, "call identity()");
  244. /**
  245. * <code>MYSQL_DEFAULT_IDSEL</code> is the default value related to the key <code>IDSEL</code> in the case the
  246. * filter value <code>MYSQL</code> is used for the {@link SqlProcessorLoader} instance creation.
  247. */
  248. FILTERED_FEATURES.get(MYSQL.name()).put(IDSEL, "select last_insert_id()");
  249. /**
  250. * <code>POSTGRESQL_DEFAULT_IDSEL</code> is the default value related to the key <code>IDSEL</code> in the case
  251. * the filter value <code>POSTGRESQL</code> is used for the {@link SqlProcessorLoader} instance creation.
  252. */
  253. // TODO $t - table name, $c - column name
  254. FILTERED_FEATURES.get(POSTGRESQL.name()).put(IDSEL, "select currval(pg_get_serial_sequence('$t','$c'))");
  255. /**
  256. * <code>INFORMIX_DEFAULT_IDSEL</code> is the default value related to the key <code>IDSEL</code> in the case
  257. * the filter value <code>INFORMIX</code> is used for the {@link SqlProcessorLoader} instance creation.
  258. */
  259. FILTERED_FEATURES.get(INFORMIX.name()).put(IDSEL, "SELECT FIRST 1 dbinfo('bigserial') FROM systables");
  260. /**
  261. * <code>INFORMIX_DEFAULT_IDSEL_Long</code> is the default value related to the key <code>IDSEL</code> in the
  262. * case the filter value <code>INFORMIX</code> is used for the {@link SqlProcessorLoader} instance creation.
  263. */
  264. // public static final String INFORMIX_DEFAULT_IDSEL_Long,
  265. // "select dbinfo('serial8') from informix.systables where tabid=1");
  266. FILTERED_FEATURES.get(INFORMIX.name()).put(IDSEL_Long,
  267. "select dbinfo('bigserial') from informix.systables where tabid=1");
  268. /**
  269. * <code>INFORMIX_DEFAULT_IDSEL_Integer</code> is the default value related to the key <code>IDSEL</code> in the
  270. * case the filter value <code>INFORMIX</code> is used for the {@link SqlProcessorLoader} instance creation.
  271. */
  272. FILTERED_FEATURES.get(INFORMIX.name()).put(IDSEL_Integer,
  273. "select dbinfo('sqlca.sqlerrd1') from informix.systables where tabid=1");
  274. /**
  275. * <code>MSSQL_DEFAULT_IDSEL</code> is the default value related to the key <code>IDSEL</code> in the case the
  276. * filter value <code>MSSQL</code> is used for the {@link SqlProcessorLoader} instance creation.
  277. * <p/>
  278. * The default value is {@link #IDSEL_JDBC}.
  279. */
  280. FILTERED_FEATURES.get(MSSQL.name()).put(IDSEL, JDBC.name());
  281. /**
  282. * /** <code>DB2_DEFAULT_IDSEL</code> is the default value related to the key <code>IDSEL</code> in the case the
  283. * filter value <code>DB2</code> is used for the {@link SqlProcessorLoader} instance creation.
  284. */
  285. FILTERED_FEATURES.get(DB2.name()).put(IDSEL, "SELECT identity_val_local() FROM SYSIBM.DUAL");
  286. /**
  287. * <code>DEFAULT_VERSION_COLUMN</code> is the default name of the column devoted to the optimistic locking.
  288. */
  289. FEATURES.put(VERSION_COLUMN, "version");
  290. /**
  291. * This is the default value related to the key <code>OPERATOR_ATTRIBUTE</code>.
  292. */
  293. FEATURES.put(OPERATOR_ATTRIBUTE, "Op");
  294. /**
  295. * This is the default value related to the key <code>OPERATOR_ATTRIBUTE_IN_MAP</code>.
  296. */
  297. FEATURES.put(OPERATOR_ATTRIBUTE_IN_MAP, "operators");
  298. /**
  299. * This is the default value related to the key <code>LOG_SQL_COMMAND_FOR_EXCEPTION</code> .
  300. */
  301. FEATURES.put(LOG_SQL_COMMAND_FOR_EXCEPTION, Boolean.FALSE);
  302. }
  303. /**
  304. * Returns filtered features. For example the features can be specific ones for given database.
  305. *
  306. * @param filter
  307. * the filter (for example the database name)
  308. * @return the specific features
  309. */
  310. public static Map<String, Object> getFilteredFeatures(String filter) {
  311. Map<String, Object> features = new HashMap<String, Object>();
  312. for (Entry<SqlFeature, Object> entry : SqlDefaultFeatures.FEATURES.entrySet()) {
  313. features.put(entry.getKey().name(), entry.getValue());
  314. }
  315. if (filter != null) {
  316. features.put(SqlFeature.FILTER.name(), filter);
  317. Map<SqlFeature, Object> filteredFeatures = SqlDefaultFeatures.FILTERED_FEATURES.get(filter);
  318. if (filteredFeatures != null) {
  319. for (Entry<SqlFeature, Object> entry : filteredFeatures.entrySet()) {
  320. features.put(entry.getKey().name(), entry.getValue());
  321. }
  322. }
  323. }
  324. return features;
  325. }
  326. }