/* * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You under the Apache License, Version 2.0 * (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License.
*/ package org.apache.catalina.session;
/** * Implementation of the {@link org.apache.catalina.Store Store} interface that stores serialized session objects in a * database. Sessions that are saved are still subject to being expired based on inactivity. * * @author Bip Thelin
*/ publicclass DataSourceStore extends StoreBase {
/** * Context name associated with this Store
*/ private String name = null;
/** * Name to register for this Store, used for logging.
*/ protectedstaticfinal String storeName = "dataSourceStore";
/** * name of the JNDI resource
*/ protected String dataSourceName = null;
/** * Context local datasource.
*/ privateboolean localDataSource = false;
/** * DataSource to use
*/ protected DataSource dataSource = null;
/** * @return the name for this Store, used for logging.
*/
@Override public String getStoreName() { return storeName;
}
/** * Set the table for this Store. * * @param sessionTable The new table
*/ publicvoid setSessionTable(String sessionTable) {
String oldSessionTable = this.sessionTable; this.sessionTable = sessionTable;
support.firePropertyChange("sessionTable", oldSessionTable, this.sessionTable);
}
/** * @return the table for this Store.
*/ public String getSessionTable() { return sessionTable;
}
/** * Set the App column for the table. * * @param sessionAppCol the column name
*/ publicvoid setSessionAppCol(String sessionAppCol) {
String oldSessionAppCol = this.sessionAppCol; this.sessionAppCol = sessionAppCol;
support.firePropertyChange("sessionAppCol", oldSessionAppCol, this.sessionAppCol);
}
/** * @return the web application name column for the table.
*/ public String getSessionAppCol() { returnthis.sessionAppCol;
}
/** * Set the Id column for the table. * * @param sessionIdCol the column name
*/ publicvoid setSessionIdCol(String sessionIdCol) {
String oldSessionIdCol = this.sessionIdCol; this.sessionIdCol = sessionIdCol;
support.firePropertyChange("sessionIdCol", oldSessionIdCol, this.sessionIdCol);
}
/** * @return the Id column for the table.
*/ public String getSessionIdCol() { returnthis.sessionIdCol;
}
/** * Set the Data column for the table * * @param sessionDataCol the column name
*/ publicvoid setSessionDataCol(String sessionDataCol) {
String oldSessionDataCol = this.sessionDataCol; this.sessionDataCol = sessionDataCol;
support.firePropertyChange("sessionDataCol", oldSessionDataCol, this.sessionDataCol);
}
/** * @return the data column for the table
*/ public String getSessionDataCol() { returnthis.sessionDataCol;
}
/** * Set the {@code Is Valid} column for the table * * @param sessionValidCol The column name
*/ publicvoid setSessionValidCol(String sessionValidCol) {
String oldSessionValidCol = this.sessionValidCol; this.sessionValidCol = sessionValidCol;
support.firePropertyChange("sessionValidCol", oldSessionValidCol, this.sessionValidCol);
}
/** * @return the {@code Is Valid} column
*/ public String getSessionValidCol() { returnthis.sessionValidCol;
}
/** * Set the {@code Max Inactive} column for the table * * @param sessionMaxInactiveCol The column name
*/ publicvoid setSessionMaxInactiveCol(String sessionMaxInactiveCol) {
String oldSessionMaxInactiveCol = this.sessionMaxInactiveCol; this.sessionMaxInactiveCol = sessionMaxInactiveCol;
support.firePropertyChange("sessionMaxInactiveCol", oldSessionMaxInactiveCol, this.sessionMaxInactiveCol);
}
/** * @return the {@code Max Inactive} column
*/ public String getSessionMaxInactiveCol() { returnthis.sessionMaxInactiveCol;
}
/** * Set the {@code Last Accessed} column for the table * * @param sessionLastAccessedCol The column name
*/ publicvoid setSessionLastAccessedCol(String sessionLastAccessedCol) {
String oldSessionLastAccessedCol = this.sessionLastAccessedCol; this.sessionLastAccessedCol = sessionLastAccessedCol;
support.firePropertyChange("sessionLastAccessedCol", oldSessionLastAccessedCol, this.sessionLastAccessedCol);
}
/** * @return the {@code Last Accessed} column
*/ public String getSessionLastAccessedCol() { returnthis.sessionLastAccessedCol;
}
/** * Set the JNDI name of a DataSource-factory to use for db access * * @param dataSourceName The JNDI name of the DataSource-factory
*/ publicvoid setDataSourceName(String dataSourceName) { if (dataSourceName == null || dataSourceName.trim().isEmpty()) {
manager.getContext().getLogger().warn(sm.getString(getStoreName() + ".missingDataSourceName")); return;
} this.dataSourceName = dataSourceName;
}
/** * @return the name of the JNDI DataSource-factory
*/ public String getDataSourceName() { returnthis.dataSourceName;
}
/** * @return if the datasource will be looked up in the webapp JNDI Context.
*/ publicboolean getLocalDataSource() { return localDataSource;
}
/** * Set to {@code true} to cause the datasource to be looked up in the webapp JNDI Context. * * @param localDataSource the new flag value
*/ publicvoid setLocalDataSource(boolean localDataSource) { this.localDataSource = localDataSource;
}
// --------------------------------------------------------- Public Methods
@Override public String[] expiredKeys() throws IOException { return keys(true);
}
@Override public String[] keys() throws IOException { return keys(false);
}
/** * Return an array containing the session identifiers of all Sessions currently saved in this Store. If there are no * such Sessions, a zero-length array is returned. * * @param expiredOnly flag, whether only keys of expired sessions should be returned * * @return array containing the list of session IDs * * @exception IOException if an input/output error occurred
*/ private String[] keys(boolean expiredOnly) throws IOException {
String keys[] = null; int numberOfTries = 2; while (numberOfTries > 0) {
String keysSql = "SELECT " + sessionIdCol + " FROM " + sessionTable + " WHERE " + sessionAppCol + " = ?"; if (expiredOnly) {
keysSql += " AND (" + sessionLastAccessedCol + " + " + sessionMaxInactiveCol + " * 1000 < ?)";
} try (PreparedStatement preparedKeysSql = _conn.prepareStatement(keysSql)) {
preparedKeysSql.setString(1, getName()); if (expiredOnly) {
preparedKeysSql.setLong(2, System.currentTimeMillis());
} try (ResultSet rst = preparedKeysSql.executeQuery()) {
List<String> tmpkeys = new ArrayList<>(); if (rst != null) { while (rst.next()) {
tmpkeys.add(rst.getString(1));
}
}
keys = tmpkeys.toArray(new String[0]); // Break out after the finally block
numberOfTries = 0;
}
}
} catch (SQLException e) {
manager.getContext().getLogger().error(sm.getString(getStoreName() + ".SQLException", e));
keys = new String[0]; // Close the connection so that it gets reopened next time
} finally {
release(_conn);
}
numberOfTries--;
} return keys;
}
/** * Return an integer containing a count of all Sessions currently saved in this Store. If there are no Sessions, * <code>0</code> is returned. * * @return the count of all sessions currently saved in this Store * * @exception IOException if an input/output error occurred
*/
@Override publicint getSize() throws IOException { int size = 0;
String sizeSql = "SELECT COUNT(" + sessionIdCol + ") FROM " + sessionTable + " WHERE " + sessionAppCol + " = ?";
int numberOfTries = 2; while (numberOfTries > 0) {
Connection _conn = getConnection();
/** * Load the Session associated with the id <code>id</code>. If no such session is found <code>null</code> is * returned. * * @param id a value of type <code>String</code> * * @return the stored <code>Session</code> * * @exception ClassNotFoundException if an error occurs * @exception IOException if an input/output error occurred
*/
@Override public Session load(String id) throws ClassNotFoundException, IOException {
StandardSession _session = null;
org.apache.catalina.Context context = getManager().getContext();
Log contextLog = context.getLogger();
int numberOfTries = 2;
String loadSql = "SELECT " + sessionIdCol + ", " + sessionDataCol + " FROM " + sessionTable + " WHERE " +
sessionIdCol + " = ? AND " + sessionAppCol + " = ?"; while (numberOfTries > 0) {
Connection _conn = getConnection(); if (_conn == null) { returnnull;
}
/** * Remove the Session with the specified session identifier from this Store, if present. If no such Session is * present, this method takes no action. * * @param id Session identifier of the Session to be removed * * @exception IOException if an input/output error occurs
*/
@Override publicvoid remove(String id) throws IOException {
int numberOfTries = 2; while (numberOfTries > 0) {
Connection _conn = getConnection();
if (_conn == null) { return;
}
try {
remove(id, _conn); // Break out after the finally block
numberOfTries = 0;
} catch (SQLException e) {
manager.getContext().getLogger().error(sm.getString(getStoreName() + ".SQLException", e));
} finally {
release(_conn);
}
numberOfTries--;
}
if (manager.getContext().getLogger().isDebugEnabled()) {
manager.getContext().getLogger().debug(sm.getString(getStoreName() + ".removing", id, sessionTable));
}
}
/** * Remove the Session with the specified session identifier from this Store, if present. If no such Session is * present, this method takes no action. * * @param id Session identifier of the Session to be removed * @param _conn open connection to be used * * @throws SQLException if an error occurs while talking to the database
*/ privatevoid remove(String id, Connection _conn) throws SQLException {
String removeSql = "DELETE FROM " + sessionTable + " WHERE " + sessionIdCol + " = ? AND " + sessionAppCol + " = ?"; try (PreparedStatement preparedRemoveSql = _conn.prepareStatement(removeSql)) {
preparedRemoveSql.setString(1, id);
preparedRemoveSql.setString(2, getName());
preparedRemoveSql.execute();
}
}
/** * Remove all of the Sessions in this Store. * * @exception IOException if an input/output error occurs
*/
@Override publicvoid clear() throws IOException {
String clearSql = "DELETE FROM " + sessionTable + " WHERE " + sessionAppCol + " = ?";
int numberOfTries = 2; while (numberOfTries > 0) {
Connection _conn = getConnection(); if (_conn == null) { return;
}
/** * Save a session to the Store. * * @param session the session to be stored * * @exception IOException if an input/output error occurs
*/
@Override publicvoid save(Session session) throws IOException {
ByteArrayOutputStream bos = null;
String saveSql = "INSERT INTO " + sessionTable + " (" + sessionIdCol + ", " + sessionAppCol + ", " +
sessionDataCol + ", " + sessionValidCol + ", " + sessionMaxInactiveCol + ", " + sessionLastAccessedCol + ") VALUES (?, ?, ?, ?, ?, ?)";
synchronized (session) { int numberOfTries = 2; while (numberOfTries > 0) {
Connection _conn = getConnection(); if (_conn == null) { return;
}
try { // If sessions already exist in DB, remove and insert again. // TODO: // * Check if ID exists in database and if so use UPDATE.
remove(session.getIdInternal(), _conn);
bos = new ByteArrayOutputStream(); try (ObjectOutputStream oos = new ObjectOutputStream(new BufferedOutputStream(bos))) {
((StandardSession) session).writeObjectData(oos);
} byte[] obs = bos.toByteArray(); int size = obs.length; try (ByteArrayInputStream bis = new ByteArrayInputStream(obs, 0, size);
InputStream in = new BufferedInputStream(bis, size);
PreparedStatement preparedSaveSql = _conn.prepareStatement(saveSql)) {
preparedSaveSql.setString(1, session.getIdInternal());
preparedSaveSql.setString(2, getName());
preparedSaveSql.setBinaryStream(3, in, size);
preparedSaveSql.setString(4, session.isValid() ? "1" : "0");
preparedSaveSql.setInt(5, session.getMaxInactiveInterval());
preparedSaveSql.setLong(6, session.getLastAccessedTime());
preparedSaveSql.execute(); // Break out after the finally block
numberOfTries = 0;
}
} catch (SQLException e) {
manager.getContext().getLogger().error(sm.getString(getStoreName() + ".SQLException", e));
} catch (IOException e) { // Ignore
} finally {
release(_conn);
}
numberOfTries--;
}
}
/** * Check the connection associated with this store, if it's <code>null</code> or closed try to reopen it. Returns * <code>null</code> if the connection could not be established. * * @return <code>Connection</code> if the connection succeeded
*/ protected Connection getConnection() {
Connection conn = null; try {
conn = open(); if (conn == null || conn.isClosed()) {
manager.getContext().getLogger().info(sm.getString(getStoreName() + ".checkConnectionDBClosed"));
conn = open(); if (conn == null || conn.isClosed()) {
manager.getContext().getLogger()
.info(sm.getString(getStoreName() + ".checkConnectionDBReOpenFail"));
}
}
} catch (SQLException ex) {
manager.getContext().getLogger()
.error(sm.getString(getStoreName() + ".checkConnectionSQLException", ex.toString()));
}
return conn;
}
/** * Open (if necessary) and return a database connection for use by this Store. * * @return database connection ready to use * * @exception SQLException if a database error occurs
*/ protected Connection open() throws SQLException { if (dataSourceName != null && dataSource == null) {
org.apache.catalina.Context context = getManager().getContext();
ClassLoader oldThreadContextCL = null; if (getLocalDataSource()) {
oldThreadContextCL = context.bind(Globals.IS_SECURITY_ENABLED, null);
}
/** * Close the specified database connection. * * @param dbConnection The connection to be closed
*/ protectedvoid close(Connection dbConnection) {
// Do nothing if the database connection is already closed if (dbConnection == null) { return;
}
// Commit if autoCommit is false try { if (!dbConnection.getAutoCommit()) {
dbConnection.commit();
}
} catch (SQLException e) {
manager.getContext().getLogger().error(sm.getString(getStoreName() + ".commitSQLException"), e);
}
// Close this database connection, and log any errors try {
dbConnection.close();
} catch (SQLException e) {
manager.getContext().getLogger().error(sm.getString(getStoreName() + ".close", e.toString())); // Just log // it here
}
}
/** * Release the connection, if it is associated with a connection pool. * * @param conn The connection to be released
*/ protectedvoid release(Connection conn) { if (dataSource != null) {
close(conn);
}
}
}
¤ Dauer der Verarbeitung: 0.20 Sekunden
(vorverarbeitet)
¤
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung ist noch experimentell.