ActiveDirectory/LDAP authentication backend extension for Sal with ldap group to business unit mapping.
This docker container extends Sal by adding a Django authentication backend with ActiveDirectory/LDAP integration.
Currently, this connector works with MS ActiveDirectory/LDAP only.
- Binding to the configured AD/LDAP server with username and password from the Sal/Django user login.
- Creating a Sal/Django user with field information from AD/LDAP (no password is stored in Django!)
- Setting Sal user profile (GA, RW, RO) based on their AD/LDAP group.
- Assigning users to Sal business units based on their AD/LDAP group.
- Updating the user profile and business unit assignment at every login of the user.
The setup is idantical to a standard Sal docker container as described in the Sal Wiki.
In addition you have to link your settings.py into the container.
$ docker run -d --name="sal"\
  -p 80:8000 \
  --link postgres-sal:db \
  -e ADMIN_PASS=pass \
  -e DB_NAME=sal \
  -e DB_USER=admin \
  -e DB_PASS=password \
  --volume /path/to/your/settings.py:/home/docker/sal/sal/settings.py:ro
  macadmins/sal-activedirectory:3.2.14Following settings can/need to be configured in the Django settings.py file to get this ActiveDirectory authentication backend to work.
See the settings_example.py for guidance.
Make sure that the Active Directory authentication is configured as authentication backend.
AUTHENTICATION_BACKENDS = [
    'server.ADConnector.ADConnector',
    'django.contrib.auth.backends.ModelBackend',
]If you don't intend to use the connector, just don't add it to your AUTHENTICATION_BACKENDS.
URL of the AD/LDAP server.
AUTH_LDAP_SERVER_URI = 'ldaps://hostname.company.com:636'Domain of the AD/LDAP server.
AUTH_LDAP_USER_DOMAIN = 'company.com'username will be converted to [email protected] (company.com = AUTH_LDAP_USER_DOMAIN) for the AD/LDAP authentication. The domain will only get appended to the username if the username does not end with the configured domain.
AD/LDAP search base for the user object.
AUTH_LDAP_USER_SEARCH = 'DC=department,DC=ads,DC=company,DC=com'It is possible to specify more than one base dn. In this case, the first valid one is taken.
AUTH_LDAP_USER_SEARCH = ('DC=ch,DC=ads,DC=company,DC=com', 'DC=uk,DC=ads,DC=company,DC=com', 'DC=us,DC=ads,DC=company,DC=com')Mapping of the AD/LDAP attributes to Django attributes. If these settings are not configured, these default values are used.
AUTH_LDAP_USER_ATTR_MAP = {
    "username": "sAMAccountName",
    "first_name": "givenName",
    "last_name": "sn",
    "email": "mail"
}If you have a self signed certificate or an unknown certificate to the Django server, you need to disable the certificate check by setting this value to True.
AUTH_LDAP_TRUST_ALL_CERTIFICATES = TrueThe parameter defaults to False, causing Sal to trust certificates with a valide certificate chain only.
Django users that where created via AD/LDAP recieve the ldap_ preffix. (username becomes ldap_username). This allows to have local Django users and AD/LDAP users in coexistence. Furthermore, local Django users and AD/LDAP users are distinguished very easy.
AUTH_LDAP_USER_PREFIX = 'ldap_'Mapping of the user profile level (GA = Global Admin, RW = Read & Write, RO = Read Only, SO = Stats Only (not implemented in Sal)) to AD/LDAP groups. Mapping is a dictionary, where the key is the user profile level and the value corresponds to the AD/LDAP group. The group can be a single group or a list/tuple of AD/LDAP groups.
AUTH_LDAP_USER_PROFILE = {
                            'RO': ('CN=all-users,OU=it,DC=ad,DC=company,DC=com',),
                            'RW': ('CN=service-desk,OU=it,DC=ad,DC=company,DC=com',
                                   'CN=mac-admins,OU=it,DC=ad,DC=company,DC=com'),
                            'GA': 'CN=admins,DC=it,DC=ad,DC=company,DC=com',
                        }The order of the user profile check is from GA to RO respectively SO. If a user is member of the GA and RO group, the assigned user profile level is GA.
Mapping of AD/LDAP groups to business units. Mapping is a dictionary, where the key is the name of the business unit and the value corresponds to the AD/LDAP group. The group can be a single group or a list/tuple of AD/LDAP groups.
AUTH_LDAP_USER_TO_BUSINESS_UNIT = {
                            '#ALL_BU': ('CN=service-desk,OU=it,DC=ad,DC=company,DC=com',
                                        'CN=sysadmins,OU=it,DC=ad,DC=company,DC=com',),
                            'CH':      ('CN=mac-admins,OU=it,DC=ch,DC=ad,DC=company,DC=com',),
                            'UK':      ('CN=mac-admins,OU=it,DC=uk,DC=ad,DC=company,DC=com',),
                            'US':       'CN=mac-admins,OU=it,DC=us,DC=ad,DC=company,DC=com',
                        }Attention: #ALL_BU is a special business unit. All users in this configured groups get access to all existing business units.
If something does not work as expected, an extensive debug logging can be turned on. This is implemented with the python logging module and can be configured in the Django settings.
LOGGING = {
    'version': 1,
    'disable_existing_loggers': False,
    'formatters': {
        'standard': {
            'format': '%(asctime)-15s %(levelname)-7s %(filename)s:%(lineno)-4d: %(message)s',
            'datefmt': '%Y-%m-%d %H:%M:%S'
        },
    },
    'handlers': {
        'file': {
            'level': 'DEBUG',
            'class': 'logging.FileHandler',
            'filename': '/tmp/sal.log',
            'formatter': 'standard',
        },
    },
    'loggers': {
        'server': {
            'handlers': ['file'],
            'level': 'DEBUG',
            'propagate': True,
        },
    },
}With this configuration everything is logged to /tmp/sal.log. Make sure that you turn off this option in production environment!
Yes: This can be accomplished with the setting AUTH_LDAP_USER_PREFIX very easily.
This configured prefix will be added to the Django username, therefore existing Django users with the same username as users in the AD/LDAP can still login.
Is it possible to have a user with readonly rights in specific business unit and with write rights in a different one?
No: Unfortunately this is not possible by design of Sal. A user does always have one user profile which is valid for all assigned business unit.
Then the default user profile is set, which is read only (RO). Defined at UserProfile._meta.get_field('level').get_default().
This is possible with business unit #ALL_BU in the AUTH_LDAP_USER_TO_BUSINESS_UNIT configuration.
The user profile and all assigned business units of a user get a reset every time a user logs in. Otherwise it is not possible to ensure that users get removed from business units they shouldn't have access anymore. Therefore it is not possible and recommended to mix the assignment between Sal and the AD/LDAP configuration.
May be! But it is not tested or guaranteed. There are some AD/LDAP specific notations used to get nested groups (memberOf:1.2.840.113556.1.4.1941:=),
therefore I would not trust on a connection to another ldap implementation.
There are always things which can be improved!
- Bind with service account. At the moment the AD/LDAP connection is initiated with the authenticated user. It could be that this user does not have the permissions to access the group memberships. Therefore AD/LDAP bind with a service account could be very useful.
- Make it work with other ldap implementations than AD/LDAP. At the moment, this authentication works with AD/LDAP only. May be someone can test and adapt it for other ldap implementation as well. Note: Take care of the nested groups.
The version is identical with the version of Sal in the Dockerfile. This version of Sal is tested and proved to work with the given AD/LDAP authentication.
It could be that new authentication features get introduced between different version. See the changelog below for details.
- Sal version 3.3.10 in docker container
- Be consistent with documentation, make AUTH_LDAP_TRUST_ALL_CERTIFICATESoptional. UpdateAUTHENTICATION_BACKENDSin settings_example.py
- Sal version 3.3.1 in docker container
- Sal version 3.2.14 in docker container
- Support multiple user scopes
- Documentation update