7 个版本
0.1.0 | 2024 年 5 月 13 日 |
---|---|
0.0.16 | 2024 年 2 月 28 日 |
0.0.15 | 2023 年 12 月 15 日 |
0.0.14 | 2023 年 9 月 25 日 |
0.0.11 | 2023 年 3 月 18 日 |
#316 在 编程语言
每月 27 次下载
41KB
255 行
cargo-typify
安装后,以下命令将 JSON Schema 文件转换为 Rust 代码
$ cargo typify my_types.json
这是对 typify
包的封装,用于命令行使用。
安装
使用 cargo install cargo-typify
安装。此命令要求已安装 rustfmt
。使用 rustup component add rustfmt 安装 rustfmt
示例
$cat id-or-name.json
{
"$schema": "https://json-schema.fullstack.org.cn/draft-07/schema#",
"definitions": {
"IdOrName": {
"oneOf": [
{
"title": "Id",
"allOf": [
{
"type": "string",
"format": "uuid"
}
]
},
{
"title": "Name",
"allOf": [
{
"$ref": "#/definitions/Name"
}
]
}
]
},
"Name": {
"title": "A name unique within the parent collection",
"description": "Names must begin with a lower case ASCII letter, be composed exclusively of lowercase ASCII, uppercase ASCII, numbers, and '-', and may not end with a '-'. Names cannot be a UUID though they may contain a UUID.",
"type": "string",
"pattern": "^(?![0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}$)^[a-z][a-z0-9-]*[a-zA-Z0-9]$",
"maxLength": 63
}
}
}
$ cargo typify id-or-name.json && cat id-or-name.rs
#![allow(clippy::redundant_closure_call)]
#![allow(clippy::needless_lifetimes)]
#![allow(clippy::match_single_binding)]
#![allow(clippy::clone_on_copy)]
use serde::{Deserialize, Serialize};
#[derive(Clone, Debug, Deserialize, Serialize)]
#[serde(untagged)]
pub enum IdOrName {
Id(uuid::Uuid),
Name(Name),
}
impl From<&IdOrName> for IdOrName {
fn from(value: &IdOrName) -> Self {
value.clone()
}
}
impl std::str::FromStr for IdOrName {
type Err = &'static str;
fn from_str(value: &str) -> Result<Self, &'static str> {
if let Ok(v) = value.parse() {
Ok(Self::Id(v))
} else if let Ok(v) = value.parse() {
Ok(Self::Name(v))
} else {
Err("string conversion failed for all variants")
}
}
}
impl std::convert::TryFrom<&str> for IdOrName {
type Error = &'static str;
fn try_from(value: &str) -> Result<Self, &'static str> {
value.parse()
}
}
impl std::convert::TryFrom<&String> for IdOrName {
type Error = &'static str;
fn try_from(value: &String) -> Result<Self, &'static str> {
value.parse()
}
}
impl std::convert::TryFrom<String> for IdOrName {
type Error = &'static str;
fn try_from(value: String) -> Result<Self, &'static str> {
value.parse()
}
}
impl ToString for IdOrName {
fn to_string(&self) -> String {
match self {
Self::Id(x) => x.to_string(),
Self::Name(x) => x.to_string(),
}
}
}
impl From<uuid::Uuid> for IdOrName {
fn from(value: uuid::Uuid) -> Self {
Self::Id(value)
}
}
impl From<Name> for IdOrName {
fn from(value: Name) -> Self {
Self::Name(value)
}
}
#[doc = "Names must begin with a lower case ASCII letter, be composed exclusively of lowercase ASCII, uppercase ASCII, numbers, and '-', and may not end with a '-'. Names cannot be a UUID though they may contain a UUID."]
#[derive(Clone, Debug, Serialize)]
pub struct Name(String);
impl std::ops::Deref for Name {
type Target = String;
fn deref(&self) -> &String {
&self.0
}
}
impl From<Name> for String {
fn from(value: Name) -> Self {
value.0
}
}
impl From<&Name> for Name {
fn from(value: &Name) -> Self {
value.clone()
}
}
impl std::str::FromStr for Name {
type Err = &'static str;
fn from_str(value: &str) -> Result<Self, &'static str> {
if value.len() > 63usize {
return Err("longer than 63 characters");
}
if regress::Regex::new("^(?![0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}$)^[a-z][a-z0-9-]*[a-zA-Z0-9]$")
.unwrap()
.find(value)
.is_none()
{
return Err("doesn't match pattern \"^(?![0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}$)^[a-z][a-z0-9-]*[a-zA-Z0-9]$\"");
}
Ok(Self(value.to_string()))
}
}
impl std::convert::TryFrom<&str> for Name {
type Error = &'static str;
fn try_from(value: &str) -> Result<Self, &'static str> {
value.parse()
}
}
impl std::convert::TryFrom<&String> for Name {
type Error = &'static str;
fn try_from(value: &String) -> Result<Self, &'static str> {
value.parse()
}
}
impl std::convert::TryFrom<String> for Name {
type Error = &'static str;
fn try_from(value: String) -> Result<Self, &'static str> {
value.parse()
}
}
impl<'de> serde::Deserialize<'de> for Name {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: serde::Deserializer<'de>,
{
String::deserialize(deserializer)?
.parse()
.map_err(|e: &'static str| <D::Error as serde::de::Error>::custom(e.to_string()))
}
}
选项
请参阅 cargo typify --help
获取完整选项列表。
使用 --output
选项可以覆盖默认输出文件(使用 .rs
替换输入文件扩展名)。使用 -
表示标准输出。
使用 --no-builder
禁用结构体构建器生成(默认为 --builder
)。构建器输出允许您编写如下代码
let xy: MyStruct = MyStruct::builder().x_coord(x).y_coord(y).try_into();
使用 --additional-derive
将指定的 derive 宏添加到所有生成的类型。可以多次指定。
依赖关系
~13–24MB
~376K SLoC