SpringBoot配置SpringDoc

This commit is contained in:
2023-10-18 16:03:14 +08:00
parent cc567b98e6
commit 0e1da3342e
12 changed files with 930 additions and 30 deletions

View File

@ -23,6 +23,14 @@
<groupId>com.jnssd</groupId>
<artifactId>spring-boot-model</artifactId>
</dependency>
<!--springdoc 官方Starter-->
<dependency>
<groupId>org.springdoc</groupId>
<artifactId>springdoc-openapi-ui</artifactId>
<version>1.7.0</version>
</dependency>
</dependencies>
</project>

View File

@ -0,0 +1,18 @@
package com.jnssd;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
/**
* <h3>spring-boot-openapi</h3>
* <p>入口</p>
*
* @author zxj
* @since 2023-10-17 10:10:28
*/
@SpringBootApplication
public class SpringDocApplication {
public static void main(String[] args) {
SpringApplication.run(SpringDocApplication.class, args);
}
}

View File

@ -0,0 +1,123 @@
package com.jnssd.config;
import io.swagger.v3.oas.models.Components;
import io.swagger.v3.oas.models.ExternalDocumentation;
import io.swagger.v3.oas.models.OpenAPI;
import io.swagger.v3.oas.models.info.Contact;
import io.swagger.v3.oas.models.info.Info;
import io.swagger.v3.oas.models.info.License;
import io.swagger.v3.oas.models.security.*;
import org.springdoc.core.GroupedOpenApi;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
/**
* <h3>spring-boot-openapi</h3>
* <p></p>
*
* @author zxj
* @since 2023-10-16 17:41:17
*/
@Configuration
@EnableConfigurationProperties(SwaggerProperties.class)
public class SpringDocConfig {
@Bean
public OpenAPI openAPI(SwaggerProperties properties) {
return new OpenAPI()
.info(createInfo(properties))
// .externalDocs(new ExternalDocumentation()
// .description("SpringDoc API功能 演示")
// .url("http://127.0.0.1:" + port + "/"))
.components(new Components()
.addSecuritySchemes("basicScheme", new SecurityScheme().type(SecurityScheme.Type.HTTP).scheme("basic").bearerFormat("JWT"))
.addSecuritySchemes("oauth2", securitySchemeOauth2Password()))
.addSecurityItem(new SecurityRequirement().addList("basicScheme")
.addList("oauth2"));
}
/**
* 文档信息
* @param properties 配置信息
*/
Info createInfo(SwaggerProperties properties){
return new Info()
.title(properties.getTitle())
.description(properties.getDescription())
.termsOfService(properties.getTermsOfServiceUrl())
.contact(new Contact().name(properties.getContact().getName()).url(properties.getContact().getUrl()).email(properties.getContact().getEmail()))
.license(new License().name("Apache 2.0").url(properties.getLicense()))
.version(properties.getVersion());
}
@Bean
public GroupedOpenApi menuApi() {
return GroupedOpenApi.builder()
.addOpenApiCustomiser(openApi -> openApi.info(new Info().title("菜单 API")
.description("菜单 API 演示")
.version("v1.0.0")
.license(new License().name("Apache 2.0").url("jnssd.com"))
.contact(new Contact().name("zxj").email("zxj@163.com").url("jnssd.com"))
.termsOfService("jnssd")
).externalDocs(new ExternalDocumentation()
.description("菜单 API功能 演示").url("jnssd.com")))
.group("menu")
.pathsToMatch("/menu/**")
.addOperationCustomizer((operation, handlerMethod) -> {
operation.addSecurityItem(new SecurityRequirement().addList("basicScheme").addList("oauth2"));
return operation;
})
.build();
}
//
// @Bean
// public GroupedOpenApi userApi() {
// return GroupedOpenApi.builder()
// .group("user")
// .pathsToMatch("/user/**")
// .build();
// }
// @Bean
// public OpenAPI customOpenAPI() {
// SecurityScheme oauth2Scheme = new SecurityScheme()
// .type(SecurityScheme.Type.OAUTH2)
// .flows(new OAuthFlows()
// .password(new OAuthFlow()
// .tokenUrl("/oauth/token")
// .scopes(new Scopes()
// .addString("read_scope", "Grants read access")
// .addString("write_scope", "Grants write access")
// .addString("admin_scope", "Grants read write and delete access")
// )
// )
// );
//
// Components components = new Components().addSecuritySchemes("OAuth2", oauth2Scheme);
// SecurityRequirement securityRequirement = new SecurityRequirement().addList("OAuth2");
//
// return new OpenAPI()
// .components(components)
// .addSecurityItem(securityRequirement);
// }
/**
* 配置oauth2密码模式
*
* @return
*/
SecurityScheme securitySchemeOauth2Password() {
return new SecurityScheme()
.type(SecurityScheme.Type.OAUTH2)
.flows(new OAuthFlows()
.password(new OAuthFlow()
.tokenUrl("/oauth/token")
.scopes(new Scopes()
.addString("read_scope", "Grants read access")
.addString("write_scope", "Grants write access")
.addString("admin_scope", "Grants read write and delete access")
)
)
);
}
}

View File

@ -0,0 +1,112 @@
// package com.jnssd.config;
//
// import io.swagger.v3.oas.models.tags.Tag;
// import org.springdoc.core.GroupedOpenApi;
// import org.springdoc.core.customizers.OpenApiCustomiser;
// import org.springdoc.core.customizers.OperationCustomizer;
// import org.springdoc.core.customizers.RouterOperationCustomizer;
// import org.springdoc.core.filters.OpenApiMethodFilter;
// import org.springframework.context.annotation.Bean;
// import org.springframework.context.annotation.Configuration;
// import org.springframework.http.HttpMethod;
//
// @Configuration
// public class SpringDocOpenApiConfig {
//
//
// @Bean
// public GroupedOpenApi publicApi() {
// return GroupedOpenApi.builder()
// .group("menu")
// .pathsToMatch("/menu/**")
// .addOperationCustomizer(customizePublicApiOperations())
// .addOpenApiCustomiser(customizePublicApiOpenApi())
// .addRouterOperationCustomizer(customizePublicApiRouterOperations())
// .addOpenApiMethodFilter(publicApiMethodFilter())
// .build();
// }
//
// @Bean
// public GroupedOpenApi privateApi() {
// return GroupedOpenApi.builder()
// .group("user")
// .pathsToMatch("/user/**")
// .addOperationCustomizer(customizePrivateApiOperations())
// .addOpenApiCustomiser(customizePrivateApiOpenApi())
// .addRouterOperationCustomizer(customizePrivateApiRouterOperations())
// .addOpenApiMethodFilter(privateApiMethodFilter())
// .build();
// }
//
// // Customizing public API operations
// private OperationCustomizer customizePublicApiOperations() {
// return (operation, handlerMethod) -> {
// // Customize the operations for the public API
// operation.operationId("publicOperation");
// operation.addTagsItem("Public API");
// return operation;
// };
// }
//
// // Customizing public API OpenAPI document
// private OpenApiCustomiser customizePublicApiOpenApi() {
// return (openApi) -> {
// // Customize the OpenAPI document for the public API
// openApi.addTagsItem(new Tag().name("Public API").description("Endpoints for public API"));
// };
// }
//
// // Customizing public API router operations
// private RouterOperationCustomizer customizePublicApiRouterOperations() {
// return (routerOperation, handlerMethod) -> {
// // Customize the router operations for the public API
// // Customize the router operation as needed
// return routerOperation;
// };
// }
//
// // Filtering public API methods
// private OpenApiMethodFilter publicApiMethodFilter() {
// return (method) -> {
// // Filter out DELETE methods from the public API
// return HttpMethod.GET.matches(method.getName());
// // return true;
// };
// }
//
// // Customizing private API operations
// private OperationCustomizer customizePrivateApiOperations() {
// return (operation, handlerMethod) -> {
// // Customize the operations for the private API
// operation.operationId("privateOperation");
// operation.addTagsItem("Private API");
// return operation;
// };
// }
//
// // Customizing private API OpenAPI document
// private OpenApiCustomiser customizePrivateApiOpenApi() {
// return (openApi) -> {
// // Customize the OpenAPI document for the private API
// openApi.addTagsItem(new Tag().name("Private API").description("Endpoints for private API"));
// };
// }
//
// // Customizing private API router operations
// private RouterOperationCustomizer customizePrivateApiRouterOperations() {
// return (routerOperation, handlerMethod) -> {
// // Customize the router operations for the private API
// // Customize the router operation as needed
// return routerOperation;
// };
// }
//
// // Filtering private API methods
// private OpenApiMethodFilter privateApiMethodFilter() {
// return (method) -> {
// // Filter out PATCH methods from the private API
// return true;
// };
// }
// }
//

View File

@ -0,0 +1,57 @@
package com.jnssd.config;
import io.swagger.v3.oas.models.info.Contact;
import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
/**
* <h3>spring-boot-openapi</h3>
* <p>Swagger配置</p>
*
* @author zxj
* @since 2023-10-18 09:57:25
*/
@Data
@ConfigurationProperties(prefix = "swagger")
public class SwaggerProperties {
/**
* 标题
**/
private String title = "";
/**
* 描述
**/
private String description = "";
/**
* 版本
**/
private String version = "";
/**
* 许可证
**/
private String license = "";
/**
* 许可证URL
**/
private String licenseUrl = "";
/**
* 服务条款URL
**/
private String termsOfServiceUrl = "";
/**
* host信息
**/
private String host = "";
/**
* 联系人信息
*/
private Contact contact = new Contact();
}

View File

@ -0,0 +1,17 @@
package com.jnssd.controller;
/**
* <h3>spring-boot-openapi</h3>
* <p></p>
*
* @author zxj
* @since 2023-10-12 17:35:26
*/
// @Controller
// public class CommonController {
//
// @GetMapping("/")
// public String index() {
// return "redirect:/swagger-ui/index.html";
// }
// }

View File

@ -2,10 +2,13 @@ package com.jnssd.controller;
import com.jnssd.model.Menu;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import java.util.List;
import java.util.Objects;
/**
* <h3>spring-boot-openapi</h3>
@ -14,39 +17,66 @@ import java.util.List;
* @author zxj
* @since 2023-10-12 16:33:23
*/
@RestController("menu")
@RestController
@RequestMapping("/menu")
public class MenuController {
final static String SUCCESS_TEXT = "操作成功!";
final static String FAIL_TEXT = "操作失败!";
List<Menu> list = new java.util.ArrayList<>();
@GetMapping("/")
public ResponseEntity<List<Menu>> getAll() {
return ResponseEntity.ok(list);
}
// 添加方法
@PostMapping("add")
public ResponseEntity<Menu> add(Menu entity) {
list.add(entity);
return ResponseEntity.ok(entity);
public ResponseEntity<String> add(Menu entity) {
try {
entity.setId(list.size() + 1);
list.add(entity);
return ResponseEntity.ok(SUCCESS_TEXT);
} catch (Exception e) {
e.printStackTrace();
return ResponseEntity.status(400).body(FAIL_TEXT);
}
}
// 添加方法
@PostMapping("update")
public ResponseEntity<Menu> update(Menu entity) {
public ResponseEntity<Object> update(Menu entity) {
try {
// 修改list下面的数据
list.replaceAll((menu) -> Objects.equals(menu.getId(), entity.getId()) ? entity : menu);
return null;
return ResponseEntity.ok(SUCCESS_TEXT);
} catch (Exception e) {
return ResponseEntity.status(400).body(FAIL_TEXT);
}
}
// 删除方法
@PostMapping("delete")
public ResponseEntity<Menu> delete(Menu entity) {
return null;
public ResponseEntity<String> delete(Menu entity) {
try {
boolean result = list.removeIf(e -> Objects.equals(e.getId(), entity.getId()));
return result ? ResponseEntity.ok(SUCCESS_TEXT) : ResponseEntity.status(400).body(FAIL_TEXT);
} catch (Exception e) {
return ResponseEntity.status(400).body(null);
}
}
// 查询方法
@PostMapping("query")
public ResponseEntity<List<Menu>> query(Menu entity) {
return null;
public ResponseEntity<Object> query(Integer id) {
try {
Menu menu = (Menu) list.stream().filter(e -> Objects.equals(e.getId(), id));
// list 下查询
return ResponseEntity.ok(menu);
} catch (Exception e) {
return ResponseEntity.status(400).body(FAIL_TEXT);
}
}
}

View File

@ -1,6 +1,14 @@
package com.jnssd.controller;
import lombok.Data;
import com.jnssd.model.Role;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import java.util.List;
import java.util.Objects;
/**
* <h3>spring-boot-openapi</h3>
@ -9,10 +17,66 @@ import lombok.Data;
* @author zxj
* @since 2023-10-12 16:32:38
*/
@Data
@RestController
@RequestMapping("/role")
public class RoleController {
private String id;
private String name;
private String description;
private String createTime;
final static String SUCCESS_TEXT = "操作成功!";
final static String FAIL_TEXT = "操作失败!";
List<Role> list = new java.util.ArrayList<>();
@GetMapping("/")
public ResponseEntity<List<Role>> getAll() {
return ResponseEntity.ok(list);
}
// 添加方法
@PostMapping("add")
public ResponseEntity<String> add(Role entity) {
try {
entity.setId(list.size() + 1);
list.add(entity);
return ResponseEntity.ok(SUCCESS_TEXT);
} catch (Exception e) {
e.printStackTrace();
return ResponseEntity.status(400).body(FAIL_TEXT);
}
}
// 添加方法
@PostMapping("update")
public ResponseEntity<Object> update(Role entity) {
try {
// 修改list下面的数据
list.replaceAll((role) -> Objects.equals(role.getId(), entity.getId()) ? entity : role);
return ResponseEntity.ok(SUCCESS_TEXT);
} catch (Exception e) {
return ResponseEntity.status(400).body(FAIL_TEXT);
}
}
// 删除方法
@PostMapping("delete")
public ResponseEntity<String> delete(Role entity) {
try {
boolean result = list.removeIf(e -> Objects.equals(e.getId(), entity.getId()));
return result ? ResponseEntity.ok(SUCCESS_TEXT) : ResponseEntity.status(400).body(FAIL_TEXT);
} catch (Exception e) {
return ResponseEntity.status(400).body(null);
}
}
// 查询方法
@PostMapping("query")
public ResponseEntity<Object> query(Integer id) {
try {
Role role = (Role) list.stream().filter(e -> Objects.equals(e.getId(), id));
// list 下查询
return ResponseEntity.ok(role);
} catch (Exception e) {
return ResponseEntity.status(400).body(FAIL_TEXT);
}
}
}

View File

@ -1,6 +1,14 @@
package com.jnssd.controller;
import lombok.Data;
import com.jnssd.model.User;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import java.util.List;
import java.util.Objects;
/**
* <h3>spring-boot-openapi</h3>
@ -9,18 +17,66 @@ import lombok.Data;
* @author zxj
* @since 2023-10-12 16:30:04
*/
@Data
@RestController
@RequestMapping("/user")
public class UserController {
private String id;
private String name;
private String password;
private String email;
private String phone;
private String address;
private String description;
private String createTime;
private String updateTime;
private String status;
final static String SUCCESS_TEXT = "操作成功!";
final static String FAIL_TEXT = "操作失败!";
List<User> list = new java.util.ArrayList<>();
@GetMapping("/")
public ResponseEntity<List<User>> getAll() {
return ResponseEntity.ok(list);
}
// 添加方法
@PostMapping("add")
public ResponseEntity<String> add(User entity) {
try {
entity.setId(list.size() + 1);
list.add(entity);
return ResponseEntity.ok(SUCCESS_TEXT);
} catch (Exception e) {
e.printStackTrace();
return ResponseEntity.status(400).body(FAIL_TEXT);
}
}
// 添加方法
@PostMapping("update")
public ResponseEntity<Object> update(User entity) {
try {
// 修改list下面的数据
list.replaceAll((user) -> Objects.equals(user.getId(), entity.getId()) ? entity : user);
return ResponseEntity.ok(SUCCESS_TEXT);
} catch (Exception e) {
return ResponseEntity.status(400).body(FAIL_TEXT);
}
}
// 删除方法
@PostMapping("delete")
public ResponseEntity<String> delete(User entity) {
try {
boolean result = list.removeIf(e -> Objects.equals(e.getId(), entity.getId()));
return result ? ResponseEntity.ok(SUCCESS_TEXT) : ResponseEntity.status(400).body(FAIL_TEXT);
} catch (Exception e) {
return ResponseEntity.status(400).body(null);
}
}
// 查询方法
@PostMapping("query")
public ResponseEntity<Object> query(Integer id) {
try {
User user = (User) list.stream().filter(e -> Objects.equals(e.getId(), id));
// list 下查询
return ResponseEntity.ok(user);
} catch (Exception e) {
return ResponseEntity.status(400).body(FAIL_TEXT);
}
}
}

View File

@ -0,0 +1,365 @@
openapi: 3.0.1
info:
title: 快速开发平台 API
description: 后台管理平台
termsOfService: "https://${novel.serverHost}"
contact:
name: jnssd
url: https://jnssd.com
email: 1845124851@jnssd.com
license:
name: Apache 2.0
url: Powered By 江南山水电
version: 1.0.0
servers:
- url: http://localhost:18080
description: Generated server url
security:
- basicScheme: []
oauth2: []
paths:
/user/update:
post:
tags:
- user-controller
operationId: update
parameters:
- name: entity
in: query
required: true
schema:
$ref: '#/components/schemas/User'
responses:
"200":
description: OK
content:
'*/*':
schema:
type: object
/user/query:
post:
tags:
- user-controller
operationId: query
parameters:
- name: id
in: query
required: true
schema:
type: integer
format: int32
responses:
"200":
description: OK
content:
'*/*':
schema:
type: object
/user/delete:
post:
tags:
- user-controller
operationId: delete
parameters:
- name: entity
in: query
required: true
schema:
$ref: '#/components/schemas/User'
responses:
"200":
description: OK
content:
'*/*':
schema:
type: string
/user/add:
post:
tags:
- user-controller
operationId: add
parameters:
- name: entity
in: query
required: true
schema:
$ref: '#/components/schemas/User'
responses:
"200":
description: OK
content:
'*/*':
schema:
type: string
/role/update:
post:
tags:
- role-controller
operationId: update_1
parameters:
- name: entity
in: query
required: true
schema:
$ref: '#/components/schemas/Role'
responses:
"200":
description: OK
content:
'*/*':
schema:
type: object
/role/query:
post:
tags:
- role-controller
operationId: query_1
parameters:
- name: id
in: query
required: true
schema:
type: integer
format: int32
responses:
"200":
description: OK
content:
'*/*':
schema:
type: object
/role/delete:
post:
tags:
- role-controller
operationId: delete_1
parameters:
- name: entity
in: query
required: true
schema:
$ref: '#/components/schemas/Role'
responses:
"200":
description: OK
content:
'*/*':
schema:
type: string
/role/add:
post:
tags:
- role-controller
operationId: add_1
parameters:
- name: entity
in: query
required: true
schema:
$ref: '#/components/schemas/Role'
responses:
"200":
description: OK
content:
'*/*':
schema:
type: string
/menu/update:
post:
tags:
- menu-controller
operationId: update_2
parameters:
- name: entity
in: query
required: true
schema:
$ref: '#/components/schemas/Menu'
responses:
"200":
description: OK
content:
'*/*':
schema:
type: object
/menu/query:
post:
tags:
- menu-controller
operationId: query_2
parameters:
- name: id
in: query
required: true
schema:
type: integer
format: int32
responses:
"200":
description: OK
content:
'*/*':
schema:
type: object
/menu/delete:
post:
tags:
- menu-controller
operationId: delete_2
parameters:
- name: entity
in: query
required: true
schema:
$ref: '#/components/schemas/Menu'
responses:
"200":
description: OK
content:
'*/*':
schema:
type: string
/menu/add:
post:
tags:
- menu-controller
operationId: add_2
parameters:
- name: entity
in: query
required: true
schema:
$ref: '#/components/schemas/Menu'
responses:
"200":
description: OK
content:
'*/*':
schema:
type: string
/user/:
get:
tags:
- user-controller
operationId: getAll
responses:
"200":
description: OK
content:
'*/*':
schema:
type: array
items:
$ref: '#/components/schemas/User'
/role/:
get:
tags:
- role-controller
operationId: getAll_1
responses:
"200":
description: OK
content:
'*/*':
schema:
type: array
items:
$ref: '#/components/schemas/Role'
/menu/:
get:
tags:
- menu-controller
operationId: getAll_2
responses:
"200":
description: OK
content:
'*/*':
schema:
type: array
items:
$ref: '#/components/schemas/Menu'
components:
schemas:
User:
type: object
properties:
id:
type: integer
format: int32
name:
type: string
password:
type: string
email:
type: string
phone:
type: string
address:
type: string
description:
type: string
createTime:
type: string
updateTime:
type: string
status:
type: string
Role:
type: object
properties:
id:
type: integer
format: int32
name:
type: string
description:
type: string
createTime:
type: string
Menu:
type: object
properties:
id:
type: integer
format: int32
name:
type: string
url:
type: string
icon:
type: string
parentId:
type: string
parentName:
type: string
parentUrl:
type: string
parentIcon:
type: string
description:
type: string
sort:
type: string
type:
type: string
status:
type: string
createTime:
type: string
updateTime:
type: string
securitySchemes:
basicScheme:
type: http
scheme: basic
bearerFormat: JWT
oauth2:
type: oauth2
flows:
password:
tokenUrl: /oauth/token
scopes:
read_scope: Grants read access
write_scope: Grants write access
admin_scope: Grants read write and delete access

View File

@ -0,0 +1,50 @@
server:
port: 18080
swagger:
title: 快速开发平台 API
description: 后台管理平台
license: Powered By 江南山水电
licenseUrl: https://${novel.serverHost}
host: ${novel.serverHost}:${server.port}
terms-of-service-url: https://${novel.serverHost}
version: 1.0.0
contact:
name: jnssd
url: https://jnssd.com
email: 1845124851@jnssd.com
springdoc:
swagger-ui:
display-request-duration: true
groups-order: DESC
operationsSorter: method
disable-swagger-default-url: true
use-root-path: true
oauth:
clientId: swagger
clientSecret: 123456
realm: realm-name
appName: jnssd
scopeSeparator: read_scope
scopes:
- read_scope
- write_scope
- admin_scope
additionalQueryStringParams: { test: "hello" }
useBasicAuthenticationWithAccessCodeGrant: true
usePkceWithAuthorizationCodeGrant: true
show-actuator: true
group-configs:
- group: role
display-name: 角色信息
paths-to-match: /role/**
- group: user
display-name: 用户信息
paths-to-match: /user/**
# - group: menu
# display-name: 菜单信息
# paths-to-match:
# - /menu/**
# - /user/**

View File

@ -1,5 +1,5 @@
server:
port: 8080
port: 28080
# ========================================================================
# 启动报错需要修改以下mvc配置 Failed to start bean 'documentationPluginsBootstrapper'