====== The last packet successfully received from the server was X milliseconds ago ======
===== Intro =====
==== This crap will be useful for you if... ====
* Have tried the well known solutions: [[http://stackoverflow.com/questions/16175398/java-communications-link-failure-error-after|solution1]], [[http://stackoverflow.com/questions/15752667/warn-sqlexceptionhelper143-sql-error-0-sqlstate-08s01-sqlexceptionhelper|solution2]]
* Have [[https://www.google.es/search?source=ig&rlz=&q=The+last+packet+successfully+received+from+the+server+was|searched in google]] for solutions to this problem and nothing works for you
__This problem is soooooo common that the solutions described above should work in most of the cases__, however, it wasn't my case:
* My application worked perfectly
* Mysql worked perfectly
* There wasn't a network error
* So... were the problem is???
==== My architecture ====
Before going on, I think it's important to describe what's my architecture. If it doesn't match yours, probably the solutions described here won't work in your case.
* Mysql as database
* C3P0 as connection pooling mechanism
* Hibernate as data access layer
* Spring to tie all together
* A batch application that runs for hours and then stops. It only have one thread: starts, do thounsands of queries and then stops.
If you have the same problem, but in a web application, this please read on; some of the things described here are still useful or at least you should check them out.
===== First approach: retention of Session objects =====
Yes, I am new in hibernate and I miss out this:
[[http://stackoverflow.com/questions/14554353/database-connection-to-mysql-times-out-even-after-setting-c3p0-testconnectiononc]]
Apparently, the retention of Hibernate Sessions can more time than the value of [[http://dev.mysql.com/doc/refman/5.5/en/server-system-variables.html|wait_timeout]] can produce this problem.
This could be my case, because it happens that I found one ''Session'' object stored as a private property inside of a singleton: this property could be held for many time after the first query is made. And the first query was made in the creation of the database.
==== Steps to reproduce the problem ====
- Configure your [[http://dev.mysql.com/doc/refman/5.5/en/server-system-variables.html|wait_timeout]] value to a low value, like five minutes or less:
wait_timeout = 60
# just in case
interactive_timeout = 60
- **Make a test.** I've built the following test, which is pretty straightforward:
@Test
public void test()
{
Period p = entityProvider.randomExistingPeriod();
// issue a query
StringBuilder hql = new StringBuilder();
hql.append( "from Table where period = :period" );
Query q = session.createQuery( hql.toString() );
q.setEntity( "period", p );
@SuppressWarnings("unchecked")
List result = q.list();
log.info( "Sleeping...." );
try
{
Thread.sleep( 10 * 60 * 1000 ); // ten minutes
}catch( InterruptedException ex )
{
log.info( "Awakening..." );
}
// issue a second query
// issue a query
StringBuilder hql2 = new StringBuilder();
hql2.append( "from Table where period = :period" );
Query q2 = session.createQuery( hql.toString() );
q2.setEntity( "period", p );
@SuppressWarnings("unchecked")
List result2 = q2.list();
assertNotNull( result );
assertNotNull( result2 );
assertTrue( result.size() > 0 );
assertTrue( result2.size() > 0 );
} // test
In other words;
- issue a valid query to the database
- wait an amount of time **greater than the wait_timeout you have configured**
- issue a valid query to the database. This second query will show up the problem
2013-09-03 15:52:53,634 DEBUG c3p0.impl.NewPooledConnection - com.mchange.v2.c3p0.impl.NewPooledConnection@c22a3b invalidated by Exception.
com.mysql.jdbc.exceptions.jdbc4.CommunicationsException: Communications link failure
The last packet successfully received from the server was 600.214 milliseconds ago. The last packet sent successfully to the server was 15 milliseconds ago.
at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
at sun.reflect.NativeConstructorAccessorImpl.newInstance(Unknown Source)
at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(Unknown Source)
at java.lang.reflect.Constructor.newInstance(Unknown Source)
at com.mysql.jdbc.Util.handleNewInstance(Util.java:411)
at com.mysql.jdbc.SQLError.createCommunicationsException(SQLError.java:1116)
at com.mysql.jdbc.MysqlIO.reuseAndReadPacket(MysqlIO.java:3090)
at com.mysql.jdbc.MysqlIO.reuseAndReadPacket(MysqlIO.java:2979)
at com.mysql.jdbc.MysqlIO.checkErrorPacket(MysqlIO.java:3520)
at com.mysql.jdbc.MysqlIO.sendCommand(MysqlIO.java:1990)
[....]
==== Solution ====
The solution is as easy as **not keeping the hibernate [[http://docs.jboss.org/hibernate/orm/3.5/javadoc/org/hibernate/Session.html|Session]] object too much in memory**. Saying this is easy, but achiving this....
For myself, it worked the following countermeasures:
* Bear always in mind that the Session object can't be kept in memory too much time, and close as quick as you can
* Avoid saving Session beans as members of classes; use it only in local variables will reduce the exposure to this problem
* Lower the [[http://dev.mysql.com/doc/refman/5.5/en/server-system-variables.html|wait_timeout]] value for your application in development. I've discovered that lower values of wait_timeout in the config file are rejected, so [[#A way to set lower values of wait_timeout only for our tests|it is best to have a small bean that issues this command upon execution for development]]: this will ensure you that only your application is enforced to work with this values instead of changing the values for all the database
To change the wait_timeout value for a particular session:
set session wait_timeout = 300;
==== A quick and dirty solution ====
In my case I have a batch application that longs for 24 hours. So putting a big value in wait_timeout will solve the problem. Here is how to do this programmatically:
mysql> set global interactive_timeout = 172800;
Query OK, 0 rows affected (0.00 sec)
mysql> set global wait_timeout = 172800;
Query OK, 0 rows affected (0.01 sec)
You have to change ''interactive_timeout'' also because, depeding on the type of session is starting, the value ''wait_timeout'' can be set to the value of ''ineractive_timeout''.
==== A way to set lower values of wait_timeout only for our tests ====
I've developed a bean who issues mysql commands to the database. With this, I've configured it to
set low ''wait_timeout'' values for the testing sessions. With this, all the tests made will use
low values, hence allowing all the possible errors arise before they reach to production.
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import org.springframework.jdbc.core.simple.SimpleJdbcTemplate;
/**
* Allows to set commands to the sql database, like set commands.
*
* @author RLUNARO
*
*/
public class MysqlConfigurator
{
private SimpleJdbcTemplate jdbc;
private List initialListOfCommands;
public SimpleJdbcTemplate getJdbc()
{
return jdbc;
}
public void setJdbc(SimpleJdbcTemplate jdbc)
{
this.jdbc = jdbc;
}
public List getInitialListOfCommands()
{
return initialListOfCommands;
}
public void setInitialListOfCommands(List initialListOfCommands)
{
this.initialListOfCommands = initialListOfCommands;
}
/**
* This method must be called upon set of the initial properties.
*/
public void init()
{
// traverse list of commands and run them
for( String command : initialListOfCommands )
{
issueStatement( command );
} // command : initialListOfCommands
}
public void issueStatement( String command )
{
jdbc.update( command );
}
public String queryVariable( String variable )
{
String sql = "show variables like ? ";
List
And here is the spring configuration I am using (see below). I am using a singleton to load the bean
at the beginning of the program and make sure that the values are set.
set session wait_timeout = 30
===== Second Approach: be careful with your open sessions =====
After this, **sometimes I have the same problem, others I get this error**.
com.mchange.v2.resourcepool.TimeoutException: A client timed out while waiting to acquire a resource from com.mchange.v2.resourcepool.BasicResourcePool@1665147 -- timeout at awaitAvailable()
at com.mchange.v2.resourcepool.BasicResourcePool.awaitAvailable(BasicResourcePool.java:1317)
at com.mchange.v2.resourcepool.BasicResourcePool.prelimCheckoutResource(BasicResourcePool.java:557)
at com.mchange.v2.resourcepool.BasicResourcePool.checkoutResource(BasicResourcePool.java:477)
at com.mchange.v2.c3p0.impl.C3P0PooledConnectionPool.checkoutPooledConnection(C3P0PooledConnectionPool.java:525)
at com.mchange.v2.c3p0.impl.AbstractPoolBackedDataSource.getConnection(AbstractPoolBackedDataSource.java:128)
at org.springframework.orm.hibernate3.LocalDataSourceConnectionProvider.getConnection(LocalDataSourceConnectionProvider.java:81)
==== HibernateTemplate must be a singleton ====
Catch it!!! I had this:
**An this is wrong.** The [[http://docs.spring.io/spring/docs/2.5.x/api/org/springframework/orm/hibernate3/HibernateTemplate.html|HibernateTemplate]] object must be declared singleton (maybe "session" in a web application to avoid multithreading issues). This saves lots of session objects and keep them alive because they are always been called across the application.
However, I am starting to think that HibernateTemplate doesn't close properly it's Session objects. If this were the case, it will explain why when I release my HibernateTemplate objects the corresponding session objects weren't released and closed.
==== You can configure also when Hibernate release its session objects ====
I've discovered these things that I think they could help in solving the problem:
* [[http://docs.jboss.org/hibernate/stable/core.old/reference/en/html/transactions-connection-release.html|configuring when hibernate releases its connections]] (and don't forget [[http://docs.jboss.org/hibernate/orm/3.3/reference/en/html/session-configuration.html|this]] and [[http://stackoverflow.com/questions/5978613/spring-3-mvc-hibernate-3-5-4-hibernatetemplate-not-closing-connections-non-tran|this]])
You can se the parameter "releaseMode" to the value "AFTER_STATEMENT" and let C3P0 get this open sessions to mysql back. With this, the open sessions that hibernate gets, are released in the quickest way. This is discouraged in the documentation, but bear in mind that if you have put a pooling mechanism, this could act as a safety net catching this connections and reusing them.
==== You can configure C3P0 to reopen used connections ====
For me, this have been the silver bullet that can fix a broken application whihout the need to go back to development to fix the issue.
The configuration parameter [[http://www.mchange.com/projects/c3p0/#unreturnedConnectionTimeout|unreturnedConnectionTimeout]] does the following:
- When C3P0 gives a connection to the application, it starts a counter
- When this counter reaches this limit, C3P0 silently replaces this connection and gives another fresh open connection
- This parameter can avoid the problem that I had that C3P0 gave a connection and it went closed because it was retained open for days
However, it is discouraged because it hides problems in the application. But if it can save your day, take it into account.
You have also this parameter: [[http://www.mchange.com/projects/c3p0/#debugUnreturnedConnectionStackTraces|debugUnreturnedConnectionStackTraces]] that can help to debug those connection that the application stores and doesn't return back until it's too late.
===== My final configuration of Spring =====
First, define C3P0 datasource:
Second, the hibernateTemplate and jdbctemplate objects of spring will be available accross the application.
**Beware!!! If you are developing a web application, these should be session instead of singleton!!!**
Finally, the sessionFactory:
com.mapfre.dga.mercurio.entity.Detailcom.mapfre.dga.mercurio.entity.Indicatorcom.mapfre.dga.mercurio.entity.Periodcom.mapfre.dga.mercurio.entity.Structcom.mapfre.dga.mercurio.entity.Valorationcom.mapfre.dga.mercurio.entity.Versionorg.hibernate.dialect.MySQL5Dialect${show-sql-in-hibernate}${format-sql-in-hibernate}
I've put plenty of comments of useful configuration parameters and lots of information, just in case.
===== Resources =====
http://www.mchange.com/projects/c3p0/
~~DISQUS~~