added IntoOwned
trait and derive macro
This commit is contained in:
parent
19243162f7
commit
c367c24606
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "torn-api-macros"
|
name = "torn-api-macros"
|
||||||
version = "0.1.1"
|
version = "0.1.2"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
authors = ["Pyrit [2111649]"]
|
authors = ["Pyrit [2111649]"]
|
||||||
license = "MIT"
|
license = "MIT"
|
||||||
|
@ -12,7 +12,7 @@ description = "Macros implementation of #[derive(ApiCategory)]"
|
||||||
proc-macro = true
|
proc-macro = true
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
syn = { version = "1", features = [ "extra-traits" ] }
|
syn = { version = "2", features = [ "parsing" ] }
|
||||||
proc-macro2 = "1"
|
proc-macro2 = "1"
|
||||||
quote = "1"
|
quote = "1"
|
||||||
convert_case = "0.5"
|
convert_case = "0.5"
|
||||||
|
|
|
@ -25,13 +25,6 @@ struct ApiAttribute {
|
||||||
with: Option<syn::Ident>,
|
with: Option<syn::Ident>,
|
||||||
}
|
}
|
||||||
|
|
||||||
fn get_lit_string(lit: syn::Lit) -> String {
|
|
||||||
match lit {
|
|
||||||
syn::Lit::Str(lit) => lit.value(),
|
|
||||||
_ => panic!("Expected api attribute to be a string"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn impl_api_category(ast: &syn::DeriveInput) -> TokenStream {
|
fn impl_api_category(ast: &syn::DeriveInput) -> TokenStream {
|
||||||
let name = &ast.ident;
|
let name = &ast.ident;
|
||||||
|
|
||||||
|
@ -41,23 +34,19 @@ fn impl_api_category(ast: &syn::DeriveInput) -> TokenStream {
|
||||||
};
|
};
|
||||||
|
|
||||||
let mut category: Option<String> = None;
|
let mut category: Option<String> = None;
|
||||||
ast.attrs
|
for attr in &ast.attrs {
|
||||||
.iter()
|
if attr.path().is_ident("api") {
|
||||||
.filter(|a| a.path.is_ident("api"))
|
attr.parse_nested_meta(|meta| {
|
||||||
.for_each(|a| {
|
if meta.path.is_ident("category") {
|
||||||
if let Ok(syn::Meta::List(l)) = a.parse_meta() {
|
let c: syn::LitStr = meta.value()?.parse()?;
|
||||||
for nested in l.nested {
|
category = Some(c.value());
|
||||||
match nested {
|
Ok(())
|
||||||
syn::NestedMeta::Meta(syn::Meta::NameValue(m))
|
} else {
|
||||||
if m.path.is_ident("category") =>
|
Err(meta.error("unknown attribute"))
|
||||||
{
|
|
||||||
category = Some(get_lit_string(m.lit))
|
|
||||||
}
|
}
|
||||||
_ => panic!("unknown api attribute"),
|
}).unwrap();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
let category = category.expect("`category`");
|
let category = category.expect("`category`");
|
||||||
|
|
||||||
|
@ -65,76 +54,41 @@ fn impl_api_category(ast: &syn::DeriveInput) -> TokenStream {
|
||||||
.variants
|
.variants
|
||||||
.iter()
|
.iter()
|
||||||
.filter_map(|variant| {
|
.filter_map(|variant| {
|
||||||
for attr in &variant.attrs {
|
let mut r#type: Option<String> = None;
|
||||||
if attr.path.is_ident("api") {
|
|
||||||
let meta = attr.parse_meta();
|
|
||||||
match meta {
|
|
||||||
Ok(syn::Meta::List(l)) => {
|
|
||||||
let mut type_: Option<String> = None;
|
|
||||||
let mut field: Option<ApiField> = None;
|
let mut field: Option<ApiField> = None;
|
||||||
let mut with: Option<String> = None;
|
let mut with: Option<proc_macro2::Ident> = None;
|
||||||
for nested in l.nested.into_iter() {
|
for attr in &variant.attrs {
|
||||||
match nested {
|
if attr.path().is_ident("api") {
|
||||||
syn::NestedMeta::Meta(syn::Meta::NameValue(m))
|
attr.parse_nested_meta(|meta| {
|
||||||
if m.path.is_ident("type") =>
|
if meta.path.is_ident("type") {
|
||||||
{
|
let t: syn::LitStr = meta.value()?.parse()?;
|
||||||
if type_.is_none() {
|
r#type = Some(t.value());
|
||||||
type_ = Some(get_lit_string(m.lit));
|
Ok(())
|
||||||
} else {
|
} else if meta.path.is_ident("with") {
|
||||||
panic!("type can only be specified once");
|
let w: syn::LitStr = meta.value()?.parse()?;
|
||||||
}
|
with = Some(quote::format_ident!("{}", w.value()));
|
||||||
}
|
Ok(())
|
||||||
syn::NestedMeta::Meta(syn::Meta::NameValue(m))
|
} else if meta.path.is_ident("field") {
|
||||||
if m.path.is_ident("with") =>
|
let f: syn::LitStr = meta.value()?.parse()?;
|
||||||
{
|
field = Some(ApiField::Property(quote::format_ident!("{}", f.value())));
|
||||||
if with.is_none() {
|
Ok(())
|
||||||
with = Some(get_lit_string(m.lit));
|
} else if meta.path.is_ident("flatten") {
|
||||||
} else {
|
|
||||||
panic!("with can only be specified once");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
syn::NestedMeta::Meta(syn::Meta::NameValue(m))
|
|
||||||
if m.path.is_ident("field") =>
|
|
||||||
{
|
|
||||||
if field.is_none() {
|
|
||||||
field = Some(ApiField::Property(quote::format_ident!(
|
|
||||||
"{}",
|
|
||||||
get_lit_string(m.lit)
|
|
||||||
)));
|
|
||||||
} else {
|
|
||||||
panic!("field/flatten can only be specified once");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
syn::NestedMeta::Meta(syn::Meta::Path(m))
|
|
||||||
if m.is_ident("flatten") =>
|
|
||||||
{
|
|
||||||
if field.is_none() {
|
|
||||||
field = Some(ApiField::Flattened);
|
field = Some(ApiField::Flattened);
|
||||||
|
Ok(())
|
||||||
} else {
|
} else {
|
||||||
panic!("field/flatten can only be specified once");
|
Err(meta.error("unsupported attribute"))
|
||||||
}
|
}
|
||||||
}
|
}).unwrap();
|
||||||
_ => panic!("Couldn't parse api attribute"),
|
let name = format_ident!("{}", variant.ident.to_string().to_case(Case::Snake));
|
||||||
}
|
|
||||||
}
|
|
||||||
let name =
|
|
||||||
format_ident!("{}", variant.ident.to_string().to_case(Case::Snake));
|
|
||||||
let raw_value = variant.ident.to_string().to_lowercase();
|
let raw_value = variant.ident.to_string().to_lowercase();
|
||||||
|
|
||||||
return Some(ApiAttribute {
|
return Some(ApiAttribute {
|
||||||
field: field.expect("one of field/flatten"),
|
field: field.expect("field or flatten attribute must be specified"),
|
||||||
name,
|
|
||||||
raw_value,
|
raw_value,
|
||||||
variant: variant.ident.clone(),
|
variant: variant.ident.clone(),
|
||||||
type_name: type_
|
type_name: r#type.expect("type must be specified").parse().unwrap(),
|
||||||
.expect("Need to specify type name")
|
name,
|
||||||
.parse()
|
with
|
||||||
.expect("failed to parse type name"),
|
})
|
||||||
with: with.map(|w| format_ident!("{}", w)),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
_ => panic!("Couldn't parse api attribute"),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
None
|
None
|
||||||
|
@ -214,3 +168,125 @@ fn impl_api_category(ast: &syn::DeriveInput) -> TokenStream {
|
||||||
|
|
||||||
gen.into()
|
gen.into()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[proc_macro_derive(IntoOwned, attributes(into_owned))]
|
||||||
|
pub fn derive_into_owned(input: TokenStream) -> TokenStream {
|
||||||
|
let ast = syn::parse(input).unwrap();
|
||||||
|
|
||||||
|
impl_into_owned(&ast)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn to_static_lt(ty: &mut syn::Type) -> bool {
|
||||||
|
let mut res = false;
|
||||||
|
match ty {
|
||||||
|
syn::Type::Path(path) => {
|
||||||
|
if let Some(syn::PathArguments::AngleBracketed(ab)) = path.path.segments.last_mut().map(|s| &mut s.arguments).as_mut() {
|
||||||
|
for mut arg in &mut ab.args {
|
||||||
|
match &mut arg {
|
||||||
|
syn::GenericArgument::Type(ty) => {
|
||||||
|
if to_static_lt(ty) {
|
||||||
|
res = true;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
syn::GenericArgument::Lifetime(lt) => {
|
||||||
|
lt.ident = syn::Ident::new("static", proc_macro2::Span::call_site());
|
||||||
|
res = true;
|
||||||
|
}
|
||||||
|
_ => ()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
syn::Type::Reference(r) => {
|
||||||
|
if let Some(lt) = r.lifetime.as_mut() {
|
||||||
|
lt.ident = syn::Ident::new("static", proc_macro2::Span::call_site());
|
||||||
|
res = true;
|
||||||
|
}
|
||||||
|
to_static_lt(&mut r.elem);
|
||||||
|
}
|
||||||
|
_ => ()
|
||||||
|
};
|
||||||
|
res
|
||||||
|
}
|
||||||
|
|
||||||
|
fn impl_into_owned(ast: &syn::DeriveInput) -> TokenStream {
|
||||||
|
let name = &ast.ident;
|
||||||
|
let (impl_generics, ty_generics, where_clause) = ast.generics.split_for_impl();
|
||||||
|
|
||||||
|
let mut identity = false;
|
||||||
|
for attr in &ast.attrs {
|
||||||
|
if attr.path().is_ident("into_owned") {
|
||||||
|
attr.parse_nested_meta(|meta| {
|
||||||
|
if meta.path.is_ident("identity") {
|
||||||
|
identity = true;
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(meta.error("unknown attribute"))
|
||||||
|
}
|
||||||
|
}).unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if identity {
|
||||||
|
return quote! {
|
||||||
|
impl #impl_generics crate::into_owned::IntoOwned for #name #ty_generics #where_clause {
|
||||||
|
type Owned = Self;
|
||||||
|
fn into_owned(self) -> Self::Owned {
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}.into()
|
||||||
|
}
|
||||||
|
|
||||||
|
let syn::Data::Struct(r#struct) = &ast.data else {
|
||||||
|
panic!("Only stucts are supported");
|
||||||
|
};
|
||||||
|
|
||||||
|
let syn::Fields::Named(named_fields) = &r#struct.fields else {
|
||||||
|
panic!("Only named fields are supported");
|
||||||
|
};
|
||||||
|
|
||||||
|
let vis = &ast.vis;
|
||||||
|
|
||||||
|
for attr in &ast.attrs {
|
||||||
|
if attr.path().is_ident("identity") {
|
||||||
|
//
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut owned_fields = Vec::with_capacity(named_fields.named.len());
|
||||||
|
let mut fields = Vec::with_capacity(named_fields.named.len());
|
||||||
|
|
||||||
|
for field in &named_fields.named {
|
||||||
|
let field_name = &field.ident.as_ref().unwrap();
|
||||||
|
let mut ty = field.ty.clone();
|
||||||
|
let vis = &field.vis;
|
||||||
|
|
||||||
|
if to_static_lt(&mut ty) {
|
||||||
|
owned_fields.push(quote! { #vis #field_name: <#ty as crate::into_owned::IntoOwned>::Owned });
|
||||||
|
fields.push(quote! { #field_name: crate::into_owned::IntoOwned::into_owned(self.#field_name) });
|
||||||
|
} else {
|
||||||
|
owned_fields.push(quote! { #vis #field_name: #ty });
|
||||||
|
fields.push(quote! { #field_name: self.#field_name });
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
let owned_name = syn::Ident::new(&format!("{}Owned", ast.ident), proc_macro2::Span::call_site());
|
||||||
|
|
||||||
|
let gen = quote! {
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
#vis struct #owned_name {
|
||||||
|
#(#owned_fields,)*
|
||||||
|
}
|
||||||
|
impl #impl_generics crate::into_owned::IntoOwned for #name #ty_generics #where_clause {
|
||||||
|
type Owned = #owned_name;
|
||||||
|
fn into_owned(self) -> Self::Owned {
|
||||||
|
#owned_name {
|
||||||
|
#(#fields,)*
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
gen.into()
|
||||||
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "torn-api"
|
name = "torn-api"
|
||||||
version = "0.5.27"
|
version = "0.5.28"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
authors = ["Pyrit [2111649]"]
|
authors = ["Pyrit [2111649]"]
|
||||||
license = "MIT"
|
license = "MIT"
|
||||||
|
@ -37,7 +37,7 @@ reqwest = { version = "0.11", default-features = false, features = [ "json" ], o
|
||||||
awc = { version = "3", default-features = false, optional = true }
|
awc = { version = "3", default-features = false, optional = true }
|
||||||
rust_decimal = { version = "1", default-features = false, optional = true, features = [ "serde" ] }
|
rust_decimal = { version = "1", default-features = false, optional = true, features = [ "serde" ] }
|
||||||
|
|
||||||
torn-api-macros = { path = "../torn-api-macros", version = "0.1.1" }
|
torn-api-macros = { path = "../torn-api-macros", version = "0.1.2" }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
actix-rt = { version = "2.7.0" }
|
actix-rt = { version = "2.7.0" }
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
use chrono::{serde::ts_seconds, DateTime, Utc};
|
use chrono::{serde::ts_seconds, DateTime, Utc};
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
|
use torn_api_macros::IntoOwned;
|
||||||
|
|
||||||
use crate::de_util;
|
use crate::de_util;
|
||||||
|
|
||||||
|
@ -36,7 +37,7 @@ pub enum StateColour {
|
||||||
Blue,
|
Blue,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Deserialize)]
|
#[derive(Debug, IntoOwned, Deserialize)]
|
||||||
pub struct Status<'a> {
|
pub struct Status<'a> {
|
||||||
pub description: &'a str,
|
pub description: &'a str,
|
||||||
#[serde(deserialize_with = "de_util::empty_string_is_none")]
|
#[serde(deserialize_with = "de_util::empty_string_is_none")]
|
||||||
|
|
|
@ -3,7 +3,7 @@ use std::collections::{BTreeMap, HashMap};
|
||||||
use chrono::{DateTime, Utc};
|
use chrono::{DateTime, Utc};
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
|
|
||||||
use torn_api_macros::ApiCategory;
|
use torn_api_macros::{ApiCategory, IntoOwned};
|
||||||
|
|
||||||
use crate::de_util::{self, null_is_empty_dict};
|
use crate::de_util::{self, null_is_empty_dict};
|
||||||
|
|
||||||
|
@ -32,7 +32,7 @@ pub enum FactionSelection {
|
||||||
|
|
||||||
pub type Selection = FactionSelection;
|
pub type Selection = FactionSelection;
|
||||||
|
|
||||||
#[derive(Debug, Clone, Deserialize)]
|
#[derive(Debug, IntoOwned, Deserialize)]
|
||||||
pub struct Member<'a> {
|
pub struct Member<'a> {
|
||||||
pub name: &'a str,
|
pub name: &'a str,
|
||||||
pub level: i16,
|
pub level: i16,
|
||||||
|
@ -42,7 +42,7 @@ pub struct Member<'a> {
|
||||||
pub last_action: LastAction,
|
pub last_action: LastAction,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Deserialize)]
|
#[derive(Debug, IntoOwned, Deserialize)]
|
||||||
pub struct FactionTerritoryWar<'a> {
|
pub struct FactionTerritoryWar<'a> {
|
||||||
pub territory_war_id: i32,
|
pub territory_war_id: i32,
|
||||||
pub territory: &'a str,
|
pub territory: &'a str,
|
||||||
|
@ -58,7 +58,7 @@ pub struct FactionTerritoryWar<'a> {
|
||||||
pub end_time: DateTime<Utc>,
|
pub end_time: DateTime<Utc>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Deserialize)]
|
#[derive(Debug, IntoOwned, Deserialize)]
|
||||||
pub struct Basic<'a> {
|
pub struct Basic<'a> {
|
||||||
#[serde(rename = "ID")]
|
#[serde(rename = "ID")]
|
||||||
pub id: i32,
|
pub id: i32,
|
||||||
|
|
79
torn-api/src/into_owned.rs
Normal file
79
torn-api/src/into_owned.rs
Normal file
|
@ -0,0 +1,79 @@
|
||||||
|
pub use torn_api_macros::IntoOwned;
|
||||||
|
|
||||||
|
pub trait IntoOwned {
|
||||||
|
type Owned;
|
||||||
|
|
||||||
|
fn into_owned(self) -> Self::Owned;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> IntoOwned for Option<T>
|
||||||
|
where
|
||||||
|
T: IntoOwned,
|
||||||
|
{
|
||||||
|
type Owned = Option<T::Owned>;
|
||||||
|
|
||||||
|
fn into_owned(self) -> Self::Owned {
|
||||||
|
self.map(IntoOwned::into_owned)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> IntoOwned for Vec<T> where T: IntoOwned {
|
||||||
|
type Owned = Vec<<T as IntoOwned>::Owned>;
|
||||||
|
|
||||||
|
fn into_owned(self) -> Self::Owned {
|
||||||
|
let mut owned = Vec::with_capacity(self.len());
|
||||||
|
for elem in self {
|
||||||
|
owned.push(elem.into_owned());
|
||||||
|
}
|
||||||
|
owned
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<K, V> IntoOwned for std::collections::HashMap<K, V> where V: IntoOwned, K: Eq + std::hash::Hash {
|
||||||
|
type Owned = std::collections::HashMap<K, <V as IntoOwned>::Owned>;
|
||||||
|
|
||||||
|
fn into_owned(self) -> Self::Owned {
|
||||||
|
self.into_iter().map(|(k, v)| (k, v.into_owned())).collect()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<K, V> IntoOwned for std::collections::BTreeMap<K, V> where V: IntoOwned, K: Eq + Ord + std::hash::Hash {
|
||||||
|
type Owned = std::collections::BTreeMap<K, <V as IntoOwned>::Owned>;
|
||||||
|
|
||||||
|
fn into_owned(self) -> Self::Owned {
|
||||||
|
self.into_iter().map(|(k, v)| (k, v.into_owned())).collect()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<Z> IntoOwned for chrono::DateTime<Z> where Z: chrono::TimeZone {
|
||||||
|
type Owned = Self;
|
||||||
|
|
||||||
|
fn into_owned(self) -> Self::Owned {
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> IntoOwned for &'a str {
|
||||||
|
type Owned = String;
|
||||||
|
|
||||||
|
fn into_owned(self) -> Self::Owned {
|
||||||
|
self.to_owned()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! impl_ident {
|
||||||
|
($name:path) => {
|
||||||
|
impl IntoOwned for $name {
|
||||||
|
type Owned = $name;
|
||||||
|
fn into_owned(self) -> Self::Owned {
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
impl_ident!(i64);
|
||||||
|
impl_ident!(i32);
|
||||||
|
impl_ident!(i16);
|
||||||
|
impl_ident!(i8);
|
||||||
|
impl_ident!(String);
|
|
@ -1,5 +1,6 @@
|
||||||
#![warn(clippy::all, clippy::perf, clippy::style, clippy::suspicious)]
|
#![warn(clippy::all, clippy::perf, clippy::style, clippy::suspicious)]
|
||||||
|
|
||||||
|
pub mod into_owned;
|
||||||
pub mod local;
|
pub mod local;
|
||||||
pub mod send;
|
pub mod send;
|
||||||
|
|
||||||
|
@ -32,6 +33,8 @@ use chrono::{DateTime, Utc};
|
||||||
use serde::{de::Error as DeError, Deserialize};
|
use serde::{de::Error as DeError, Deserialize};
|
||||||
use thiserror::Error;
|
use thiserror::Error;
|
||||||
|
|
||||||
|
pub use into_owned::IntoOwned;
|
||||||
|
|
||||||
pub struct ApiResponse {
|
pub struct ApiResponse {
|
||||||
pub value: serde_json::Value,
|
pub value: serde_json::Value,
|
||||||
}
|
}
|
||||||
|
@ -156,7 +159,7 @@ where
|
||||||
from: None,
|
from: None,
|
||||||
to: None,
|
to: None,
|
||||||
comment: None,
|
comment: None,
|
||||||
phantom: std::marker::PhantomData::default(),
|
phantom: Default::default(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,7 @@ use serde::{
|
||||||
};
|
};
|
||||||
use std::collections::{BTreeMap, HashMap};
|
use std::collections::{BTreeMap, HashMap};
|
||||||
|
|
||||||
use torn_api_macros::ApiCategory;
|
use torn_api_macros::{ApiCategory, IntoOwned};
|
||||||
|
|
||||||
use crate::de_util;
|
use crate::de_util;
|
||||||
|
|
||||||
|
@ -41,7 +41,7 @@ pub enum Gender {
|
||||||
Enby,
|
Enby,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, IntoOwned)]
|
||||||
pub struct Faction<'a> {
|
pub struct Faction<'a> {
|
||||||
pub faction_id: i32,
|
pub faction_id: i32,
|
||||||
pub faction_name: &'a str,
|
pub faction_name: &'a str,
|
||||||
|
@ -133,7 +133,7 @@ where
|
||||||
deserializer.deserialize_struct("Faction", FIELDS, FactionVisitor)
|
deserializer.deserialize_struct("Faction", FIELDS, FactionVisitor)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Deserialize)]
|
#[derive(Debug, IntoOwned, Deserialize)]
|
||||||
pub struct Basic<'a> {
|
pub struct Basic<'a> {
|
||||||
pub player_id: i32,
|
pub player_id: i32,
|
||||||
pub name: &'a str,
|
pub name: &'a str,
|
||||||
|
@ -142,7 +142,8 @@ pub struct Basic<'a> {
|
||||||
pub status: Status<'a>,
|
pub status: Status<'a>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Eq, Deserialize)]
|
#[derive(Debug, Clone, IntoOwned, PartialEq, Eq, Deserialize)]
|
||||||
|
#[into_owned(identity)]
|
||||||
pub struct Discord {
|
pub struct Discord {
|
||||||
#[serde(rename = "userID")]
|
#[serde(rename = "userID")]
|
||||||
pub user_id: i32,
|
pub user_id: i32,
|
||||||
|
@ -188,7 +189,8 @@ pub enum EliminationTeam {
|
||||||
CapsLockCrew,
|
CapsLockCrew,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone, IntoOwned)]
|
||||||
|
#[into_owned(identity)]
|
||||||
pub enum Competition {
|
pub enum Competition {
|
||||||
Elimination {
|
Elimination {
|
||||||
score: i32,
|
score: i32,
|
||||||
|
@ -327,7 +329,7 @@ where
|
||||||
deserializer.deserialize_option(CompetitionVisitor)
|
deserializer.deserialize_option(CompetitionVisitor)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Deserialize)]
|
#[derive(Debug, IntoOwned, Deserialize)]
|
||||||
pub struct Profile<'a> {
|
pub struct Profile<'a> {
|
||||||
pub player_id: i32,
|
pub player_id: i32,
|
||||||
pub name: &'a str,
|
pub name: &'a str,
|
||||||
|
|
Loading…
Reference in a new issue