org.jooq.impl.DivideBy Maven / Gradle / Ivy
/**
* Copyright (c) 2009-2014, Data Geekery GmbH (http://www.datageekery.com)
* All rights reserved.
*
* This work is dual-licensed
* - under the Apache Software License 2.0 (the "ASL")
* - under the jOOQ License and Maintenance Agreement (the "jOOQ License")
* =============================================================================
* You may choose which license applies to you:
*
* - If you're using this work with Open Source databases, you may choose
* either ASL or jOOQ License.
* - If you're using this work with at least one commercial database, you must
* choose jOOQ License
*
* For more information, please visit http://www.jooq.org/licenses
*
* Apache Software License 2.0:
* -----------------------------------------------------------------------------
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
* jOOQ License and Maintenance Agreement:
* -----------------------------------------------------------------------------
* Data Geekery grants the Customer the non-exclusive, timely limited and
* non-transferable license to install and use the Software under the terms of
* the jOOQ License and Maintenance Agreement.
*
* This library is distributed with a LIMITED WARRANTY. See the jOOQ License
* and Maintenance Agreement for more details: http://www.jooq.org/licensing
*/
package org.jooq.impl;
import static org.jooq.impl.DSL.condition;
import static org.jooq.impl.DSL.exists;
import static org.jooq.impl.DSL.notExists;
import static org.jooq.impl.DSL.selectDistinct;
import static org.jooq.impl.DSL.selectOne;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import org.jooq.Condition;
import org.jooq.DivideByOnConditionStep;
import org.jooq.DivideByOnStep;
import org.jooq.Field;
import org.jooq.Operator;
import org.jooq.QueryPart;
import org.jooq.Record;
import org.jooq.Select;
import org.jooq.Table;
/**
* @author Lukas Eder
*/
class DivideBy
implements
DivideByOnStep,
DivideByOnConditionStep {
private final Table> dividend;
private final Table> divisor;
private final ConditionProviderImpl condition;
private final QueryPartList> returning;
DivideBy(Table> dividend, Table> divisor) {
this.dividend = dividend;
this.divisor = divisor;
this.condition = new ConditionProviderImpl();
this.returning = new QueryPartList>();
}
// ------------------------------------------------------------------------
// XXX: Table API
// ------------------------------------------------------------------------
/**
* Transform the relational division operation into SQL.
*
* Various nice examples of how this can be achieved are found here: http://www.simple-talk.com/sql/t-sql-programming/divided-we-stand-the-
* sql-of-relational-division/
*/
private final Table table() {
ConditionProviderImpl selfJoin = new ConditionProviderImpl();
List> select = new ArrayList>();
Table> outer = dividend.as("dividend");
for (Field> field : returning) {
Field> outerField = outer.field(field);
// Fields from the RETURNING clause are added AS-IS to the SELECT
// statement, if they're not contained in the dividend table
if (outerField == null) {
select.add(field);
}
// Fields from the RETURNING clause need proper aliasing if they're
// contained in the dividend table
else {
select.add(outerField);
selfJoin(selfJoin, outer, dividend, field);
}
}
// Apply relational division using double-nested NOT EXISTS clauses
// There are more efficient ways to express division in SQL, but those
// are hard to simulate with jOOQ
return selectDistinct(select)
.from(outer)
.whereNotExists(
selectOne()
.from(divisor)
.whereNotExists(
selectOne()
.from(dividend)
.where(selfJoin)
.and(condition)))
.asTable();
}
/**
* Extracted method for type-safety
*/
@SuppressWarnings("deprecation")
private final void selfJoin(org.jooq.ConditionProvider selfJoin, Table> outer, Table> inner, Field field) {
Field outerField = outer.field(field);
Field innerField = inner.field(field);
if (outerField == null || innerField == null) {
return;
}
else {
selfJoin.addConditions(outerField.equal(innerField));
}
}
// ------------------------------------------------------------------------
// XXX: DivideBy API
// ------------------------------------------------------------------------
@Override
public final DivideByOnConditionStep on(Condition... conditions) {
condition.addConditions(conditions);
return this;
}
@Override
public final DivideByOnConditionStep on(Field c) {
return on(condition(c));
}
@Override
public final DivideByOnConditionStep on(String sql) {
and(sql);
return this;
}
@Override
public final DivideByOnConditionStep on(String sql, Object... bindings) {
and(sql, bindings);
return this;
}
@Override
public final DivideByOnConditionStep on(String sql, QueryPart... parts) {
and(sql, parts);
return this;
}
@Override
public final Table returning(Field>... fields) {
return returning(Arrays.asList(fields));
}
@Override
public final Table returning(Collection extends Field>> fields) {
returning.addAll(fields);
return table();
}
@Override
public final DivideByOnConditionStep and(Condition c) {
condition.addConditions(c);
return this;
}
@Override
public final DivideByOnConditionStep and(Field c) {
return and(condition(c));
}
@Override
public final DivideByOnConditionStep and(String sql) {
return and(condition(sql));
}
@Override
public final DivideByOnConditionStep and(String sql, Object... bindings) {
return and(condition(sql, bindings));
}
@Override
public final DivideByOnConditionStep and(String sql, QueryPart... parts) {
return and(condition(sql, parts));
}
@Override
public final DivideByOnConditionStep andNot(Condition c) {
return and(c.not());
}
@Override
public final DivideByOnConditionStep andNot(Field c) {
return andNot(condition(c));
}
@Override
public final DivideByOnConditionStep andExists(Select> select) {
return and(exists(select));
}
@Override
public final DivideByOnConditionStep andNotExists(Select> select) {
return and(notExists(select));
}
@Override
public final DivideByOnConditionStep or(Condition c) {
condition.addConditions(Operator.OR, c);
return this;
}
@Override
public final DivideByOnConditionStep or(Field c) {
return or(condition(c));
}
@Override
public final DivideByOnConditionStep or(String sql) {
return or(condition(sql));
}
@Override
public final DivideByOnConditionStep or(String sql, Object... bindings) {
return or(condition(sql, bindings));
}
@Override
public final DivideByOnConditionStep or(String sql, QueryPart... parts) {
return or(condition(sql, parts));
}
@Override
public final DivideByOnConditionStep orNot(Condition c) {
return or(c.not());
}
@Override
public final DivideByOnConditionStep orNot(Field c) {
return orNot(condition(c));
}
@Override
public final DivideByOnConditionStep orExists(Select> select) {
return or(exists(select));
}
@Override
public final DivideByOnConditionStep orNotExists(Select> select) {
return or(notExists(select));
}
}