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

com.github.molcikas.photon.query.PhotonQuery Maven / Gradle / Ivy

package com.github.molcikas.photon.query;

import com.github.molcikas.photon.Photon;
import com.github.molcikas.photon.blueprints.entity.EntityBlueprint;
import com.github.molcikas.photon.blueprints.entity.EntityBlueprintBuilder;
import com.github.molcikas.photon.blueprints.entity.EntityClassDiscriminator;
import com.github.molcikas.photon.blueprints.entity.MappedClassBlueprint;
import com.github.molcikas.photon.blueprints.table.ColumnDataType;
import com.github.molcikas.photon.converters.Convert;
import com.github.molcikas.photon.converters.Converter;
import com.github.molcikas.photon.exceptions.PhotonException;
import org.apache.commons.lang3.StringUtils;

import java.lang.reflect.Field;
import java.sql.Connection;
import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.stream.Collectors;

public class PhotonQuery
{
    private static final String parameterRegex = "'?\"?:[A-Za-z0-9_]+'?\"?";
    private static final Pattern parameterPattern = Pattern.compile(parameterRegex);

    private final String sqlText;
    private final boolean populateGeneratedKeys;
    private final Connection connection;
    private final Photon photon;
    private final List mappedClasses;

    private List parameters;
    private EntityClassDiscriminator entityClassDiscriminator;
    private List generatedKeys;

    private final Map customFieldHydraters;

    public PhotonQuery(String sqlText, boolean populateGeneratedKeys, Connection connection, Photon photon)
    {
        this.sqlText = sqlText;
        this.populateGeneratedKeys = populateGeneratedKeys;
        this.connection = connection;
        this.photon = photon;
        this.parameters = new ArrayList<>();
        this.mappedClasses = new ArrayList<>();

        Matcher parameterMatcher = parameterPattern.matcher(sqlText);
        while(parameterMatcher.find())
        {
            String parameterName = sqlText.substring(parameterMatcher.start() + 1, parameterMatcher.end());
            parameters.add(new PhotonSqlParameter(parameterName));
        }

        this.customFieldHydraters = new HashMap<>();
    }

    /**
     * Retrieve the list of generated keys. This should only be called if populateGeneratedKeys is true for this query,
     * otherwise an exception will be thrown.
     *
     * @return - The list of generated keys.
     */
    public List getGeneratedKeys()
    {
        if(!populateGeneratedKeys)
        {
            throw new PhotonException("Cannot get generated keys because the query was created with populateGeneratedKeys set to false.");
        }
        return generatedKeys;
    }

    /**
     * Adds a parameter to the current query.
     *
     * @param parameter - The name of the parameter. Must match the name used in the SQL text for this query.
     * @param value - The parameter value
     * @return - The photon query (for chaining)
     */
    public PhotonQuery addParameter(String parameter, Object value)
    {
        boolean foundMatch = false;

        for(PhotonSqlParameter photonSqlParameter : parameters)
        {
            if(StringUtils.equals(photonSqlParameter.getName(), parameter))
            {
                photonSqlParameter.assignValue(value, photon.getOptions());
                foundMatch = true;
            }
        }
        if(!foundMatch)
        {
            throw new PhotonException("The parameter '%s' is not in the SQL query: \n%s", parameter, sqlText);
        }
        return this;
    }

    public PhotonQuery withMappedClass(Class mappedClass)
    {
        mappedClasses.add(new MappedClassBlueprint(mappedClass, true, null));
        return this;
    }

    public PhotonQuery withMappedClass(Class mappedClass, List includedFields)
    {
        mappedClasses.add(new MappedClassBlueprint(mappedClass, false, includedFields));
        return this;
    }

    public PhotonQuery withClassDiscriminator(EntityClassDiscriminator entityClassDiscriminator)
    {
        this.entityClassDiscriminator = entityClassDiscriminator;
        return this;
    }

    /**
     * Adds a parameter to the current query.
     *
     * @param parameter - The name of the parameter. Must match the name used in the SQL text for this query.
     * @param value - The parameter value
     * @param dataType - the data type of parameter. See java.sql.Types.
     * @return - The photon query (for chaining)
     */
    public PhotonQuery addParameter(String parameter, Object value, ColumnDataType dataType)
    {
        boolean foundMatch = false;

        for(PhotonSqlParameter photonSqlParameter : parameters)
        {
            if(StringUtils.equals(photonSqlParameter.getName(), parameter))
            {
                photonSqlParameter.assignValue(value, dataType);
                foundMatch = true;
            }
        }
        if(!foundMatch)
        {
            throw new PhotonException("The parameter '%s' is not in the SQL query: \n%s", parameter, sqlText);
        }
        return this;
    }

    /**
     * Execute the query and return a scalar value. The value is the first value in the first row.
     *
     * @param scalarClass - The scalar value's class
     * @param  - The scalar value's class
     * @return - The scalar value, or null if there were no results
     */
    public  T fetchScalar(Class scalarClass)
    {
        PhotonPreparedStatement photonPreparedStatement = prepareStatement();
        List results = photonPreparedStatement.executeQuery();
        if(results.isEmpty())
        {
            return null;
        }
        Object firstValue = results.get(0).getFirstValue();
        Converter converter = Convert.getConverterIfExists(scalarClass);
        if(converter == null)
        {
            return (T) firstValue;
        }
        return converter.convert(firstValue);
    }

    /**
     * Execute the query and return a scalar list. The list consists of the first value in each row.
     *
     * @param scalarClass - The scalar value's class
     * @param  - The scalar value's class
     * @return - The scalar value list
     */
    public  List fetchScalarList(Class scalarClass)
    {
        PhotonPreparedStatement photonPreparedStatement = prepareStatement();
        List results = photonPreparedStatement.executeQuery();
        if(results.isEmpty())
        {
            return Collections.emptyList();
        }
        List databaseValues = results.stream().map(PhotonQueryResultRow::getFirstValue).collect(Collectors.toList());
        Converter converter = Convert.getConverterIfExists(scalarClass);
        if(converter == null)
        {
            return databaseValues;
        }
        return (List) databaseValues.stream().map(v -> converter.convert(v)).collect(Collectors.toList());
    }

    /**
     * Executes the query and returns the result mapped to the specified class. Only the first row is mapped. Columns
     * are mapped to fields by name.
     *
     * @param classToFetch - The class to map the results into
     * @param  - The class to map the results into
     * @return - An instance of the class with the values in the result set, or null if there were no results
     */
    public  T fetch(Class classToFetch)
    {
        List entities = fetchList(classToFetch);
        return entities.size() > 0 ? entities.get(0) : null;
    }

    /**
     * Executes the query and returns the result with each row mapped to the specified class. Columns are mapped
     * to fields by name.
     *
     * @param classToFetch - The class to map the results into
     * @param  - The class to map the results into
     * @return - A list of instances of the class with the values in the result set
     */
    public  List fetchList(Class classToFetch)
    {
        PhotonPreparedStatement photonPreparedStatement = prepareStatement();

        EntityBlueprintBuilder entityBlueprintBuilder = new EntityBlueprintBuilder(classToFetch, photon);
        entityBlueprintBuilder.withClassDiscriminator(entityClassDiscriminator);

        for(MappedClassBlueprint blueprint : mappedClasses)
        {
            entityBlueprintBuilder.withMappedClass(
                blueprint.getMappedClass(),
                blueprint.getIncludedFields().stream().map(Field::getName).collect(Collectors.toList())
            );
        }

        for(Map.Entry entry : customFieldHydraters.entrySet())
        {
            entityBlueprintBuilder.withFieldHydrater(entry.getKey(), entry.getValue());
        }

        EntityBlueprint entityBlueprint = entityBlueprintBuilder.build();

        List rows = photonPreparedStatement
            .executeQuery(entityBlueprint.getAllColumnNames());
        List> populatedEntities = rows
            .stream()
            .map(r -> new PopulatedEntity(entityBlueprint, r, false))
            .collect(Collectors.toList());

        return populatedEntities
            .stream()
            .map(PopulatedEntity::getEntityInstance)
            .collect(Collectors.toList());
    }

    /**
     * Executes the query and returns the number of rows updated.
     *
     * @return - The number of rows updated
     */
    public int executeUpdate()
    {
        PhotonPreparedStatement photonPreparedStatement = prepareStatement();
        return photonPreparedStatement.executeUpdate();
    }

    /**
     * Executes the query and returns the number of rows inserted.
     *
     * @return - The number of rows inserted
     */
    public int executeInsert()
    {
        PhotonPreparedStatement photonPreparedStatement = prepareStatement();
        int rowsUpdated = photonPreparedStatement.executeInsert();
        if(populateGeneratedKeys)
        {
            generatedKeys = photonPreparedStatement.getGeneratedKeys();
        }
        return rowsUpdated;
    }

    /**
     * Sets a custom converter for hydrating a database value into an entity field value.
     *
     * @param fieldName - the entity field name
     * @param fieldHydrater - the converter for doing the field hydration
     * @return - builder for chaining
     */
    public PhotonQuery withFieldHydrater(String fieldName, Converter fieldHydrater)
    {
        customFieldHydraters.put(fieldName, fieldHydrater);
        return this;
    }

    List getParameters()
    {
        return parameters;
    }

    String getSqlTextWithQuestionMarks()
    {
        return sqlText.replaceAll(parameterRegex, "?");
    }

    private PhotonPreparedStatement prepareStatement()
    {
        PhotonPreparedStatement photonPreparedStatement = new PhotonPreparedStatement(
            getSqlTextWithQuestionMarks(),
            populateGeneratedKeys,
            connection,
            photon.getOptions()
        );

        for(PhotonSqlParameter photonSqlParameter : parameters)
        {
            photonPreparedStatement.setNextParameter(photonSqlParameter);
        }

        return photonPreparedStatement;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy