import os from django.contrib.auth.hashers import make_password from django.core import validators from django.core.mail import send_mail from django.db import models from django.utils.functional import cached_property from django.utils.translation import ugettext_lazy as _ from orchestra.core import services from . import settings class SystemUserQuerySet(models.QuerySet): def create_user(self, username, password='', **kwargs): user = super(SystemUserQuerySet, self).create(username=username, **kwargs) user.set_password(password) user.save(update_fields=['password']) return user class SystemUser(models.Model): """ System users """ username = models.CharField(_("username"), max_length=64, unique=True, help_text=_("Required. 30 characters or fewer. Letters, digits and ./-/_ only."), validators=[validators.RegexValidator(r'^[\w.-]+$', _("Enter a valid username."), 'invalid')]) password = models.CharField(_("password"), max_length=128) account = models.ForeignKey('accounts.Account', verbose_name=_("Account"), related_name='systemusers') home = models.CharField(_("home"), max_length=256, blank=True, help_text=_("Home directory relative to account's ~main_user")) shell = models.CharField(_("shell"), max_length=32, choices=settings.SYSTEMUSERS_SHELLS, default=settings.SYSTEMUSERS_DEFAULT_SHELL) groups = models.ManyToManyField('self', blank=True, help_text=_("A new group will be created for the user. " "Which additional groups would you like them to be a member of?")) is_main = models.BooleanField(_("is main"), default=False) is_active = models.BooleanField(_("active"), default=True, help_text=_("Designates whether this account should be treated as active. " "Unselect this instead of deleting accounts.")) objects = SystemUserQuerySet.as_manager() def __unicode__(self): return self.username @cached_property def active(self): a = type(self).account.field.model try: return self.is_active and self.account.is_active except type(self).account.field.rel.to.DoesNotExist: return self.is_active def set_password(self, raw_password): self.password = make_password(raw_password) def get_home(self): if self.is_main: context = { 'username': self.username, } basehome = settings.SYSTEMUSERS_HOME % context else: basehome = self.account.systemusers.get(is_main=True).get_home() basehome = basehome.replace('/./', '/') home = os.path.join(basehome, self.home) # Chrooting home = home.split('/') home.insert(-2, '.') return '/'.join(home) ## TODO user deletion and group handling. #class SystemGroup(models.Model): # name = models.CharField(_("name"), max_length=64, unique=True, # help_text=_("Required. 30 characters or fewer. Letters, digits and ./-/_ only."), # validators=[validators.RegexValidator(r'^[\w.-]+$', # _("Enter a valid group name."), 'invalid')]) # account = models.ForeignKey('accounts.Account', verbose_name=_("Account"), # related_name='systemgroups') # # def __unicode__(self): # return self.name services.register(SystemUser)