36 个版本 (19 个破坏性更改)
0.20.1 | 2024 年 2 月 15 日 |
---|---|
0.19.0 | 2024 年 2 月 3 日 |
0.18.4 | 2023 年 12 月 27 日 |
0.14.0 | 2023 年 10 月 31 日 |
0.1.8 | 2022 年 12 月 27 日 |
#754 in 网页编程
每月 323 次下载
用于 2 crates
1MB
1K SLoC
oasgen
- OpenAPI 规范生成器
oasgen
是一个库,可以从 Rust 服务器代码(或任何异步函数)生成 OpenAPI 3.0 规范。它支持
actix
- actix-webaxum
- axum- 无框架 - 如果您只想将 Rust 函数注册为生成 OpenAPI 规范文件。
欢迎为支持其他网页框架做出贡献!
示例
// Actix-web example
use actix_web::web::Json;
use actix_web::{App, HttpServer};
use oasgen::{oasgen, OaSchema, Server};
use serde::{Deserialize, Serialize};
#[derive(OaSchema, Deserialize)]
pub struct SendCode {
pub mobile: String,
}
#[derive(Serialize, OaSchema, Debug)]
pub struct SendCodeResponse {
pub found_account: bool,
}
#[oasgen]
async fn send_code(_body: Json<SendCode>) -> Json<SendCodeResponse> {
Json(SendCodeResponse {
found_account: false,
})
}
#[tokio::main]
async fn main() {
let server = Server::actix().post("/send-code", send_code).freeze();
HttpServer::new(move || App::new().service(server.clone().into_service()))
.bind(("127.0.0.1", 5000))
.unwrap()
.run()
.await
.unwrap()
}
// axum example
use oasgen::{OaSchema, Server, oasgen};
use axum::{Json, routing};
use serde::{Deserialize, Serialize};
#[derive(OaSchema, Deserialize)]
pub struct SendCode {
pub mobile: String,
}
#[derive(Serialize, OaSchema, Debug)]
pub struct SendCodeResponse {
pub found_account: bool,
}
#[oasgen]
async fn send_code(_body: Json<SendCode>) -> Json<SendCodeResponse> {
Json(SendCodeResponse { found_account: false })
}
#[tokio::main]
async fn main() {
let server = Server::axum()
.post("/send-code", send_code)
.freeze();
let router = axum::Router::new()
.route("/healthcheck", routing::get(|| async { "OK" }))
.merge(server.into_router());
axum::Server::bind(&"0.0.0.0:5000".parse().unwrap())
.serve(router.into_make_service())
.await
.unwrap();
}
要编译 axum 示例,请使用以下依赖项
[dependencies]
axum = "0.6"
oasgen = { version = "0.19.0", features = ["axum"] }
serde = { version = "1.0.196", features = ["derive"] }
tokio = { version = "1.36.0", features = ["full"] }
安装
[dependencies]
# At minimum, you probably want a server feature installed (axum, actix) to support that framework
oasgen = { version = "..", features = []}
有几种功能可以激活其他库
actix
- actix-webaxum
- axumswagger-ui
- swagger uiuuid
- uuidchrono
- chronotime
- timesqlx
- sqlx
自定义生成的规范
您可以通过多种方式自定义生成的规范。
直接访问 OpenAPI 结构
您可以直接访问 OpenAPI 结构,因此您可以按任何方式自定义它。
let mut server = Server::new();
server.openapi.info.title = "My API".to_string();
server.openapi.components.schemas.insert("MySchema".to_string(), Schema::new_object());
server
.get("/my-route", my_handler)
.freeze();
请注意,您必须在调用 .freeze()
(该操作将 OpenAPI 结构移动到 Arc 以在线程之间共享)之前进行任何更改。
自定义模式
您可以通过手动编写 OaSchema 的实现来自定义任何模式,而不是使用 derive 进行自定义。如果您这样做,请在结构定义之后调用 register_schema
以将其添加到规范中。
use oasgen::{OaSchema, Schema, register_schema};
pub struct User {
pub id: i32,
pub name: String,
}
impl OaSchema for User {
fn schema() -> Schema {
let mut schema = Schema::new_object();
schema.properties_mut().insert("id", Schema::new_integer());
schema.properties_mut().insert("name", Schema::new_string());
schema
}
}
register_schema!("User", &|| User::schema());
从技术上来说,您根本不需要实现 OaSchema。您可以将任何返回 Schema
的任意闭包传递给 register_schema 宏。
您还可以自定义操作
async fn my_server_handler() {
// ...
}
// You must use the fully qualified path to the function.
// You can simplify this slightly by passing in `concat!(module_path!(), "::my_server_handler")`
register_operation!("my_server_crate::path::to::my_server_handler", &|| {
let mut operation = Operation::default();
operation.summary = Some("My summary".to_string());
// ...
operation
});
属性
oasgen
定义了自己的属性,并尊重 serde
属性。它还使用 docstrings 作为描述。您可以在 macro/src/attr.rs
中查看所有属性。查看这些结构的相关文档,并查看下面的示例。
#[derive(OaSchema)]
pub struct User {
pub id: i32,
pub name: String,
// Because oasgen respects serde attributes, this will not appear in the spec.
#[serde(skip)]
pub password_hash: String,
// This will be in the response (because there's no serde(skip), but it will not show up in the OpenAPI spec.
#[oasgen(skip)]
pub internal_id: i32,
}
#[oasgen(
tags("auth", "users"),
summary = "This is a short summary"),
deprecated = true,
operation_id = "my_operation_id",
description = "This is a long description and will override the docstring of the function",
)]
async fn my_server_handler() {
// ...
}
将规范写入文件
您可以直接访问 OpenAPI
结构。您可以使用 serde
将其写入文件、stdout 等。
我们提供了一个辅助函数 write_and_exit_if_env_var_set
,它与基本的构建过程很好地集成。
let server = Server::new()
// your routes
.write_and_exit_if_env_var_set("./openapi.yaml")
// .freeze() here, if you're mounting to a server.
如果设置了 OASGEN_WRITE_SPEC=1
,它将规范写入路径,然后退出。
在您的构建过程中,构建可执行文件,设置环境变量以输出规范后运行一次,然后再次运行不带环境变量的以正常启动服务器。
显示规范的路由
[!注意]
需要swagger-ui
功能
有一些内置函数可以创建显示原始规范的路由,或显示规范的 Swagger UI 页面。
let mut server = oasgen::Server::axum()
.post("/auth/register_password", auth::register_password) // example route
.route_yaml_spec("/openapi.yaml") // the spec will be available at /openapi.yaml
.route_json_spec("/openapi.json") // the spec will be available at /openapi.json
.swagger_ui("/openapi/"); // the swagger UI will be available at /openapi/.
// NOTE: The trailing slash is required, as is calling either `route_yaml_spec()` or `route_json_spec()` before `swagger_ui()`.
如果您需要自定义这些路由,可以直接使用 OpenAPI 结构的克隆。它位于 Arc 中,因此克隆成本低。
let mut server = oasgen::Server::axum()
.post("/auth/register_password", auth::register_password) // example route
.freeze();
let spec = server.openapi.clone();
let router = axum::Router::new()
.merge(server.into_router())
.route("/alt/route/to/openapi.yaml", get(|| {
let spec = spec.clone();
async {
serde_yaml::to_string(spec).unwrap()
}
}))
;
依赖关系
~9-25MB
~328K SLoC