Apache Pool не может вернуть объект в контроллере / сервисе Spring
Я пытаюсь использовать apache pool2 для объединения некоторых ресурсов. У меня он отлично работает в модульном тесте, но когда я пытаюсь использовать его весной 3, я получаю ошибку.
Все остальное работает с контроллером и сервисом. Я мог получить доступ к конечной точке до того, как добавил код пула, и служба автонастройки не пуста. Все это связано вместе с использованием context: scan-scan
Если я устанавливаю точку останова в методе контроллера, я вижу, что объект, возвращаемый из заимствования, является org.apache.cxf.jaxws.JaxWsClientProxy@15de00c . Затем проверка объекта в активном пуле дает org.apache.cxf.jaxws.JaxWsClientProxy@15de00c
Итак, мой вопрос таков: почему это работает в модульном тесте, но не работает в контроллере / сервисе пружины
контроллер:
@Controller
public class TestController() {
@Autowired
private TestService testService
@RequestMapping(value="/test", method=RequestMethod.GET)
public ModelAndView getTest() throws Exception {
GenericObjectPool<MyObj> pool = testService.getPool();
pool.returnObject(pool.borrowObject());
return new ModelAndView("jsp/test", "command", new TestObj()); //not really relevant yet
}
}
и сервис:
@Service
public class TestService implements DisposableBean {
GenericObjectPool<MyObj> pool;
public TestService () {
pool = new GenericObjectPool<MyObj>(new MyObjPooledObjectFactory());
}
public GenericObjectPool<MyObj> getPool() {
return pool;
}
public void setPool(GenericObjectPool<MyObj> pool) {
this.pool = pool;
}
@Override
public void destroy() throws Exception {
LOG.info("DESTROYING Service");
this.pool.close();
}
}
Фабрика:
MyObjPooledObjectFactory extends BasePooledObjectFactory<MyObj> {
@Override
public MyObjc create() throws Exception {
MyObj myObj = expensive call.
return myObj;
}
@Override
public PooledObject<MyObj> wrap(MyObj obj) {
return new DefaultPooledObject<MyObj>(obj);
}
}
@Override
public void destroyObject(PooledObject<MyObj > p) throws Exception {
super.destroyObject(p);
p.releaseConnection();
}
и наконец у меня есть
@Test
public void testMe() {
TestService service = new TestService();
GenericObjectPool<MyObj> pool = service.getPool();
try {
pool.returnObject(pool.borrowObject());
} catch (Exception e) {
e.printStackTrace();
fail();
}
}
я получаю ошибку:
java.lang.IllegalStateException: Returned object not currently part of this pool
at org.apache.commons.pool2.impl.GenericObjectPool.returnObject(GenericObjectPool.java:537)
at com.example.controller.TestController.getTest(TestController.java:56)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:597)
at org.springframework.web.method.support.InvocableHandlerMethod.invoke(InvocableHandlerMethod.java:213)
at org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:126)
at org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:96)
at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandlerMethod(RequestMappingHandlerAdapter.java:617)
at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:578)
at org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:80)
at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:923)
at org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:852)
at org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:882)
at org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:779)
at javax.servlet.http.HttpServlet.service(HttpServlet.java:707)
at javax.servlet.http.HttpServlet.service(HttpServlet.java:821)
at weblogic.servlet.internal.StubSecurityHelper$ServletServiceAction.run(StubSecurityHelper.java:227)
at weblogic.servlet.internal.StubSecurityHelper.invokeServlet(StubSecurityHelper.java:125)
at weblogic.servlet.internal.ServletStubImpl.execute(ServletStubImpl.java:300)
at weblogic.servlet.internal.TailFilter.doFilter(TailFilter.java:27)
at weblogic.servlet.internal.FilterChainImpl.doFilter(FilterChainImpl.java:57)
at org.springframework.web.filter.CharacterEncodingFilter.doFilterInternal(CharacterEncodingFilter.java:89)
at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:76)
at weblogic.servlet.internal.FilterChainImpl.doFilter(FilterChainImpl.java:57)
at weblogic.servlet.internal.WebAppServletContext$ServletInvocationAction.wrapRun(WebAppServletContext.java:3715)
at weblogic.servlet.internal.WebAppServletContext$ServletInvocationAction.run(WebAppServletContext.java:3681)
at weblogic.security.acl.internal.AuthenticatedSubject.doAs(AuthenticatedSubject.java:321)
at weblogic.security.service.SecurityManager.runAs(SecurityManager.java:120)
at weblogic.servlet.internal.WebAppServletContext.securedExecute(WebAppServletContext.java:2277)
at weblogic.servlet.internal.WebAppServletContext.execute(WebAppServletContext.java:2183)
at weblogic.servlet.internal.ServletRequestImpl.run(ServletRequestImpl.java:1454)
at weblogic.work.ExecuteThread.execute(ExecuteThread.java:207)
at weblogic.work.ExecuteThread.run(ExecuteThread.java:176)
Кроме того, кажется, не поможет, если я заменю @Autowired TestService новым TestService().
Ну, я не знаю, почему это терпит неудачу, но я сузил это до совершенно другой проблемы:
if (!myObj.equals(myObj)) {
throw new IllegalStateException("WTF, Why is this not equal to itself!");
}
выбросит исключение. Так что теперь мне нужно выяснить, что происходит с этим jaxWsClientProxy
2 ответа
У меня та же проблема, когда я пытаюсь опрашивать клиентский порт jaxws. Основная причина в том же методе equals не работает для объекта Proxy. Если мы хотим сравнить идентичность объекта внутри нашего метода equals() (то есть с "=="), то это не сработает, потому что мы сравниваем обернутый объект с прокси-объектом, что, очевидно, ложно. Чтобы решить эту проблему, я ввел объект-оболочку вокруг прокси, где определяю поле "id", которое можно использовать в методах equals() и hashCode().
Код
public class DefaultCommonsPooledObject<T> extends DefaultPooledObject<T> {
private final UUID id = UUID.randomUUID();
/**
* Create a new instance that wraps the provided object so that the pool can track the state of the pooled object.
*
* @param object The object to wrap
*/
public DefaultCommonsPooledObject(T object) {
super(object);
}
@Override
public boolean equals(Object o) {
if (this == o)
return true;
if (!(o instanceof DefaultCommonsPooledObject))
return false;
DefaultCommonsPooledObject that = (DefaultCommonsPooledObject) o;
if (id != null ? !id.equals(that.id) : that.id != null)
return false;
return true;
}
@Override
public int hashCode() {
return id != null ? id.hashCode() : 0;
}
}
Я видел ту же самую странность с прокси порта jaxws, где (obj == obj)
это правда, но (obj.equals(obj))
был ложным Итак, как вы упомянули в комментарии, мне пришлось обернуть служебный порт Jaxws, чтобы переопределить equals и hashcode, прежде чем помещать объект порта в пул.
Класс Groovy Proxy делает это довольно простым. Вот прокси, который я использую:
class JaxwsPortProxy<T> extends Proxy {
private final UUID poolId = UUID.randomUUID()
@Override
boolean equals(Object o) {
if (this == o)
return true
if (!(o instanceof JaxwsPortProxy))
return false
JaxwsPortProxy that = (JaxwsPortProxy) o
poolId == null ? that.poolId == null : poolId.equals(that.poolId)
}
@Override
int hashCode() {
poolId?.hashCode() ?: 0
}
}
И затем конец моего метода BasePooledObjectFactory.create() просто оборачивает служебный порт следующим образом:
return new JaxwsPortProxy<RuleServiceWS>().wrap(ruleService)
Спасибо за помощь в получении этого решения!