FR [ 1981760 ] Improve Query class
This commit is contained in:
parent
8168ac255b
commit
16e64af5c4
|
@ -20,8 +20,6 @@ package org.compiere.model;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import org.compiere.util.Env;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* Iterator implementation to fetch PO one at a time using a prefetch ID list.
|
* Iterator implementation to fetch PO one at a time using a prefetch ID list.
|
||||||
|
@ -89,15 +87,12 @@ public class POIterator<T extends PO> implements Iterator<T> {
|
||||||
public T get(int index) {
|
public T get(int index) {
|
||||||
if (index <= (idList.size() - 1)) {
|
if (index <= (idList.size() - 1)) {
|
||||||
Object[] ids = idList.get(index);
|
Object[] ids = idList.get(index);
|
||||||
if (ids.length == 1 && ids[0] instanceof Integer) {
|
if (ids.length == 1 && (ids[0] instanceof Number)) {
|
||||||
return (T) table.getPO((Integer)ids[0], trxName);
|
return (T) table.getPO( ((Number)ids[0]).intValue(), trxName);
|
||||||
} else {
|
} else {
|
||||||
if (keyWhereClause == null) {
|
if (keyWhereClause == null) {
|
||||||
String[] keys = table.getKeyColumns();
|
String[] keys = table.getKeyColumns();
|
||||||
POInfo info = POInfo.getPOInfo(Env.getCtx(), table.getAD_Table_ID(), trxName);
|
StringBuffer sqlBuffer = new StringBuffer();
|
||||||
if (info == null) return null;
|
|
||||||
StringBuffer sqlBuffer = info.buildSelect();
|
|
||||||
sqlBuffer.append(" WHERE ");
|
|
||||||
for (int i = 0; i < keys.length; i++) {
|
for (int i = 0; i < keys.length; i++) {
|
||||||
if (i > 0)
|
if (i > 0)
|
||||||
sqlBuffer.append(" AND ");
|
sqlBuffer.append(" AND ");
|
||||||
|
|
|
@ -22,8 +22,8 @@ import java.sql.ResultSet;
|
||||||
import java.sql.SQLException;
|
import java.sql.SQLException;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.List;
|
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
|
import java.util.List;
|
||||||
import java.util.logging.Level;
|
import java.util.logging.Level;
|
||||||
|
|
||||||
import org.compiere.util.CLogger;
|
import org.compiere.util.CLogger;
|
||||||
|
@ -60,6 +60,12 @@ public class Query {
|
||||||
this.trxName = trxName;
|
this.trxName = trxName;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Query(String tableName, String whereClause, String trxName) {
|
||||||
|
this(MTable.get(Env.getCtx(), tableName), whereClause, trxName);
|
||||||
|
if (this.table == null)
|
||||||
|
throw new IllegalArgumentException("Table Name Not Found - "+tableName);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set query parameters
|
* Set query parameters
|
||||||
* @param parameters
|
* @param parameters
|
||||||
|
@ -108,33 +114,14 @@ public class Query {
|
||||||
*/
|
*/
|
||||||
public <T extends PO> List<T> list() throws DBException {
|
public <T extends PO> List<T> list() throws DBException {
|
||||||
List<T> list = new ArrayList<T>();
|
List<T> list = new ArrayList<T>();
|
||||||
|
String sql = buildSQL(null);
|
||||||
POInfo info = POInfo.getPOInfo(Env.getCtx(), table.getAD_Table_ID(), trxName);
|
|
||||||
if (info == null) return null;
|
|
||||||
StringBuffer sqlBuffer = info.buildSelect();
|
|
||||||
if (whereClause != null && whereClause.trim().length() > 0)
|
|
||||||
sqlBuffer.append(" WHERE ").append(whereClause);
|
|
||||||
if (orderBy != null && orderBy.trim().length() > 0)
|
|
||||||
sqlBuffer.append(" Order By ").append(orderBy);
|
|
||||||
String sql = sqlBuffer.toString();
|
|
||||||
if (applyAccessFilter) {
|
|
||||||
MRole role = MRole.getDefault();
|
|
||||||
sql = role.addAccessSQL(sql, table.getTableName(), true, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
PreparedStatement pstmt = null;
|
PreparedStatement pstmt = null;
|
||||||
ResultSet rs = null;
|
ResultSet rs = null;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
pstmt = DB.prepareStatement (sql, trxName);
|
pstmt = DB.prepareStatement (sql, trxName);
|
||||||
if (parameters != null && parameters.length > 0)
|
rs = createResultSet(pstmt);
|
||||||
{
|
|
||||||
for (int i = 0; i < parameters.length; i++)
|
|
||||||
{
|
|
||||||
pstmt.setObject(i+1, parameters[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
rs = pstmt.executeQuery ();
|
|
||||||
while (rs.next ())
|
while (rs.next ())
|
||||||
{
|
{
|
||||||
T po = (T)table.getPO(rs, trxName);
|
T po = (T)table.getPO(rs, trxName);
|
||||||
|
@ -152,12 +139,69 @@ public class Query {
|
||||||
return list;
|
return list;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return first PO that match query criteria
|
||||||
|
* @return PO
|
||||||
|
* @throws DBException
|
||||||
|
*/
|
||||||
|
public <T extends PO> T first() throws DBException {
|
||||||
|
T po = null;
|
||||||
|
String sql = buildSQL(null);
|
||||||
|
|
||||||
|
PreparedStatement pstmt = null;
|
||||||
|
ResultSet rs = null;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
pstmt = DB.prepareStatement (sql, trxName);
|
||||||
|
rs = createResultSet(pstmt);
|
||||||
|
if (rs.next ())
|
||||||
|
{
|
||||||
|
po = (T)table.getPO(rs, trxName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (SQLException e)
|
||||||
|
{
|
||||||
|
log.log(Level.SEVERE, sql, e);
|
||||||
|
throw new DBException(e);
|
||||||
|
} finally {
|
||||||
|
DB.close(rs, pstmt);
|
||||||
|
rs = null; pstmt = null;
|
||||||
|
}
|
||||||
|
return po;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Count items that match query criteria
|
||||||
|
* @return count
|
||||||
|
* @throws DBException
|
||||||
|
*/
|
||||||
|
public int count() throws DBException
|
||||||
|
{
|
||||||
|
int count = -1;
|
||||||
|
String sql = buildSQL(new StringBuffer("SELECT COUNT(*) FROM ").append(table.getTableName()));
|
||||||
|
PreparedStatement pstmt = null;
|
||||||
|
ResultSet rs = null;
|
||||||
|
try {
|
||||||
|
pstmt = DB.prepareStatement(sql, this.trxName);
|
||||||
|
rs = createResultSet(pstmt);
|
||||||
|
rs.next();
|
||||||
|
count = rs.getInt(1);
|
||||||
|
}
|
||||||
|
catch (SQLException e) {
|
||||||
|
throw new DBException(e);
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
DB.close(rs, pstmt);
|
||||||
|
}
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return an Iterator implementation to fetch one PO at a time. The implementation first retrieve
|
* Return an Iterator implementation to fetch one PO at a time. The implementation first retrieve
|
||||||
* all IDS that match the query criteria and issue sql query to fetch the PO when caller want to
|
* all IDS that match the query criteria and issue sql query to fetch the PO when caller want to
|
||||||
* fetch the next PO. This minimize memory usage but it is slower than the list method.
|
* fetch the next PO. This minimize memory usage but it is slower than the list method.
|
||||||
* @return Iterator
|
* @return Iterator
|
||||||
* @throws SQLException
|
* @throws DBException
|
||||||
*/
|
*/
|
||||||
public <T extends PO> Iterator<T> iterate() throws DBException {
|
public <T extends PO> Iterator<T> iterate() throws DBException {
|
||||||
String[] keys = table.getKeyColumns();
|
String[] keys = table.getKeyColumns();
|
||||||
|
@ -168,29 +212,15 @@ public class Query {
|
||||||
sqlBuffer.append(keys[i]);
|
sqlBuffer.append(keys[i]);
|
||||||
}
|
}
|
||||||
sqlBuffer.append(" FROM ").append(table.getTableName());
|
sqlBuffer.append(" FROM ").append(table.getTableName());
|
||||||
if (whereClause != null && whereClause.trim().length() > 0)
|
String sql = buildSQL(sqlBuffer);
|
||||||
sqlBuffer.append(" WHERE ").append(whereClause);
|
|
||||||
if (orderBy != null && orderBy.trim().length() > 0)
|
|
||||||
sqlBuffer.append(" Order By ").append(orderBy);
|
|
||||||
String sql = sqlBuffer.toString();
|
|
||||||
if (applyAccessFilter) {
|
|
||||||
MRole role = MRole.getDefault();
|
|
||||||
sql = role.addAccessSQL(sql, table.getTableName(), true, false);
|
|
||||||
}
|
|
||||||
PreparedStatement pstmt = null;
|
PreparedStatement pstmt = null;
|
||||||
ResultSet rs = null;
|
ResultSet rs = null;
|
||||||
List<Object[]> idList = new ArrayList<Object[]>();
|
List<Object[]> idList = new ArrayList<Object[]>();
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
pstmt = DB.prepareStatement (sql, trxName);
|
pstmt = DB.prepareStatement (sql, trxName);
|
||||||
if (parameters != null && parameters.length > 0)
|
rs = createResultSet(pstmt);
|
||||||
{
|
|
||||||
for (int i = 0; i < parameters.length; i++)
|
|
||||||
{
|
|
||||||
pstmt.setObject(i+1, parameters[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
rs = pstmt.executeQuery ();
|
|
||||||
while (rs.next ())
|
while (rs.next ())
|
||||||
{
|
{
|
||||||
Object[] ids = new Object[keys.length];
|
Object[] ids = new Object[keys.length];
|
||||||
|
@ -215,12 +245,37 @@ public class Query {
|
||||||
* Return a simple wrapper over a jdbc resultset. It is the caller responsibility to
|
* Return a simple wrapper over a jdbc resultset. It is the caller responsibility to
|
||||||
* call the close method to release the underlying database resources.
|
* call the close method to release the underlying database resources.
|
||||||
* @return POResultSet
|
* @return POResultSet
|
||||||
* @throws SQLException
|
* @throws DBException
|
||||||
*/
|
*/
|
||||||
public <T extends PO> POResultSet<T> scroll() throws DBException {
|
public <T extends PO> POResultSet<T> scroll() throws DBException {
|
||||||
|
String sql = buildSQL(null);
|
||||||
|
PreparedStatement pstmt = null;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
pstmt = DB.prepareStatement (sql, trxName);
|
||||||
|
ResultSet rs = createResultSet(pstmt);
|
||||||
|
return new POResultSet<T>(table, pstmt, rs, trxName);
|
||||||
|
}
|
||||||
|
catch (SQLException e)
|
||||||
|
{
|
||||||
|
log.log(Level.SEVERE, sql, e);
|
||||||
|
throw new DBException(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build SQL Clause
|
||||||
|
* @param selectClause optional; if null the select clause will be build according to POInfo
|
||||||
|
* @return final SQL
|
||||||
|
*/
|
||||||
|
private final String buildSQL(StringBuffer selectClause) {
|
||||||
|
if (selectClause == null) {
|
||||||
POInfo info = POInfo.getPOInfo(Env.getCtx(), table.getAD_Table_ID(), trxName);
|
POInfo info = POInfo.getPOInfo(Env.getCtx(), table.getAD_Table_ID(), trxName);
|
||||||
if (info == null) return null;
|
if (info == null)
|
||||||
StringBuffer sqlBuffer = info.buildSelect();
|
throw new IllegalStateException("No POInfo found for AD_Table_ID="+table.getAD_Table_ID());
|
||||||
|
selectClause = info.buildSelect();
|
||||||
|
}
|
||||||
|
StringBuffer sqlBuffer = new StringBuffer(selectClause);
|
||||||
if (whereClause != null && whereClause.trim().length() > 0)
|
if (whereClause != null && whereClause.trim().length() > 0)
|
||||||
sqlBuffer.append(" WHERE ").append(whereClause);
|
sqlBuffer.append(" WHERE ").append(whereClause);
|
||||||
if (orderBy != null && orderBy.trim().length() > 0)
|
if (orderBy != null && orderBy.trim().length() > 0)
|
||||||
|
@ -230,11 +285,11 @@ public class Query {
|
||||||
MRole role = MRole.getDefault();
|
MRole role = MRole.getDefault();
|
||||||
sql = role.addAccessSQL(sql, table.getTableName(), true, false);
|
sql = role.addAccessSQL(sql, table.getTableName(), true, false);
|
||||||
}
|
}
|
||||||
|
return sql;
|
||||||
|
}
|
||||||
|
|
||||||
PreparedStatement pstmt = null;
|
private final ResultSet createResultSet (PreparedStatement pstmt) throws SQLException
|
||||||
try
|
|
||||||
{
|
{
|
||||||
pstmt = DB.prepareStatement (sql, trxName);
|
|
||||||
if (parameters != null && parameters.length > 0)
|
if (parameters != null && parameters.length > 0)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < parameters.length; i++)
|
for (int i = 0; i < parameters.length; i++)
|
||||||
|
@ -242,13 +297,7 @@ public class Query {
|
||||||
pstmt.setObject(i+1, parameters[i]);
|
pstmt.setObject(i+1, parameters[i]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ResultSet rs = pstmt.executeQuery ();
|
return pstmt.executeQuery();
|
||||||
return new POResultSet<T>(table, pstmt, rs, trxName);
|
|
||||||
}
|
|
||||||
catch (SQLException e)
|
|
||||||
{
|
|
||||||
log.log(Level.SEVERE, sql, e);
|
|
||||||
throw new DBException(e);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,6 +17,7 @@ public class FunctionalTestSuite {
|
||||||
suite.addTestSuite(POTest.class);
|
suite.addTestSuite(POTest.class);
|
||||||
suite.addTestSuite(MStorageTest.class);
|
suite.addTestSuite(MStorageTest.class);
|
||||||
suite.addTestSuite(MSysConfigTest.class);
|
suite.addTestSuite(MSysConfigTest.class);
|
||||||
|
suite.addTestSuite(QueryTest.class);
|
||||||
//$JUnit-END$
|
//$JUnit-END$
|
||||||
return suite;
|
return suite;
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,128 @@
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package test.functional;
|
||||||
|
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.compiere.model.MTable;
|
||||||
|
import org.compiere.model.POResultSet;
|
||||||
|
import org.compiere.model.Query;
|
||||||
|
import org.compiere.util.DBException;
|
||||||
|
|
||||||
|
import test.AdempiereTestCase;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test {@link org.compiere.model.Query} class
|
||||||
|
* @author Teo Sarca, SC ARHIPAC SERVICE SRL
|
||||||
|
*/
|
||||||
|
public class QueryTest extends AdempiereTestCase {
|
||||||
|
|
||||||
|
public void testQuery_NoTable() throws Exception {
|
||||||
|
boolean exThrowed = false;
|
||||||
|
try {
|
||||||
|
new Query("NO_TABLE_DEFINED", null, getTrxName());
|
||||||
|
}
|
||||||
|
catch (RuntimeException e) {
|
||||||
|
exThrowed = true;
|
||||||
|
}
|
||||||
|
assertTrue("No Error Was Throwed", exThrowed);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testList() throws Exception {
|
||||||
|
List<MTable> list = new Query("AD_Table", "TableName IN (?,?)", getTrxName())
|
||||||
|
.setParameters(new Object[]{"C_Invoice", "M_InOut"})
|
||||||
|
.setOrderBy("TableName")
|
||||||
|
.list();
|
||||||
|
assertEquals("Invalid list size", 2, list.size());
|
||||||
|
assertEquals("Invalid object 1", list.get(0).getTableName(), "C_Invoice");
|
||||||
|
assertEquals("Invalid object 2", list.get(1).getTableName(), "M_InOut");
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testScroll() throws Exception {
|
||||||
|
POResultSet<MTable> rs = null;
|
||||||
|
try {
|
||||||
|
rs = new Query("AD_Table", "TableName IN (?,?)", getTrxName())
|
||||||
|
.setParameters(new Object[]{"C_Invoice", "M_InOut"})
|
||||||
|
.setOrderBy("TableName")
|
||||||
|
.scroll();
|
||||||
|
int i = 0;
|
||||||
|
for(MTable t = rs.next(); t != null; t = rs.next()) {
|
||||||
|
if (i == 0) {
|
||||||
|
assertEquals("Invalid object "+i, "C_Invoice", t.getTableName());
|
||||||
|
}
|
||||||
|
else if (i == 1) {
|
||||||
|
assertEquals("Invalid object "+i, "M_InOut", t.getTableName());
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
assertFalse("More objects retrived than expected", true);
|
||||||
|
}
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
if (rs != null)
|
||||||
|
rs.close();
|
||||||
|
rs = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testIterate() throws Exception {
|
||||||
|
Iterator<MTable> it = new Query("AD_Table", "TableName IN (?,?)", getTrxName())
|
||||||
|
.setParameters(new Object[]{"C_Invoice", "M_InOut"})
|
||||||
|
.setOrderBy("TableName")
|
||||||
|
.iterate();
|
||||||
|
int i = 0;
|
||||||
|
while(it.hasNext()) {
|
||||||
|
MTable t = it.next();
|
||||||
|
if (i == 0) {
|
||||||
|
assertEquals("Invalid object "+i, "C_Invoice", t.getTableName());
|
||||||
|
}
|
||||||
|
else if (i == 1) {
|
||||||
|
assertEquals("Invalid object "+i, "M_InOut", t.getTableName());
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
assertFalse("More objects retrived than expected", true);
|
||||||
|
}
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testCount() throws Exception {
|
||||||
|
int count = new Query("AD_Table", "TableName IN (?,?)", getTrxName())
|
||||||
|
.setParameters(new Object[]{"C_Invoice", "M_InOut"})
|
||||||
|
.setOrderBy("TableName")
|
||||||
|
.count();
|
||||||
|
assertEquals("Invalid count", 2, count);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testCount_BadSQL() throws Exception {
|
||||||
|
boolean exThrowed = false;
|
||||||
|
try {
|
||||||
|
new Query("AD_Table", "TableName IN (?,?) AND BAD_SQL", getTrxName())
|
||||||
|
.setParameters(new Object[]{"C_Invoice", "M_InOut"})
|
||||||
|
.setOrderBy("TableName")
|
||||||
|
.count();
|
||||||
|
}
|
||||||
|
catch (DBException e) {
|
||||||
|
exThrowed = true;
|
||||||
|
}
|
||||||
|
assertTrue("No Error Was Throwed", exThrowed);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testCount_NoValues() throws Exception {
|
||||||
|
int count = new Query("AD_Table", "1=2", getTrxName()).count();
|
||||||
|
assertEquals("Counter should be ZERO", 0, count);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testFirst() throws Exception {
|
||||||
|
MTable t = new Query("AD_Table", "TableName IN (?,?)", getTrxName())
|
||||||
|
.setParameters(new Object[]{"C_Invoice", "M_InOut"})
|
||||||
|
.setOrderBy("TableName")
|
||||||
|
.first();
|
||||||
|
assertEquals("Invalid object", "C_Invoice", t.getTableName());
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue