r6506: Rename parameter to GetDomPwInfo.
[samba.git] / source / scripting / swig / samr.py
index 5fd87da3ad94bc7c38b2a5e545db182c26460cb5..b6b1577c023493d60bd913d0ba5c2d3aaae03182 100644 (file)
@@ -3,18 +3,21 @@ import dcerpc
 def sid_to_string(sid):
     """Convert a Python dictionary SID to a string SID."""
 
-    result = 'S-%d' % sid['sid_rev_num']
+    result = 'S-%d' % sid.sid_rev_num
 
-    ia = sid['id_auth']
+    result = result + '-%u' % \
+             (dcerpc.uint8_array_getitem(sid.id_auth, 5) +
+              (dcerpc.uint8_array_getitem(sid.id_auth, 4) << 8) + 
+              (dcerpc.uint8_array_getitem(sid.id_auth, 3) << 16) +
+              (dcerpc.uint8_array_getitem(sid.id_auth, 2) << 24))
     
-    result = result + '-%u' % (ia[5] + (ia[4] << 8) + (ia[3] << 16) + \
-             (ia[2] << 24))
-    
-    for i in range(0, sid['num_auths']):
-        result = result + '-%u' % sid['sub_auths'][i]
+    for i in range(0, sid.num_auths):
+        result = result + '-%u' % \
+                 dcerpc.uint32_array_getitem(sid.sub_auths, i)
 
     return result
 
+
 def string_to_sid(string):
     """Convert a string SID to a Python dictionary SID.  Throws a
     ValueError if the SID string was badly formed."""
@@ -72,12 +75,31 @@ def string_to_sid(string):
         sub_auths.append(int(sa))
 
         string = string[match.end():]
+
+    sid = dcerpc.dom_sid()
+    sid.sid_rev_num = sid_rev_num
+    sid.id_auth = dcerpc.new_uint8_array(6)
+    for i in range(6):
+        dcerpc.uint8_array_setitem(sid.id_auth, i, id_auth[i])
+    sid.num_auths = num_auths
+    sid.sub_auths = dcerpc.new_uint32_array(num_auths)
+    for i in range(num_auths):
+        dcerpc.uint32_array_setitem(sid.sub_auths, i, sub_auths[i])
+
+    return sid
+
+
+def call_fn(fn, pipe, args):
+    """Wrap up a RPC call and throw an exception is an error was returned."""
     
-    print map(type, sub_auths)
+    result = fn(pipe, args);
+
+    if result & 0xc0000000L:
+        raise dcerpc.NTSTATUS(result, dcerpc.nt_errstr(result));
 
-    return {'sid_rev_num': sid_rev_num, 'id_auth': id_auth,
-            'num_auths': num_auths, 'sub_auths': sub_auths}
+    return result;
 
+    
 class SamrHandle:
 
     def __init__(self, pipe, handle):
@@ -87,79 +109,480 @@ class SamrHandle:
 
     def __del__(self):
 
-        r = {}
-        r['handle'] = self.handle
+        if self.handle is not None:
+            self.Close()
+
+    def Close(self):
+                    
+        r = dcerpc.samr_Close()
+        r.data_in.handle = self.handle
 
-        dcerpc.samr_Close(self.pipe, r)
+        call_fn(dcerpc.dcerpc_samr_Close, self.pipe, r)
 
+        self.handle = None
+
+    def QuerySecurity(self, sec_info = 7):
+
+        r = dcerpc.samr_QuerySecurity()
+        r.data_in.handle = self.handle
+        r.data_in.sec_info = sec_info
+
+        call_fn(dcerpc.dcerpc_samr_QuerySecurity, self.pipe, r)
+
+        return r.data_out.sdbuf
+
+    def SetSecurity(self, sdbuf, sec_info = 7):
+
+        r = dcerpc.samr_SetSecurity()
+        r.data_in.handle = self.handle
+        r.data_in.sec_info = sec_info
+        r.data_in.sdbuf = sdbuf
+
+        call_fn(dcerpc.dcerpc_samr_SetSecurity, self.pipe, r)
+
+        
 class ConnectHandle(SamrHandle):
 
     def EnumDomains(self):
 
-        r = {}
-        r['connect_handle'] = self.handle
-        r['resume_handle'] = 0
-        r['buf_size'] = -1
+        r = dcerpc.samr_EnumDomains()
+        r.data_in.connect_handle = self.handle
+        r.data_in.resume_handle = 0
+        r.data_in.buf_size = -1
 
         domains = []
 
         while 1:
 
-            result = dcerpc.samr_EnumDomains(self.pipe, r)
+            call_fn(dcerpc.dcerpc_samr_EnumDomains, self.pipe, r)
 
-            domains = domains + result['sam']['entries']
+            for i in range(r.data_out.sam.count):
+                domains.append(dcerpc.samr_SamEntry_array_getitem(
+                    r.data_out.sam.entries, i).name.string)
 
-            if result['result'] == dcerpc.STATUS_MORE_ENTRIES:
-                r['resume_handle'] = result['resume_handle']
-                continue
+            # TODO: Handle more entries here
 
             break
 
-        return map(lambda x: x['name']['name'], domains)
+        return domains
 
     def LookupDomain(self, domain_name):
 
-        r = {}
-        r['connect_handle'] = self.handle
-        r['domain'] = {}
-        r['domain']['name_len'] = 0
-        r['domain']['name_size'] = 0
-        r['domain']['name'] = domain_name
+        r = dcerpc.samr_LookupDomain()
+        r.data_in.connect_handle = self.handle
+        r.data_in.domain_name = dcerpc.samr_String()
+        r.data_in.domain_name.string = domain_name
 
-        result = dcerpc.samr_LookupDomain(self.pipe, r)
+        call_fn(dcerpc.dcerpc_samr_LookupDomain, self.pipe, r)
 
-        return sid_to_string(result['sid'])
+        return sid_to_string(r.data_out.sid);
 
     def OpenDomain(self, domain_sid, access_mask = 0x02000000):
 
-        r = {}
-        r['connect_handle'] = self.handle
-        r['access_mask'] = access_mask
-        r['sid'] = string_to_sid(domain_sid)
+        r = dcerpc.samr_OpenDomain()
+        r.data_in.connect_handle = self.handle
+        r.data_in.access_mask = access_mask
+        r.data_in.sid = string_to_sid(domain_sid)
+
+        call_fn(dcerpc.dcerpc_samr_OpenDomain, self.pipe, r)
 
-        result = dcerpc.samr_OpenDomain(self.pipe, r)
+        return DomainHandle(self.pipe, r.data_out.domain_handle)
+
+    def Shutdown(self):
+
+        r = dcerpc.samr_Shutdown()
+        r.data_in.connect_handle = self.handle
+
+        call_fn(dcerpc.dcerpc_samr_Shutdown, self.pipe, r)
+
+    def GetDomPwInfo(self, domain_name):
+
+        r = dcerpc.samr_GetDomPwInfo()
+        r.data_in.domain_name = dcerpc.samr_String()
+        r.data_in.domain_name.string = domain_name
+
+        call_fn(dcerpc.dcerpc_samr_GetDomPwInfo, self.pipe, r)
+
+        return r.data_out.info
 
-        return DomainHandle(pipe, result['domain_handle'])
 
 class DomainHandle(SamrHandle):
 
     def QueryDomainInfo(self, level = 2):
 
-        r = {}
-        r['domain_handle'] = self.domain_handle
-        r['level'] = level
+        r = dcerpc.samr_QueryDomainInfo()
+        r.data_in.domain_handle = self.handle
+        r.data_in.level = level
+
+        call_fn(dcerpc.dcerpc_samr_QueryDomainInfo, self.pipe, r)
+
+        return getattr(r.data_out.info, 'info%d' % level)
+
+    def QueryDomainInfo2(self, level = 2):
+
+        r = dcerpc.samr_QueryDomainInfo2()
+        r.data_in.domain_handle = self.handle
+        r.data_in.level = level
+
+        call_fn(dcerpc.dcerpc_samr_QueryDomainInfo2, self.pipe, r)
+
+        return getattr(r.data_out.info, 'info%d' % level)       
+
+    def EnumDomainGroups(self):
+
+        r = dcerpc.samr_EnumDomainGroups()
+        r.data_in.domain_handle = self.handle
+        r.data_in.resume_handle = 0
+        r.data_in.max_size = 1000
+
+        call_fn(dcerpc.dcerpc_samr_EnumDomainGroups, self.pipe, r)
+
+        groups = []
+
+        if r.data_out.sam.entries:
+            for i in range(r.data_out.sam.count):
+                groups.append(dcerpc.samr_SamEntry_array_getitem(
+                    r.data_out.sam.entries, i).name.string)
+
+        return groups
+
+    def EnumDomainAliases(self):
+
+        r = dcerpc.samr_EnumDomainAliases()
+        r.data_in.domain_handle = self.handle
+        r.data_in.resume_handle = 0
+        # acct_flags in SamrEnumerateAliasesInDomain has probably
+        # no meaning so use 0xffffffff like W2K
+        r.data_in.acct_flags = 0xffffffffL
+
+        call_fn(dcerpc.dcerpc_samr_EnumDomainAliases, self.pipe, r)
+
+        aliases = []
+
+        if r.data_out.sam.entries:
+            for i in range(r.data_out.sam.count):
+                aliases.append(dcerpc.samr_SamEntry_array_getitem(
+                    r.data_out.sam.entries, i).name.string)
+
+        return aliases
+
+    def EnumDomainUsers(self, user_account_flags = 16):
+
+        r = dcerpc.samr_EnumDomainUsers()
+        r.data_in.domain_handle = self.handle
+        r.data_in.resume_handle = 0
+        r.data_in.acct_flags = user_account_flags
+        r.data_in.max_size = 1000
+
+        call_fn(dcerpc.dcerpc_samr_EnumDomainUsers, self.pipe, r)
+
+        users = []
+
+        if r.data_out.sam.entries:
+            for i in range(r.data_out.sam.count):
+                users.append(dcerpc.samr_SamEntry_array_getitem(
+                    r.data_out.sam.entries, i).name.string)
+
+        return users
+
+    def CreateUser(self, account_name, access_mask = 0x02000000):
+
+        r = dcerpc.samr_CreateUser()
+        r.data_in.domain_handle = self.handle
+        r.data_in.account_name = dcerpc.samr_String()
+        r.data_in.account_name.string = account_name
+        r.data_in.access_mask = access_mask
+
+        call_fn(dcerpc.dcerpc_samr_CreateUser, self.pipe, r)
+
+        return (r.data_out.user_handle,
+                dcerpc.uint32_array_getitem(r.data_out.rid, 0))
+
+    def CreateUser2(self, account_name, acct_flags = 0x00000010,
+                    access_mask = 0x02000000):
+
+        r = dcerpc.samr_CreateUser2()
+        r.data_in.domain_handle = self.handle
+        r.data_in.account_name = dcerpc.samr_String()
+        r.data_in.account_name.string = account_name
+        r.data_in.acct_flags = acct_flags
+        r.data_in.access_mask = access_mask
+
+        call_fn(dcerpc.dcerpc_samr_CreateUser2, self.pipe, r)
+
+        return (r.data_out.user_handle,
+                dcerpc.uint32_array_getitem(r.data_out.access_granted, 0),
+                dcerpc.uint32_array_getitem(r.data_out.rid, 0))
+
+    def OpenUser(self, rid, access_mask = 0x02000000):
+
+        r = dcerpc.samr_OpenUser()
+        r.data_in.domain_handle = self.handle
+        r.data_in.access_mask = access_mask
+        r.data_in.rid = rid
+
+        call_fn(dcerpc.dcerpc_samr_OpenUser, self.pipe, r)
+
+        return UserHandle(self.pipe, r.data_out.user_handle)
+
+    def OpenGroup(self, rid, access_mask = 0x02000000):
+
+        r = dcerpc.samr_OpenGroup()
+        r.data_in.domain_handle = self.handle
+        r.data_in.access_mask = access_mask
+        r.data_in.rid = rid
+
+        call_fn(dcerpc.dcerpc_samr_OpenGroup, self.pipe, r)
+
+        return GroupHandle(pipe, r.data_out.group_handle)
+
+    def OpenAlias(self, rid, access_mask = 0x02000000):
+
+        r = dcerpc.samr_OpenAlias()
+        r.data_in.domain_handle = self.handle
+        r.data_in.access_mask = access_mask
+        r.data_in.rid = rid
+
+        call_fn(dcerpc.dcerpc_samr_OpenAlias, self.pipe, r)
+
+        return AliasHandle(self.pipe, r.data_out.alias_handle)
 
-        result = dcerpc.samr_QueryDomainInfo(pipe, r)
+    def CreateDomAlias(self, alias_name, access_mask = 0x02000000):
 
-        return result
+        r = dcerpc.samr_CreateDomAlias()
+        r.data_in.domain_handle = self.handle
+        r.data_in.alias_name = dcerpc.samr_String()
+        r.data_in.alias_name.string = alias_name
+        r.data_in.access_mask = access_mask
 
-def Connect(pipe, system_name = None, access_mask = 0x02000000):
+        call_fn(dcerpc.dcerpc_samr_CreateDomAlias, self.pipe, r)
+
+        return (AliasHandle(self.pipe, r.data_out.alias_handle),
+                r.data_out.rid)    
+
+    def RidToSid(self, rid):
+
+        r = dcerpc.samr_RidToSid()
+        r.data_in.domain_handle = self.handle
+
+        call_fn(dcerpc.dcerpc_samr_RidToSid, self.pipe, r)
+
+        return sid_to_string(r.data_out.sid)
+
+    def RemoveMemberFromForeignDomain(self, sid):
+
+        r = dcerpc.samr_RemoveMemberFromForeignDomain()
+        r.data_in.domain_handle = self.handle
+        r.data_in.sid = sid
+
+        call_fn(dcerpc.dcerpc_samr_RemoveMemberFromForeignDomain, self.pipe, r)
+
+    def LookupNames(self, names):
+
+        r = dcerpc.samr_LookupNames()
+        r.data_in.domain_handle = self.handle
+        r.data_in.num_names = len(names)
+        r.data_in.names = dcerpc.new_samr_String_array(len(names))
+
+        for i in range(len(names)):
+            s = dcerpc.samr_String()
+            s.string = names[i]
+            dcerpc.samr_String_array_setitem(r.data_in.names, i, s)
+
+        call_fn(dcerpc.dcerpc_samr_LookupNames, self.pipe, r)
+
+        return ([dcerpc.uint32_array_getitem(r.data_out.rids.ids, i) for i in range(r.data_out.rids.count)],
+                [dcerpc.uint32_array_getitem(r.data_out.types.ids, i) for i in range(r.data_out.types.count)])
+
+    def CreateDomainGroup(self, domain_name, access_mask = 0x02000000):
+
+        r = dcerpc.samr_CreateDomainGroup()
+        r.data_in.domain_handle = self.handle
+        r.data_in.name = dcerpc.samr_String()
+        r.data_in.name.string = domain_name
+        r.data_in.access_mask = access_mask
+
+        call_fn(dcerpc.dcerpc_samr_CreateDomainGroup, self.pipe, r)
+
+    def GetAliasMembership(self, sids):
+
+        r = dcerpc.samr_GetAliasMembership()
+        r.data_in.domain_handle = self.handle
+        r.data_in.sids = dcerpc.lsa_SidArray()
+        r.data_in.sids.num_sids = len(sids)
+        r.data_in.sids.sids = dcerpc.new_lsa_SidPtr_array(len(sids))
+
+        for i in range(len(sids)):
+            s = dcerpc.lsa_SidPtr()
+            s.sid = string_to_sid(sids[i])
+            dcerpc.lsa_SidPtr_array_setitem(r.data_in.sids.sids, i, s)
+
+        call_fn(dcerpc.dcerpc_samr_GetAliasMembership, self.pipe, r)
+
+        return [r.ids[x] x in range(r.count)]
+
+class UserHandle(SamrHandle):
+
+    def DeleteUser(self):
+
+        r = dcerpc.samr_DeleteUser()
+        r.data_in.user_handle = self.handle
+        
+        call_fn(dcerpc.dcerpc_samr_DeleteUser, self.pipe, r)
+
+        self.handle = None
+    
+
+class GroupHandle(SamrHandle):
+    pass
+    
+
+class AliasHandle(SamrHandle):
+
+    def DeleteDomAlias(self):
+
+        r = dcerpc.samr_DeleteDomAlias()
+        r.data_in.alias_handle = self.handle
+
+        call_fn(dcerpc.dcerpc_samr_DeleteDomAlias, self.pipe, r)
+
+        self.handle = None
+
+    def QueryAliasInfo(self, level = 1):
+
+        r = dcerpc.samr_QueryAliasInfo()
+        r.data_in.alias_handle = self.handle
+        r.data_in.level = level
+
+        call_fn(dcerpc.dcerpc_samr_QueryAliasInfo, self.pipe, r)
+
+        return r.data_out.info
+
+    def SetAliasInfo(self, level, info):
+
+        r = dcerpc.samr_SetAliasInfo()
+        r.data_in.alias_handle = self.handle
+        r.data_in.level = level
+        r.data_in.info = info
+
+        call_fn(dcerpc.dcerpc_samr_SetAliasInfo, self.pipe, r)
+
+    def AddAliasMember(self, sid):
+
+        r = dcerpc.samr_AddAliasMember()
+        r.data_in.alias_handle = self.handle
+        r.data_in.sid = string_to_sid(sid)
+
+        call_fn(dcerpc.dcerpc_samr_AddAliasMember, self.pipe, r)
+
+    def AddMultipleMembersToAlias(self, sids):
+
+        r = dcerpc.samr_AddMultipleMembersToAlias()
+        r.data_in.alias_handle = self.handle
+        r.data_in.sids = dcerpc.lsa_SidArray()
+        r.data_in.sids.num_sids = len(sids)
+        r.data_in.sids.sids = dcerpc.new_lsa_SidPtr_array(len(sids))
+
+        for i in range(len(sids)):
+            s = dcerpc.lsa_SidPtr()
+            s.sid = string_to_sid(sids[i])
+            dcerpc.lsa_SidPtr_array_setitem(r.data_in.sids.sids, i, s)
+
+        call_fn(dcerpc.dcerpc_samr_AddMultipleMembersToAlias, self.pipe, r)
+
+def Connect(pipe, access_mask = 0x02000000):
+
+    r = dcerpc.samr_Connect()
+    r.data_in.system_name = dcerpc.new_uint16_array(1)
+    dcerpc.uint16_array_setitem(r.data_in.system_name, 0, ord('\\'))
+    r.data_in.access_mask = access_mask
+
+    call_fn(dcerpc.dcerpc_samr_Connect, pipe, r)
+
+    return ConnectHandle(pipe, r.data_out.connect_handle)
+
+
+def Connect2(pipe, system_name = '', access_mask = 0x02000000):
     """Connect to the SAMR pipe."""
 
-    r = {}
-    r['system_name'] = system_name
-    r['access_mask'] = access_mask
+    r = dcerpc.samr_Connect2()
+    r.data_in.system_name = system_name
+    r.data_in.access_mask = access_mask
+
+    call_fn(dcerpc.dcerpc_samr_Connect2, pipe, r)
+
+    return ConnectHandle(pipe, r.data_out.connect_handle)
+
+
+def Connect3(pipe, system_name = '', access_mask = 0x02000000):
+
+    r = dcerpc.samr_Connect3()
+    r.data_in.system_name = system_name
+    r.data_in.unknown = 0
+    r.data_in.access_mask = access_mask
+
+    call_fn(dcerpc.dcerpc_samr_Connect3, pipe, r)
+
+    return ConnectHandle(pipe, r.data_out.connect_handle)
+
 
-    result = dcerpc.samr_Connect2(pipe, r)
+def Connect4(pipe, system_name = '', access_mask = 0x02000000):
 
-    return ConnectHandle(pipe, result['connect_handle'])
+    r = dcerpc.samr_Connect4()
+    r.data_in.system_name = system_name
+    r.data_in.unknown = 0
+    r.data_in.access_mask = access_mask
+
+    call_fn(dcerpc.dcerpc_samr_Connect4, pipe, r)
+
+    return ConnectHandle(pipe, r.data_out.connect_handle)
+
+
+def Connect5(pipe, system_name = '', access_mask = 0x02000000):
+
+    r = dcerpc.samr_Connect5()
+    r.data_in.system_name = system_name
+    r.data_in.access_mask = access_mask
+    r.data_in.level = 1
+    r.data_in.info = dcerpc.new_samr_ConnectInfo_array(1)
+    r.data_in.info.unknown1 = 0
+    r.data_in.info.unknown2 = 0
+
+    call_fn(dcerpc.dcerpc_samr_Connect5, pipe, r)
+
+    return ConnectHandle(pipe, r.data_out.connect_handle)
+    
+# QueryGroupInfo
+# SetGroupInfo
+# AddGroupMember
+# DeleteDomainGroup
+# DeleteGroupMember
+# QueryGroupMember
+# SetMemberAttributesofGroup
+# AddAliasMember
+# DeleteAliasMember
+# GetMembersinAlias
+# QueryUserInfo
+# SetUserInfo
+# ChangePasswordUser
+# GetGroupsForUser
+# QueryDisplayInfo
+# GetDisplayEnumerationIndex
+# TestPrivateFunctionsDomain
+# TestPrivateFunctionsUser
+# GetUserPwInfo
+# RemoveMemberFromForeignDomain
+# QueryDomainInfo2
+# QueryUserInfo2
+# QueryDisplayInfo2
+# GetDisplayEnumerationIndex2
+# QueryDisplayInfo3
+# RemoveMultipleMembersFromAlias
+# OemChangePasswordUser2
+# ChangePasswordUser2
+# SetUserInfo2
+# SetBootKeyInformation
+# GetBootKeyInformation
+# ChangePasswordUser3
+# SetDsrmPassword
+# ValidatePassword