mod codegen;
mod error;
mod input;
mod rename;
mod schema;
use std::path::PathBuf;
use proc_macro::TokenStream;
use syn::parse_macro_input;
use crate::{
error::ForgeError,
input::{AttrInput, MacroInput},
schema::{apply_rename, infer_top_level},
};
#[proc_macro]
pub fn json_forge(input: TokenStream) -> TokenStream {
let MacroInput {
path,
name,
vis,
data_vis,
embedded,
rename_all,
} = parse_macro_input!(input as MacroInput);
let data_vis = data_vis.as_ref().unwrap_or(&vis);
#[cfg(not(feature = "embedded"))]
if embedded == Some(true) {
return ForgeError::call_site(
"`embedded = true` requires the `embedded` Cargo feature to be enabled",
)
.to_compile_error()
.into();
}
let manifest_dir = match std::env::var("CARGO_MANIFEST_DIR") {
Ok(v) => PathBuf::from(v),
Err(_) => {
return ForgeError::call_site("CARGO_MANIFEST_DIR env var is not set")
.to_compile_error()
.into();
},
};
let path_str = path.value();
let abs_path = manifest_dir.join(&path_str);
let json_text = match std::fs::read_to_string(&abs_path) {
Ok(t) => t,
Err(e) => {
return ForgeError::spanned(
path.span(),
format!("failed to read `{}`: {e}", abs_path.display()),
)
.to_compile_error()
.into();
},
};
let root: serde_json::Value = match serde_json::from_str(&json_text) {
Ok(v) => v,
Err(e) => {
return ForgeError::spanned(path.span(), format!("JSON parse error: {e}"))
.to_compile_error()
.into();
},
};
let top_level = match infer_top_level(&root) {
Ok(t) => t,
Err(e) => return e.to_compile_error().into(),
};
let top_level = match rename_all {
Some(rule) => apply_rename(top_level, rule),
None => top_level,
};
#[cfg(feature = "embedded")]
if embedded != Some(false) {
return match codegen::embedded::generate(&vis, data_vis, &name, &top_level, &root) {
Ok(ts) => ts.into(),
Err(e) => e.to_compile_error().into(),
};
}
let _ = embedded; codegen::runtime::generate(&vis, data_vis, &name, &top_level, &path_str).into()
}
#[proc_macro_attribute]
pub fn json_forge_typed(attr: TokenStream, item: TokenStream) -> TokenStream {
let attr_input = match syn::parse::<AttrInput>(attr) {
Ok(a) => a,
Err(e) => return e.to_compile_error().into(),
};
let struct_item = match syn::parse::<syn::ItemStruct>(item) {
Ok(s) => s,
Err(e) => return e.to_compile_error().into(),
};
let AttrInput {
path,
data_vis,
embedded,
rename_all,
} = attr_input;
let struct_vis = &struct_item.vis;
let data_vis = data_vis.as_ref().unwrap_or(struct_vis);
#[cfg(not(feature = "embedded"))]
if embedded == Some(true) {
return ForgeError::call_site(
"`embedded = true` requires the `embedded` Cargo feature to be enabled",
)
.to_compile_error()
.into();
}
let manifest_dir = match std::env::var("CARGO_MANIFEST_DIR") {
Ok(v) => PathBuf::from(v),
Err(_) => {
return ForgeError::call_site("CARGO_MANIFEST_DIR env var is not set")
.to_compile_error()
.into();
},
};
let path_str = path.value();
let abs_path = manifest_dir.join(&path_str);
let json_text = match std::fs::read_to_string(&abs_path) {
Ok(t) => t,
Err(e) => {
return ForgeError::spanned(
path.span(),
format!("failed to read `{}`: {e}", abs_path.display()),
)
.to_compile_error()
.into();
},
};
let root: serde_json::Value = match serde_json::from_str(&json_text) {
Ok(v) => v,
Err(e) => {
return ForgeError::spanned(path.span(), format!("JSON parse error: {e}"))
.to_compile_error()
.into();
},
};
#[cfg(feature = "embedded")]
let use_embedded = embedded != Some(false);
#[cfg(not(feature = "embedded"))]
let use_embedded = false;
match codegen::user_defined::generate_with_text(
&struct_item,
data_vis,
&root,
&json_text,
use_embedded,
rename_all,
) {
Ok(ts) => ts.into(),
Err(e) => e.to_compile_error().into(),
}
}