moxy json介绍_使用MOXy 2.5.1快速且有点脏的JSON模式生成

moxy json介绍

因此,这些天我正在为即将推出的Oracle云服务开发新的REST API,因此我需要做的事情之一就是能够为模型中的bean自动生成JSON模式。 我正在使用MOXy从POJO生成JSON,从EclipseLink 2.5.1版本开始,它现在具有从bean模型生成JSON模式的能力。

将来会有一个更加正式的解决方案集成到Jersey 2.x; 但是如果您想尝试一下,此解决方案目前就可以使用。

因此,我们需要设置的第一类是模型处理器,这是非常内部的Jersey类,它允许我们使用额外的方法和资源来修改资源模型。 我们可以向模型中的每个资源添加JsonSchemaHandler ,以完成生成新模式的艰苦工作。 由于这是一个简单的POC,因此这里没有缓存,如果要在生产代码中使用它,请注意这一点。

import com.google.common.collect.Lists;import example.Bean;import java.io.IOException;
import java.io.StringWriter;import java.text.SimpleDateFormat;import java.util.Date;
import java.util.List;import javax.inject.Inject;import javax.ws.rs.HttpMethod;
import javax.ws.rs.WebApplicationException;
import javax.ws.rs.container.ContainerRequestContext;
import javax.ws.rs.core.Configuration;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;import javax.xml.bind.JAXBException;
import javax.xml.bind.SchemaOutputResolver;
import javax.xml.transform.Result;
import javax.xml.transform.stream.StreamResult;import org.eclipse.persistence.jaxb.JAXBContext;import org.glassfish.jersey.process.Inflector;
import org.glassfish.jersey.server.ExtendedUriInfo;
import org.glassfish.jersey.server.model.ModelProcessor;
import org.glassfish.jersey.server.model.ResourceMethod;
import org.glassfish.jersey.server.model.ResourceModel;
import org.glassfish.jersey.server.model.RuntimeResource;
import org.glassfish.jersey.server.model.internal.ModelProcessorUtil;
import org.glassfish.jersey.server.wadl.internal.WadlResource;public class JsonSchemaModelProcessor implements ModelProcessor {private static final MediaType JSON_SCHEMA_TYPE = MediaType.valueOf("application/schema+json");private final List<ModelProcessorUtil.Method> methodList;public JsonSchemaModelProcessor() {methodList = Lists.newArrayList();methodList.add(new ModelProcessorUtil.Method("$schema", HttpMethod.GET, MediaType.WILDCARD_TYPE, JSON_SCHEMA_TYPE,JsonSchemaHandler.class));}@Overridepublic ResourceModel processResourceModel(ResourceModel resourceModel, Configuration configuration) {return ModelProcessorUtil.enhanceResourceModel(resourceModel, true, methodList, true).build();}@Overridepublic ResourceModel processSubResource(ResourceModel resourceModel, Configuration configuration) {return ModelProcessorUtil.enhanceResourceModel(resourceModel, true, methodList, true).build();}public static class JsonSchemaHandler implements Inflector<ContainerRequestContext, Response> {private final String lastModified = new SimpleDateFormat(WadlResource.HTTPDATEFORMAT).format(new Date());@Injectprivate ExtendedUriInfo extendedUriInfo;@Overridepublic Response apply(ContainerRequestContext containerRequestContext) {// Find the resource that we are decorating, then work out the// return type on the first GETList<RuntimeResource> ms = extendedUriInfo.getMatchedRuntimeResources();List<ResourceMethod> rms = ms.get(1).getResourceMethods();Class responseType = null;found:for (ResourceMethod rm : rms) {if ("GET".equals(rm.getHttpMethod())) {responseType = (Class) rm.getInvocable().getResponseType();break found;}}if (responseType == null) {throw new WebApplicationException("Cannot resolve type for schema generation");}//try {JAXBContext context = (JAXBContext) JAXBContext.newInstance(responseType);StringWriter sw = new StringWriter();final StreamResult sr = new StreamResult(sw);context.generateJsonSchema(new SchemaOutputResolver() {@Overridepublic Result createOutput(String namespaceUri, String suggestedFileName) throws IOException {return sr;}}, responseType);return Response.ok().type(JSON_SCHEMA_TYPE).header("Last-modified", lastModified).entity(sw.toString()).build();} catch (JAXBException jaxb) {throw new WebApplicationException(jaxb);}}}}

请注意, JsonSchemaHandler代码中的非常简单的启发式方法假定每个资源都有1:1映射到单个JSON Schema元素。 当然,这可能不适用于您的特定应用程序。

现在,我们已经在一个已知的位置生成了架构,我们需要将其告知客户,我们要做的第一件事是确保当用户在特定资源上调用OPTIONS时,有一个合适的链接头:

import java.io.IOException;import javax.ws.rs.container.ContainerRequestContext;
import javax.ws.rs.container.ContainerResponseContext;
import javax.ws.rs.container.ContainerResponseFilter;
import javax.ws.rs.core.Context;
import javax.ws.rs.core.Link;
import javax.ws.rs.core.UriInfo;public class JsonSchemaResponseFilter implements ContainerResponseFilter {@Contextprivate UriInfo uriInfo;@Overridepublic void filter(ContainerRequestContext containerRequestContext,ContainerResponseContext containerResponseContext) throws IOException {String method = containerRequestContext.getMethod();if ("OPTIONS".equals(method)) {Link schemaUriLink =Link.fromUriBuilder(uriInfo.getRequestUriBuilder().path("$schema")).rel("describedBy").build();containerResponseContext.getHeaders().add("Link", schemaUriLink);}}
}

由于这是JAX-RS 2.x,因此我们正在与之合作,将所有内容捆绑为一个功能:

import javax.ws.rs.core.Feature;
import javax.ws.rs.core.FeatureContext;public class JsonSchemaFeature implements Feature {@Overridepublic boolean configure(FeatureContext featureContext) {if (!featureContext.getConfiguration().isRegistered(JsonSchemaModelProcessor.class)) {featureContext.register(JsonSchemaModelProcessor.class);featureContext.register(JsonSchemaResponseFilter.class);return true;}return false;}
}

我不会展示我的整个POJO类集。 但是很快,这就是模式生成代码所需的带有@GET方法的Resource类:

import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;
import javax.ws.rs.core.MediaType;@Path("/bean")
public class BeanResource {@GET@Produces(MediaType.APPLICATION_JSON)public Bean getBean() {return new Bean();}
}

最后,如果对资源执行GET,将看到以下内容:

GET .../resources/bean
Content-Type: application/json{"message" : "hello","other" : {"message" : "OtherBean"},"strings" : ["one","two","three","four"]
}

和选项:

OPTIONS .../resources/bean
Content-Type: text/plain
Link: <http://.../resources/bean/$schema>; rel="describedBy"GET, OPTIONS, HEAD

最后,如果您解析模式资源:

GET .../resources/bean/$schema
Content-Type: application/schema+json{"$schema" : "http://json-schema.org/draft-04/schema#","title" : "example.Bean","type" : "object","properties" : {"message" : {"type" : "string"},"other" : {"$ref" : "#/definitions/OtherBean"},"strings" : {"type" : "array","items" : {"type" : "string"}}},"additionalProperties" : false,"definitions" : {"OtherBean" : {"type" : "object","properties" : {"message" : {"type" : "string"}},"additionalProperties" : false}}
}

这里有很多工作要做,特别是根据我不久前转发到Jersey 2.xa的声明性链接注释生成超媒体扩展。 但这确实指向解决方案,我们可以运用各种解决方案使某些事情现在起作用。

翻译自: https://www.javacodegeeks.com/2014/04/quick-and-a-bit-dirty-json-schema-generation-with-moxy-2-5-1.html

moxy json介绍

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/341390.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C++迭代器(Iterator)详解

迭代器&#xff08;Iterator&#xff09;的介绍 背景&#xff1a; 指针可以用来遍历存储空间连续的数据结构&#xff0c;但是对于存储空间费连续的&#xff0c;就需要寻找一个行为类似指针的类&#xff0c;来对非数组的数据结构进行遍历。定义&#xff1a;迭代器是一种检查容器…

用键盘怎么直接打出小于等于和大于等于

≤&#xff1a;按住Alt&#xff0c;然后再按数字键盘41436≥&#xff1a;按住Alt&#xff0c;然后再按数字键盘41437

Java中的责任链设计模式

在本教程中&#xff0c;我们将学习如何在Java中实现责任链模式。 “责任链”设计模式涉及到拥有一系列负责处理请求的对象链。 当客户端发送请求时&#xff0c;第一个处理程序将尝试处理该请求。 如果可以处理&#xff0c;则请求处理在此结束。 但是&#xff0c;如果处理程序无…

C++ vector 遍历修改汇总

通过迭代器、下标操作符、范围for三种方式遍历vector内部元素以及不同遍历方式修改元素。 #include<iostream> #include<vector> using namespace std; int main() {vector<int> ivec;for(int i 0; i < 10; i)ivec.push_back(i);//遍历vector元素//1: 利…

C++11 for区间遍历:vector几种遍历方式

近来&#xff0c;基本上所有现代编程语言都有一种对一个区间写for循环的便捷方式。最终&#xff0c;C也有了相同的概念&#xff1b;你可以给循环提供一个容器&#xff0c;它帮你迭代。 example: #include <iostream> #include <vector> #include <map> usi…

指针基础及实例

指针基础 指针&#xff08;变量&#xff09;是C/C支持的一种变量。指针变量的值是一个地址&#xff0c;该地址是一个16进制的数。指针变量指向该地址所对应的变量。 指针声明 //*号表示指针变量。**号表示二级指针&#xff08;指向指针的指针&#xff09;&#xff0c;只有由它…

硒测试可实现有效的测试自动化

尽管移动应用程序的市场份额有大幅增长&#xff0c;但网络应用程序仍然以大量的用户群为主导。 在部署之前&#xff0c;企业正在快速关注Web应用程序的质量。 这是测试发挥重要作用的地方。 与某些可以自动化的功能测试不同&#xff0c;UI测试仍然主要是手动过程。 但是将测试自…

C++ map用法总结(整理)

1&#xff0c;map简介 map是STL的一个关联容器&#xff0c;它提供一对一的hash。 第一个可以称为关键字(key)&#xff0c;每个关键字只能在map中出现一次&#xff1b;第二个可能称为该关键字的值(value)&#xff1b; map以模板(泛型)方式实现&#xff0c;可以存储任意类型的…

InterruptedException和中断线程的解释

如果没有将InterruptedException检查为异常&#xff0c;则可能甚至没人会注意到它–实际上&#xff0c;这些年来可以防止出现几个错误。 但是由于必须对其进行处理&#xff0c;因此许多人不正确或不加考虑地处理它。 让我们以一个线程的简单示例为例&#xff0c;该线程定期进行…

pat 乙级 1020 月饼(C++实现)

题目 月饼是中国人在中秋佳节时吃的一种传统食品&#xff0c;不同地区有许多不同风味的月饼。现给定所有种类月饼的库存量、总售价、以及市场的最大需求量&#xff0c;请你计算可以获得的最大收益是多少。 注意&#xff1a;销售时允许取出一部分库存。样例给出的情形是这样的&…

Python的from和import用法

import使一个变量名引用整个模块对象&#xff0c;因此必须通过模块名称来得到该模块的属性,比如我们导入一个数学计算的模块 math&#xff1a; import mathprint math.pi #导出圆周率的值>>>3.14159265359我们导入math模块&#xff0c;在python模块学习中我们会知道&a…

带有Spring Cloud Config和JHipster的Java微服务

朋友不允许朋友写用户身份验证。 厌倦了管理自己的用户&#xff1f; 立即尝试Okta的API和Java SDK。 在几分钟之内即可对任何应用程序中的用户进行身份验证&#xff0c;管理和保护。 如今&#xff0c;使用Java和Spring Boot开发微服务架构非常流行。 它绝对是Java生态系统中最…

pat 乙级 1026 程序运行时间(C++)

题目 要获得一个 C 语言程序的运行时间&#xff0c;常用的方法是调用头文件 time.h&#xff0c;其中提供了 clock() 函数&#xff0c;可以捕捉从程序开始运行到 clock() 被调用时所耗费的时间。这个时间单位是 clock tick&#xff0c;即“时钟打点”。同时还有一个常数 CLK_TC…

什么是命名空间 为什么C++需要使用命名空间

引言&#xff1a; 问题&#xff1a;学习过C&#xff0c;然后再学c的人可能会这么问&#xff0c;我#include<iostream>之后&#xff0c;不就是把头文件拉进来了吗&#xff0c;而cin cout等函数不就是iostream.h里面的函数吗&#xff0c;我使用cin cout时&#xff0c;编译器…

pat 乙级 1023 组个最小数(C++)

题目 给定数字 0-9 各若干个。你可以以任意顺序排列这些数字&#xff0c;但必须全部使用。目标是使得最后得到的数尽可能小&#xff08;注意 0 不能做首位&#xff09;。例如&#xff1a;给定两个 0&#xff0c;两个 1&#xff0c;三个 5&#xff0c;一个 8&#xff0c;我们得…

C++谷歌命名规范

1.文件命名规则 文件名全部小写&#xff0c;可以含下划线或连字符&#xff0c;按项目约定命名,且尽量保证文件名明确。比如&#xff1a;cmd_save_player_info_class.cc &#xff0c;my_use_full_class.cc 定义类的文件名一般是成对出现&#xff0c;如&#xff1a;foo_bar.h f…

pat 乙级 1028 人口普查(C++)

题目 某城镇进行人口普查&#xff0c;得到了全体居民的生日。现请你写个程序&#xff0c;找出镇上最年长和最年轻的人。 这里确保每个输入的日期都是合法的&#xff0c;但不一定是合理的——假设已知镇上没有超过 200 岁的老人&#xff0c;而今天是 2014 年 9 月 6 日&#xf…

c++ 为什么要按它们声明的顺序初始化成员变量?

我今天正在写一些代码&#xff0c;并得到一个奇怪的编译错误&#xff0c;这似乎是由初始化成员变量的顺序不同于他们声明的。 class Test {int a;int b;public:Test() : b(1), a(2) {} };int main() {Test test;return 0; } 然后&#xff0c;如果我编译它与-Werror -Wall&…

如何在Java中使用Lombok删除样板安装程序获取器

你好朋友&#xff0c; 一次又一次反对Java的观点之一是&#xff0c;我们必须写很多样板 我们简单的POJO类的setter和getter形式的代码&#xff0c;不必要地增加了 我们代码的长度。 为了解决这个问题&#xff0c;有一个名为Project Lombok的开源项目可以解决这个问题。 通过…

Python 列表list与数组array的区别

1. 列表list与数组array的定义&#xff1a; 列表是由一系列按特定顺序排列的元素组成&#xff0c;可以将任何东西加入列表中&#xff0c;其中的元素之间没有任何关系&#xff1b; Python中的列表(list)用于顺序存储结构。它可以方便、高效的的添加删除元素&#xff0c;并且列…