opendp 0.14.2-dev.20260401.2

A library of differential privacy algorithms for the statistical analysis of sensitive private data.
use polars::prelude::*;

use crate::core::{Function, MetricSpace, StabilityMap, Transformation};
use crate::domains::{ExprDomain, OuterMetric, WildExprDomain};
use crate::error::Fallible;

#[cfg(test)]
mod test;

/// Make a Transformation that returns a `col(column_name)` expression for a Lazy Frame.
///
/// # Arguments
/// * `input_domain` - Domain of the expression to be applied.
/// * `input_metric` - The metric under which neighboring LazyFrames are compared.
/// * `expr` - A column expression.
pub fn make_expr_col<M: OuterMetric>(
    input_domain: WildExprDomain,
    input_metric: M,
    expr: Expr,
) -> Fallible<Transformation<WildExprDomain, M, ExprDomain, M>>
where
    M::Distance: Clone,
    (WildExprDomain, M): MetricSpace,
    (ExprDomain, M): MetricSpace,
{
    let Expr::Column(col_name) = expr else {
        return fallible!(MakeTransformation, "Expected col() expression");
    };

    let input_columns = input_domain
        .columns
        .iter()
        .map(|column| column.name.to_string())
        .collect::<Vec<_>>()
        .join(", ");
    let output_domain = ExprDomain {
        column: (input_domain.columns.iter())
            .find(|s| s.name == col_name)
            .ok_or_else(|| err!(MakeTransformation, "unrecognized column '{col_name}' in output domain; expected one of: {input_columns}"))?
            .clone(),
        context: input_domain.context.clone(),
    };

    Transformation::new(
        input_domain,
        input_metric.clone(),
        output_domain,
        input_metric,
        Function::from_expr(col(&*col_name)),
        StabilityMap::new(Clone::clone),
    )
}