From a59fcaadb5c928a243f198c91d69313234e07afe Mon Sep 17 00:00:00 2001 From: ulrich Date: Mon, 22 Jan 2024 13:56:18 +0000 Subject: [PATCH] Teile von PersistenceManager in andere Klassen ausgelagert --- src/de/uhilger/baselink/Eraser.java | 99 +++ src/de/uhilger/baselink/Selector.java | 315 ++++++++++++ src/de/uhilger/baselink/Inserter.java | 100 ++++ src/de/uhilger/baselink/Script.java | 216 ++++++++ src/de/uhilger/baselink/ListConverter.java | 130 +++++ src/de/uhilger/baselink/DBActor.java | 11 src/de/uhilger/baselink/PersistenceManager.java | 484 +----------------- src/de/uhilger/baselink/Updater.java | 98 +++ 8 files changed, 1,006 insertions(+), 447 deletions(-) diff --git a/src/de/uhilger/baselink/DBActor.java b/src/de/uhilger/baselink/DBActor.java new file mode 100644 index 0000000..ea91911 --- /dev/null +++ b/src/de/uhilger/baselink/DBActor.java @@ -0,0 +1,11 @@ +package de.uhilger.baselink; + +/** + * + * @author Ulrich Hilger + */ +public class DBActor { + + protected PersistenceManager pm; + +} diff --git a/src/de/uhilger/baselink/Eraser.java b/src/de/uhilger/baselink/Eraser.java new file mode 100644 index 0000000..484ed2a --- /dev/null +++ b/src/de/uhilger/baselink/Eraser.java @@ -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; + } + +} diff --git a/src/de/uhilger/baselink/Inserter.java b/src/de/uhilger/baselink/Inserter.java new file mode 100644 index 0000000..696131a --- /dev/null +++ b/src/de/uhilger/baselink/Inserter.java @@ -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; + } + + +} diff --git a/src/de/uhilger/baselink/ListConverter.java b/src/de/uhilger/baselink/ListConverter.java new file mode 100644 index 0000000..79375fe --- /dev/null +++ b/src/de/uhilger/baselink/ListConverter.java @@ -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; + } + + +} diff --git a/src/de/uhilger/baselink/PersistenceManager.java b/src/de/uhilger/baselink/PersistenceManager.java index eb12e31..c11ec96 100644 --- a/src/de/uhilger/baselink/PersistenceManager.java +++ b/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.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(); - } - } - 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, params); } - /** * 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; - } } diff --git a/src/de/uhilger/baselink/Script.java b/src/de/uhilger/baselink/Script.java new file mode 100644 index 0000000..3444746 --- /dev/null +++ b/src/de/uhilger/baselink/Script.java @@ -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; + } + +} diff --git a/src/de/uhilger/baselink/Selector.java b/src/de/uhilger/baselink/Selector.java new file mode 100644 index 0000000..2ebc1a8 --- /dev/null +++ b/src/de/uhilger/baselink/Selector.java @@ -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; + } + +} diff --git a/src/de/uhilger/baselink/Updater.java b/src/de/uhilger/baselink/Updater.java new file mode 100644 index 0000000..19f7e88 --- /dev/null +++ b/src/de/uhilger/baselink/Updater.java @@ -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; + } + +} -- Gitblit v1.9.3