I'm running my application in tomee and trying to test my jax-rs web services via jersey test framework.
I added maven dependencies as below:
<dependency>
<groupId>org.glassfish.jersey.test-framework</groupId>
<artifactId>jersey-test-framework-core</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.glassfish.jersey.test-framework.providers</groupId>
<artifactId>jersey-test-framework-provider-grizzly2</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.glassfish.jersey.inject</groupId>
<artifactId>jersey-hk2</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.glassfish.jersey.core</groupId>
<artifactId>jersey-common</artifactId>
<version>${jersey.version}</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.glassfish.jersey.media</groupId>
<artifactId>jersey-media-json-jackson</artifactId>
<scope>test</scope>
</dependency>
Inside my resource class i have
@Inject
private AuthenticationTokenService authenticationTokenService;
Inside my service class i have:
@Inject
private AuthenticationTokenIssuer tokenIssuer;
@Inject
private AuthenticationTokenParser tokenParser;
inside these two classes i have
@Inject
private AuthenticationTokenSettings settings;
I guess the my problem comes because of the member within AuthenticationTokenSettings
@Inject
@Config("token")
private String token;
these are injected with the qualifier @Config from a file via a Properties object.
I've written an InjectionResolver class to resolve
@Singleton
public static class ConfigInjectionResolver implements InjectionResolver<Configurable>
{
private static Properties properties;
public ConfigurableInjectionResolver()
{
properties = new Properties();
InputStream stream = ConfigurationProducer.class.getResourceAsStream("/application.properties");
if (stream == null)
{
throw new RuntimeException("Cannot find application.properties configuration file.");
}
try
{
this.properties.load(stream);
}
catch (final IOException e)
{
throw new RuntimeException("Configuration file cannot be loaded.");
}
}
@Override
public Object resolve(Injectee injectee, ServiceHandle<?> handle)
{
if (String.class == injectee.getRequiredType())
{
AnnotatedElement elem = injectee.getParent();
if (elem instanceof Constructor)
{
Constructor ctor = (Constructor) elem;
Configurable config = (Configurable) ctor.getParameterAnnotations()[injectee.getPosition()][0];
return properties.get(config.value());
}
else
{
Configurable configurable = elem.getAnnotation(Configurable.class);
return properties.get(configurable.value());
}
}
return null;
}
@Override
public boolean isConstructorParameterIndicator()
{
return true;
}
@Override
public boolean isMethodParameterIndicator()
{
return false;
}
}
In my overridden configure method i created the abstract binder like this.
ResourceConfig resourceConfig = new ResourceConfig(
AuthenticationResource.class);
resourceConfig.register(new AbstractBinder()
{
@Override
protected void configure()
{
bind(AuthenticationTokenService.class).
to(AuthenticationResource.class);
bind(AuthenticationTokenIssuer.class).
to(AuthenticationTokenService.class);
bind(AuthenticationTokenParser.class).
to(AuthenticationTokenService.class);
bind(AuthenticationTokenSettings.class).
to(AuthenticationTokenIssuer.class);
bind(AuthenticationTokenSettings.class).
to(AuthenticationTokenParser.class);
bind(ConfigurableInjectionResolver.class)
.to(new TypeLiteral<InjectionResolver<Configurable>>()
{
})
.in(Singleton.class);
.
.
.
.
I have exception as below:
Apr 15, 2020 2:05:15 AM org.glassfish.jersey.internal.Errors logErrors
WARNING: The following warnings have been detected: WARNING: HK2 service reification failed for [com.transport.webservices.security.service.AuthenticationTokenService] with an exception:
MultiException stack 1 of 2
java.lang.NoSuchMethodException: Could not find a suitable constructor in com.transport.webservices.security.service.AuthenticationTokenService class.
at org.glassfish.jersey.inject.hk2.JerseyClassAnalyzer.getConstructor(JerseyClassAnalyzer.java:168)
.
.
MultiException stack 2 of 2
java.lang.IllegalArgumentException: Errors were discovered while reifying SystemDescriptor(
implementation=com.transport.webservices.security.service.AuthenticationTokenService
contracts={com.transport.webservices.security.api.resource.AuthenticationResource}
scope=org.glassfish.hk2.api.PerLookup
qualifiers={}
descriptorType=CLASS
descriptorVisibility=NORMAL
metadata=
rank=0
loader=org.glassfish.hk2.utilities.binding.AbstractBinder$2@71a88a3e
proxiable=null
proxyForSameScope=null
analysisName=null
id=142
locatorId=0
identityHashCode=1099971211
reified=false)
at org.jvnet.hk2.internal.SystemDescriptor.reify(SystemDescriptor.java:681)
.
.
WARNING: HK2 service reification failed for [com.transport.webservices.security.service.AuthenticationTokenService] with an exception:
MultiException stack 1 of 2
java.lang.NoSuchMethodException: Could not find a suitable constructor in com.transport.webservices.security.service.AuthenticationTokenService class.
.
.
MultiException stack 2 of 2
java.lang.IllegalArgumentException: Errors were discovered while reifying SystemDescriptor(
implementation=com.transport.webservices.security.service.AuthenticationTokenService
contracts={com.transport.webservices.security.api.resource.AuthenticationResource}
scope=org.glassfish.hk2.api.PerLookup
qualifiers={}
descriptorType=CLASS
descriptorVisibility=NORMAL
metadata=
rank=0
loader=org.glassfish.hk2.utilities.binding.AbstractBinder$2@71a88a3e
proxiable=null
proxyForSameScope=null
analysisName=null
id=142
locatorId=0
identityHashCode=1099971211
reified=false)
at org.jvnet.hk2.internal.SystemDescriptor.reify(SystemDescriptor.java:681)
.
.
WARNING: Unknown HK2 failure detected:
MultiException stack 1 of 3
org.glassfish.hk2.api.UnsatisfiedDependencyException: There was no object available for injection at SystemInjecteeImpl(requiredType=AuthenticationTokenSettings,parent=AuthenticationTokenIssuer,qualifiers={},position=-1,optional=false,self=false,unqualified=null,672817351)
.
.
MultiException stack 2 of 3
java.lang.IllegalArgumentException: While attempting to resolve the dependencies of com.transport.webservices.security.service.AuthenticationTokenIssuer errors were found
.
.
MultiException stack 3 of 3
java.lang.IllegalStateException: Unable to perform operation: resolve on com.transport.webservices.security.service.AuthenticationTokenIssuer
.
.
WARNING: Unknown HK2 failure detected:
MultiException stack 1 of 5
org.glassfish.hk2.api.UnsatisfiedDependencyException: There was no object available for injection at SystemInjecteeImpl(requiredType=AuthenticationTokenSettings,parent=AuthenticationTokenIssuer,qualifiers={},position=-1,optional=false,self=false,unqualified=null,672817351)
.
.
MultiException stack 2 of 5
java.lang.IllegalArgumentException: While attempting to resolve the dependencies of com.transport.webservices.security.service.AuthenticationTokenIssuer errors were found
.
.
MultiException stack 3 of 5
java.lang.IllegalStateException: Unable to perform operation: resolve on com.transport.webservices.security.service.AuthenticationTokenIssuer
.
.
MultiException stack 4 of 5
java.lang.IllegalArgumentException: While attempting to resolve the dependencies of com.transport.webservices.security.api.resource.AuthenticationResource errors were found
.
.
MultiException stack 5 of 5
java.lang.IllegalStateException: Unable to perform operation: resolve on com.transport.webservices.security.api.resource.AuthenticationResource
.
.
java.lang.AssertionError: Should return status 200 expected:<200> but was:<500>
at org.junit.Assert.fail(Assert.java:88)
In addition to this i had to add the below maven dependencies because my existing soap classes gave up compiling after i addded maven dependencies for jersey testing which i could not understand why. I don't know i fthe problem is about this.
<dependency>
<groupId>javax.xml.bind</groupId>
<artifactId>jaxb-api</artifactId>
<version>2.3.1</version>
</dependency>
<dependency>
<groupId>javax.xml.ws</groupId>
<artifactId>jaxws-api</artifactId>
<version>2.3.1</version>
</dependency>
<dependency>
<groupId>javax.jws</groupId>
<artifactId>javax.jws-api</artifactId>
<version>1.1</version>
</dependency>
I created a class extending AnnotationLiteral
as below:
public static class ConfigurableQualifierLiteral extends AnnotationLiteral<Configurable> implements Configurable
{
String value;
ConfigurableQualifierLiteral(String value)
{
this.value = value;
}
@Override
public String value()
{
return value;
}
}
Then in my AbstractBinder
i've bound my annotated members as below:
bind(Long.valueOf(36000)).to(new TypeLiteral<Long>()
{
}).qualifiedBy(new TestAuthenticationResource.ConfigurableQualifierLiteral("authentication.jwt.validFor"));