IDEMPIERE-1145 Trx and Trx Monitor is not thread safe.
This commit is contained in:
parent
f4c2a30f44
commit
58ab8b352d
|
@ -22,10 +22,10 @@ import java.sql.Savepoint;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
import java.util.logging.Level;
|
import java.util.logging.Level;
|
||||||
|
|
||||||
|
@ -63,16 +63,11 @@ public class Trx
|
||||||
* @param createNew if false, null is returned if not found
|
* @param createNew if false, null is returned if not found
|
||||||
* @return Transaction or null
|
* @return Transaction or null
|
||||||
*/
|
*/
|
||||||
public static synchronized Trx get (String trxName, boolean createNew)
|
public static Trx get (String trxName, boolean createNew)
|
||||||
{
|
{
|
||||||
if (trxName == null || trxName.length() == 0)
|
if (trxName == null || trxName.length() == 0)
|
||||||
throw new IllegalArgumentException ("No Transaction Name");
|
throw new IllegalArgumentException ("No Transaction Name");
|
||||||
|
|
||||||
if (s_cache == null)
|
|
||||||
{
|
|
||||||
s_cache = new HashMap<String,Trx>(10);
|
|
||||||
}
|
|
||||||
|
|
||||||
Trx retValue = (Trx)s_cache.get(trxName);
|
Trx retValue = (Trx)s_cache.get(trxName);
|
||||||
if (retValue == null && createNew)
|
if (retValue == null && createNew)
|
||||||
{
|
{
|
||||||
|
@ -83,9 +78,9 @@ public class Trx
|
||||||
} // get
|
} // get
|
||||||
|
|
||||||
/** Transaction Cache */
|
/** Transaction Cache */
|
||||||
private static Map<String,Trx> s_cache = null; // create change listener
|
private static final Map<String,Trx> s_cache = new ConcurrentHashMap<String, Trx>();
|
||||||
|
|
||||||
private static Trx.TrxMonitor s_monitor = new Trx.TrxMonitor();
|
private static final Trx.TrxMonitor s_monitor = new Trx.TrxMonitor();
|
||||||
|
|
||||||
private List<TrxEventListener> listeners = new ArrayList<TrxEventListener>();
|
private List<TrxEventListener> listeners = new ArrayList<TrxEventListener>();
|
||||||
|
|
||||||
|
@ -171,7 +166,7 @@ public class Trx
|
||||||
* @param createNew if true, create new connection if the trx does not have one created yet
|
* @param createNew if true, create new connection if the trx does not have one created yet
|
||||||
* @return connection
|
* @return connection
|
||||||
*/
|
*/
|
||||||
public Connection getConnection(boolean createNew)
|
public synchronized Connection getConnection(boolean createNew)
|
||||||
{
|
{
|
||||||
if (log.isLoggable(Level.ALL))log.log(Level.ALL, "Active=" + isActive() + ", Connection=" + m_connection);
|
if (log.isLoggable(Level.ALL))log.log(Level.ALL, "Active=" + isActive() + ", Connection=" + m_connection);
|
||||||
|
|
||||||
|
@ -179,7 +174,7 @@ public class Trx
|
||||||
{
|
{
|
||||||
if (createNew)
|
if (createNew)
|
||||||
{
|
{
|
||||||
if (s_cache == null || !s_cache.containsKey(m_trxName))
|
if (!s_cache.containsKey(m_trxName))
|
||||||
{
|
{
|
||||||
new Exception("Illegal to getConnection for Trx that is not register.").printStackTrace();
|
new Exception("Illegal to getConnection for Trx that is not register.").printStackTrace();
|
||||||
return null;
|
return null;
|
||||||
|
@ -272,7 +267,7 @@ public class Trx
|
||||||
* @param throwException if true, re-throws exception
|
* @param throwException if true, re-throws exception
|
||||||
* @return true if success, false if failed or transaction already rollback
|
* @return true if success, false if failed or transaction already rollback
|
||||||
*/
|
*/
|
||||||
public boolean rollback(boolean throwException) throws SQLException
|
public synchronized boolean rollback(boolean throwException) throws SQLException
|
||||||
{
|
{
|
||||||
//local
|
//local
|
||||||
try
|
try
|
||||||
|
@ -351,7 +346,7 @@ public class Trx
|
||||||
* @param throwException if true, re-throws exception
|
* @param throwException if true, re-throws exception
|
||||||
* @return true if success
|
* @return true if success
|
||||||
**/
|
**/
|
||||||
public boolean commit(boolean throwException) throws SQLException
|
public synchronized boolean commit(boolean throwException) throws SQLException
|
||||||
{
|
{
|
||||||
//local
|
//local
|
||||||
try
|
try
|
||||||
|
@ -410,8 +405,7 @@ public class Trx
|
||||||
*/
|
*/
|
||||||
public synchronized boolean close()
|
public synchronized boolean close()
|
||||||
{
|
{
|
||||||
if (s_cache != null)
|
s_cache.remove(getTrxName());
|
||||||
s_cache.remove(getTrxName());
|
|
||||||
|
|
||||||
//local
|
//local
|
||||||
if (m_connection == null)
|
if (m_connection == null)
|
||||||
|
@ -459,7 +453,7 @@ public class Trx
|
||||||
* @return Savepoint
|
* @return Savepoint
|
||||||
* @throws SQLException
|
* @throws SQLException
|
||||||
*/
|
*/
|
||||||
public Savepoint setSavepoint(String name) throws SQLException {
|
public synchronized Savepoint setSavepoint(String name) throws SQLException {
|
||||||
if (m_connection == null)
|
if (m_connection == null)
|
||||||
getConnection();
|
getConnection();
|
||||||
|
|
||||||
|
@ -479,7 +473,7 @@ public class Trx
|
||||||
* @throws SQLException
|
* @throws SQLException
|
||||||
* @see {@link Connection#releaseSavepoint(Savepoint)}
|
* @see {@link Connection#releaseSavepoint(Savepoint)}
|
||||||
*/
|
*/
|
||||||
public void releaseSavepoint(Savepoint savepoint) throws SQLException
|
public synchronized void releaseSavepoint(Savepoint savepoint) throws SQLException
|
||||||
{
|
{
|
||||||
if (DB.isOracle())
|
if (DB.isOracle())
|
||||||
{
|
{
|
||||||
|
@ -519,9 +513,6 @@ public class Trx
|
||||||
*/
|
*/
|
||||||
public static Trx[] getActiveTransactions()
|
public static Trx[] getActiveTransactions()
|
||||||
{
|
{
|
||||||
if (s_cache == null)
|
|
||||||
return new Trx[0];
|
|
||||||
|
|
||||||
Collection<Trx> collections = s_cache.values();
|
Collection<Trx> collections = s_cache.values();
|
||||||
Trx[] trxs = new Trx[collections.size()];
|
Trx[] trxs = new Trx[collections.size()];
|
||||||
collections.toArray(trxs);
|
collections.toArray(trxs);
|
||||||
|
@ -626,11 +617,15 @@ public class Trx
|
||||||
* @param listener
|
* @param listener
|
||||||
*/
|
*/
|
||||||
public void addTrxEventListener(TrxEventListener listener) {
|
public void addTrxEventListener(TrxEventListener listener) {
|
||||||
listeners.add(listener);
|
synchronized (listeners) {
|
||||||
|
listeners.add(listener);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean removeTrxEventListener(TrxEventListener listener) {
|
public boolean removeTrxEventListener(TrxEventListener listener) {
|
||||||
return listeners.remove(listener);
|
synchronized (listeners) {
|
||||||
|
return listeners.remove(listener);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static class TrxMonitor implements Runnable
|
static class TrxMonitor implements Runnable
|
||||||
|
@ -638,7 +633,7 @@ public class Trx
|
||||||
|
|
||||||
public void run()
|
public void run()
|
||||||
{
|
{
|
||||||
if (Trx.s_cache != null && !Trx.s_cache.isEmpty())
|
if (!Trx.s_cache.isEmpty())
|
||||||
{
|
{
|
||||||
Trx[] trxs = Trx.s_cache.values().toArray(new Trx[0]);
|
Trx[] trxs = Trx.s_cache.values().toArray(new Trx[0]);
|
||||||
for(int i = 0; i < trxs.length; i++)
|
for(int i = 0; i < trxs.length; i++)
|
||||||
|
|
Loading…
Reference in New Issue