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