Skip to main content

endpoint_sec/event/
event_od_group_set.rs

1//! [`EventOdGroupSet`]
2
3use std::ffi::OsStr;
4
5use endpoint_sec_sys::{
6    es_event_od_group_set_t, es_od_member_id_array_t, es_od_member_id_array_t_anon0, es_od_member_type_t,
7    es_string_token_t,
8};
9
10use crate::Process;
11
12/// Notification that a group had it's members initialised or replaced.
13#[doc(alias = "es_event_od_group_set_t")]
14pub struct EventOdGroupSet<'a> {
15    /// The raw reference.
16    pub(crate) raw: &'a es_event_od_group_set_t,
17    /// The version of the message.
18    pub(crate) version: u32,
19}
20
21impl<'a> EventOdGroupSet<'a> {
22    /// Process that instigated operation (XPC caller).
23    #[inline(always)]
24    pub fn instigator(&self) -> Process<'a> {
25        // Safety: 'a tied to self, object obtained through ES
26        Process::new(unsafe { self.raw.instigator.as_ref() }, self.version)
27    }
28
29    /// Result code for the operation.
30    #[inline(always)]
31    pub fn error_code(&self) -> i32 {
32        self.raw.error_code
33    }
34
35    /// The group to which members were set.
36    #[inline(always)]
37    pub fn group_name(&self) -> &'a OsStr {
38        // Safety: 'a tied to self, object obtained through ES
39        unsafe { self.raw.group_name.as_os_str() }
40    }
41
42    /// Array of new members.
43    #[inline(always)]
44    pub fn members(&self) -> OdMemberIdArray<'a> {
45        OdMemberIdArray {
46            // Safety: 'a tied to self, object obtained through ES
47            raw: unsafe { self.raw.members.as_ref() },
48        }
49    }
50
51    /// OD node being mutated.
52    ///
53    /// Typically one of "/Local/Default", "/LDAPv3/<server>" or "/Active Directory/<domain>".
54    #[inline(always)]
55    pub fn node_name(&self) -> &'a OsStr {
56        // Safety: 'a tied to self, object obtained through ES
57        unsafe { self.raw.node_name.as_os_str() }
58    }
59
60    /// Optional. If node_name is "/Local/Default", this is, the path of the database against which
61    /// OD is authenticating.
62    #[inline(always)]
63    pub fn db_path(&self) -> Option<&'a OsStr> {
64        if self.node_name() == OsStr::new("/Local/Default") {
65            // Safety: 'a tied to self, object obtained through ES
66            Some(unsafe { self.raw.db_path.as_os_str() })
67        } else {
68            None
69        }
70    }
71}
72
73// Safety: safe to send across threads: does not contain any interior mutability nor depend on current thread state
74unsafe impl Send for EventOdGroupSet<'_> {}
75// Safety: safe to share across threads: does not contain any interior mutability nor depend on current thread state
76unsafe impl Sync for EventOdGroupSet<'_> {}
77
78impl_debug_eq_hash_with_functions!(EventOdGroupSet<'a> with version; instigator, error_code, group_name, members, node_name, db_path);
79
80/// An array of group member identities.
81#[doc(alias = "es_od_member_id_array_t")]
82pub struct OdMemberIdArray<'a> {
83    /// The raw reference.
84    pub(crate) raw: &'a es_od_member_id_array_t,
85}
86
87impl<'a> OdMemberIdArray<'a> {
88    /// Indicates the type of the members, and how they are identified.
89    #[inline(always)]
90    pub fn member_type(&self) -> es_od_member_type_t {
91        self.raw.member_type
92    }
93
94    /// The number of elements.
95    #[inline(always)]
96    pub fn member_count(&self) -> usize {
97        self.raw.member_count
98    }
99
100    /// The members identity, as its raw value.
101    #[inline(always)]
102    pub fn raw_member_array(&self) -> &'a es_od_member_id_array_t_anon0 {
103        &self.raw.member_array
104    }
105
106    /// Iterator over the relevant union value based on the member type.
107    #[inline(always)]
108    pub fn members<'arr>(&'arr self) -> Option<OdMemberIdArrayIters<'arr, 'a>> {
109        let res = match self.member_type() {
110            es_od_member_type_t::ES_OD_MEMBER_TYPE_USER_NAME => {
111                OdMemberIdArrayIters::UserName(OdMemberIdArrayNames::new(self))
112            },
113            es_od_member_type_t::ES_OD_MEMBER_TYPE_USER_UUID => {
114                OdMemberIdArrayIters::UserUuid(OdMemberIdArrayUuids::new(self))
115            },
116            es_od_member_type_t::ES_OD_MEMBER_TYPE_GROUP_UUID => {
117                OdMemberIdArrayIters::GroupUuid(OdMemberIdArrayUuids::new(self))
118            },
119            _ => return None,
120        };
121        Some(res)
122    }
123}
124
125// Safety: safe to send across threads: does not contain any interior mutability nor depend on current thread state
126unsafe impl Send for OdMemberIdArray<'_> {}
127
128impl_debug_eq_hash_with_functions!(OdMemberIdArray<'a>; member_type, member_count);
129
130/// One of the possible iterator for [`OdMemberIdArray`]
131pub enum OdMemberIdArrayIters<'arr, 'raw> {
132    /// Users, designated by name
133    UserName(OdMemberIdArrayNames<'arr, 'raw>),
134    /// Users, designated by UUID
135    UserUuid(OdMemberIdArrayUuids<'arr, 'raw>),
136    /// Groups, designated by UUID
137    GroupUuid(OdMemberIdArrayUuids<'arr, 'raw>),
138}
139
140/// Read the `idx` name of `raw`
141///
142/// # Safety
143///
144/// Must be called with a valid member array for which `idx` is in range `0..raw.member_count` and the
145/// member type is correct.
146unsafe fn read_nth_name(raw: &es_od_member_id_array_t, idx: usize) -> es_string_token_t {
147    std::ptr::read(raw.member_array.names.as_ptr().add(idx))
148}
149
150make_event_data_iterator!(
151    OdMemberIdArray;
152    /// Iterator over the names in an [`OdMemberIdArray`]
153    OdMemberIdArrayNames with member_count (usize);
154    &'raw OsStr;
155    read_nth_name,
156    super::as_os_str,
157);
158
159/// Read the `idx` uuid of `raw`
160///
161/// # Safety
162///
163/// Must be called with a valid member array for which `idx` is in range `0..raw.member_count` and the
164/// member type is correct.
165unsafe fn read_nth_uuid(raw: &es_od_member_id_array_t, idx: usize) -> libc::uuid_t {
166    std::ptr::read(raw.member_array.uuids.as_ptr().add(idx))
167}
168
169make_event_data_iterator!(
170    OdMemberIdArray;
171    /// Iterator over the uuids in an [`OdMemberIdArray`]
172    OdMemberIdArrayUuids with member_count (usize);
173    libc::uuid_t;
174    read_nth_uuid,
175    std::convert::identity,
176);