extern crate proc_macro;
use proc_macro::TokenStream;
#[proc_macro]
pub fn core_riff(_: TokenStream) -> TokenStream {
    quote::quote! {
    #[soroban_sdk::contract]
    pub struct Contract__;
    #[soroban_sdk::contractimpl]
    impl Contract__ {
            pub fn owner_get(env: soroban_sdk::Env) -> Option<soroban_sdk::Address> {
                env.storage().instance().get(&Self::owner_key())
            }
            pub fn owner_set(env: soroban_sdk::Env, new_owner: soroban_sdk::Address) {
                Self::owner_get(env.clone()).as_ref().map(soroban_sdk::Address::require_auth);
                env.storage().instance().set(&Self::owner_key(), &new_owner);
            }
            pub fn redeploy(env: soroban_sdk::Env, wasm_hash: soroban_sdk::BytesN<32>) {
                Self::owner_get(env.clone()).as_ref().map(soroban_sdk::Address::require_auth);
                env.deployer().update_current_contract_wasm(wasm_hash);
            }
            fn owner_key() -> soroban_sdk::Symbol {
                soroban_sdk::symbol_short!("OWNER")
            }
    }
        }
    .into()
}
#[proc_macro]
pub fn dev_deploy(_: TokenStream) -> TokenStream {
    quote::quote! {
    #[soroban_sdk::contract]
    pub struct DevDeploy__;
    #[soroban_sdk::contractimpl]
    impl DevDeploy__ {
        pub fn dev_deploy(env: soroban_sdk::Env, wasm: soroban_sdk::Bytes) {
            let wasm_hash = env.deployer().upload_contract_wasm(wasm);
            env.deployer().update_current_contract_wasm(wasm_hash);
        }
    }
        }
    .into()
}
#[proc_macro]
pub fn import_contract_interface(tokens: TokenStream) -> TokenStream {
    let (name, file, _) = file_and_id(&tokens);
    quote::quote! {
        mod #name {
            use super::*;
            use soroban_sdk::TryFromVal;
            soroban_sdk::contractimport!(file = #file);
        }
    }
    .into()
}
#[proc_macro]
pub fn import_contract(tokens: TokenStream) -> TokenStream {
    let (name, file, Some(id)) = file_and_id(&tokens) else {
        panic!("contract_id.txt not found")
    };
    quote::quote! {
        mod #name {
            use super::*;
            use soroban_sdk::TryFromVal;
            soroban_sdk::contractimport!(file = #file);
            pub fn new(env: &Env) -> Client {
                let bytes: soroban_sdk::BytesN<32> = soroban_sdk::Bytes::from_slice(&env, &[#(#id),*]).try_into().unwrap();
                let contract_id =  &soroban_sdk::Address::from_contract_id(&bytes);
                Client::new(env, contract_id)
            }
        }
    }
    .into()
}
fn file_and_id(tokens: &TokenStream) -> (syn::Ident, String, Option<[u8; 32]>) {
    let dir = smartdeploy_build::target_dir().expect("failed to find target_dir");
    let wasm = smartdeploy_build::wasm_location(&tokens.to_string(), Some(&dir))
        .expect("failed to file wasm");
    let name =
        syn::parse::<syn::Ident>(tokens.clone()).expect("The input must be a valid identifier");
    let binding = wasm.canonicalize().expect("cannot canonicalize");
    let file = binding.to_str().unwrap();
    let id = std::fs::read_to_string(binding.parent().unwrap().join("contract_id.txt"))
        .ok()
        .and_then(|s| stellar_strkey::Contract::from_string(&s).ok());
    (name, file.to_string(), id.map(|id| id.0))
}