生成kafka内部请求与响应的接口文档

/**
*/
package com.code260.ss.kafka10demo; import java.io.File;
import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.net.URL;
import java.net.URLDecoder;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.List;
import java.util.jar.JarEntry;
import java.util.jar.JarFile; import org.apache.kafka.common.protocol.types.BoundField;
import org.apache.kafka.common.protocol.types.Schema;
import org.apache.kafka.common.requests.AbstractRequest;
import org.apache.kafka.common.requests.AbstractResponse; /**
* @since 2019-08-07
*
*/
public class GenKafkaRequestDoc { /**
* @param args
* @throws IOException
* @throws ClassNotFoundException
*/
public static void main(String[] args) throws IOException, ClassNotFoundException { collect(AbstractRequest.class);
System.out.println("=======");
collect(AbstractResponse.class);
} /**
* @param sedClazz
* @throws IOException
* @throws ClassNotFoundException
*/
public static void collect(Class<?> sedClazz) throws IOException, ClassNotFoundException {
List<Class<?>> children = getAllChildrenClass("org.apache.kafka", sedClazz);
List<Class<?>> childrenOfAbstractRequest = filter(children, sedClazz);
for (Class<?> rquestClazz : childrenOfAbstractRequest) {
System.out.println("\n## " + rquestClazz.getSimpleName());
try {
Schema[] schemas = (Schema[]) rquestClazz.getDeclaredMethod("schemaVersions").invoke(null);
for (int i = 0; i < schemas.length; i++) {
Schema schema = schemas[i];
System.out.println("\n### version:" + i);
StringBuffer sb = new StringBuffer();
sb.append(
"<table><tr><td style=\"width:160px\">name</td><td>type</td><td style=\"width:80px\">defaultValue</td><td>docString</td></tr>");
for (int j = 0; j < schema.fields().length; j++) {
BoundField field = schema.fields()[j]; sb.append("<tr><td>").append(field.def.name).append("</td>");
sb.append("<td>").append(field.def.type).append("</td>");
sb.append("<td>").append(field.def.defaultValue).append("</td>");
sb.append("<td>").append(field.def.docString).append("</td></tr>").append("\n"); }
sb.append("</table> \n \n ");
System.out.println(sb.toString());
}
// System.out.println(schemas);
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (SecurityException e) {
e.printStackTrace();
}
}
} private static List<Class<?>> filter(List<Class<?>> children, Class<?> parentClazz) {
List<Class<?>> result = new ArrayList<Class<?>>();
if (children == null || children.isEmpty()) {
return result;
}
for (Class<?> child : children) {
if (isAssignedFrom(child, parentClazz)) {
result.add(child);
}
}
return result;
} private static boolean isAssignedFrom(Class<?> currentClazz, Class<?> targetParentClazz) {
if (targetParentClazz == null || currentClazz == null) {
return false;
} // Class<?>[] parentInterfaces = currentClazz.getInterfaces();
Class<?> parentClazz = currentClazz.getSuperclass();
while (parentClazz != null) {
if (parentClazz.equals(targetParentClazz)) {
return true;
} else {
parentClazz = parentClazz.getSuperclass();
}
} Class<?>[] parentInterfaces = currentClazz.getInterfaces();
return isAssignedFrom(parentInterfaces, targetParentClazz);
} private static boolean isAssignedFrom(Class<?>[] parentInterfaces, Class<?> targetParentClazz) {
if (parentInterfaces == null) {
return false;
}
for (Class<?> ppInterface : parentInterfaces) {
if (ppInterface != null && ppInterface.equals(targetParentClazz)) {
return true;
} else {
boolean result = isAssignedFrom(ppInterface.getInterfaces(), targetParentClazz);
if (result) {
return true;
}
}
}
return false;
} private static List<Class<?>> getAllChildrenClass(String scanPackageStr, Class<?> parentClazz)
throws IOException, ClassNotFoundException {
List<Class<?>> children = new ArrayList<Class<?>>();
Enumeration<URL> packageLocations = parentClazz.getClassLoader().getResources(scanPackageStr.replace(".", "/"));
while (packageLocations.hasMoreElements()) {
URL packageLocation = packageLocations.nextElement();
String protocol = packageLocation.getProtocol();
if ("file".equals(protocol)) {
String filePath = URLDecoder.decode(packageLocation.getFile(), "UTF-8");
findAndAddClassesInPackageByFile(filePath, scanPackageStr, children);
} else if ("jar".equals(protocol)) {
findAndAddClassesInPackageByJar(packageLocation.getPath(), scanPackageStr, children); }
}
return children;
} private static void findAndAddClassesInPackageByJar(String packageLocation, String packageName,
List<Class<?>> clazzList) throws IOException, ClassNotFoundException {
String[] segs = packageLocation.split("!");
String jarFilePath = segs[0].substring(segs[0].indexOf("/"));
jarFilePath = URLDecoder.decode(jarFilePath, "UTF-8");
String packagePath = segs[1].substring(1);
JarFile jarFile = null;
try {
jarFile = new JarFile(jarFilePath);
Enumeration<JarEntry> entrys = jarFile.entries();
while (entrys.hasMoreElements()) {
JarEntry jarEntry = entrys.nextElement();
String entryName = jarEntry.getName();
if (entryName.endsWith(".class")) {
if (entryName.startsWith(packagePath)) {
entryName = entryName.replace("/", ".").substring(0, entryName.lastIndexOf("."));
Class<?> clazz = Thread.currentThread().getContextClassLoader().loadClass(entryName);
clazzList.add(clazz);
}
}
}
} catch (Exception e) {
e.printStackTrace();
}
finally {
if (jarFile != null) {
jarFile.close();
}
} } private static void findAndAddClassesInPackageByFile(String filePath, String packageName,
List<Class<?>> clazzList) {
File file = new File(filePath);
File[] children = file.listFiles();
for (File child : children) {
if (child.isDirectory()) {
findAndAddClassesInPackageByFile(child.getAbsolutePath(), packageName + "." + child.getName(),
clazzList);
}
if (child.isFile()) {
if (!child.getName().endsWith(".class")) {
continue;
}
}
String childName = child.getName();
if (!childName.endsWith(".class")) {
continue;
}
String className = packageName + "." + childName.substring(0, childName.length() - ".class".length());
try {
Class<?> clazz = Thread.currentThread().getContextClassLoader().loadClass(className);
clazzList.add(clazz);
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
} }

生成kafka内部请求与响应的接口文档的更多相关文章

  1. Swagger 接口文档规范

    导语: 相信无论是前端还是后端开发,都或多或少地被接口文档折磨过.前端经常抱怨后端给的接口文档与实际情况不一致.后端又觉得编写及维护接口文档会耗费不少精力,经常来不及更新.其实无论是前端调用后端,还是 ...

  2. Markdown写接口文档,自动添加TOC

    上回说到,用Impress.js代替PPT来做项目展示.这回换Markdown来做接口文档好了.(不敢说代替Word,只能说个人感觉更为方便)当然,还要辅之以Git,来方便版本管理. Markdown ...

  3. Django框架深入了解_05 (Django中的缓存、Django解决跨域流程(非简单请求,简单请求)、自动生成接口文档)

    一.Django中的缓存: 前戏: 在动态网站中,用户所有的请求,服务器都会去数据库中进行相应的增,删,查,改,渲染模板,执行业务逻辑,最后生成用户看到的页面. 当一个网站的用户访问量很大的时候,每一 ...

  4. SpringBoot整合knife4j框架(可生成离线接口文档),并设置接口请求头token默认值

    功能和swagger类似 官网地址:https://doc.xiaominfo.com/knife4j/ 这个框架可以设置返回字段的描述 引入依赖 <dependency> <gro ...

  5. drf07 过滤 排序 分页 异常处理 自动生成接口文档

    4. 过滤Filtering 对于列表数据可能需要根据字段进行过滤,我们可以通过添加django-fitlter扩展来增强支持. pip install django-filter 在配置文件sett ...

  6. spring boot:用swagger3生成接口文档,支持全局通用参数(swagger 3.0.0 / spring boot 2.3.2)

    一,什么是swagger? 1,  Swagger 是一个规范和完整的文档框架, 用于生成.描述.调用和可视化 RESTful 风格的 Web 服务文档 官方网站: https://swagger.i ...

  7. day74:drf:drf其他功能:认证/权限/限流/过滤/排序/分页/异常处理&自动生成接口文档

    目录 1.django-admin 2.认证:Authentication 3.权限:Permissions 4.限流:Throttling 5.过滤:Filtering 6.排序:OrderingF ...

  8. SpringBoot接口 - 如何生成接口文档之非侵入方式(通过注释生成)Smart-Doc?

    通过Swagger系列可以快速生成API文档,但是这种API文档生成是需要在接口上添加注解等,这表明这是一种侵入式方式: 那么有没有非侵入式方式呢, 比如通过注释生成文档? 本文主要介绍非侵入式的方式 ...

  9. asp.net core 使用 swagger 生成接口文档

    参考地址:http://www.cnblogs.com/daxnet/p/6181366.html http://www.jianshu.com/p/fa5a9b76f3ed 微软参考文档:https ...

随机推荐

  1. 题解 洛谷 P2280 【[HNOI2003]激光炸弹】

    这道题因为要求价值最大值,所以正方形应尽可能多覆盖目标,因此所得的正方形四个顶点一定在格点上. 经过分析后,我们就可以知道,该题做法就是用二维前缀和进行事前预处理,然后一个一个枚举每个点覆盖到的总价值 ...

  2. 数字货币交易所(火币为例)如何使用二次验证码/虚拟MFA/两步验证/谷歌验证器?

    一般点账户名——设置——安全设置中开通虚拟MFA两步验证 具体步骤见链接  数字货币交易所(火币为例)如何使用二次验证码/虚拟MFA/两步验证/谷歌验证器? 二次验证码小程序于谷歌身份验证器APP的优 ...

  3. python爬虫入门(1)----- requests

    介绍 requests是python实现的简单易用的HTTP库,使用起来比urllib简洁很多 基本使用 requests.get("http://www.baidu.com") ...

  4. DJANGO-天天生鲜项目从0到1-008-列表页

    本项目基于B站UP主‘神奇的老黄’的教学视频‘天天生鲜Django项目’,视频讲的非常好,推荐新手观看学习 https://www.bilibili.com/video/BV1vt41147K8?p= ...

  5. feign.FeignException: status 404 reading xxService#xxmethod

    做乐优商城授权中心出错 public interface UserApi { @GetMapping("query") public User queryUser( @Reques ...

  6. Python列表截取

    Python列表截取: 使用索引下标查看列表元素: lst = ['a','b','c','d','e','f','g','h'] print(lst[0]) # a print(lst[3]) # ...

  7. PHP xml_get_current_line_number() 函数

    定义和用法 xml_get_current_line_number() 函数获取 XML 解析器的当前行号.高佣联盟 www.cgewang.com 如果成功,该函数则返回当前行号.如果失败,则返回 ...

  8. Skill 脚本演示 ycAlignAll.il

    https://www.cnblogs.com/yeungchie/ ycAlignAll.il 将版图整体对齐至 指定象限 / 原点,可以忽略 Label 干扰带来的 offGrid 的风险. 回到 ...

  9. 5.20 省选模拟赛 求和 组合数的性质 EGF CRT

    LINK:求和 绝妙的一道题目.没做绝对亏了. 对于第一个subtask 考虑直接递推出组合数. 对于第二个subtask 考虑EGF 设两个EGF 都只含偶数项指标且系数为1的那种 一个到n一个到m ...

  10. 4.9 省选模拟赛 划分序列 二分 结论 树状数组优化dp

    显然发现可以二分. 对于n<=100暴力dp f[i][j]表示前i个数分成j段对于当前的答案是否可行. 可以发现这个dp是可以被优化的 sum[i]-sum[j]<=mid sum[i] ...