2018-11-16 08:10:35 +00:00
|
|
|
"""passbook saml_idp Models"""
|
2019-03-08 11:47:50 +00:00
|
|
|
from django.contrib.postgres.fields import ArrayField
|
2018-11-16 08:10:35 +00:00
|
|
|
from django.db import models
|
2018-12-26 16:21:20 +00:00
|
|
|
from django.shortcuts import reverse
|
2018-11-26 16:18:56 +00:00
|
|
|
from django.utils.translation import gettext as _
|
2019-10-01 09:24:10 +01:00
|
|
|
from structlog import get_logger
|
2018-11-16 08:10:35 +00:00
|
|
|
|
2019-03-08 11:47:50 +00:00
|
|
|
from passbook.core.models import PropertyMapping, Provider
|
2018-12-26 16:21:20 +00:00
|
|
|
from passbook.lib.utils.reflection import class_to_path, path_to_class
|
2019-10-07 15:33:48 +01:00
|
|
|
from passbook.providers.saml.base import Processor
|
2018-11-16 08:10:35 +00:00
|
|
|
|
2019-10-04 09:08:53 +01:00
|
|
|
LOGGER = get_logger()
|
2019-04-29 20:39:41 +01:00
|
|
|
|
2018-11-16 08:10:35 +00:00
|
|
|
|
2018-11-26 16:17:32 +00:00
|
|
|
class SAMLProvider(Provider):
|
2018-11-16 08:10:35 +00:00
|
|
|
"""Model to save information about a Remote SAML Endpoint"""
|
|
|
|
|
2018-11-26 16:17:32 +00:00
|
|
|
name = models.TextField()
|
2018-11-16 08:10:35 +00:00
|
|
|
acs_url = models.URLField()
|
2019-04-18 10:30:21 +01:00
|
|
|
audience = models.TextField(default='')
|
2018-11-16 08:10:35 +00:00
|
|
|
processor_path = models.CharField(max_length=255, choices=[])
|
2018-12-09 22:06:14 +00:00
|
|
|
issuer = models.TextField()
|
|
|
|
assertion_valid_for = models.IntegerField(default=86400)
|
|
|
|
signing = models.BooleanField(default=True)
|
|
|
|
signing_cert = models.TextField()
|
|
|
|
signing_key = models.TextField()
|
2018-11-16 08:10:35 +00:00
|
|
|
|
2019-10-07 15:33:48 +01:00
|
|
|
form = 'passbook.providers.saml.forms.SAMLProviderForm'
|
2018-12-26 16:21:20 +00:00
|
|
|
_processor = None
|
2018-11-26 21:40:10 +00:00
|
|
|
|
2018-11-16 08:10:35 +00:00
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
2018-12-09 22:06:14 +00:00
|
|
|
self._meta.get_field('processor_path').choices = get_provider_choices()
|
2018-11-16 08:10:35 +00:00
|
|
|
|
2018-12-26 16:21:20 +00:00
|
|
|
@property
|
|
|
|
def processor(self):
|
2018-12-26 16:26:17 +00:00
|
|
|
"""Return selected processor as instance"""
|
2018-12-26 16:21:20 +00:00
|
|
|
if not self._processor:
|
2019-04-18 10:30:21 +01:00
|
|
|
try:
|
|
|
|
self._processor = path_to_class(self.processor_path)(self)
|
2019-10-08 13:34:59 +01:00
|
|
|
except ImportError as exc:
|
2019-04-29 20:39:41 +01:00
|
|
|
LOGGER.warning(exc)
|
2019-04-18 10:30:21 +01:00
|
|
|
self._processor = None
|
2018-12-26 16:21:20 +00:00
|
|
|
return self._processor
|
|
|
|
|
2018-11-16 08:10:35 +00:00
|
|
|
def __str__(self):
|
2019-03-07 13:14:49 +00:00
|
|
|
return "SAML Provider %s" % self.name
|
2018-11-16 12:08:37 +00:00
|
|
|
|
2018-12-26 16:21:20 +00:00
|
|
|
def link_download_metadata(self):
|
|
|
|
"""Get link to download XML metadata for admin interface"""
|
2019-02-27 13:47:11 +00:00
|
|
|
try:
|
2018-12-26 20:56:08 +00:00
|
|
|
# pylint: disable=no-member
|
2019-10-07 17:36:09 +01:00
|
|
|
return reverse('passbook_providers_saml:saml-metadata',
|
2018-12-26 20:56:08 +00:00
|
|
|
kwargs={'application': self.application.slug})
|
2019-02-27 13:47:11 +00:00
|
|
|
except Provider.application.RelatedObjectDoesNotExist:
|
|
|
|
return None
|
2018-12-26 16:21:20 +00:00
|
|
|
|
2018-11-26 16:17:32 +00:00
|
|
|
class Meta:
|
|
|
|
|
|
|
|
verbose_name = _('SAML Provider')
|
|
|
|
verbose_name_plural = _('SAML Providers')
|
2018-12-09 22:06:14 +00:00
|
|
|
|
|
|
|
|
2019-03-08 11:47:50 +00:00
|
|
|
class SAMLPropertyMapping(PropertyMapping):
|
|
|
|
"""SAML Property mapping, allowing Name/FriendlyName mapping to a list of strings"""
|
|
|
|
|
|
|
|
saml_name = models.TextField()
|
|
|
|
friendly_name = models.TextField(default=None, blank=True, null=True)
|
|
|
|
values = ArrayField(models.TextField())
|
|
|
|
|
2019-10-07 15:33:48 +01:00
|
|
|
form = 'passbook.providers.saml.forms.SAMLPropertyMappingForm'
|
2019-03-08 11:47:50 +00:00
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return "SAML Property Mapping %s" % self.saml_name
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
|
|
|
verbose_name = _('SAML Property Mapping')
|
|
|
|
verbose_name_plural = _('SAML Property Mappings')
|
|
|
|
|
2018-12-09 22:06:14 +00:00
|
|
|
def get_provider_choices():
|
|
|
|
"""Return tuple of class_path, class name of all providers."""
|
|
|
|
return [(class_to_path(x), x.__name__) for x in Processor.__subclasses__()]
|