- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我做了一个 ParamConverter
它提供了一个 Instant
(日期)当给定一个格式化为 Instant 的 native ISO-8601 或自纪元以来的整数毫秒数的字符串时。这工作正常,但我还需要能够支持其他日期格式(客户很挑剔)。
避免经典dd/mm/yyyy
对比mm/dd/yyyy
歧义,我想让客户指定他们的首选格式作为请求的一部分*。例如:
GET http://api.example.com/filter?since=01/02/2000&dateformat=dd/mm/yyyy
传递给一个看起来像这样的方法:
@GET
String getFilteredList( final @QueryParam( "since" ) Instant since ) {
...
}
(为清楚起见省略了时间和时区部分)
所以我想要我的 ParamConverter<Instant>
能够阅读 dateformat
参数。
我已经能够使用过滤器的组合来设置 ContainerRequestContext
属性(property)和一个AbstractValueFactoryProvider
做类似的事情,但这需要参数应用自定义注释,并且不允许它与 QueryParam/FormParam/等一起使用,这使得它的用处大大降低。
有没有办法从 ParamConverter 中获取其他参数或请求对象本身?
[*] 在现实世界中,这将来自一系列预先批准的格式,但现在假设它们正在向 DateTimeFormatter
提供输入
为清楚起见,这是我的代码:
public class InstantParameterProvider implements ParamConverterProvider {
private static final ParamConverter<Instant> INSTANT_CONVERTER =
new ParamConverter<Instant>( ) {
@Override public final T fromString( final String value ) {
// This is where I would like to get the other parameter's value
// Is it possible?
}
@Override public final String toString( final T value ) {
return value.toString( );
}
};
@SuppressWarnings( "unchecked" )
@Override public <T> ParamConverter<T> getConverter(
final Class<T> rawType,
final Type genericType,
final Annotation[] annotations
) {
if( rawType == Instant.class ) {
return (ParamConverter<T>) INSTANT_CONVERTER;
}
return null;
}
}
最佳答案
如前所述here ,关键是用 javax.inject.Provider
注入(inject)一些上下文对象,这允许我们懒惰地检索对象。自 ParamConverterProvider
是一个由 Jersey 管理的组件,我们应该可以注入(inject)其他组件。
问题是我们需要的组件将在请求范围内。为了解决这个问题,我们注入(inject) javax.inject.Provider<UriInfo>
进入供应商。当我们实际调用 get()
时在Provider
获取 UriInfo
的实际实例,它将在请求中。这同样适用于任何其他需要请求范围的组件。
例如
public class InstantParamProvider implements ParamConverterProvider {
@Inject
private javax.inject.Provider<UriInfo> uriInfoProvider;
@Override
public <T> ParamConverter<T> getConverter(Class<T> rawType,
Type genericType,
Annotation[] annotations) {
if (rawType != Instant.class) return null;
return new ParamConverter<T>() {
@Override
public T fromString(String value) {
UriInfo uriInfo = uriInfoProvider.get();
String format = uriInfo.getQueryParameters().getFirst("date-format");
if (format == null) {
throw new WebApplicationException(Response.status(400)
.entity("data-format query parameter required").build());
} else {
try {
// parse and return here
} catch (Exception ex) {
throw new WebApplicationException(
Response.status(400).entity("Bad format " + format).build());
}
}
}
@Override
public String toString(T value) {
return value.toString();
}
};
}
}
这是一个使用 Jersey 测试框架的完整示例
import java.lang.annotation.Annotation;
import java.lang.reflect.Type;
import java.time.LocalDate;
import java.time.format.DateTimeFormatter;
import java.util.logging.Logger;
import javax.inject.Inject;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.QueryParam;
import javax.ws.rs.WebApplicationException;
import javax.ws.rs.core.Response;
import javax.ws.rs.core.UriInfo;
import javax.ws.rs.ext.ParamConverter;
import javax.ws.rs.ext.ParamConverterProvider;
import org.glassfish.jersey.filter.LoggingFilter;
import org.glassfish.jersey.server.ResourceConfig;
import org.glassfish.jersey.test.JerseyTest;
import org.junit.Test;
import static org.junit.Assert.*;
import static org.junit.matchers.JUnitMatchers.*;
public class LocalDateTest extends JerseyTest {
public static class LocalDateParamProvider implements ParamConverterProvider {
@Inject
private javax.inject.Provider<UriInfo> uriInfoProvider;
@Override
public <T> ParamConverter<T> getConverter(Class<T> rawType,
Type genericType,
Annotation[] annotations) {
if (rawType != LocalDate.class) {
return null;
}
return new ParamConverter<T>() {
@Override
public T fromString(String value) {
UriInfo uriInfo = uriInfoProvider.get();
String format = uriInfo.getQueryParameters().getFirst("date-format");
if (format == null) {
throw new WebApplicationException(Response.status(400)
.entity("date-format query parameter required").build());
} else {
try {
return (T) LocalDate.parse(value, DateTimeFormatter.ofPattern(format));
// parse and return here
} catch (Exception ex) {
throw new WebApplicationException(
Response.status(400).entity("Bad format " + format).build());
}
}
}
@Override
public String toString(T value) {
return value.toString();
}
};
}
}
@Path("localdate")
public static class LocalDateResource {
@GET
public String get(@QueryParam("since") LocalDate since) {
return since.format(DateTimeFormatter.ofPattern("MM/dd/yyyy"));
}
}
@Override
public ResourceConfig configure() {
return new ResourceConfig(LocalDateResource.class)
.register(LocalDateParamProvider.class)
.register(new LoggingFilter(Logger.getAnonymousLogger(), true));
}
@Test
public void should_return_bad_request_with_bad_format() {
Response response = target("localdate")
.queryParam("since", "09/20/2015")
.queryParam("date-format", "yyyy/MM/dd")
.request().get();
assertEquals(400, response.getStatus());
assertThat(response.readEntity(String.class), containsString("format yyyy/MM/dd"));
response.close();
}
@Test
public void should_return_bad_request_with_no_date_format() {
Response response = target("localdate")
.queryParam("since", "09/20/2015")
.request().get();
assertEquals(400, response.getStatus());
assertThat(response.readEntity(String.class), containsString("query parameter required"));
response.close();
}
@Test
public void should_succeed_with_correct_format() {
Response response = target("localdate")
.queryParam("since", "09/20/2015")
.queryParam("date-format", "MM/dd/yyyy")
.request().get();
assertEquals(200, response.getStatus());
assertThat(response.readEntity(String.class), containsString("09/20/2015"));
response.close();
}
}
这是测试依赖
<dependency>
<groupId>org.glassfish.jersey.test-framework.providers</groupId>
<artifactId>jersey-test-framework-provider-grizzly2</artifactId>
<version>${jersey2.version}</version>
<scope>test</scope>
</dependency>
关于java - 读取 jersey 的 ParamConverter 中的另一个参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27584161/
我在我的 Controller 中使用 symfony @ParamConverter 来通过它的 slug 获取一个实体,但我没有使用“findBySlug()”方法,而是使用了一个更具体的方法“f
我有几条以 /experiment/{id}/... 开头的路由,我厌倦了重写相同的逻辑来检索登录用户的实验。我想我可以重构我的代码,但我猜 @ParamConverter 会是更好的解决方案。 我将
我有一个类似的 Action : /** * @Security("is_granted('ROLE_USER_EDITOR')") * @Route("/{email}") * @Method
所以我想创建一个自定义ParamConverter在 jax-rs 中: public class IntParamConverter implements ParamConverter{ @
我正在使用 Jersey 版本 2.27 创建 Jersey 应用程序。 我的资源之一是使用 @QueryParam 读取查询参数值。 我正在创建 ParamConverter 和 ParamConv
这是我遇到的异常: No result was found for query although at least one row was expected. 当在数据库中找不到用户 ID 时,我基本
我做了一个 ParamConverter它提供了一个 Instant (日期)当给定一个格式化为 Instant 的 native ISO-8601 或自纪元以来的整数毫秒数的字符串时。这工作正常,但
每个标题,从 ParamConverter 抛出的异常没有按照我期望的方式处理。 使用 ExceptionMapper: @Provider public class MyExceptionMappe
我发现 Symfony4 具有类似的博客示例,如 https://symfony.com/doc/current/routing.html 中描述的然后我添加了一条新路线来添加/blog/about
我正在尝试使用 FOSRestBundle 的请求正文转换器,但我当前的实现似乎不起作用。 我正在使用 Symfony2、Propel、AngularJS(它将数据发送到服务器) 我做错了什么? co
我有一个 Controller ,它使用路由、方法、模板和 ParamConverter 的注释。 如果我在下一页加载时对文件进行任何更改(甚至是空格更改或注释),则会发生以下错误 Cannot im
当我尝试访问其中一个路由时出现问题,出现此错误:“@ParamConverter 注释找不到 App\Entity\CompanyUser 对象” 很多人都有同样的问题,但我能看到的解决方案都不能解决
问题仅出现在调试工具栏中。 这是我怀疑导致问题的代码 /** * @Route("/{category_slug}/{slug}", name="content_show") * @Par
是否可以使用 Symfony2 中的@ParamConverter 注释将通过 $_POST 请求发送的参数转换为实体? symfony2 文档中给出的所有示例都从路由中定义的参数转换实体。 设置类似
我有这个 Controller : /** * {@inheritdoc} * * @Route("entity/{domain_host}") * @ParamConverter("enti
我创建了一个名为 search.html.twig 的新边界类,但是当我转到 URL ( http://localhost:8000/shrubs/search) 时,出现以下错误: ERROR -
我需要使用 PUT 方法实现一个 API,我想在我的 Controller 中使用 ParamConverter 来查找现有实体对象,或者如果实体对象不存在,则创建一个新实体对象。 但是,如果在存储库
我编写了一个 ParamConverterProvider 和一个 ParamConverter,以便让 JAX-RS (Jersey) 在 HeaderParam 中实例化一个枚举。转换器看起来像这
我是一名优秀的程序员,十分优秀!