use crate::core::{Function, FunctionEvaluationResult};
use crate::lang::lir::{Bindings, EvalContext};
use crate::package::Package;
use crate::runtime::{Output, RuntimeError};
use crate::runtime::{PackagePath, World};
use crate::value::{RationaleResult, RuntimeValue};
use ariadne::Cache;
use base64::alphabet::STANDARD;
use base64::engine::general_purpose::STANDARD_NO_PAD as PEM_ENGINE;
use base64::Engine;
use std::cell::RefCell;
use std::future::Future;
use std::pin::Pin;
use std::rc::Rc;
use std::str::from_utf8;
use std::sync::Arc;
pub fn package() -> Package {
let mut pkg = Package::new(PackagePath::from_parts(vec!["pem"]));
pkg.register_function("AsCertificate".into(), AsCertificate);
pkg
}
#[allow(clippy::upper_case_acronyms)]
#[derive(Debug)]
pub struct AsCertificate;
impl Function for AsCertificate {
fn order(&self) -> u8 {
128
}
fn call<'v>(
&'v self,
input: Rc<RuntimeValue>,
ctx: &'v mut EvalContext,
bindings: &'v Bindings,
world: &'v World,
) -> Pin<Box<dyn Future<Output = Result<FunctionEvaluationResult, RuntimeError>> + 'v>> {
Box::pin(async move {
let bytes = if let Some(inner) = input.try_get_octets() {
inner
} else {
return Ok(Output::None.into());
};
let contents = PEM_ENGINE.encode(&bytes);
let mut result = String::with_capacity(contents.len() + 128);
result.push_str("-----BEGIN CERTIFICATE-----\n");
for c in contents.as_bytes().chunks(64) {
result.push_str(from_utf8(c).unwrap());
result.push('\n');
}
result.push_str("-----END CERTIFICATE-----\n");
Ok(Output::Transform(Rc::new(result.into())).into())
})
}
}