2 # -*- coding: utf-8 -*-
3 # Originally based on ./sam.py
10 sys.path.insert(0, "bin/python")
12 from samba.tests.subunitrun import SubunitOptions, TestProgram
14 import samba.getopt as options
16 from samba.auth import system_session
18 from samba.samdb import SamDB
19 from samba.dcerpc import misc
21 from samba.dcerpc import drsuapi, misc, drsblobs
22 from samba.drs_utils import drs_DsBind
23 from samba.ndr import ndr_unpack, ndr_pack
30 class LATestException(Exception):
34 class LATests(drs_base.DrsBaseTestCase):
37 super(LATests, self).setUp()
38 # DrsBaseTestCase sets up self.ldb_dc1, self.ldb_dc2
39 # we're only using one
40 self.samdb = self.ldb_dc1
42 self.base_dn = self.samdb.domain_dn()
43 self.ou = "OU=la,%s" % self.base_dn
46 self.samdb.delete(self.ou, ['tree_delete:1'])
47 except ldb.LdbError as e:
49 self.samdb.add({'objectclass': 'organizationalUnit',
52 self.dc_guid = self.samdb.get_invocation_id()
53 self.drs, self.drs_handle = self._ds_bind(self.dnsname_dc1)
56 super(LATests, self).tearDown()
58 self.samdb.delete(self.ou, ['tree_delete:1'])
59 except ldb.LdbError as e:
62 def delete_user(self, user):
63 self.samdb.delete(user['dn'])
64 del self.users[self.users.index(user)]
66 def add_object(self, cn, objectclass):
67 dn = "CN=%s,%s" % (cn, self.ou)
68 self.samdb.add({'cn': cn,
69 'objectclass': objectclass,
74 def add_objects(self, n, objectclass, prefix=None):
79 dns.append(self.add_object("%s%d" % (prefix, i + 1),
83 def add_linked_attribute(self, src, dest, attr='member'):
85 m.dn = ldb.Dn(self.samdb, src)
86 m[attr] = ldb.MessageElement(dest, ldb.FLAG_MOD_ADD, attr)
89 def remove_linked_attribute(self, src, dest, attr='member'):
91 m.dn = ldb.Dn(self.samdb, src)
92 m[attr] = ldb.MessageElement(dest, ldb.FLAG_MOD_DELETE, attr)
95 def attr_search(self, obj, expected, attr, scope=ldb.SCOPE_BASE):
97 req8 = self._exop_req8(dest_dsa=None,
98 invocation_id=self.dc_guid,
100 exop=drsuapi.DRSUAPI_EXOP_REPL_OBJ)
102 level, ctr = self.drs.DsGetNCChanges(self.drs_handle, 8, req8)
103 expected_attid = getattr(drsuapi, 'DRSUAPI_ATTID_' + attr)
106 for link in ctr.linked_attributes:
107 if link.attid == expected_attid:
108 unpacked = ndr_unpack(drsuapi.DsReplicaObjectIdentifier3,
110 active = link.flags & drsuapi.DRSUAPI_DS_LINKED_ATTRIBUTE_FLAG_ACTIVE
111 links.append((str(unpacked.dn), bool(active)))
116 def assert_forward_links(self, obj, expected, attr='member'):
117 results = self.attr_search(obj, expected, attr)
118 self.assertEqual(len(results), len(expected))
121 self.assertTrue(k in expected)
122 self.assertEqual(expected[k], v, "%s active flag should be %d, not %d" %
125 def get_object_guid(self, dn):
126 res = self.samdb.search(dn,
127 scope=ldb.SCOPE_BASE,
128 attrs=['objectGUID'])
129 return str(misc.GUID(res[0]['objectGUID'][0]))
131 def test_links_all_delete_group(self):
132 u1, u2 = self.add_objects(2, 'user', 'u_all_del_group')
133 g1, g2 = self.add_objects(2, 'group', 'g_all_del_group')
134 g2guid = self.get_object_guid(g2)
136 self.add_linked_attribute(g1, u1)
137 self.add_linked_attribute(g2, u1)
138 self.add_linked_attribute(g2, u2)
140 self.samdb.delete(g2)
141 self.assert_forward_links(g1, {u1: True})
142 res = self.samdb.search('<GUID=%s>' % g2guid,
143 scope=ldb.SCOPE_BASE,
144 controls=['show_deleted:1'])
146 self.assert_forward_links(new_dn, {})
149 def test_la_links_delete_link(self):
150 u1, u2 = self.add_objects(2, 'user', 'u_del_link')
151 g1, g2 = self.add_objects(2, 'group', 'g_del_link')
153 self.add_linked_attribute(g1, u1)
154 self.add_linked_attribute(g2, u1)
155 self.add_linked_attribute(g2, u2)
157 self.remove_linked_attribute(g2, u1)
159 self.assert_forward_links(g1, {u1: True})
160 self.assert_forward_links(g2, {u1: False, u2: True})
162 self.add_linked_attribute(g2, u1)
163 self.remove_linked_attribute(g2, u2)
164 self.assert_forward_links(g2, {u1: True, u2: False})
165 self.remove_linked_attribute(g2, u1)
166 self.assert_forward_links(g2, {u1: False, u2: False})
168 def test_la_links_delete_user(self):
169 u1, u2 = self.add_objects(2, 'user', 'u_del_user')
170 g1, g2 = self.add_objects(2, 'group', 'g_del_user')
172 self.add_linked_attribute(g1, u1)
173 self.add_linked_attribute(g2, u1)
174 self.add_linked_attribute(g2, u2)
176 self.samdb.delete(u1)
178 self.assert_forward_links(g1, {})
179 self.assert_forward_links(g2, {u2: True})