All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jooq.impl.Fields Maven / Gradle / Ivy

There is a newer version: 8.1.2
Show newest version
/*
 * Copyright (c) 2009-2016, Data Geekery GmbH (http://www.datageekery.com)
 * All rights reserved.
 *
 * 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.
 *
 * Other licenses:
 * -----------------------------------------------------------------------------
 * Commercial licenses for this work are available. These replace the above
 * ASL 2.0 and offer limited warranties, support, maintenance, and commercial
 * database integrations.
 *
 * For more information, please visit: http://www.jooq.org/licenses
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 */

package org.jooq.impl;

import static org.jooq.impl.Tools.EMPTY_FIELD;
import static org.jooq.impl.Tools.indexOrFail;

import java.sql.SQLWarning;
import java.util.Collection;

import org.jooq.Clause;
import org.jooq.Context;
import org.jooq.DataType;
import org.jooq.Field;
import org.jooq.Name;
import org.jooq.Record;
import org.jooq.RecordType;
import org.jooq.Table;
import org.jooq.TableField;
import org.jooq.tools.JooqLogger;

/**
 * A simple wrapper for Field[], providing some useful lookup
 * methods
 *
 * @author Lukas Eder
 */
final class Fields extends AbstractQueryPart implements RecordType {

    private static final long       serialVersionUID = -6911012275707591576L;
    private static final JooqLogger log              = JooqLogger.getLogger(Fields.class);
    Field[]                      fields;

    Fields(Field... fields) {
        this.fields = fields;
    }

    Fields(Collection> fields) {
        this.fields = fields.toArray(EMPTY_FIELD);
    }

    @Override
    public final int size() {
        return fields.length;
    }

    @Override
    @SuppressWarnings("unchecked")
    public final  Field field(Field field) {
        if (field == null)
            return null;

        // [#4540] Try finding a match by identity
        for (Field f : fields)
            if (f == field)
                return (Field) f;

        // [#1802] Try finding an exact match (e.g. exact matching qualified name)
        for (Field f : fields)
            if (f.equals(field))
                return (Field) f;

        // [#4283] table / column matches are better than only column matches
        Field columnMatch = null;
        Field columnMatch2 = null;

        String tableName = tableName(field);
        String fieldName = field.getName();

        for (Field f : fields) {
            String fName = f.getName();

            if (tableName != null) {
                String tName = tableName(f);

                if (tName != null && tableName.equals(tName) && fName.equals(fieldName))
                    return (Field) f;
            }

            // In case no exact match was found, return the first field with matching name
            if (fName.equals(fieldName)) {
                if (columnMatch == null)
                    columnMatch = f;
                else
                    // [#4476] [#4477] This might be unintentional from a user
                    //                 perspective, e.g. when ambiguous ID columns are present.
                    // [#5578] Finish the loop, though, as we might have an exact match
                    //         despite some ambiguity
                    columnMatch2 = f;
            }
        }

        if (columnMatch2 != null)
            if (log.isInfoEnabled())
                log.info("Ambiguous match found for " + fieldName + ". Both " + columnMatch + " and " + columnMatch2 + " match.", new SQLWarning());

        return (Field) columnMatch;
    }

    private final String tableName(Field field) {
        if (field instanceof TableField) {
            Table table = ((TableField) field).getTable();

            if (table != null) {
                return table.getName();
            }
        }

        return null;
    }

    @Override
    public final Field field(String fieldName) {
        if (fieldName == null)
            return null;

        Field columnMatch = null;

        for (Field f : fields)
            if (f.getName().equals(fieldName))
                if (columnMatch == null)
                    columnMatch = f;
                else
                    // [#4476] [#4477] [#5046] This might be unintentional from a user
                    // perspective, e.g. when ambiguous ID columns are present.
                    log.info("Ambiguous match found for " + fieldName + ". Both " + columnMatch + " and " + f + " match.", new SQLWarning());

        return columnMatch;
    }

    @Override
    public final  Field field(String fieldName, Class type) {
        Field result = field(fieldName);
        return result == null ? null : result.coerce(type);
    }

    @Override
    public final  Field field(String fieldName, DataType dataType) {
        Field result = field(fieldName);
        return result == null ? null : result.coerce(dataType);
    }

    @Override
    public final Field field(Name name) {
        if (name == null)
            return null;

        return field(DSL.field(name));
    }

    @Override
    public final  Field field(Name fieldName, Class type) {
        Field result = field(fieldName);
        return result == null ? null : result.coerce(type);
    }

    @Override
    public final  Field field(Name fieldName, DataType dataType) {
        Field result = field(fieldName);
        return result == null ? null : result.coerce(dataType);
    }

    @Override
    public final Field field(int index) {
        if (index >= 0 && index < fields.length) {
            return fields[index];
        }

        return null;
    }

    @Override
    public final  Field field(int fieldIndex, Class type) {
        Field result = field(fieldIndex);
        return result == null ? null : result.coerce(type);
    }

    @Override
    public final  Field field(int fieldIndex, DataType dataType) {
        Field result = field(fieldIndex);
        return result == null ? null : result.coerce(dataType);
    }

    @Override
    public final Field[] fields() {
        return fields;
    }

    @Override
    public final Field[] fields(Field... f) {
        Field[] result = new Field[f.length];

        for (int i = 0; i < result.length; i++)
            result[i] = field(f[i]);

        return result;
    }

    @Override
    public final Field[] fields(String... f) {
        Field[] result = new Field[f.length];

        for (int i = 0; i < result.length; i++)
            result[i] = field(f[i]);

        return result;
    }

    @Override
    public final Field[] fields(Name... f) {
        Field[] result = new Field[f.length];

        for (int i = 0; i < result.length; i++)
            result[i] = field(f[i]);

        return result;
    }

    @Override
    public final Field[] fields(int... f) {
        Field[] result = new Field[f.length];

        for (int i = 0; i < result.length; i++)
            result[i] = field(f[i]);

        return result;
    }

    @Override
    public final int indexOf(Field field) {

        // Get an exact match, or a field with a similar name
        Field compareWith = field(field);

        if (compareWith != null) {
            int size = fields.length;

            // [#4540] Match by identity first
            for (int i = 0; i < size; i++)
                if (fields[i] == compareWith)
                    return i;

            for (int i = 0; i < size; i++)
                if (fields[i].equals(compareWith))
                    return i;
        }

        return -1;
    }

    @Override
    public final int indexOf(String fieldName) {
        return indexOf(field(fieldName));
    }

    @Override
    public final int indexOf(Name fieldName) {
        return indexOf(field(fieldName));
    }

    @Override
    public final Class[] types() {
        int size = fields.length;
        Class[] result = new Class[size];

        for (int i = 0; i < size; i++) {
            result[i] = field(i).getType();
        }

        return result;
    }

    @Override
    public final Class type(int fieldIndex) {
        return fieldIndex >= 0 && fieldIndex < size() ? field(fieldIndex).getType() : null;
    }

    @Override
    public final Class type(String fieldName) {
        return type(indexOrFail(this, fieldName));
    }

    @Override
    public final Class type(Name fieldName) {
        return type(indexOrFail(this, fieldName));
    }

    @Override
    public final DataType[] dataTypes() {
        int size = fields.length;
        DataType[] result = new DataType[size];

        for (int i = 0; i < size; i++) {
            result[i] = field(i).getDataType();
        }

        return result;
    }

    @Override
    public final DataType dataType(int fieldIndex) {
        return fieldIndex >= 0 && fieldIndex < size() ? field(fieldIndex).getDataType() : null;
    }

    @Override
    public final DataType dataType(String fieldName) {
        return dataType(indexOrFail(this, fieldName));
    }

    @Override
    public final DataType dataType(Name fieldName) {
        return dataType(indexOrFail(this, fieldName));
    }

    final int[] indexesOf(Field... f) {
        int[] result = new int[f.length];

        for (int i = 0; i < f.length; i++) {
            result[i] = indexOrFail(this, f[i]);
        }

        return result;
    }

    final int[] indexesOf(String... fieldNames) {
        int[] result = new int[fieldNames.length];

        for (int i = 0; i < fieldNames.length; i++) {
            result[i] = indexOrFail(this, fieldNames[i]);
        }

        return result;
    }

    final int[] indexesOf(Name... fieldNames) {
        int[] result = new int[fieldNames.length];

        for (int i = 0; i < fieldNames.length; i++) {
            result[i] = indexOrFail(this, fieldNames[i]);
        }

        return result;
    }


    @Override
    public final void accept(Context ctx) {
        ctx.visit(new QueryPartList>(fields));
    }

    @Override
    public final Clause[] clauses(Context ctx) {
        return null;
    }

    // -------------------------------------------------------------------------
    // XXX: List-like API
    // -------------------------------------------------------------------------

    final void add(Field f) {
        Field[] result = new Field[fields.length + 1];

        System.arraycopy(fields, 0, result, 0, fields.length);
        result[fields.length] = f;

        fields = result;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy