/* * JBoss, Home of Professional Open Source. * See the COPYRIGHT.txt file distributed with this work for information * regarding copyright ownership. Some portions may be licensed * to Red Hat, Inc. under one or more contributor license agreements. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA * 02110-1301 USA. */ package org.teiid.translator.hive; import static org.teiid.translator.TypeFacility.RUNTIME_NAMES.*; import java.sql.Connection; import java.util.ArrayList; import java.util.Arrays; import java.util.List; import org.teiid.core.types.DataTypeManager; import org.teiid.language.*; import org.teiid.language.Join.JoinType; import org.teiid.language.SortSpecification.Ordering; import org.teiid.metadata.AggregateAttributes; import org.teiid.translator.ExecutionContext; import org.teiid.translator.SourceSystemFunctions; import org.teiid.translator.Translator; import org.teiid.translator.TranslatorException; import org.teiid.translator.jdbc.AliasModifier; import org.teiid.translator.jdbc.FunctionModifier; import org.teiid.util.Version; @Translator(name="impala", description="A translator for Coludera's Impala based database on HDFS") public class ImpalaExecutionFactory extends BaseHiveExecutionFactory { public static String IMPALA = "impala"; //$NON-NLS-1$ public static final Version TWO_0 = Version.getVersion("2.0"); //$NON-NLS-1$ public static final Version ONE_2_1 = Version.getVersion("1.2.1"); //$NON-NLS-1$ @Override public void start() throws TranslatorException { super.start(); convert.addTypeMapping("tinyint", FunctionModifier.BYTE); //$NON-NLS-1$ convert.addTypeMapping("smallint", FunctionModifier.SHORT); //$NON-NLS-1$ convert.addTypeMapping("int", FunctionModifier.INTEGER); //$NON-NLS-1$ convert.addTypeMapping("bigint", FunctionModifier.BIGINTEGER, FunctionModifier.LONG); //$NON-NLS-1$ convert.addTypeMapping("boolean", FunctionModifier.BOOLEAN); //$NON-NLS-1$ convert.addTypeMapping("double", FunctionModifier.DOUBLE); //$NON-NLS-1$ convert.addTypeMapping("float", FunctionModifier.FLOAT); //$NON-NLS-1$ convert.addTypeMapping("string", FunctionModifier.STRING); //$NON-NLS-1$ convert.addTypeMapping("timestamp", FunctionModifier.TIMESTAMP); //$NON-NLS-1$ registerFunctionModifier(SourceSystemFunctions.CONVERT, convert); registerFunctionModifier(SourceSystemFunctions.LCASE, new AliasModifier("lower")); //$NON-NLS-1$ registerFunctionModifier(SourceSystemFunctions.UCASE, new AliasModifier("upper")); //$NON-NLS-1$ registerFunctionModifier(SourceSystemFunctions.SUBSTRING, new AliasModifier("substr")); //$NON-NLS-1$ registerFunctionModifier(SourceSystemFunctions.CURDATE, new AliasModifier("unix_timestamp")); //$NON-NLS-1$ registerFunctionModifier(SourceSystemFunctions.IFNULL, new AliasModifier("isnull")); //$NON-NLS-1$ registerFunctionModifier("string_agg", new AliasModifier("group_concat")); //$NON-NLS-1$ //$NON-NLS-2$ addPushDownFunction(IMPALA, "lower", STRING, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "upper", STRING, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "positive", INTEGER, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "positive", DOUBLE, DOUBLE); //$NON-NLS-1$ addPushDownFunction(IMPALA, "negitive", INTEGER, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "negitive", DOUBLE, DOUBLE); //$NON-NLS-1$ addPushDownFunction(IMPALA, "ln", DOUBLE, DOUBLE); //$NON-NLS-1$ addPushDownFunction(IMPALA, "reverse", STRING, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "space", STRING, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "hex", STRING, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "unhex", STRING, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "bin", STRING, LONG); //$NON-NLS-1$ //date functions addPushDownFunction(IMPALA, "add_months", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "adddate", TIMESTAMP, TIMESTAMP, INTEGER ); //$NON-NLS-1$ addPushDownFunction(IMPALA, "date_add", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "date_part", INTEGER, STRING, TIMESTAMP); //$NON-NLS-1$ addPushDownFunction(IMPALA, "date_sub", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "datediff", INTEGER, STRING, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "day", INTEGER, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "dayofyear", INTEGER, TIMESTAMP); //$NON-NLS-1$ addPushDownFunction(IMPALA, "days_add", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "days_add", TIMESTAMP, TIMESTAMP , LONG ); //$NON-NLS-1$ addPushDownFunction(IMPALA, "days_sub", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "days_sub", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "extract", INTEGER, TIMESTAMP, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "from_unixtime", STRING, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "from_unixtime", STRING, LONG, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "from_utc_timestamp", TIMESTAMP, TIMESTAMP, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "hour",INTEGER, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "hours_add", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "hours_add", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "hours_sub", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "hours_sub", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "microseconds_sub", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "microseconds_sub", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "milliseconds_add", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "milliseconds_add", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "milliseconds_sub", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "milliseconds_sub", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "minute", INTEGER, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "minutes_add", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "minutes_add", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "minutes_sub", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "minutes_sub", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "month", INTEGER, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "months_add", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "months_add", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "months_sub", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "months_sub", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "nanoseconds_add", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "nanoseconds_add", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "nanoseconds_sub", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "nanoseconds_sub", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "second", INTEGER, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "seconds_add", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "seconds_add", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "seconds_sub", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "seconds_sub", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "subdate", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "to_date", STRING, TIMESTAMP); //$NON-NLS-1$ addPushDownFunction(IMPALA, "to_utc_timestamp", TIMESTAMP, TIMESTAMP, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "trunc", TIMESTAMP, TIMESTAMP, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "unix_timestamp", INTEGER, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "unix_timestamp", INTEGER, STRING, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "unix_timestamp", INTEGER, TIMESTAMP); //$NON-NLS-1$ addPushDownFunction(IMPALA, "weekofyear", INTEGER, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "weeks_add", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "weeks_add", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "weeks_sub", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "weeks_sub", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "years_add", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "years_add", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "years_sub", TIMESTAMP, TIMESTAMP, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "years_sub", TIMESTAMP, TIMESTAMP, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "conv", STRING, LONG, INTEGER, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "greatest", STRING, STRING, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "greatest", TIMESTAMP, TIMESTAMP, TIMESTAMP); //$NON-NLS-1$ addPushDownFunction(IMPALA, "greatest", LONG, LONG, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "least", STRING, STRING, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "least", TIMESTAMP, TIMESTAMP, TIMESTAMP); //$NON-NLS-1$ addPushDownFunction(IMPALA, "least", LONG, LONG, LONG); //$NON-NLS-1$ addPushDownFunction(IMPALA, "log2", STRING, DOUBLE); //$NON-NLS-1$ addPushDownFunction(IMPALA, "pow", DOUBLE, DOUBLE); //$NON-NLS-1$ addPushDownFunction(IMPALA, "quotient", INTEGER, INTEGER, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "radians", DOUBLE, DOUBLE); //$NON-NLS-1$ addPushDownFunction(IMPALA, "sign", INTEGER, DOUBLE); //$NON-NLS-1$ addPushDownFunction(IMPALA, "parse_url", STRING, STRING, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "regexp_extract", STRING, STRING, STRING, INTEGER); //$NON-NLS-1$ addPushDownFunction(IMPALA, "regexp_replace", STRING, STRING, STRING, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "group_concat", STRING, STRING, STRING).setAggregateAttributes(new AggregateAttributes()); //$NON-NLS-1$ addPushDownFunction(IMPALA, "concat_ws", STRING, STRING, STRING).setVarArgs(true); //$NON-NLS-1$ addPushDownFunction(IMPALA, "concat", STRING, STRING).setVarArgs(true); //$NON-NLS-1$ addPushDownFunction(IMPALA, "initcap", STRING, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "instr", INTEGER, STRING, STRING); //$NON-NLS-1$ addPushDownFunction(IMPALA, "find_in_set", INTEGER, STRING, STRING); //$NON-NLS-1$ } @Override public void initCapabilities(Connection connection) throws TranslatorException { super.initCapabilities(connection); //supported data types post-Impala v2 if (getVersion().compareTo(TWO_0) >= 0) { convert.addTypeMapping("decimal", FunctionModifier.BIGDECIMAL); //$NON-NLS-1$ convert.addTypeMapping("char(1)", FunctionModifier.CHAR); //$NON-NLS-1$ convert.addTypeMapping("varchar", FunctionModifier.STRING); //$NON-NLS-1$ } } @Override public List<String> getSupportedFunctions() { List<String> supportedFunctions = new ArrayList<String>(); supportedFunctions.addAll(super.getSupportedFunctions()); supportedFunctions.add(SourceSystemFunctions.ABS); supportedFunctions.add(SourceSystemFunctions.ACOS); //supportedFunctions.add(SourceSystemFunctions.ARRAY_GET); supportedFunctions.add(SourceSystemFunctions.ASIN); supportedFunctions.add(SourceSystemFunctions.ASCII); supportedFunctions.add(SourceSystemFunctions.ATAN); //supportedFunctions.add(SourceSystemFunctions.BITAND); //supportedFunctions.add(SourceSystemFunctions.BITNOT); //supportedFunctions.add(SourceSystemFunctions.BITOR); //supportedFunctions.add(SourceSystemFunctions.BITXOR); supportedFunctions.add(SourceSystemFunctions.CEILING); supportedFunctions.add(SourceSystemFunctions.COALESCE); supportedFunctions.add(SourceSystemFunctions.CONCAT); supportedFunctions.add(SourceSystemFunctions.COS); supportedFunctions.add(SourceSystemFunctions.CONVERT); supportedFunctions.add(SourceSystemFunctions.CURDATE); // supportedFunctions.add(SourceSystemFunctions.CURTIME); supportedFunctions.add(SourceSystemFunctions.DEGREES); supportedFunctions.add(SourceSystemFunctions.DAYNAME); supportedFunctions.add(SourceSystemFunctions.DAYOFMONTH); supportedFunctions.add(SourceSystemFunctions.DAYOFWEEK); supportedFunctions.add(SourceSystemFunctions.FROM_UNIXTIME); supportedFunctions.add(SourceSystemFunctions.EXP); supportedFunctions.add(SourceSystemFunctions.FLOOR); supportedFunctions.add(SourceSystemFunctions.HOUR); supportedFunctions.add(SourceSystemFunctions.IFNULL); supportedFunctions.add(SourceSystemFunctions.LENGTH); supportedFunctions.add(SourceSystemFunctions.LOCATE); supportedFunctions.add(SourceSystemFunctions.LCASE); //lower supportedFunctions.add(SourceSystemFunctions.LPAD); supportedFunctions.add(SourceSystemFunctions.LTRIM); supportedFunctions.add(SourceSystemFunctions.LOG); supportedFunctions.add(SourceSystemFunctions.LOG10); supportedFunctions.add(SourceSystemFunctions.MINUTE); supportedFunctions.add(SourceSystemFunctions.MOD); //pmod supportedFunctions.add(SourceSystemFunctions.MONTH); supportedFunctions.add(SourceSystemFunctions.NOW); supportedFunctions.add(SourceSystemFunctions.POWER); supportedFunctions.add(SourceSystemFunctions.PI); supportedFunctions.add(SourceSystemFunctions.RADIANS); supportedFunctions.add(SourceSystemFunctions.RAND); supportedFunctions.add(SourceSystemFunctions.REPEAT); supportedFunctions.add(SourceSystemFunctions.ROUND); supportedFunctions.add(SourceSystemFunctions.RPAD); supportedFunctions.add(SourceSystemFunctions.RTRIM); supportedFunctions.add(SourceSystemFunctions.SECOND); supportedFunctions.add(SourceSystemFunctions.SIN); supportedFunctions.add(SourceSystemFunctions.SQRT); supportedFunctions.add(SourceSystemFunctions.SUBSTRING); //substr supportedFunctions.add(SourceSystemFunctions.TAN); supportedFunctions.add(SourceSystemFunctions.TRIM); supportedFunctions.add(SourceSystemFunctions.UCASE); supportedFunctions.add(SourceSystemFunctions.YEAR); return supportedFunctions; } @Override public boolean supportsCommonTableExpressions() { return true; // WITH clause } @Override public boolean supportsElementaryOlapOperations() { /* Impala supports window functions From Cloudera doc: http://www.cloudera.com/documentation/archive/impala/2-x/2-1-x/topics/impala_analytic_functions.html#window_clause_unique_1 */ return getVersion().compareTo(TWO_0) >= 0; } @Override public boolean supportsHaving() { /* * From Coludera DOC, different from Hive * Performs a filter operation on a SELECT query, by examining the results of * aggregation functions rather than testing each individual table row. Thus * always used in conjunction with a function such as COUNT(), SUM(), AVG(), * MIN(), or MAX(), and typically with the GROUP BY clause also. */ return true; } @Override public boolean supportsRowLimit() { /* * In Impala 1.2.1 and higher, you can combine a LIMIT clause with an OFFSET clause * to produce a small result set that is different from a top-N query */ return true; } @Override public boolean supportsRowOffset() { return getVersion().compareTo(ONE_2_1) >= 0; } @Override public org.teiid.translator.ExecutionFactory.NullOrder getDefaultNullOrder() { return NullOrder.HIGH; } @Override public boolean supportsOrderByNullOrdering() { return true; } @Override public org.teiid.translator.ExecutionFactory.SupportedJoinCriteria getSupportedJoinCriteria() { return SupportedJoinCriteria.ANY; } @Override public boolean requiresLeftLinearJoin() { return true; } @Override public List<?> translateCommand(Command command, ExecutionContext context) { if (command instanceof Select) { Select select = (Select)command; if (select.getLimit() != null && select.getLimit().getRowOffset() != 0 && select.getOrderBy() == null) { select.setOrderBy(new OrderBy(Arrays.asList(new SortSpecification(Ordering.ASC, new Literal(1, DataTypeManager.DefaultDataClasses.INTEGER))))); } //compensate for an impala issue - https://issues.jboss.org/browse/TEIID-3743 if (select.getGroupBy() == null && select.getHaving() == null) { boolean rewrite = false; String distinctVal = null; for (DerivedColumn col : select.getDerivedColumns()) { if (col.getExpression() instanceof AggregateFunction && ((AggregateFunction)col.getExpression()).isDistinct()) { if (distinctVal == null) { distinctVal = ((AggregateFunction)col.getExpression()).getParameters().toString(); } else if (!((AggregateFunction)col.getExpression()).getParameters().toString().equals(distinctVal)){ rewrite = true; break; } } } if (rewrite) { Select top = new Select(); top.setWith(select.getWith()); top.setDerivedColumns(new ArrayList<DerivedColumn>()); top.setFrom(new ArrayList<TableReference>()); //rewrite as a cross join of single groups Select viewSelect = new Select(); viewSelect.setFrom(select.getFrom()); viewSelect.setDerivedColumns(new ArrayList<DerivedColumn>()); viewSelect.setWhere(select.getWhere()); distinctVal = null; int viewCount = 0; NamedTable view = new NamedTable("v" + viewCount++, null, null); //$NON-NLS-1$ for (int i = 0; i < select.getDerivedColumns().size(); i++) { DerivedColumn col = select.getDerivedColumns().get(i); if (col.getExpression() instanceof AggregateFunction && ((AggregateFunction)col.getExpression()).isDistinct()) { if (distinctVal == null) { distinctVal = ((AggregateFunction)col.getExpression()).getParameters().toString(); } else if (!((AggregateFunction)col.getExpression()).getParameters().toString().equals(distinctVal)){ DerivedTable dt = new DerivedTable(viewSelect, view.getName()); if (top.getFrom().isEmpty()) { top.getFrom().add(dt); } else { Join join = new Join(top.getFrom().remove(0), dt, JoinType.CROSS_JOIN, null); top.getFrom().add(join); } view = new NamedTable("v" + viewCount++, null, null); //$NON-NLS-1$ viewSelect = new Select(); viewSelect.setFrom(select.getFrom()); viewSelect.setDerivedColumns(new ArrayList<DerivedColumn>()); viewSelect.setWhere(select.getWhere()); distinctVal = ((AggregateFunction)col.getExpression()).getParameters().toString(); } } col.setAlias("c" + i); //$NON-NLS-1$ top.getDerivedColumns().add(new DerivedColumn(null, new ColumnReference(view, col.getAlias(), null, col.getExpression().getType()))); viewSelect.getDerivedColumns().add(col); } DerivedTable dt = new DerivedTable(viewSelect, view.getName()); Join join = new Join(top.getFrom().remove(0), dt, JoinType.CROSS_JOIN, null); top.getFrom().add(join); return Arrays.asList(top); } } } return super.translateCommand(command, context); } @Override public List<?> translate(LanguageObject obj, ExecutionContext context) { if (obj instanceof WithItem) { WithItem item = (WithItem)obj; List<ColumnReference> cols = item.getColumns(); item.setColumns(null); Select select = item.getSubquery().getProjectedQuery(); List<DerivedColumn> selectClause = select.getDerivedColumns(); for (int i = 0; i < cols.size(); i++) { selectClause.get(i).setAlias(cols.get(i).getName()); } } return super.translate(obj, context); } @Override protected boolean usesDatabaseVersion() { return true; } @Override public List<?> translateLimit(Limit limit, ExecutionContext context) { if (limit.getRowOffset() > 0) { return Arrays.asList("LIMIT ", limit.getRowLimit(), " OFFSET ", limit.getRowOffset()); //$NON-NLS-1$ //$NON-NLS-2$ } return null; } @Override public String translateLiteralDate(java.sql.Date dateValue) { return '\'' + formatDateValue(dateValue) + '\''; } @Override public boolean supportsGroupByMultipleDistinctAggregates() { return false; } @Override public boolean supportsStringAgg() { return true; } }