golem-common 1.3.1

Shared code between Golem services
// Copyright 2024-2025 Golem Cloud
//
// Licensed under the Golem Source License v1.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://license.golem.cloud/LICENSE
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

use lenient_bool::LenientBool;
use shadow_rs::{BuildPattern, SdResult, ShadowBuilder};
use std::fs::File;
use std::io::Write;
use std::path::Path;
use std::process::Command;

static SKIP_ENV_VAR_NAME: &str = "GOLEM_BUILD_SKIP_SHADOW";

fn main() {
    if should_skip() {
        println!("cargo::warning=skipping shadow build");
        return;
    }

    ShadowBuilder::builder()
        .hook(append_write_git_describe_tags_hook)
        .build_pattern(BuildPattern::Lazy)
        .build()
        .unwrap();
}

fn should_skip() -> bool {
    let skip_requested = std::env::var(SKIP_ENV_VAR_NAME)
        .ok()
        .and_then(|value| value.parse::<LenientBool>().ok())
        .unwrap_or_default()
        .0;

    if !skip_requested {
        return false;
    }

    let out = Path::new(&std::env::var("OUT_DIR").unwrap()).join("shadow.rs");
    out.exists()
}

fn append_write_git_describe_tags_hook(file: &File) -> SdResult<()> {
    append_write_git_describe_tags(file)?;
    Ok(())
}

fn append_write_git_describe_tags(mut file: &File) -> SdResult<()> {
    let output = Command::new("git")
        .args(["describe", "--tags", "--match", "v*"])
        .output()?;

    let version = {
        if !output.status.success() {
            println!("cargo::warning=git describe failed, using fallback version 0.0.0");
            for line in String::from_utf8_lossy(&output.stdout).lines() {
                println!("cargo::warning=git stdout: {line}");
            }
            for line in String::from_utf8_lossy(&output.stderr).lines() {
                println!("cargo::warning=git stderr: {line}");
            }

            "0.0.0".to_string()
        } else {
            let version = String::from_utf8(output.stdout)?.trim().to_string();
            let version = version
                .strip_prefix("v")
                .map(|v| v.to_string())
                .unwrap_or_else(|| version);
            println!("cargo::warning=git describe result: {version}");
            version
        }
    };

    let git_describe_tags = format!(
        r#"#[allow(clippy::all, clippy::pedantic, clippy::restriction, clippy::nursery)]
pub const GIT_DESCRIBE_TAGS: &str = "{version}";"#
    );
    writeln!(file, "{git_describe_tags}")?;

    Ok(())
}