polars-plan 0.54.3

Lazy query engine for the Polars DataFrame library
use polars_ops::series::ClosedInterval;
#[cfg(feature = "temporal")]
use polars_time::ClosedWindow;

use super::*;

/// Generate a range of integers.
///
/// Alias for `int_range`.
pub fn arange(start: Expr, end: Expr, step: i64, dtype: DataType) -> Expr {
    int_range(start, end, step, dtype)
}

/// Generate a range of integers.
pub fn int_range(start: Expr, end: Expr, step: i64, dtype: impl Into<DataTypeExpr>) -> Expr {
    Expr::n_ary(
        RangeFunction::IntRange {
            step,
            dtype: dtype.into(),
        },
        vec![start, end],
    )
}

/// Generate a range of integers for each row of the input columns.
pub fn int_ranges(start: Expr, end: Expr, step: Expr, dtype: impl Into<DataTypeExpr>) -> Expr {
    Expr::n_ary(
        RangeFunction::IntRanges {
            dtype: dtype.into(),
        },
        vec![start, end, step],
    )
}

#[cfg(feature = "dtype-date")]
/// Create a date range column from `start`, `end`, and `interval`, and expressions.
pub fn date_range(
    start: Option<Expr>,
    end: Option<Expr>,
    interval: Option<Duration>,
    num_samples: Option<Expr>,
    closed: ClosedWindow,
) -> PolarsResult<Expr> {
    let (input, arg_type) = DateRangeArgs::parse(start, end, interval, num_samples)?;

    polars_ensure!(
        interval.is_none_or(|i| i.is_full_days()),
        ComputeError: "`interval` input for `date_range` must consist of full days, got: {:?}", interval.unwrap(),
    );

    // Only "both" is supported for date_range(start, end, num_samples).
    polars_ensure!(
            !(arg_type == DateRangeArgs::StartEndSamples && closed != ClosedWindow::Both),
            InvalidOperation: "date_range does not support 'left', 'right', or 'none' for the \
                'closed' parameter when 'start', 'end', and 'num_samples' is provided.",
    );

    Ok(Expr::n_ary(
        RangeFunction::DateRange {
            interval,
            closed,
            arg_type,
        },
        input,
    ))
}

#[cfg(feature = "dtype-date")]
/// Create a column of date ranges from `start`, `end`, `interval`, and `num_samples` expressions.
pub fn date_ranges(
    start: Option<Expr>,
    end: Option<Expr>,
    interval: Option<Duration>,
    num_samples: Option<Expr>,
    closed: ClosedWindow,
) -> PolarsResult<Expr> {
    let (input, arg_type) = DateRangeArgs::parse(start, end, interval, num_samples)?;

    polars_ensure!(
        interval.is_none_or(|i| i.is_full_days()),
        ComputeError: "`interval` input for `date_range` must consist of full days, got: {:?}", interval.unwrap(),
    );

    // Only "both" is supported for date_ranges(start, end, num_samples).
    polars_ensure!(
        !(arg_type == DateRangeArgs::StartEndSamples && closed != ClosedWindow::Both),
        InvalidOperation: "date_range does not support 'left', 'right', or 'none' for the \
            'closed' parameter when 'start', 'end', and 'num_samples' is provided.",
    );

    Ok(Expr::n_ary(
        RangeFunction::DateRanges {
            interval,
            closed,
            arg_type,
        },
        input,
    ))
}

/// Create a datetime range from `start`, `end`, `interval`, and `num_samples` expressions.
#[cfg(feature = "dtype-datetime")]
pub fn datetime_range(
    start: Option<Expr>,
    end: Option<Expr>,
    interval: Option<Duration>,
    num_samples: Option<Expr>,
    closed: ClosedWindow,
    time_unit: Option<TimeUnit>,
    time_zone: Option<TimeZone>,
) -> PolarsResult<Expr> {
    let (input, arg_type) = DateRangeArgs::parse(start, end, interval, num_samples)?;
    Ok(Expr::n_ary(
        RangeFunction::DatetimeRange {
            interval,
            closed,
            time_unit,
            time_zone,
            arg_type,
        },
        input,
    ))
}

/// Create a column of datetime ranges from `start`, `end`, `interval`, and `num_samples` expressions.
#[cfg(feature = "dtype-datetime")]
pub fn datetime_ranges(
    start: Option<Expr>,
    end: Option<Expr>,
    interval: Option<Duration>,
    num_samples: Option<Expr>,
    closed: ClosedWindow,
    time_unit: Option<TimeUnit>,
    time_zone: Option<TimeZone>,
) -> PolarsResult<Expr> {
    let (input, arg_type) = DateRangeArgs::parse(start, end, interval, num_samples)?;
    Ok(Expr::n_ary(
        RangeFunction::DatetimeRanges {
            interval,
            closed,
            time_unit,
            time_zone,
            arg_type,
        },
        input,
    ))
}

/// Generate a time range.
#[cfg(feature = "dtype-time")]
pub fn time_range(start: Expr, end: Expr, interval: Duration, closed: ClosedWindow) -> Expr {
    Expr::n_ary(
        RangeFunction::TimeRange { interval, closed },
        vec![start, end],
    )
}

/// Create a column of time ranges from a `start` and `stop` expression.
#[cfg(feature = "dtype-time")]
pub fn time_ranges(start: Expr, end: Expr, interval: Duration, closed: ClosedWindow) -> Expr {
    Expr::n_ary(
        RangeFunction::TimeRanges { interval, closed },
        vec![start, end],
    )
}

/// Generate a series of equally-spaced points.
pub fn linear_space(start: Expr, end: Expr, num_samples: Expr, closed: ClosedInterval) -> Expr {
    Expr::n_ary(
        RangeFunction::LinearSpace { closed },
        vec![start, end, num_samples],
    )
}

/// Create a column of linearly-spaced sequences from 'start', 'end', and 'num_samples' expressions.
pub fn linear_spaces(
    start: Expr,
    end: Expr,
    num_samples: Expr,
    closed: ClosedInterval,
    as_array: bool,
) -> PolarsResult<Expr> {
    let mut input = Vec::<Expr>::with_capacity(3);
    input.push(start);
    input.push(end);
    let array_width = if as_array {
        Some(num_samples.extract_usize().map_err(|_| {
            polars_err!(InvalidOperation: "'as_array' is only valid when 'num_samples' is a constant integer")
        })?)
    } else {
        input.push(num_samples);
        None
    };

    Ok(Expr::n_ary(
        RangeFunction::LinearSpaces {
            closed,
            array_width,
        },
        input,
    ))
}