use bacnet_types::enums::{ObjectType, PropertyIdentifier};
use bacnet_types::error::Error;
use bacnet_types::primitives::{ObjectIdentifier, PropertyValue, StatusFlags};
use std::borrow::Cow;
use crate::common::{self, read_common_properties};
use crate::traits::BACnetObject;
pub struct NotificationForwarderObject {
oid: ObjectIdentifier,
name: String,
description: String,
status_flags: StatusFlags,
out_of_service: bool,
reliability: u32,
pub process_identifier_filter: Vec<u32>,
pub subscribed_recipients: u32,
pub local_forwarding_only: bool,
pub event_detection_enable: bool,
}
impl NotificationForwarderObject {
pub fn new(instance: u32, name: impl Into<String>) -> Result<Self, Error> {
let oid = ObjectIdentifier::new(ObjectType::NOTIFICATION_FORWARDER, instance)?;
Ok(Self {
oid,
name: name.into(),
description: String::new(),
status_flags: StatusFlags::empty(),
out_of_service: false,
reliability: 0,
process_identifier_filter: Vec::new(),
subscribed_recipients: 0,
local_forwarding_only: false,
event_detection_enable: true,
})
}
}
impl BACnetObject for NotificationForwarderObject {
fn object_identifier(&self) -> ObjectIdentifier {
self.oid
}
fn object_name(&self) -> &str {
&self.name
}
fn read_property(
&self,
property: PropertyIdentifier,
array_index: Option<u32>,
) -> Result<PropertyValue, Error> {
if let Some(result) = read_common_properties!(self, property, array_index) {
return result;
}
match property {
p if p == PropertyIdentifier::OBJECT_TYPE => Ok(PropertyValue::Enumerated(
ObjectType::NOTIFICATION_FORWARDER.to_raw(),
)),
p if p == PropertyIdentifier::PROCESS_IDENTIFIER_FILTER => Ok(PropertyValue::List(
self.process_identifier_filter
.iter()
.map(|id| PropertyValue::Unsigned(*id as u64))
.collect(),
)),
p if p == PropertyIdentifier::SUBSCRIBED_RECIPIENTS => {
Ok(PropertyValue::Unsigned(self.subscribed_recipients as u64))
}
p if p == PropertyIdentifier::LOCAL_FORWARDING_ONLY => {
Ok(PropertyValue::Boolean(self.local_forwarding_only))
}
p if p == PropertyIdentifier::EVENT_DETECTION_ENABLE => {
Ok(PropertyValue::Boolean(self.event_detection_enable))
}
p if p == PropertyIdentifier::RECIPIENT_LIST => {
Ok(PropertyValue::List(Vec::new())) }
p if p == PropertyIdentifier::PROCESS_IDENTIFIER_FILTER => {
Ok(PropertyValue::List(Vec::new()))
}
_ => Err(common::unknown_property_error()),
}
}
fn write_property(
&mut self,
property: PropertyIdentifier,
_array_index: Option<u32>,
value: PropertyValue,
_priority: Option<u8>,
) -> Result<(), Error> {
if property == PropertyIdentifier::LOCAL_FORWARDING_ONLY {
if let PropertyValue::Boolean(v) = value {
self.local_forwarding_only = v;
return Ok(());
}
return Err(common::invalid_data_type_error());
}
if property == PropertyIdentifier::EVENT_DETECTION_ENABLE {
if let PropertyValue::Boolean(v) = value {
self.event_detection_enable = v;
return Ok(());
}
return Err(common::invalid_data_type_error());
}
if let Some(result) =
common::write_out_of_service(&mut self.out_of_service, property, &value)
{
return result;
}
if let Some(result) = common::write_description(&mut self.description, property, &value) {
return result;
}
Err(common::write_access_denied_error())
}
fn property_list(&self) -> Cow<'static, [PropertyIdentifier]> {
static PROPS: &[PropertyIdentifier] = &[
PropertyIdentifier::OBJECT_IDENTIFIER,
PropertyIdentifier::OBJECT_NAME,
PropertyIdentifier::DESCRIPTION,
PropertyIdentifier::OBJECT_TYPE,
PropertyIdentifier::STATUS_FLAGS,
PropertyIdentifier::OUT_OF_SERVICE,
PropertyIdentifier::RELIABILITY,
PropertyIdentifier::PROCESS_IDENTIFIER_FILTER,
PropertyIdentifier::SUBSCRIBED_RECIPIENTS,
PropertyIdentifier::LOCAL_FORWARDING_ONLY,
PropertyIdentifier::EVENT_DETECTION_ENABLE,
];
Cow::Borrowed(PROPS)
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn create_notification_forwarder() {
let nf = NotificationForwarderObject::new(1, "NF-1").unwrap();
assert_eq!(
nf.object_identifier().object_type(),
ObjectType::NOTIFICATION_FORWARDER
);
assert_eq!(nf.object_identifier().instance_number(), 1);
assert_eq!(nf.object_name(), "NF-1");
}
#[test]
fn object_type() {
let nf = NotificationForwarderObject::new(1, "NF").unwrap();
let val = nf
.read_property(PropertyIdentifier::OBJECT_TYPE, None)
.unwrap();
assert_eq!(
val,
PropertyValue::Enumerated(ObjectType::NOTIFICATION_FORWARDER.to_raw())
);
}
#[test]
fn process_identifier_filter_empty() {
let nf = NotificationForwarderObject::new(1, "NF").unwrap();
let val = nf
.read_property(PropertyIdentifier::PROCESS_IDENTIFIER_FILTER, None)
.unwrap();
if let PropertyValue::List(items) = val {
assert!(items.is_empty());
} else {
panic!("Expected List");
}
}
#[test]
fn process_identifier_filter_with_values() {
let mut nf = NotificationForwarderObject::new(1, "NF").unwrap();
nf.process_identifier_filter.push(100);
nf.process_identifier_filter.push(200);
let val = nf
.read_property(PropertyIdentifier::PROCESS_IDENTIFIER_FILTER, None)
.unwrap();
if let PropertyValue::List(items) = val {
assert_eq!(items.len(), 2);
assert_eq!(items[0], PropertyValue::Unsigned(100));
assert_eq!(items[1], PropertyValue::Unsigned(200));
} else {
panic!("Expected List");
}
}
#[test]
fn local_forwarding_only_default() {
let nf = NotificationForwarderObject::new(1, "NF").unwrap();
let val = nf
.read_property(PropertyIdentifier::LOCAL_FORWARDING_ONLY, None)
.unwrap();
assert_eq!(val, PropertyValue::Boolean(false));
}
#[test]
fn write_local_forwarding_only() {
let mut nf = NotificationForwarderObject::new(1, "NF").unwrap();
nf.write_property(
PropertyIdentifier::LOCAL_FORWARDING_ONLY,
None,
PropertyValue::Boolean(true),
None,
)
.unwrap();
let val = nf
.read_property(PropertyIdentifier::LOCAL_FORWARDING_ONLY, None)
.unwrap();
assert_eq!(val, PropertyValue::Boolean(true));
}
#[test]
fn write_local_forwarding_only_wrong_type() {
let mut nf = NotificationForwarderObject::new(1, "NF").unwrap();
let result = nf.write_property(
PropertyIdentifier::LOCAL_FORWARDING_ONLY,
None,
PropertyValue::Unsigned(1),
None,
);
assert!(result.is_err());
}
#[test]
fn event_detection_enable_default() {
let nf = NotificationForwarderObject::new(1, "NF").unwrap();
let val = nf
.read_property(PropertyIdentifier::EVENT_DETECTION_ENABLE, None)
.unwrap();
assert_eq!(val, PropertyValue::Boolean(true));
}
#[test]
fn write_event_detection_enable() {
let mut nf = NotificationForwarderObject::new(1, "NF").unwrap();
nf.write_property(
PropertyIdentifier::EVENT_DETECTION_ENABLE,
None,
PropertyValue::Boolean(false),
None,
)
.unwrap();
let val = nf
.read_property(PropertyIdentifier::EVENT_DETECTION_ENABLE, None)
.unwrap();
assert_eq!(val, PropertyValue::Boolean(false));
}
#[test]
fn subscribed_recipients_default() {
let nf = NotificationForwarderObject::new(1, "NF").unwrap();
let val = nf
.read_property(PropertyIdentifier::SUBSCRIBED_RECIPIENTS, None)
.unwrap();
assert_eq!(val, PropertyValue::Unsigned(0));
}
#[test]
fn property_list() {
let nf = NotificationForwarderObject::new(1, "NF").unwrap();
let props = nf.property_list();
assert!(props.contains(&PropertyIdentifier::PROCESS_IDENTIFIER_FILTER));
assert!(props.contains(&PropertyIdentifier::SUBSCRIBED_RECIPIENTS));
assert!(props.contains(&PropertyIdentifier::LOCAL_FORWARDING_ONLY));
assert!(props.contains(&PropertyIdentifier::EVENT_DETECTION_ENABLE));
}
}