2015-05-21 08:33:56 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
from __future__ import with_statement
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
|
|
|
import pytest
|
|
|
|
import sys
|
|
|
|
from sets import Set
|
2015-07-05 22:01:09 +02:00
|
|
|
from ldapcherry.backend.backendLdap import Backend, DelUserDontExists, CaFileDontExist
|
2015-07-05 22:48:24 +02:00
|
|
|
from ldapcherry.exceptions import *
|
2015-06-18 00:23:00 +02:00
|
|
|
from disable import travis_disabled
|
2015-05-21 08:33:56 +02:00
|
|
|
import cherrypy
|
2015-05-26 00:33:36 +02:00
|
|
|
import logging
|
|
|
|
import ldap
|
2015-05-21 08:33:56 +02:00
|
|
|
|
|
|
|
cfg = {
|
2015-05-25 18:52:14 +02:00
|
|
|
'module' : 'ldapcherry.backend.ldap',
|
2015-05-27 21:56:55 +02:00
|
|
|
'groupdn' : 'ou=groups,dc=example,dc=org',
|
2015-05-25 18:52:14 +02:00
|
|
|
'userdn' : 'ou=People,dc=example,dc=org',
|
|
|
|
'binddn' : 'cn=dnscherry,dc=example,dc=org',
|
|
|
|
'password' : 'password',
|
2015-06-17 20:51:21 +02:00
|
|
|
'uri' : 'ldap://ldap.dnscherry.org:390',
|
2015-07-05 22:01:09 +02:00
|
|
|
'ca' : './tests/test_env/etc/ldapcherry/TEST-cacert.pem',
|
2015-05-25 18:52:14 +02:00
|
|
|
'starttls' : 'off',
|
|
|
|
'checkcert' : 'off',
|
|
|
|
'user_filter_tmpl' : '(uid=%(username)s)',
|
|
|
|
'group_filter_tmpl' : '(member=%(userdn)s)',
|
2015-05-25 19:30:41 +02:00
|
|
|
'search_filter_tmpl' : '(|(uid=%(searchstring)s*)(sn=%(searchstring)s*))',
|
2015-05-26 00:33:36 +02:00
|
|
|
'objectclasses' : 'top, person, organizationalPerson, simpleSecurityObject, posixAccount',
|
|
|
|
'dn_user_attr' : 'uid',
|
2015-06-17 00:39:03 +02:00
|
|
|
'group_attr.member' : "%(dn)s",
|
2015-06-17 20:30:26 +02:00
|
|
|
'timeout' : 10,
|
2015-05-21 08:33:56 +02:00
|
|
|
}
|
|
|
|
|
2015-05-26 00:33:36 +02:00
|
|
|
def syslog_error(msg='', context='',
|
|
|
|
severity=logging.INFO, traceback=False):
|
|
|
|
pass
|
|
|
|
|
2015-05-21 21:40:13 +02:00
|
|
|
cherrypy.log.error = syslog_error
|
2015-05-22 20:05:24 +02:00
|
|
|
attr = ['shéll', 'shell', 'cn', 'uid', 'uidNumber', 'gidNumber', 'home', 'userPassword', 'givenName', 'email', 'sn']
|
2015-05-21 21:40:13 +02:00
|
|
|
|
2015-05-21 08:33:56 +02:00
|
|
|
class TestError(object):
|
|
|
|
|
|
|
|
def testNominal(self):
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-21 08:33:56 +02:00
|
|
|
return True
|
|
|
|
|
2015-06-17 22:20:21 +02:00
|
|
|
def testConnectSSLNoCheck(self):
|
|
|
|
cfg2 = cfg.copy()
|
|
|
|
cfg2['uri'] = 'ldaps://ldap.ldapcherry.org:637'
|
|
|
|
cfg2['checkcert'] = 'off'
|
|
|
|
inv = Backend(cfg2, cherrypy.log, 'ldap', attr, 'uid')
|
|
|
|
ldap = inv._connect()
|
|
|
|
ldap.simple_bind_s(inv.binddn, inv.bindpassword)
|
2015-06-17 00:06:30 +02:00
|
|
|
|
2015-05-21 08:33:56 +02:00
|
|
|
def testConnect(self):
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-22 01:16:53 +02:00
|
|
|
ldap = inv._connect()
|
|
|
|
ldap.simple_bind_s(inv.binddn, inv.bindpassword)
|
2015-05-21 08:33:56 +02:00
|
|
|
return True
|
|
|
|
|
|
|
|
def testConnectSSL(self):
|
2015-05-22 01:16:53 +02:00
|
|
|
cfg2 = cfg.copy()
|
2015-06-17 19:22:57 +02:00
|
|
|
cfg2['uri'] = 'ldaps://ldap.dnscherry.org:637'
|
2015-05-22 01:16:53 +02:00
|
|
|
cfg2['checkcert'] = 'on'
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg2, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-22 01:16:53 +02:00
|
|
|
ldap = inv._connect()
|
|
|
|
ldap.simple_bind_s(inv.binddn, inv.bindpassword)
|
2015-05-21 08:33:56 +02:00
|
|
|
|
2015-05-22 19:38:41 +02:00
|
|
|
def testLdapUnavaible(self):
|
|
|
|
cfg2 = cfg.copy()
|
|
|
|
cfg2['uri'] = 'ldaps://notaldap:637'
|
|
|
|
cfg2['checkcert'] = 'on'
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg2, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-22 19:38:41 +02:00
|
|
|
try:
|
2015-07-05 22:01:09 +02:00
|
|
|
ldapc = inv._connect()
|
2015-05-22 19:38:41 +02:00
|
|
|
ldapc.simple_bind_s(inv.binddn, inv.bindpassword)
|
2015-05-26 00:33:36 +02:00
|
|
|
except ldap.SERVER_DOWN as e:
|
2015-07-05 22:48:24 +02:00
|
|
|
return
|
2015-05-22 19:38:41 +02:00
|
|
|
else:
|
|
|
|
raise AssertionError("expected an exception")
|
|
|
|
|
2015-07-05 22:01:09 +02:00
|
|
|
def testMissingCA(self):
|
|
|
|
cfg2 = cfg.copy()
|
|
|
|
cfg2['uri'] = 'ldaps://ldap.dnscherry.org:637'
|
|
|
|
cfg2['checkcert'] = 'on'
|
|
|
|
cfg2['ca'] = './test/cfg/not_a_ca.crt'
|
|
|
|
try:
|
|
|
|
inv = Backend(cfg2, cherrypy.log, 'ldap', attr, 'uid')
|
|
|
|
ldapc = inv._connect()
|
|
|
|
except CaFileDontExist as e:
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
raise AssertionError("expected an exception")
|
|
|
|
|
2015-05-22 01:16:53 +02:00
|
|
|
def testConnectSSLWrongCA(self):
|
|
|
|
cfg2 = cfg.copy()
|
|
|
|
cfg2['uri'] = 'ldaps://ldap.ldapcherry.org:637'
|
|
|
|
cfg2['checkcert'] = 'on'
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg2, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-22 01:16:53 +02:00
|
|
|
ldapc = inv._connect()
|
|
|
|
try:
|
|
|
|
ldapc.simple_bind_s(inv.binddn, inv.bindpassword)
|
2015-05-26 00:33:36 +02:00
|
|
|
except ldap.SERVER_DOWN as e:
|
2015-05-22 01:16:53 +02:00
|
|
|
assert e[0]['info'] == 'TLS: hostname does not match CN in peer certificate'
|
2015-07-05 22:01:09 +02:00
|
|
|
else:
|
|
|
|
raise AssertionError("expected an exception")
|
2015-05-22 01:16:53 +02:00
|
|
|
|
2015-06-17 22:51:33 +02:00
|
|
|
def testConnectStartTLS(self):
|
|
|
|
cfg2 = cfg.copy()
|
|
|
|
cfg2['uri'] = 'ldap://ldap.ldapcherry.org:390'
|
|
|
|
cfg2['checkcert'] = 'off'
|
|
|
|
cfg2['starttls'] = 'on'
|
|
|
|
cfg2['ca'] = './test/cfg/ca.crt'
|
|
|
|
inv = Backend(cfg2, cherrypy.log, 'ldap', attr, 'uid')
|
|
|
|
ldapc = inv._connect()
|
2015-07-05 22:01:09 +02:00
|
|
|
ldapc.simple_bind_s(inv.binddn, inv.bindpassword)
|
2015-06-17 22:51:33 +02:00
|
|
|
|
2015-05-21 08:33:56 +02:00
|
|
|
def testAuthSuccess(self):
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-22 01:33:15 +02:00
|
|
|
ret = inv.auth('jwatson', 'passwordwatson')
|
|
|
|
assert ret == True
|
2015-05-21 08:33:56 +02:00
|
|
|
|
|
|
|
def testAuthFailure(self):
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-22 01:33:15 +02:00
|
|
|
res = inv.auth('notauser', 'password') or inv.auth('jwatson', 'notapassword')
|
|
|
|
assert res == False
|
2015-05-21 08:33:56 +02:00
|
|
|
|
|
|
|
def testMissingParam(self):
|
|
|
|
cfg2 = {}
|
|
|
|
return True
|
|
|
|
try:
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg2, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-21 08:33:56 +02:00
|
|
|
except MissingKey:
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
raise AssertionError("expected an exception")
|
|
|
|
|
|
|
|
def testGetUser(self):
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-22 20:05:24 +02:00
|
|
|
ret = inv.get_user('jwatson')
|
2015-05-26 22:51:29 +02:00
|
|
|
expected = {'uid': 'jwatson', 'cn': 'John Watson', 'sn': 'watson'}
|
2015-05-22 20:05:24 +02:00
|
|
|
assert ret == expected
|
2015-05-25 19:30:41 +02:00
|
|
|
|
2015-06-17 00:39:03 +02:00
|
|
|
def testGetGroups(self):
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-27 21:56:55 +02:00
|
|
|
ret = inv.get_groups('jwatson')
|
|
|
|
expected = ['cn=itpeople,ou=Groups,dc=example,dc=org']
|
|
|
|
assert ret == expected
|
|
|
|
|
2015-06-17 00:39:03 +02:00
|
|
|
def testAddDeleteGroups(self):
|
|
|
|
inv = Backend(cfg, cherrypy.log, 'ldap', attr, 'uid')
|
|
|
|
groups = [
|
|
|
|
'cn=hrpeople,ou=Groups,dc=example,dc=org',
|
|
|
|
'cn=itpeople,ou=Groups,dc=example,dc=org',
|
|
|
|
]
|
|
|
|
inv.add_to_groups('jwatson', groups)
|
|
|
|
ret = inv.get_groups('jwatson')
|
|
|
|
print ret
|
|
|
|
inv.del_from_groups('jwatson', ['cn=hrpeople,ou=Groups,dc=example,dc=org'])
|
|
|
|
inv.del_from_groups('jwatson', ['cn=hrpeople,ou=Groups,dc=example,dc=org'])
|
|
|
|
assert ret == ['cn=itpeople,ou=Groups,dc=example,dc=org', 'cn=hrpeople,ou=Groups,dc=example,dc=org']
|
|
|
|
|
|
|
|
|
2015-05-26 00:33:36 +02:00
|
|
|
def testSearchUser(self):
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-25 19:30:41 +02:00
|
|
|
ret = inv.search('smith')
|
2015-07-05 22:48:24 +02:00
|
|
|
expected = {'ssmith': {'sn': 'smith', 'uid': 'ssmith', 'cn': 'Sheri Smith', 'userPassword': 'passwordsmith'}, 'jsmith': {'sn': 'Smith', 'uid': 'jsmith', 'cn': 'John Smith', 'userPassword': 'passwordsmith'}}
|
2015-05-25 19:30:41 +02:00
|
|
|
assert ret == expected
|
2015-05-26 00:33:36 +02:00
|
|
|
|
|
|
|
def testAddUser(self):
|
2015-07-29 00:09:32 +02:00
|
|
|
try:
|
|
|
|
inv.del_user(u'test☭')
|
|
|
|
except:
|
|
|
|
pass
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-26 00:33:36 +02:00
|
|
|
user = {
|
2015-07-29 00:09:32 +02:00
|
|
|
'uid': u'test☭',
|
2015-05-26 00:33:36 +02:00
|
|
|
'sn': 'test',
|
|
|
|
'cn': 'test',
|
|
|
|
'userPassword': 'test',
|
|
|
|
'uidNumber': '42',
|
|
|
|
'gidNumber': '42',
|
|
|
|
'homeDirectory': '/home/test/'
|
|
|
|
}
|
|
|
|
inv.add_user(user)
|
2015-07-29 00:09:32 +02:00
|
|
|
inv.del_user(u'test☭')
|
2015-05-26 00:33:36 +02:00
|
|
|
|
2015-06-16 21:32:14 +02:00
|
|
|
def testModifyUser(self):
|
|
|
|
inv = Backend(cfg, cherrypy.log, 'ldap', attr, 'uid')
|
|
|
|
user = {
|
2015-07-29 00:09:32 +02:00
|
|
|
'uid': u'test☭',
|
2015-06-16 21:32:14 +02:00
|
|
|
'sn': 'test',
|
|
|
|
'cn': 'test',
|
|
|
|
'userPassword': 'test',
|
|
|
|
'uidNumber': '42',
|
|
|
|
'gidNumber': '42',
|
|
|
|
'homeDirectory': '/home/test/'
|
|
|
|
}
|
|
|
|
inv.add_user(user)
|
2015-07-29 00:09:32 +02:00
|
|
|
inv.set_attrs(u'test☭', {'gecos': 'test2', 'homeDirectory': '/home/test/'})
|
|
|
|
inv.del_user(u'test☭')
|
2015-06-16 21:32:14 +02:00
|
|
|
|
2015-05-26 00:33:36 +02:00
|
|
|
def testAddUserDuplicate(self):
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-26 00:33:36 +02:00
|
|
|
user = {
|
|
|
|
'uid': 'test',
|
|
|
|
'sn': 'test',
|
|
|
|
'cn': 'test',
|
|
|
|
'uidNumber': '42',
|
|
|
|
'userPassword': 'test',
|
|
|
|
'gidNumber': '42',
|
|
|
|
'homeDirectory': '/home/test/'
|
|
|
|
}
|
|
|
|
try:
|
|
|
|
inv.add_user(user)
|
|
|
|
inv.add_user(user)
|
|
|
|
except ldap.ALREADY_EXISTS:
|
|
|
|
inv.del_user('test')
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
inv.del_user('test')
|
|
|
|
raise AssertionError("expected an exception")
|
|
|
|
|
2015-05-28 09:51:19 +02:00
|
|
|
def testDelUserDontExists(self):
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-28 09:51:19 +02:00
|
|
|
try:
|
|
|
|
inv.del_user('test')
|
|
|
|
inv.del_user('test')
|
|
|
|
except DelUserDontExists:
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
raise AssertionError("expected an exception")
|
|
|
|
|
2015-05-28 09:56:25 +02:00
|
|
|
def testGetUser(self):
|
2015-05-31 18:40:35 +02:00
|
|
|
inv = Backend(cfg, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-28 09:56:25 +02:00
|
|
|
ret = inv.get_user('jwatson')
|
2015-07-05 22:48:24 +02:00
|
|
|
expected = {'uid': 'jwatson', 'objectClass': 'inetOrgPerson', 'carLicense': 'HERCAR 125', 'sn': 'watson', 'mail': 'j.watson@example.com', 'homePhone': '555-111-2225', 'cn': 'John Watson', 'userPassword': u'passwordwatson'}
|
2015-05-28 09:56:25 +02:00
|
|
|
assert ret == expected
|
2015-05-28 09:51:19 +02:00
|
|
|
|
2015-05-31 18:40:35 +02:00
|
|
|
def testAddUserMissingMustattribute(self):
|
|
|
|
inv = Backend(cfg, cherrypy.log, 'ldap', attr, 'uid')
|
2015-05-26 00:33:36 +02:00
|
|
|
user = {
|
|
|
|
'uid': 'test',
|
|
|
|
'sn': 'test',
|
|
|
|
'cn': 'test',
|
|
|
|
'userPassword': 'test',
|
|
|
|
'gidNumber': '42',
|
|
|
|
'homeDirectory': '/home/test/'
|
|
|
|
}
|
|
|
|
try:
|
|
|
|
inv.add_user(user)
|
|
|
|
except ldap.OBJECT_CLASS_VIOLATION:
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
inv.del_user('test')
|
|
|
|
raise AssertionError("expected an exception")
|