use std::process::Stdio;
use nftables::{helper::NftablesError, schema::Nftables};
use tokio::{io::AsyncWriteExt, process::Command};
const NFT_DEFAULT_PROGRAM: &str = "nft";
pub async fn apply_ruleset(
    nftables: impl AsRef<Nftables>,
    program: Option<&str>,
    args: Option<Vec<&str>>,
) -> Result<(), NftablesError> {
    let payload =
        serde_json::to_string(nftables.as_ref()).map_err(NftablesError::NftInvalidJson)?;
    apply_ruleset_raw(payload, program, args).await
}
pub async fn apply_ruleset_raw(
    payload: String,
    program: Option<&str>,
    args: Option<Vec<&str>>,
) -> Result<(), NftablesError> {
    let program = program.unwrap_or(NFT_DEFAULT_PROGRAM);
    let mut command = Command::new(program);
    command.arg("-j").arg("-f").arg("-");
    if let Some(args) = args {
        command.args(args);
    }
    command
        .stdin(Stdio::piped())
        .stdout(Stdio::piped())
        .stderr(Stdio::piped());
    let mut child = command.spawn().map_err(|err| NftablesError::NftExecution {
        program: program.to_owned(),
        inner: err,
    })?;
    let mut stdin = child
        .stdin
        .take()
        .expect("Stdin was piped to Tokio but could not be retrieved");
    stdin
        .write_all(payload.as_bytes())
        .await
        .map_err(|err| NftablesError::NftExecution {
            program: program.to_owned(),
            inner: err,
        })?;
    drop(stdin);
    match child.wait_with_output().await {
        Ok(output) if output.status.success() => Ok(()),
        Ok(output) => {
            let stdout = read(program, output.stdout)?;
            let stderr = read(program, output.stderr)?;
            Err(NftablesError::NftFailed {
                program: program.to_owned(),
                hint: "applying ruleset".to_string(),
                stdout,
                stderr,
            })
        }
        Err(err) => Err(NftablesError::NftExecution {
            program: program.to_owned(),
            inner: err,
        }),
    }
}
pub async fn get_current_ruleset(
    program: Option<&str>,
    args: Option<Vec<&str>>,
) -> Result<Nftables, NftablesError> {
    let output = get_current_ruleset_raw(program, args).await?;
    serde_json::from_str(&output).map_err(NftablesError::NftInvalidJson)
}
pub async fn get_current_ruleset_raw(
    program: Option<&str>,
    args: Option<Vec<&str>>,
) -> Result<String, NftablesError> {
    let program = program.unwrap_or(NFT_DEFAULT_PROGRAM);
    let mut command = Command::new(program);
    command.arg("-j").arg("list").arg("ruleset");
    if let Some(args) = args {
        command.args(args);
    }
    let output = command
        .output()
        .await
        .map_err(|err| NftablesError::NftExecution {
            program: program.to_owned(),
            inner: err,
        })?;
    let stdout = read(program, output.stdout)?;
    if !output.status.success() {
        let stderr = read(program, output.stderr)?;
        return Err(NftablesError::NftFailed {
            program: program.to_owned(),
            hint: "getting the current ruleset".to_string(),
            stdout,
            stderr,
        });
    }
    Ok(stdout)
}
#[inline]
fn read(program: &str, stream: Vec<u8>) -> Result<String, NftablesError> {
    String::from_utf8(stream).map_err(|err| NftablesError::NftOutputEncoding {
        program: program.to_owned(),
        inner: err,
    })
}