nu-command 0.41.0

CLI for nushell
Documentation
use std::marker::PhantomData;

use crate::prelude::*;
use nu_errors::ShellError;
use nu_protocol::{ColumnPath, Primitive, SyntaxShape, UntaggedValue, Value};
use nu_protocol::{ShellTypeName, Signature};
use nu_source::Tag;

pub trait HashDigest: digest::Digest {
    fn name() -> &'static str;
    fn examples() -> Vec<Example>;
}

pub struct SubCommand<D: HashDigest> {
    name_string: String,
    usage_string: String,
    phantom: PhantomData<D>,
}

impl<D: HashDigest> Default for SubCommand<D> {
    fn default() -> Self {
        Self {
            name_string: format!("hash {}", D::name()),
            usage_string: format!("{} encode a value", D::name()),
            phantom: PhantomData,
        }
    }
}

impl<D> WholeStreamCommand for SubCommand<D>
where
    D: HashDigest + Send + Sync,
    digest::Output<D>: core::fmt::LowerHex,
{
    fn name(&self) -> &str {
        &self.name_string
    }

    fn signature(&self) -> Signature {
        Signature::build(self.name()).rest(
            "rest",
            SyntaxShape::ColumnPath,
            format!("optionally {} encode data by column paths", D::name()),
        )
    }

    fn usage(&self) -> &str {
        &self.usage_string
    }

    fn examples(&self) -> Vec<Example> {
        D::examples()
    }

    fn run(&self, args: CommandArgs) -> Result<OutputStream, ShellError> {
        let column_paths: Vec<ColumnPath> = args.rest(0)?;

        Ok(args
            .input
            .map(move |v| {
                if column_paths.is_empty() {
                    action::<D>(&v, v.tag())
                } else {
                    let mut ret = v;

                    for path in &column_paths {
                        ret = ret.swap_data_by_column_path(
                            path,
                            Box::new(move |old| action::<D>(old, old.tag())),
                        )?;
                    }

                    Ok(ret)
                }
            })
            .into_input_stream())
    }
}

pub fn action<D>(input: &Value, tag: Tag) -> Result<Value, ShellError>
where
    D: HashDigest,
    digest::Output<D>: core::fmt::LowerHex,
{
    match &input.value {
        UntaggedValue::Primitive(Primitive::String(s)) => {
            let digest_result = D::digest(s.as_bytes());
            Ok(UntaggedValue::string(&format!("{:x}", digest_result)).into_value(tag))
        }
        UntaggedValue::Primitive(Primitive::Binary(bytes)) => {
            let digest_result = D::digest(bytes);
            Ok(UntaggedValue::string(&format!("{:x}", digest_result)).into_value(tag))
        }
        other => {
            let got = format!("got {}", other.type_name());
            Err(ShellError::labeled_error(
                format!("value is not supported for hashing as {}", D::name()),
                got,
                tag.span,
            ))
        }
    }
}