java2Schema——jsonschema-generator使用

从JAVA代码生成Schema:https://github.com/victools/jsonschema-generator

引入依赖

        <dependency>
            <groupId>com.github.victools</groupId>
            <artifactId>jsonschema-generator</artifactId>
            <version>4.28.0</version>
        </dependency>

代码开发

pojo 生成schema 工具类

import com.fasterxml.jackson.databind.JsonNode;
import com.github.victools.jsonschema.generator.OptionPreset;
import com.github.victools.jsonschema.generator.SchemaGenerator;
import com.github.victools.jsonschema.generator.SchemaGeneratorConfig;
import com.github.victools.jsonschema.generator.SchemaGeneratorConfigBuilder;
import com.github.victools.jsonschema.generator.SchemaVersion;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.io.FileUtils;
import org.apache.commons.lang3.StringUtils;
import org.springframework.util.FileCopyUtils;
import org.springframework.web.multipart.MultipartFile;

import java.io.File;
import java.io.FileReader;
import java.io.LineNumberReader;
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.util.UUID;

@Slf4j
public class GenerateSchema {
    public static Class<?> getClass(String dirs, String rootClassName) {
        CustomClassLoader myClassLoader = (CustomClassLoader) AccessController.doPrivileged((PrivilegedAction) () -> new CustomClassLoader(dirs));
        return myClassLoader.findClass(rootClassName);
    }

    public static JsonNode getSchema(Class<?> type) {
        SchemaGeneratorConfigBuilder configBuilder = new SchemaGeneratorConfigBuilder(SchemaVersion.DRAFT_2020_12, OptionPreset.PLAIN_JSON);
        SchemaGeneratorConfig config = configBuilder.build();
        SchemaGenerator generator = new SchemaGenerator(config);
        try {
            return generator.generateSchema(type);
        } catch (Exception exception) {
            log.error("generate jsonSchema fail: ", exception);
        }
        return null;
    }

    public static JsonNode pojo2Schema(MultipartFile[] files, String rootClass) {
        String fileSeparator = File.separator;
        String dirs = System.getProperty("user.dir") + fileSeparator + "upload" + fileSeparator + UUID.randomUUID() + fileSeparator;
        File dir = new File(dirs);
        if (!dir.exists()) {
            boolean mkdirs = dir.mkdirs();
            log.info("mkdirs result:" + mkdirs);
        }
        try {
            String rootClassName = null;
            for (MultipartFile file : files) {
                String originalFilename = file.getOriginalFilename();
                String path = dirs + originalFilename;
                File newFile = FileUtils.getFile(path);
                if (!newFile.getParentFile().exists()) {
                    boolean mkdirs = newFile.getParentFile().mkdirs();
                    log.info("mkdirs result:" + mkdirs);
                }
                // 保存文件
                file.transferTo(newFile);
                // 这里重写类路径
                if (!getClassPath(newFile)) {
                    return null;
                }
                // 编译成class
                Tools.javac(path);
                // 将生成的class文件移动到对应包名的路径下 com.fawkes.PreCheck 则是 com/fawkes
                String classPath = path.replace(".java", ".class");
                String className = "org.test.entity";
                if (StringUtils.isNotBlank(originalFilename) && originalFilename.startsWith(rootClass)) {
                    rootClassName = className + "." + rootClass;
                }
                String replace = dirs + className.replace(".", fileSeparator);
                File file2 = new File(replace);
                if (!file2.exists()) {
                    boolean mkdirs = file2.mkdirs();
                    log.info("mkdirs result:" + mkdirs);
                }
                String name = replace + fileSeparator + originalFilename;
                String replace1 = name.replace(".java", ".class");
                File file1 = new File(replace1);
                if (!file1.exists()) {
                    boolean createRes = file1.createNewFile();
                    log.info("createRes :" + createRes);
                }
                FileCopyUtils.copy(new File(classPath), file1);
            }
            // 获取类
            Class<?> clazz = getClass(dirs, rootClassName);
            return getSchema(clazz);
        } catch (Exception exception) {
            log.info("java2Schema error ", exception);
        } finally {
            // 删除生成的文件和文件夹
            forceDelete(dir);
        }
        return null;
    }

    /**
     * 读取文件 package 那一行
     */
    public static boolean getClassPath(File file) {
        try (FileReader in = new FileReader(file); LineNumberReader reader = new LineNumberReader(in)) {
            reader.setLineNumber(1);
            String s;
            while ((s = reader.readLine()) != null) {
                if (s.startsWith("package")) {
                    return s.equals("package org.test.entity;");
                }
            }
        } catch (Exception exception) {
            log.error("readAppointedLineNumber error", exception);
        }
        return false;
    }

    public static void forceDelete(File file) {
        if (file != null && file.exists()) {
            try {
                FileUtils.forceDelete(file);
            } catch (Exception exception) {
                log.error("forceDelete error ", exception);
            }
        }
    }
}

编译Java类

import lombok.extern.slf4j.Slf4j;

import javax.tools.JavaCompiler;
import javax.tools.JavaFileObject;
import javax.tools.StandardJavaFileManager;
import javax.tools.ToolProvider;
import java.io.IOException;

@Slf4j
public abstract class Tools {

    /**
     * 编译java类
     *
     * @param writerPath 存放java文件的路径
     */
    public static void javac(String writerPath) {
        // java编译器
        JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
        // 文件管理器,参数1:diagnosticListener  监听器,监听编译过程中出现的错误
        try(StandardJavaFileManager manager = compiler.getStandardFileManager(null, null, null)) {
            // java文件转换到java对象,可以是多个文件
            Iterable<? extends JavaFileObject> it = manager.getJavaFileObjects(writerPath);
            // 编译任务,可以编译多个文件
            JavaCompiler.CompilationTask tool = compiler.getTask(null, manager, null, null, null, it);
            // 执行任务
            tool.call();
        } catch (IOException exception) {
            log.error("javac error ",exception);
        }
    }
}

前端请求

handleUpload = () => {
    const { fileList, rootClass } = this.state;
    const formData = new FormData();
    fileList.forEach(file => {
      formData.append('files[]', file);
    });
    formData.append('rootClass', rootClass);

    reqwest({
      url: apiPrefix + '/biz/model/java2Schema',
      method: 'post',
      processData: false,
      data: formData,
      headers: {
        'X-CSRF-TOKEN': getCsrfToken(),
      },
      success: (data) => {
        if (data.code === 200) {
          this.setState({
            fileList: [],
            rootClass: '',
            visible: false
          });
          if (data.data) {
            this.props.dispatch({
              type: "jsonSchemaEditorVisual/changeEditorSchemaAction",
              payload: {
                value: data.data
              },
            })
          }
        } else {
          message.error("文件格式不规范,请重新上传")
        }
      },
      error: () => {
      },
    });
  };

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

win11安装SQL Server 2012 企业版

系列文章目录 提示&#xff1a;这里可以添加系列文章的所有文章的目录&#xff0c;目录需要自己手动添加 提示&#xff1a;写完文章后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 系列文章目录前言一、硬件要求二、软件安装参考&#xff1…

uniapp开发的小程序toast被键盘遮挡提示内容无法完全显示问题解决

文章目录 问题描述问题解决参考链接&#xff1a; 问题描述 在开发抖音小程序后&#xff0c;当用户提交反馈后&#xff0c;调用了系统的toast来显示是否提交成功&#xff0c;结果被系统的键盘给盖住&#xff0c;无法显示完全。 即&#xff0c;简单来说&#xff1a;Toast会被弹…

韩顺平0基础学Java——第4天

p45—p71 老天鹅&#xff0c;居然能中断这么久&#xff0c;唉...学不完了要 API API:application programing interface应用程序编程接口 www.matools.com 可以理解成Python的调包...c的头文件对吧 字符型 char用单引号 String用双引号 char本质上是个整数&#xff0c…

AutoTable, Hibernate自动建立表替代方案

痛点 之前一直使用JPA为主要ORM技术栈&#xff0c;主要是因为Mybatis没有实体逆向建表功能。虽然Mybatis有从数据库建立实体&#xff0c;但是实际应用却没那么美好&#xff1a;当实体变更时&#xff0c;往往不会单独再建立一个数据库重新生成表&#xff0c;然后把表再逆向为实…

Pygame简单入门教程(绘制Rect、控制移动、碰撞检测、Github项目源代码)

Pygame简明教程 引言&#xff1a;本教程中的源码已上传个人Github: GItHub链接 视频教程推荐&#xff1a;YouTube教程–有点过于简单了 官方文档推荐&#xff1a;虽然写的一般&#xff0c;但还是推荐&#xff01; Navigator~ Pygame简明教程安装pygame一、代码框架二、案件输入…

小红书释放被封手机号 无限注册

前几年抖音也可以释放被封手机号 那时候都不重视 导致现在被封手机号想释放 基本不可能的 或者就是最少几百块 有专业的人帮你通过某些信息差释放 本教程是拆解 小红书被封手机号怎么释放&#xff0c;从今年开始&#xff0c;被封的手机号无法注销了 所以很困扰 那么本教程来…

如何区分APP页面是H5还是原生页面?

刚刚接触手机测试的同学&#xff0c;或多或少都有过这样的疑问&#xff1a;APP页面哪些是H5页面&#xff1f;哪些是原生页面?单凭肉眼&#xff0c;简直太难区分了&#xff01;我总结了6个小技巧&#xff0c;希望能帮大家答疑解惑。 1、看断网的情况 断开网络&#xff0c;显示…

【生信技能树】拿到表达矩阵之后,如何使用ggplot2绘图系统绘制箱线图?

拿到表达矩阵之后&#xff0c;如何使用ggplot2绘图系统绘制箱线图&#xff1f; 目录 预备知识 绘制箱线图示例 预备知识 1.pivot_longer函数 pivot_longer 是tidyr包中的一个函数&#xff0c;用于将数据框&#xff08;data frame&#xff09;从宽格式转换为长格式。在宽格…

CPU、GPU,那NPU是,神经网络到底能做什么!

人工智能时代即将到来。随着人工智能的不断推进&#xff0c;英特尔、AMD和高通等公司也在着眼于各种硬件配置方面。随着NPU&#xff08;神经网络处理器&#xff09;的引入&#xff0c;人工智能的应用过程将被加快。 苹果在其芯片中使用NPU已经很多年了&#xff0c;所以NPU并不是…

《深入Linux内核架构》第4章 进程虚拟内存(2)

目录 4.3 内存映射原理 4.4 数据结构 4.4.1 树和链表 4.4.2 虚拟内存区域VMA的表示 4.4.3 相关数据结构 本专栏文章将有70篇左右&#xff0c;欢迎关注&#xff0c;查看后续文章。 本节讲VMA结构体struct vm_area_struct和struct address_space。 4.3 内存映射原理 所有进…

k8s概述及核心组件

一、k8s概述 1.1 引言 docker compose 单机编排工具 有企业在用 docker swarm 能够在多台主机中构建一个docker集群 基本淘汰集群化管理处理工具 容器 微服务封装 dockerfile 编写成镜像 然后进行发布 dockerfile 可以写成shell脚本&#xff08;函数做调…

【Linux网络编程】HTTPS协议

【Linux网络编程】HTTPS协议 目录 【Linux网络编程】HTTPS协议HTTPS介绍加密常见的加密方式HTTPS的工作过程探究&#xff08;重点&#xff09;常见问题完整流程总结 作者&#xff1a;爱写代码的刚子 时间&#xff1a;2024.5.9 前言&#xff1a;本篇博客将会介绍HTTPS协议 HTTPS…

【记录】常见的前端设计系统(Design System)

解释一下设计系统的定义&#xff0c;以及在国内&#xff0c;都有那些优秀的设计系统可以学习&#xff0c;希望可以帮到大家。 什么是设计系统&#xff08;Design System)&#xff1f; 设计系统&#xff08;Design System&#xff09;是一套综合性的指导原则、组件和规则&…

VBA技术资料MF152:列出工作表中所有单元格的注释

我给VBA的定义&#xff1a;VBA是个人小型自动化处理的有效工具。利用好了&#xff0c;可以大大提高自己的工作效率&#xff0c;而且可以提高数据的准确度。“VBA语言専攻”提供的教程一共九套&#xff0c;分为初级、中级、高级三大部分&#xff0c;教程是对VBA的系统讲解&#…

Linux进程——Linux环境变量

前言&#xff1a;在结束完上一篇的命令行参数时&#xff0c;我们简单的了解了一下Linux中的环境变量PATH&#xff0c;而环境变量不只有PATH&#xff0c;关于更多环境变量的知识我们将在本篇展开&#xff01; 本篇主要内容&#xff1a; 常见的环境变量 获取环境变量的三种方式 本…

GORM数据库连接池对接Prometheus

一、背景与介绍 Golang的database/sql包定了关于操作数据库的相关接口&#xff0c;但是没有去做对应数据库的实现。这些实现是预留给开发者或者对应厂商进行实现的。 其中让我比较关注的是Golang的sql包有没有实现连接池pool的机制呢? 毕竟Golang是静态语言&#xff0c;类似J…

pwn(一)前置技能

以下是pwn中的题目&#xff08;漏洞&#xff09;类型&#xff1a; 关于pwn的学习&#xff1a; 一.什么是pwn&#xff1f;&#xff08;二进制的漏洞&#xff09; "Pwn"是一个俚语&#xff0c;起源于电子游戏社区&#xff0c;经常在英语中用作网络或电子游戏文化中的…

AI中转站计费平台系统源码一站式解决方案安装说明

AI中转站计费平台系统源码一站式解决方案安装说明 功能 | Features AI 联网功能 AI online searching service 多账户均衡负载 Multi-account load balancing HTTP2 Stream 实时响应功能 HTTP2 Stream real-time response function 节流和鉴权体系 Throttling and authenticati…

GitHub中Asterank源码python修改成C++(本人python不太会)

GitHub - typpo/asterank: asteroid database, interactive visualizations, and discovery tools 主要目的是在进行多元线性回归的时候将枚举型转换为数值型 python: # # The constants used in calculations for the values of asteroids. ## General constants GENERAL_I…

基于Detectron2的计算机视觉实践

书籍&#xff1a;Hands-On Computer Vision with Detectron2: Develop object detection and segmentation models with a code and visualization approach 作者&#xff1a;Van Vung Pham&#xff0c;Tommy Dang 出版&#xff1a;Packt Publishing 书籍下载-《基于Detectr…
最新文章