1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
use crate::errors::*;

use crate::components::Component;
use crate::base::{Value, ValueProperties, DataType, IndexKey};
use crate::utilities::prepend;
use crate::{base, Warnable};
use crate::proto;
use crate::components::transforms::propagate_binary_shape;
use indexmap::map::IndexMap;

impl Component for proto::Filter {
    fn propagate_property(
        &self,
        _privacy_definition: &Option<proto::PrivacyDefinition>,
        _public_arguments: IndexMap<base::IndexKey, &Value>,
        properties: base::NodeProperties,
        node_id: u32
    ) -> Result<Warnable<ValueProperties>> {
        let mut data_property = properties.get::<base::IndexKey>(&"data".into())
            .ok_or("data: missing")?.array()
            .map_err(prepend("data:"))?.clone();

        if !data_property.releasable {
            data_property.assert_is_not_aggregated()?;
        }

        let mask_property = properties.get::<IndexKey>(&"mask".into())
            .ok_or("mask: missing")?.array()
            .map_err(prepend("mask:"))?.clone();

        if !mask_property.releasable {
            mask_property.assert_is_not_aggregated()?;
        }

        if mask_property.data_type != DataType::Bool {
            return Err("mask: must be boolean".into())
        }

        if mask_property.num_columns()? != 1 {
            return Err("mask: number of columns must be one".into())
        }

        propagate_binary_shape(&data_property, &mask_property)?;

        // the number of records is not known after filtering rows
        data_property.num_records = None;

        // This exists to prevent binary ops on non-conformable arrays from being approved
        data_property.dataset_id = Some(node_id as i64);

        // no longer know if the data has a nonzero number of records
        data_property.is_not_empty = false;

        Ok(ValueProperties::Array(data_property).into())
    }
}