polars_plan/dsl/
struct_.rsuse super::*;
use crate::plans::conversion::is_regex_projection;
pub struct StructNameSpace(pub(crate) Expr);
impl StructNameSpace {
pub fn field_by_index(self, index: i64) -> Expr {
self.0
.map_private(FunctionExpr::StructExpr(StructFunction::FieldByIndex(
index,
)))
.with_function_options(|mut options| {
options.flags |= FunctionFlags::ALLOW_RENAME;
options
})
}
pub fn field_by_names<I, S>(self, names: I) -> Expr
where
I: IntoIterator<Item = S>,
S: Into<PlSmallStr>,
{
self.field_by_names_impl(names.into_iter().map(|x| x.into()).collect())
}
fn field_by_names_impl(self, names: Arc<[PlSmallStr]>) -> Expr {
self.0
.map_private(FunctionExpr::StructExpr(StructFunction::MultipleFields(
names,
)))
.with_function_options(|mut options| {
options.flags |= FunctionFlags::ALLOW_RENAME;
options
})
}
pub fn field_by_name(self, name: &str) -> Expr {
if name == "*" || is_regex_projection(name) {
return self.field_by_names([name]);
}
self.0
.map_private(FunctionExpr::StructExpr(StructFunction::FieldByName(
name.into(),
)))
.with_function_options(|mut options| {
options.flags |= FunctionFlags::ALLOW_RENAME;
options
})
}
pub fn rename_fields<I, S>(self, names: I) -> Expr
where
I: IntoIterator<Item = S>,
S: Into<PlSmallStr>,
{
self._rename_fields_impl(names.into_iter().map(|x| x.into()).collect())
}
pub fn _rename_fields_impl(self, names: Arc<[PlSmallStr]>) -> Expr {
self.0
.map_private(FunctionExpr::StructExpr(StructFunction::RenameFields(
names,
)))
}
#[cfg(feature = "json")]
pub fn json_encode(self) -> Expr {
self.0
.map_private(FunctionExpr::StructExpr(StructFunction::JsonEncode))
}
pub fn with_fields(self, fields: Vec<Expr>) -> PolarsResult<Expr> {
fn materialize_field(this: &Expr, field: Expr) -> PolarsResult<Expr> {
field.try_map_expr(|e| match e {
Expr::Field(names) => {
let this = this.clone().struct_();
Ok(if names.len() == 1 {
this.field_by_name(names[0].as_ref())
} else {
this.field_by_names_impl(names)
})
},
Expr::Exclude(_, _) => {
polars_bail!(InvalidOperation: "'exclude' not allowed in 'field'")
},
_ => Ok(e),
})
}
let mut new_fields = Vec::with_capacity(fields.len());
new_fields.push(Default::default());
for e in fields.into_iter().map(|e| materialize_field(&self.0, e)) {
new_fields.push(e?)
}
new_fields[0] = self.0;
Ok(Expr::Function {
input: new_fields,
function: FunctionExpr::StructExpr(StructFunction::WithFields),
options: FunctionOptions {
collect_groups: ApplyOptions::ElementWise,
flags: FunctionFlags::default()
| FunctionFlags::PASS_NAME_TO_APPLY
| FunctionFlags::INPUT_WILDCARD_EXPANSION,
..Default::default()
},
})
}
}