Skip to content

Commit

Permalink
depr(python): Deprecate parsing string inputs as literals for `when-t…
Browse files Browse the repository at this point in the history
…hen-otherwise` (#10122)
  • Loading branch information
stinodego committed Jul 28, 2023
1 parent 4381c04 commit 6f50fb8
Show file tree
Hide file tree
Showing 9 changed files with 261 additions and 135 deletions.
101 changes: 56 additions & 45 deletions polars/polars-lazy/polars-plan/src/dsl/arity.rs
@@ -1,78 +1,91 @@
use super::*;

/// Intermediate state of `when(..).then(..).otherwise(..)` expression.
/// Utility struct for the `when-then-otherwise` expression.
///
/// Represents the state of the expression after [when] is called.
///
/// In this state, `then` must be called to continue to finish the expression.
#[derive(Clone)]
pub struct When {
predicate: Expr,
condition: Expr,
}

/// Intermediate state of `when(..).then(..).otherwise(..)` expression.
/// Utility struct for the `when-then-otherwise` expression.
///
/// Represents the state of the expression after `when(...).then(...)` is called.
#[derive(Clone)]
pub struct Then {
predicate: Expr,
then: Expr,
condition: Expr,
statement: Expr,
}

/// Intermediate state of a chained `when(..).then(..).otherwise(..)` expression.
/// Utility struct for the `when-then-otherwise` expression.
///
/// Represents the state of the expression after an additional `when` is called.
///
/// In this state, `then` must be called to continue to finish the expression.
#[derive(Clone)]
pub struct ChainedWhen {
predicates: Vec<Expr>,
thens: Vec<Expr>,
conditions: Vec<Expr>,
statements: Vec<Expr>,
}

/// Intermediate state of a chained `when(..).then(..).otherwise(..)` expression.
/// Utility struct for the `when-then-otherwise` expression.
///
/// Represents the state of the expression after an additional `then` is called.
#[derive(Clone)]
pub struct ChainedThen {
predicates: Vec<Expr>,
thens: Vec<Expr>,
conditions: Vec<Expr>,
statements: Vec<Expr>,
}

impl When {
/// Add a condition to the `when-then-otherwise` expression.
pub fn then<E: Into<Expr>>(self, expr: E) -> Then {
Then {
predicate: self.predicate,
then: expr.into(),
condition: self.condition,
statement: expr.into(),
}
}
}

impl Then {
pub fn when<E: Into<Expr>>(self, predicate: E) -> ChainedWhen {
/// Attach a statement to the corresponding condition.
pub fn when<E: Into<Expr>>(self, condition: E) -> ChainedWhen {
ChainedWhen {
predicates: vec![self.predicate, predicate.into()],
thens: vec![self.then],
conditions: vec![self.condition, condition.into()],
statements: vec![self.statement],
}
}

pub fn otherwise<E: Into<Expr>>(self, expr: E) -> Expr {
Expr::Ternary {
predicate: Box::new(self.predicate),
truthy: Box::new(self.then),
falsy: Box::new(expr.into()),
}
/// Define a default for the `when-then-otherwise` expression.
pub fn otherwise<E: Into<Expr>>(self, statement: E) -> Expr {
ternary_expr(self.condition, self.statement, statement.into())
}
}

impl ChainedWhen {
pub fn then<E: Into<Expr>>(mut self, expr: E) -> ChainedThen {
self.thens.push(expr.into());
pub fn then<E: Into<Expr>>(mut self, statement: E) -> ChainedThen {
self.statements.push(statement.into());
ChainedThen {
predicates: self.predicates,
thens: self.thens,
conditions: self.conditions,
statements: self.statements,
}
}
}

impl ChainedThen {
pub fn when<E: Into<Expr>>(mut self, predicate: E) -> ChainedWhen {
self.predicates.push(predicate.into());
/// Add another condition to the `when-then-otherwise` expression.
pub fn when<E: Into<Expr>>(mut self, condition: E) -> ChainedWhen {
self.conditions.push(condition.into());

ChainedWhen {
predicates: self.predicates,
thens: self.thens,
conditions: self.conditions,
statements: self.statements,
}
}

/// Define a default for the `when-then-otherwise` expression.
pub fn otherwise<E: Into<Expr>>(self, expr: E) -> Expr {
// we iterate the preds/ exprs last in first out
// and nest them.
Expand All @@ -98,31 +111,29 @@ impl ChainedThen {
// which will be used in the next layer `outer`
//

let pred_iter = self.predicates.into_iter().rev();
let mut then_iter = self.thens.into_iter().rev();
let conditions_iter = self.conditions.into_iter().rev();
let mut statements_iter = self.statements.into_iter().rev();

let mut otherwise = expr.into();

for e in pred_iter {
otherwise = Expr::Ternary {
predicate: Box::new(e),
truthy: Box::new(
then_iter
.next()
.expect("expr expected, did you call when().then().otherwise?"),
),
falsy: Box::new(otherwise),
}
for e in conditions_iter {
otherwise = ternary_expr(
e,
statements_iter
.next()
.expect("expr expected, did you call when().then().otherwise?"),
otherwise,
);
}

otherwise
}
}

/// Start a `when(..).then(..).otherwise(..)` expression
pub fn when<E: Into<Expr>>(predicate: E) -> When {
/// Start a `when-then-otherwise` expression.
pub fn when<E: Into<Expr>>(condition: E) -> When {
When {
predicate: predicate.into(),
condition: condition.into(),
}
}

Expand Down
6 changes: 5 additions & 1 deletion py-polars/polars/expr/expr.py
Expand Up @@ -8598,6 +8598,7 @@ def map_dict(
Dictionary containing the before/after values to map.
default
Value to use when the remapping dict does not contain the lookup value.
Accepts expression input. Non-expression inputs are parsed as literals.
Use ``pl.first()``, to keep the original value.
return_dtype
Set return dtype to override automatic return dtype determination.
Expand Down Expand Up @@ -8923,6 +8924,9 @@ def inner_with_default(s: Series) -> Series:
is_keys=False,
)

default_parsed = self._from_pyexpr(
parse_as_expression(default, str_as_lit=True)
)
return (
(
df.lazy()
Expand All @@ -8942,7 +8946,7 @@ def inner_with_default(s: Series) -> Series:
.select(
F.when(F.col(is_remapped_column).is_not_null())
.then(F.col(remap_value_column))
.otherwise(default)
.otherwise(default_parsed)
.alias(column)
)
)
Expand Down

0 comments on commit 6f50fb8

Please sign in to comment.