src/de/uhilger/baselink/DBActor.java | ●●●●● patch | view | raw | blame | history | |
src/de/uhilger/baselink/Eraser.java | ●●●●● patch | view | raw | blame | history | |
src/de/uhilger/baselink/Inserter.java | ●●●●● patch | view | raw | blame | history | |
src/de/uhilger/baselink/ListConverter.java | ●●●●● patch | view | raw | blame | history | |
src/de/uhilger/baselink/PersistenceManager.java | ●●●●● patch | view | raw | blame | history | |
src/de/uhilger/baselink/Script.java | ●●●●● patch | view | raw | blame | history | |
src/de/uhilger/baselink/Selector.java | ●●●●● patch | view | raw | blame | history | |
src/de/uhilger/baselink/Updater.java | ●●●●● patch | view | raw | blame | history |
src/de/uhilger/baselink/DBActor.java
New file @@ -0,0 +1,11 @@ package de.uhilger.baselink; /** * * @author Ulrich Hilger */ public class DBActor { protected PersistenceManager pm; } src/de/uhilger/baselink/Eraser.java
New file @@ -0,0 +1,99 @@ package de.uhilger.baselink; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.SQLException; import java.util.logging.Level; import java.util.logging.Logger; /** * * @author Ulrich Hilger */ public class Eraser extends DBActor { private static final Logger logger = Logger.getLogger(Eraser.class.getName()); public Eraser(PersistenceManager pm) { this.pm = pm; } /** * Delete a given object from the database * * <p>Use this method for single deletes. In cases where * several subsequent deletes for objects of the same class * are required the use of method <code>delete(Object, Record)</code> * is recommended instead to minimise instantiation * overhead.</p> * * @param o object to delete * @return the deleted object */ public Object delete(Object o) { return delete(o, new GenericRecord(o.getClass())); } /** * Delete a given object from the database * * <p>Use this method for single deletes. In cases where * several subsequent deletes for objects of the same class * are required the use of method <code>delete(Connection, Object, Record)</code> * is recommended instead to minimise instantiation * overhead.</p> * * @param c the connection to use, expected to be open and established * @param o object to delete * @return the deleted object */ public Object delete(Connection c, Object o) { return delete(c, o, new GenericRecord(o.getClass())); } /** * Delete a given object from the database * @param o object to delete * @param record reference to object to use to map between object and database * @return the deleted object */ public Object delete(Object o, Record record) { Connection c = null; try { c = pm.getConnection(); o = delete(c, o, record); c.close(); c = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeConnectionFinally(c); } return o; } /** * Delete a given object from the database * @param c the connection to use, expected to be open and established * @param o object to delete * @param record reference to object to use to map between object and database * @return the deleted object */ public Object delete(Connection c, Object o, Record record) { Object deletedObject = null; PreparedStatement ps = null; try { ps = record.getDeleteStatment(c, o); ps.executeUpdate(); ps.close(); ps = null; deletedObject = o; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeStatementFinally(ps); } return deletedObject; } } src/de/uhilger/baselink/Inserter.java
New file @@ -0,0 +1,100 @@ package de.uhilger.baselink; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.SQLException; import java.util.logging.Level; import java.util.logging.Logger; /** * * @author Ulrich Hilger */ public class Inserter extends DBActor { private static final Logger logger = Logger.getLogger(Inserter.class.getName()); public Inserter(PersistenceManager pm) { this.pm = pm; } /** * Add an object to the database * * <p>Use this method for single inserts. In cases where * several subsequent inserts for objects of the same class * are required the use of method <code>insert(Object, Record)</code> * is recommended instead to minimise instantiation * overhead.</p> * * @param o the object to add * @return the added object */ public Object insert(Object o) { return insert(o, new GenericRecord(o.getClass())); } /** * Add an object to the database * * <p>Use this method for single inserts. In cases where * several subsequent inserts for objects of the same class * are required the use of method <code>insert(Connection, Object, Record)</code> * is recommended instead to minimise instantiation * overhead.</p> * * @param c the connection to use, expected to be open and established * @param o the object to add * @return the object added to the database */ public Object insert(Connection c, Object o) { return insert(c, o, new GenericRecord(o.getClass())); } /** * Add an object to the database * @param o object to add * @param record reference to object to use to map between object and database * @return the added object */ public Object insert(Object o, Record record) { Connection c = null; try { c = pm.getConnection(); o = insert(c, o, record); c.close(); c = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeConnectionFinally(c); } return o; } /** * Add an object to the database * @param c the connection to use, expected to be open and established * @param o object to add * @param record reference to object to use to map between object and database * @return the object that was added */ public Object insert(Connection c, Object o, Record record) { Object addedObject = null; PreparedStatement ps = null; try { ps = record.getInsertStatment(c, o); ps.executeUpdate(); ps.close(); ps = null; addedObject = o; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeStatementFinally(ps); } return addedObject; } } src/de/uhilger/baselink/ListConverter.java
New file @@ -0,0 +1,130 @@ package de.uhilger.baselink; import static de.uhilger.baselink.PersistenceManager.NULL_STR; import java.sql.Blob; import java.sql.ResultSet; import java.sql.ResultSetMetaData; import java.sql.SQLException; import java.sql.Types; import java.util.ArrayList; import java.util.List; import java.util.Map; import java.util.TreeMap; import java.util.logging.Logger; /** * * @author Ulrich Hilger */ public class ListConverter { private static final Logger logger = Logger.getLogger(ListConverter.class.getName()); /** * Helper method that converts a ResultSet into a list of lists, one per row, * each row is a list of strings * * @param rs result set to convert * @param includeBlobs true when blob columns should be returned, false if not * @return a list of list objects, one for each record. An element in the * list can be accessed with list.get(recordno).get(fieldno), each element is of type String. * This first row has the field names * @throws java.sql.SQLException */ public List<List<String>> toList(ResultSet rs, boolean includeBlobs) throws SQLException { List<List<String>> rows = new ArrayList<>(); ResultSetMetaData meta = rs.getMetaData(); int columnCount = meta.getColumnCount(); List<String> header = new ArrayList<>(); for (int i = 1; i <= columnCount; i++) { header.add(meta.getColumnName(i)); } rows.add(header); while (rs.next()) { List<String> row = new ArrayList<>(); for (int i = 1; i <= columnCount; i++) { String data = null; if (meta.getColumnType(i) == Types.BLOB) { if (includeBlobs) { Blob blob = rs.getBlob(i); if(blob != null) { data = new String(blob.getBytes((long) 1, (int) blob.length())); } } } else { Object o = rs.getObject(i); if(o != null) { data = o.toString(); } else { data = NULL_STR; } logger.finest(data); } row.add(data); } rows.add(row); } return rows; } /** * Helper method that maps a ResultSet into a list of maps, one per row * * @param rs database content to transform * @param wantedColumnNames list of columns names to include in the result map * @return list of maps, one per column row, with column names as keys * @throws SQLException if the connection fails */ public List<Map<String, Object>> toList(ResultSet rs, List<String> wantedColumnNames) throws SQLException { // TODO BLOB handling List<Map<String, Object>> rows = new ArrayList<>(); int numWantedColumns = wantedColumnNames.size(); while (rs.next()) { Map<String, Object> row = new TreeMap<>(); for (int i = 0; i < numWantedColumns; ++i) { String columnName = wantedColumnNames.get(i); Object value = rs.getObject(columnName); if (value != null) { row.put(columnName, value); //Logger.getLogger(Logger.GLOBAL_LOGGER_NAME).finest(columnName + " " + value); } else { row.put(columnName, ""); } } rows.add(row); } return rows; } /** * Helper method that converts a ResultSet into a list of maps, one per row * * @param rs database content to transform * @return list of maps, one per row, with column name as the key * @throws SQLException if the connection fails */ public List<Map<String, Object>> toList(ResultSet rs) throws SQLException { List<String> wantedColumnNames = getColumnNames(rs); return toList(rs, wantedColumnNames); } /** * Return all column names as a list of strings * * @param database query result set * @return list of column name strings * @throws SQLException if the query fails */ private List<String> getColumnNames(ResultSet rs) throws SQLException { List<String> columnNames = new ArrayList<String>(); ResultSetMetaData meta = rs.getMetaData(); int numColumns = meta.getColumnCount(); for (int i = 1; i <= numColumns; ++i) { String cName = meta.getColumnName(i); columnNames.add(cName); } return columnNames; } } src/de/uhilger/baselink/PersistenceManager.java
@@ -17,19 +17,14 @@ */ package de.uhilger.baselink; import java.sql.Blob; import java.sql.Connection; import java.sql.DriverManager; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.ResultSetMetaData; import java.sql.SQLException; import java.sql.Statement; import java.sql.Types; import java.util.ArrayList; import java.util.List; import java.util.Map; import java.util.TreeMap; import java.util.logging.Level; import java.util.logging.Logger; @@ -166,7 +161,7 @@ * @return the deleted object */ public Object delete(Object o) { return delete(o, new GenericRecord(o.getClass())); return new Eraser(this).delete(o, new GenericRecord(o.getClass())); } /** @@ -183,7 +178,7 @@ * @return the deleted object */ public Object delete(Connection c, Object o) { return delete(c, o, new GenericRecord(o.getClass())); return new Eraser(this).delete(c, o, new GenericRecord(o.getClass())); } /** @@ -193,18 +188,7 @@ * @return the deleted object */ public Object delete(Object o, Record record) { Connection c = null; try { c = getConnection(); o = delete(c, o, record); c.close(); c = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeConnectionFinally(c); } return o; return new Eraser(this).delete(o, record); } /** @@ -215,20 +199,7 @@ * @return the deleted object */ public Object delete(Connection c, Object o, Record record) { Object deletedObject = null; PreparedStatement ps = null; try { ps = record.getDeleteStatment(c, o); ps.executeUpdate(); ps.close(); ps = null; deletedObject = o; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeStatementFinally(ps); } return deletedObject; return new Eraser(this).delete(c, o, record); } /* ---------------------- inserts -------------------- */ @@ -246,7 +217,7 @@ * @return the added object */ public Object insert(Object o) { return insert(o, new GenericRecord(o.getClass())); return new Inserter(this).insert(o, new GenericRecord(o.getClass())); } /** @@ -263,7 +234,7 @@ * @return the object added to the database */ public Object insert(Connection c, Object o) { return insert(c, o, new GenericRecord(o.getClass())); return new Inserter(this).insert(c, o, new GenericRecord(o.getClass())); } /** @@ -273,18 +244,7 @@ * @return the added object */ public Object insert(Object o, Record record) { Connection c = null; try { c = getConnection(); o = insert(c, o, record); c.close(); c = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeConnectionFinally(c); } return o; return new Inserter(this).insert(o, record); } /** @@ -295,20 +255,7 @@ * @return the object that was added */ public Object insert(Connection c, Object o, Record record) { Object addedObject = null; PreparedStatement ps = null; try { ps = record.getInsertStatment(c, o); ps.executeUpdate(); ps.close(); ps = null; addedObject = o; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeStatementFinally(ps); } return addedObject; return new Inserter(this).insert(c, o, record); } /* --------------------------------- updates --------------------- */ @@ -326,7 +273,7 @@ * @return the object that was updated */ public Object update(Object o) { return update(o, new GenericRecord(o.getClass())); return new Updater(this).update(o, new GenericRecord(o.getClass())); } /** @@ -343,7 +290,7 @@ * @return the object that was updated */ public Object update(Connection c, Object o) { return update(c, o, new GenericRecord(o.getClass())); return new Updater(this).update(c, o, new GenericRecord(o.getClass())); } /** @@ -353,18 +300,7 @@ * @return the object that was updated */ public Object update(Object o, Record record) { Connection c = null; try { c = getConnection(); o = update(c, o, record); c.close(); c = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeConnectionFinally(c); } return o; return new Updater(this).update(o, record); } /** @@ -375,20 +311,7 @@ * @return the object that was updated */ public Object update(Connection c, Object o, Record record) { Object updatedObject = null; PreparedStatement ps = null; try { ps = record.getUpdateStatment(c, o); ps.executeUpdate(); ps.close(); ps = null; updatedObject = o; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeStatementFinally(ps); } return updatedObject; return new Updater(this).update(c, o, record); } /* --------------- selects ---------------- */ @@ -400,7 +323,7 @@ * @return a list of objects that match the given query */ public List<Object> select(String sql, Record record) { return select(sql, record, Record.WITH_BLOBS); return new Selector(this).select(sql, record, Record.WITH_BLOBS); } /** @@ -411,19 +334,7 @@ * @return a list of objects that match the given query */ public List<Object> select(String sql, Record record, boolean includeBlobs) { Connection c = null; ArrayList<Object> list = new ArrayList<Object>(); try { c = getConnection(); list = (ArrayList<Object>) select(c, sql, record, includeBlobs); c.close(); c = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeConnectionFinally(c); } return list; return new Selector(this).select(sql, record, includeBlobs); } /** @@ -435,19 +346,7 @@ * @return a list of objects that match the given query */ public List<Object> select(String sql, Record record, boolean includeBlobs, Object... params) { Connection c = null; ArrayList<Object> list = new ArrayList<Object>(); try { c = getConnection(); list = (ArrayList<Object>) select(c, sql, record, includeBlobs, params); c.close(); c = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeConnectionFinally(c); } return list; return new Selector(this).select(sql, record, includeBlobs, params); } /** @@ -459,29 +358,7 @@ * @return a list of objects that match the given query */ public List<Object> select(Connection c, String sql, Record record, boolean includeBlobs) { PreparedStatement ps = null; ResultSet rs = null; ArrayList<Object> list = new ArrayList<Object>(); try { ps = c.prepareStatement(sql, ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); rs = ps.executeQuery(); if (rs.first()) { while (!rs.isAfterLast()) { list.add(record.toObject(rs, includeBlobs)); rs.next(); } } rs.close(); rs = null; ps.close(); ps = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeResultSetFinally(rs); closeStatementFinally(ps); } return list; return new Selector(this).select(c, sql, record, includeBlobs); } /** @@ -494,32 +371,8 @@ * @return a list of objects that match the given query */ public List<Object> select(Connection c, String sql, Record record, boolean includeBlobs, Object... params) { PreparedStatement ps = null; ResultSet rs = null; ArrayList<Object> list = new ArrayList<Object>(); try { //ps = c.prepareStatement(sql, ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); ps = buildQuery(c, sql, params); rs = ps.executeQuery(); if (rs.first()) { while (!rs.isAfterLast()) { list.add(record.toObject(rs, includeBlobs)); rs.next(); return new Selector(this).select(c, sql, record, includeBlobs, params); } } rs.close(); rs = null; ps.close(); ps = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeResultSetFinally(rs); closeStatementFinally(ps); } return list; } /** * Select a list of objects that match a given SQL statement @@ -528,20 +381,7 @@ * list can be accessed with list.get(recordno).get("fieldname") */ public List<Map<String, Object>> select(String sql) { //Logger.getLogger(Logger.GLOBAL_LOGGER_NAME).finest(sql); Connection c = null; List<Map<String, Object>> list = null; try { c = getConnection(); list = select(c, sql); c.close(); c = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeConnectionFinally(c); } return list; return new Selector(this).select(sql); } /** @@ -552,19 +392,7 @@ * list can be accessed with list.get(recordno).get(fieldno), each element is of type String */ public List<List<String>> select(String sql, boolean includeBlobs) { Connection c = null; List<List<String>> list = null; try { c = getConnection(); list = select(c, sql, includeBlobs); c.close(); c = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeConnectionFinally(c); } return list; return new Selector(this).select(sql, includeBlobs); } /** @@ -575,24 +403,7 @@ * list can be accessed with list.get(recordno).get("fieldname") */ public List<Map<String, Object>> select(Connection c, String sql) { PreparedStatement ps = null; ResultSet rs = null; List<Map<String, Object>> list = null; try { ps = c.prepareStatement(sql, ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); rs = ps.executeQuery(); list = toList(rs); rs.close(); rs = null; ps.close(); ps = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeResultSetFinally(rs); closeStatementFinally(ps); } return list; return new Selector(this).select(c, sql); } /** @@ -604,24 +415,7 @@ * list can be accessed with list.get(recordno).get(fieldno), each element is of type String */ public List<List<String>> select(Connection c, String sql, boolean includeBlobs) { PreparedStatement ps = null; ResultSet rs = null; List<List<String>> list = null; try { ps = c.prepareStatement(sql, ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); rs = ps.executeQuery(); list = toList(rs, includeBlobs); rs.close(); rs = null; ps.close(); ps = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeResultSetFinally(rs); closeStatementFinally(ps); } return list; return new Selector(this).select(c, sql, includeBlobs); } /** @@ -633,19 +427,7 @@ * list can be accessed with list.get(recordno).get(fieldno), each element is of type String */ public List<List<String>> select(String sql, boolean includeBlobs, Object... params) { Connection c = null; List<List<String>> list = null; try { c = getConnection(); list = select(c, sql, includeBlobs, params); c.close(); c = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeConnectionFinally(c); } return list; return new Selector(this).select(sql, includeBlobs, params); } /** @@ -658,24 +440,7 @@ * list can be accessed with list.get(recordno).get(fieldno), each element is of type String */ public List<List<String>> select(Connection c, String sql, boolean includeBlobs, Object... params) { PreparedStatement ps = null; ResultSet rs = null; List<List<String>> list = null; try { ps = buildQuery(c, sql, params); rs = ps.executeQuery(); list = toList(rs, includeBlobs); rs.close(); rs = null; ps.close(); ps = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeResultSetFinally(rs); closeStatementFinally(ps); } return list; return new Selector(this).select(c, sql, includeBlobs, params); } /* ------------------ generic SQL execution ---------- */ @@ -686,19 +451,7 @@ * @return a list of generated keys */ public List<Map<String, Object>> executeWithKeys(String sql) { List<Map<String, Object>> keys = null; Connection c = null; try { c = getConnection(); keys = executeWithKeys(c, sql); c.close(); c = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeConnectionFinally(c); } return keys; return new Script(this).executeWithKeys(sql); } /** @@ -708,20 +461,7 @@ * @return a list of generated keys */ public List<Map<String, Object>> executeWithKeys(Connection c, String sql) { List<Map<String, Object>> keys = null; Statement s = null; try { s = c.createStatement(); s.executeUpdate(sql, Statement.RETURN_GENERATED_KEYS); keys = toList(s.getGeneratedKeys()); s.close(); s = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeStatementFinally(s); } return keys; return new Script(this).executeWithKeys(c, sql); } /** @@ -730,19 +470,7 @@ * @return the number of records affected by this statement or -1 if none */ public int execute(String sql) { int numRows = -1; Connection c = null; try { c = getConnection(); numRows = execute(c, sql); c.close(); c = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeConnectionFinally(c); } return numRows; return new Script(this).execute(sql); } /** @@ -752,19 +480,7 @@ * @return the number of records affected by this statement or -1 if none */ public int execute(Connection c, String sql) { int numRows = -1; Statement s = null; try { s = c.createStatement(); numRows = s.executeUpdate(sql); s.close(); s = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeStatementFinally(s); } return numRows; return new Script(this).execute(c, sql); } /** @@ -774,19 +490,7 @@ * @return the number of records affected by this statement or -1 if none */ public int execute(String sql, Object... params) { int numRows = -1; Connection c = null; try { c = getConnection(); numRows = execute(c, sql, params); c.close(); c = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeConnectionFinally(c); } return numRows; return new Script(this).execute(sql, params); } /** @@ -797,23 +501,7 @@ * @return the number of records affected by this statement or -1 if none */ public int execute(Connection c, String sql, Object... params) { int numRows = -1; Statement s = null; try { //s = c.createStatement(); s = buildQuery(c, sql, params); if(s != null && s instanceof PreparedStatement) { PreparedStatement ps = (PreparedStatement) s; numRows = ps.executeUpdate(); ps.close(); } s = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeStatementFinally(s); } return numRows; return new Script(this).execute(c, sql, params); } /** @@ -826,24 +514,7 @@ * about the number of affected rows */ public int[] executeScript(String sqlScript) { int[] ergebnisse = null; Connection c = null; Statement s = null; try { c = getConnection(); s = c.createStatement(); String[] sqlKommandos = sqlScript.split(";"); for(int i = 0; i < sqlKommandos.length; i++) { s.addBatch(sqlKommandos[i]); } ergebnisse = s.executeBatch(); } catch(Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeStatementFinally(s); closeConnectionFinally(c); } return ergebnisse; return new Script(this).executeScript(sqlScript); } /** @@ -857,21 +528,7 @@ * about the number of affected rows */ public int[] executeScript(Connection c, String sqlScript) { int[] ergebnisse = null; Statement s = null; try { s = c.createStatement(); String[] sqlKommandos = sqlScript.split(";"); for(int i = 0; i < sqlKommandos.length; i++) { s.addBatch(sqlKommandos[i]); } ergebnisse = s.executeBatch(); } catch(Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { closeStatementFinally(s); } return ergebnisse; return new Script(this).executeScript(c, sqlScript); } /* -------- transactions ------------ */ @@ -1007,39 +664,7 @@ * This first row has the field names */ public List<List<String>> toList(ResultSet rs, boolean includeBlobs) throws SQLException { List<List<String>> rows = new ArrayList<List<String>>(); ResultSetMetaData meta = rs.getMetaData(); int columnCount = meta.getColumnCount(); List<String> header = new ArrayList<String>(); for (int i = 1; i <= columnCount; i++) { header.add(meta.getColumnName(i)); } rows.add(header); while (rs.next()) { List<String> row = new ArrayList<String>(); for (int i = 1; i <= columnCount; i++) { String data = null; if (meta.getColumnType(i) == Types.BLOB) { if (includeBlobs) { Blob blob = rs.getBlob(i); if(blob != null) { data = new String(blob.getBytes((long) 1, (int) blob.length())); } } } else { Object o = rs.getObject(i); if(o != null) { data = o.toString(); } else { data = NULL_STR; } logger.finest(data.toString()); } row.add(data); } rows.add(row); } return rows; return new ListConverter().toList(rs, includeBlobs); } /** @@ -1049,9 +674,8 @@ * @return list of maps, one per row, with column name as the key * @throws SQLException if the connection fails */ private List<Map<String, Object>> toList(ResultSet rs) throws SQLException { List<String> wantedColumnNames = getColumnNames(rs); return toList(rs, wantedColumnNames); public List<Map<String, Object>> toList(ResultSet rs) throws SQLException { return new ListConverter().toList(rs); } /** @@ -1062,42 +686,8 @@ * @return list of maps, one per column row, with column names as keys * @throws SQLException if the connection fails */ private List<Map<String, Object>> toList(ResultSet rs, List<String> wantedColumnNames) throws SQLException { // TODO BLOB handling List<Map<String, Object>> rows = new ArrayList<Map<String, Object>>(); int numWantedColumns = wantedColumnNames.size(); while (rs.next()) { Map<String, Object> row = new TreeMap<String, Object>(); for (int i = 0; i < numWantedColumns; ++i) { String columnName = wantedColumnNames.get(i); Object value = rs.getObject(columnName); if (value != null) { row.put(columnName, value); //Logger.getLogger(Logger.GLOBAL_LOGGER_NAME).finest(columnName + " " + value); } else { row.put(columnName, ""); } } rows.add(row); } return rows; public List<Map<String, Object>> toList(ResultSet rs, List<String> wantedColumnNames) throws SQLException { return new ListConverter().toList(rs, wantedColumnNames); } /** * Return all column names as a list of strings * * @param database query result set * @return list of column name strings * @throws SQLException if the query fails */ private List<String> getColumnNames(ResultSet rs) throws SQLException { List<String> columnNames = new ArrayList<String>(); ResultSetMetaData meta = rs.getMetaData(); int numColumns = meta.getColumnCount(); for (int i = 1; i <= numColumns; ++i) { String cName = meta.getColumnName(i); columnNames.add(cName); } return columnNames; } } src/de/uhilger/baselink/Script.java
New file @@ -0,0 +1,216 @@ package de.uhilger.baselink; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.SQLException; import java.sql.Statement; import java.util.List; import java.util.Map; import java.util.logging.Level; import java.util.logging.Logger; /** * * @author Ulrich Hilger */ public class Script extends DBActor { private static final Logger logger = Logger.getLogger(Script.class.getName()); public Script(PersistenceManager pm) { this.pm = pm; } /** * Execute an SQL statement and return keys generated in the database * @param sql the statement to execute * @return a list of generated keys */ public List<Map<String, Object>> executeWithKeys(String sql) { List<Map<String, Object>> keys = null; Connection c = null; try { c = pm.getConnection(); keys = executeWithKeys(c, sql); c.close(); c = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeConnectionFinally(c); } return keys; } /** * Execute an SQL statement and return keys generated in the database * @param c database connection to use * @param sql the statement to execute * @return a list of generated keys */ public List<Map<String, Object>> executeWithKeys(Connection c, String sql) { List<Map<String, Object>> keys = null; Statement s = null; try { s = c.createStatement(); s.executeUpdate(sql, Statement.RETURN_GENERATED_KEYS); keys = pm.toList(s.getGeneratedKeys()); s.close(); s = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeStatementFinally(s); } return keys; } /** * Execute an SQL statement * @param sql the statement to execute * @return the number of records affected by this statement or -1 if none */ public int execute(String sql) { int numRows = -1; Connection c = null; try { c = pm.getConnection(); numRows = execute(c, sql); c.close(); c = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeConnectionFinally(c); } return numRows; } /** * Execute an SQL statement * @param c database connection to use * @param sql the statement to execute * @return the number of records affected by this statement or -1 if none */ public int execute(Connection c, String sql) { int numRows = -1; Statement s = null; try { s = c.createStatement(); numRows = s.executeUpdate(sql); s.close(); s = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeStatementFinally(s); } return numRows; } /** * Execute an SQL statement * @param sql the SQL string with ? at the position of params * @param params list of parameters in the order they appear in the SQL string * @return the number of records affected by this statement or -1 if none */ public int execute(String sql, Object... params) { int numRows = -1; Connection c = null; try { c = pm.getConnection(); numRows = execute(c, sql, params); c.close(); c = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeConnectionFinally(c); } return numRows; } /** * Execute an SQL statement * @param c database connection to use * @param sql the SQL string with ? at the position of params * @param params list of parameters in the order they appear in the SQL string * @return the number of records affected by this statement or -1 if none */ public int execute(Connection c, String sql, Object... params) { int numRows = -1; Statement s = null; try { //s = c.createStatement(); s = pm.buildQuery(c, sql, params); if(s != null && s instanceof PreparedStatement) { try (PreparedStatement ps = (PreparedStatement) s) { numRows = ps.executeUpdate(); } } s = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeStatementFinally(s); } return numRows; } /** * Execute an SQL script * @param sqlScript the SQL script to execute * @return an array of row counts referring to the number of affected rows of * each sql statement in the script in the order of SQL commands in the script * Statement.EXECUTE_FAILED indicates failure of single steps in the script * Statement.SUCCESS_NO_INFO indicates successful execution without information * about the number of affected rows */ public int[] executeScript(String sqlScript) { int[] ergebnisse = null; Connection c = null; Statement s = null; try { c = pm.getConnection(); s = c.createStatement(); String[] sqlKommandos = sqlScript.split(";"); for (String sqlKommando : sqlKommandos) { s.addBatch(sqlKommando); } ergebnisse = s.executeBatch(); } catch(SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeStatementFinally(s); pm.closeConnectionFinally(c); } return ergebnisse; } /** * Execute an SQL script * @param c the Connection object to use * @param sqlScript the SQL script to execute * @return an array of row counts referring to the number of affected rows of * each sql statement in the script in the order of SQL commands in the script * Statement.EXECUTE_FAILED indicates failure of single steps in the script * Statement.SUCCESS_NO_INFO indicates successful execution without information * about the number of affected rows */ public int[] executeScript(Connection c, String sqlScript) { int[] ergebnisse = null; Statement s = null; try { s = c.createStatement(); String[] sqlKommandos = sqlScript.split(";"); for (String sqlKommando : sqlKommandos) { s.addBatch(sqlKommando); } ergebnisse = s.executeBatch(); } catch(SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeStatementFinally(s); } return ergebnisse; } } src/de/uhilger/baselink/Selector.java
New file @@ -0,0 +1,315 @@ package de.uhilger.baselink; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; import java.util.ArrayList; import java.util.List; import java.util.Map; import java.util.logging.Level; import java.util.logging.Logger; /** * * @author Ulrich Hilger */ public class Selector extends DBActor { private static final Logger logger = Logger.getLogger(Selector.class.getName()); public Selector(PersistenceManager pm) { this.pm = pm; } /* --------------- selects ---------------- */ /** * Select a list of objects through a given SQL statement * @param sql sql query string that designates the requested objects * @param record object to use to map db records to objects * @return a list of objects that match the given query */ public List<Object> select(String sql, Record record) { return select(sql, record, Record.WITH_BLOBS); } /** * Select a list of objects through a given SQL statement * @param sql sql query string that designates the requested objects * @param record object to use to map db records to objects * @param includeBlobs true when BLOB contents are to be retrieved, false if not * @return a list of objects that match the given query */ public List<Object> select(String sql, Record record, boolean includeBlobs) { Connection c = null; ArrayList<Object> list = new ArrayList<>(); try { c = pm.getConnection(); list = (ArrayList<Object>) select(c, sql, record, includeBlobs); c.close(); c = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeConnectionFinally(c); } return list; } /** * Select a list of objects through a given SQL statement * @param sql sql query string that designates the requested objects * @param record object to use to map db records to objects * @param includeBlobs true when BLOB contents are to be retrieved, false if not * @param params list of parameters in the order they appear in the SQL string * @return a list of objects that match the given query */ public List<Object> select(String sql, Record record, boolean includeBlobs, Object... params) { Connection c = null; ArrayList<Object> list = new ArrayList<>(); try { c = pm.getConnection(); list = (ArrayList<Object>) select(c, sql, record, includeBlobs, params); c.close(); c = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeConnectionFinally(c); } return list; } /** * Select a list of objects that match a given SQL statement * @param sql sql query string that designates the requested objects * @return a list of map objects, one for each record. An element in the * list can be accessed with list.get(recordno).get("fieldname") */ public List<Map<String, Object>> select(String sql) { //Logger.getLogger(Logger.GLOBAL_LOGGER_NAME).finest(sql); Connection c = null; List<Map<String, Object>> list = null; try { c = pm.getConnection(); list = select(c, sql); c.close(); c = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeConnectionFinally(c); } return list; } /** * Select a list of objects that match a given SQL statement * @param sql sql query string that designates the requested objects * @param includeBlobs true when content of blob coloumns should be returned, false if not * @return a list of list objects, one for each record. An element in the * list can be accessed with list.get(recordno).get(fieldno), each element is of type String */ public List<List<String>> select(String sql, boolean includeBlobs) { Connection c = null; List<List<String>> list = null; try { c = pm.getConnection(); list = select(c, sql, includeBlobs); c.close(); c = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeConnectionFinally(c); } return list; } /** * Select a list of objects that match a given SQL statement * @param sql sql query string that designates the requested objects with ? at the position of params * @param includeBlobs true when content of blob coloumns should be returned, false if not * @param params list of parameters in the order they appear in the SQL string * @return a list of list objects, one for each record. An element in the * list can be accessed with list.get(recordno).get(fieldno), each element is of type String */ public List<List<String>> select(String sql, boolean includeBlobs, Object... params) { Connection c = null; List<List<String>> list = null; try { c = pm.getConnection(); list = select(c, sql, includeBlobs, params); c.close(); c = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeConnectionFinally(c); } return list; } /* ---------- mit Connection ---------- */ /** * Select a list of objects that match a given SQL statement * @param c the database connection to use for this query, expected to be established and open already * @param sql sql query string that designates the requested objects * @return a list of map objects, one for each record. An element in the * list can be accessed with list.get(recordno).get("fieldname") */ public List<Map<String, Object>> select(Connection c, String sql) { PreparedStatement ps = null; ResultSet rs = null; List<Map<String, Object>> list = null; try { ps = c.prepareStatement(sql, ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); rs = ps.executeQuery(); list = pm.toList(rs); rs.close(); rs = null; ps.close(); ps = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeResultSetFinally(rs); pm.closeStatementFinally(ps); } return list; } /** * Select a list of objects that match a given SQL statement * @param c the database connection to use for this query, expected to be established and open already * @param sql sql query string that designates the requested objects * @param includeBlobs true when content of blob coloumns should be returned, false if not * @return a list of list objects, one for each record. An element in the * list can be accessed with list.get(recordno).get(fieldno), each element is of type String */ public List<List<String>> select(Connection c, String sql, boolean includeBlobs) { PreparedStatement ps = null; ResultSet rs = null; List<List<String>> list = null; try { ps = c.prepareStatement(sql, ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); rs = ps.executeQuery(); list = pm.toList(rs, includeBlobs); rs.close(); rs = null; ps.close(); ps = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeResultSetFinally(rs); pm.closeStatementFinally(ps); } return list; } /** * Select a list of objects that match a given SQL statement * @param c the database connection to use for this query, expected to be established and open already * @param sql sql query string that designates the requested objects with ? at the position of params * @param includeBlobs true when content of blob coloumns should be returned, false if not * @param params list of parameters in the order they appear in the SQL string * @return a list of list objects, one for each record. An element in the * list can be accessed with list.get(recordno).get(fieldno), each element is of type String */ public List<List<String>> select(Connection c, String sql, boolean includeBlobs, Object... params) { PreparedStatement ps = null; ResultSet rs = null; List<List<String>> list = null; try { ps = pm.buildQuery(c, sql, params); rs = ps.executeQuery(); list = pm.toList(rs, includeBlobs); rs.close(); rs = null; ps.close(); ps = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeResultSetFinally(rs); pm.closeStatementFinally(ps); } return list; } /** * Select a list of objects that match a given SQL statement * @param c the database connection to use for this query, expected to be established and open already * @param sql sql query string that designates the requested objects * @param record object to use to map db records to objects * @param includeBlobs true when BLOB contents are to be retrieved, false if not * @return a list of objects that match the given query */ public List<Object> select(Connection c, String sql, Record record, boolean includeBlobs) { PreparedStatement ps = null; ResultSet rs = null; ArrayList<Object> list = new ArrayList<>(); try { ps = c.prepareStatement(sql, ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); rs = ps.executeQuery(); if (rs.first()) { while (!rs.isAfterLast()) { list.add(record.toObject(rs, includeBlobs)); rs.next(); } } rs.close(); rs = null; ps.close(); ps = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeResultSetFinally(rs); pm.closeStatementFinally(ps); } return list; } /** * Select a list of objects that match a given SQL statement * @param c the database connection to use for this query, expected to be established and open already * @param sql sql query string that designates the requested objects * @param record object to use to map db records to objects * @param includeBlobs true when BLOB contents are to be retrieved, false if not * @param params list of parameters in the order they appear in the SQL string * @return a list of objects that match the given query */ public List<Object> select(Connection c, String sql, Record record, boolean includeBlobs, Object... params) { PreparedStatement ps = null; ResultSet rs = null; ArrayList<Object> list = new ArrayList<>(); try { //ps = c.prepareStatement(sql, ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); ps = pm.buildQuery(c, sql, params); rs = ps.executeQuery(); if (rs.first()) { while (!rs.isAfterLast()) { list.add(record.toObject(rs, includeBlobs)); rs.next(); } } rs.close(); rs = null; ps.close(); ps = null; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeResultSetFinally(rs); pm.closeStatementFinally(ps); } return list; } } src/de/uhilger/baselink/Updater.java
New file @@ -0,0 +1,98 @@ package de.uhilger.baselink; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.SQLException; import java.util.logging.Level; import java.util.logging.Logger; /** * * @author Ulrich Hilger */ public class Updater extends DBActor { private static final Logger logger = Logger.getLogger(Updater.class.getName()); public Updater(PersistenceManager pm) { this.pm = pm; } /** * update an object in the database * * <p>Use this method for single updates. In cases where * several subsequent updates for objects of the same class * are required the use of method <code>update(Object, Record)</code> * is recommended instead to minimise instantiation * overhead.</p> * * @param o object to update * @return the object that was updated */ public Object update(Object o) { return update(o, new GenericRecord(o.getClass())); } /** * update an object in the database * * <p>Use this method for single updates. In cases where * several subsequent updates for objects of the same class * are required the use of method <code>update(Connection, Object, Record)</code> * is recommended instead to minimise instantiation * overhead.</p> * * @param c the connection to use, expected to be open and established * @param o object to update * @return the object that was updated */ public Object update(Connection c, Object o) { return update(c, o, new GenericRecord(o.getClass())); } /** * update an object in the database * @param o object to update * @param record reference to object to use to map between object and database * @return the object that was updated */ public Object update(Object o, Record record) { Connection c = null; try { c = pm.getConnection(); o = update(c, o, record); c.close(); c = null; } catch (SQLException ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeConnectionFinally(c); } return o; } /** * update an object in the database * @param c the connection to use, expected to be open and established * @param o object to update * @param record reference to object to use to map between object and database * @return the object that was updated */ public Object update(Connection c, Object o, Record record) { Object updatedObject = null; PreparedStatement ps = null; try { ps = record.getUpdateStatment(c, o); ps.executeUpdate(); ps.close(); ps = null; updatedObject = o; } catch (Exception ex) { logger.log(Level.SEVERE, ex.getLocalizedMessage(), ex); } finally { pm.closeStatementFinally(ps); } return updatedObject; } }