nu-completion-script 0.1.3

A script for generating nu completions from Fish completions
mod fetch;
mod generate;
use std::{
    path::Path,
    process::{Command, Stdio},
};

use anyhow::Result;
use beau_collector::BeauCollector as _;
use log::{debug, error, trace, warn};

use crate::config::Config;
pub(crate) use fetch::fetch_latest_patch_set;
pub(crate) use generate::generate_patches;

pub(crate) fn patch(source: impl AsRef<Path>, patch: impl AsRef<Path>) -> Result<()> {
    let source = source.as_ref().as_os_str();
    let patch = patch.as_ref().as_os_str();
    let result = Command::new("patch")
        .arg(source)
        .arg(patch)
        .stdout(Stdio::piped())
        .output()?;

    let patch_output = String::from_utf8_lossy(&result.stdout);
    result
        .status
        .exit_ok()
        .map(|_| {
            debug!(
                source = source.to_string_lossy(),
                patch = patch.to_string_lossy(),
                patch_output = patch_output;
                "successfully patched"
            );
        })
        .map_err(|status| {
            error!(
                source = source.to_string_lossy(),
                patch = patch.to_string_lossy(),
                patch_output = patch_output,
                status = result.status.code();
                "error patching"
            );
            status.into()
        })
}

/// Pass all sources specified in [`Config::sources()`].
///
/// ## Errors:
/// If [`Config::fail_fast()`] is true, this returns an error if any `patch`
/// command fails. Otherwise, any errors encountered patching individual files
/// are collected together into a single error.
pub(crate) fn patch_all() -> Result<()> {
    let mut errors = vec![Ok(())];
    for source in Config::sources() {
        trace!(source = source.to_string_lossy(); "checking for patches");
        let source: &Path = source.as_ref();
        let _patch = source.with_extension("patch");
        let Some(patch_file) = _patch.file_name() else {
            debug!(source = source.to_string_lossy(); "failed to get file name with patch extension");
            continue;
        };
        let patch_file = Config::patch_dir().join(patch_file);
        if patch_file.exists() {
            debug!(
                patch_file = patch_file.to_string_lossy(),
                source = source.to_string_lossy();
                "found patch file"
            );
            let def = source.with_extension("nu");
            let def = def
                .file_name()
                .expect("to be able to extract file name from path"); // this is already checked for
            let def = Config::output_dir().join(def);
            if def.exists() {
                if let Err(error) = patch(def, patch_file) {
                    if Config::fail_fast() {
                        return Err(error);
                    } else {
                        errors.push(Err(error));
                    }
                }
            } else {
                warn!(
                    source = source.to_string_lossy(),
                    patch_file = patch_file.to_string_lossy();
                    "source and patch found, but no converted definition. Perhaps conversion failed?"
                );
            }
        } else {
            trace!(source = source.to_string_lossy(), patch = patch_file.to_string_lossy(); "no patches found");
        }
    }
    errors.into_iter().bcollect()
}