VMM: Dropped support for Dovecot versions before v1.2.0.
# -*- coding: UTF-8 -*-# Copyright (c) 2007 - 2013, Pascal Volk# See COPYING for distribution information.""" VirtualMailManager.domain ~~~~~~~~~~~~~~~~~~~~~~~~~ Virtual Mail Manager's Domain class to manage e-mail domains."""importosimportrefromrandomimportchoicefromVirtualMailManager.constantsimport \ACCOUNT_AND_ALIAS_PRESENT,DOMAIN_ALIAS_EXISTS,DOMAIN_EXISTS, \DOMAIN_INVALID,DOMAIN_TOO_LONG,NO_SUCH_DOMAIN,VMM_ERRORfromVirtualMailManager.commonimportvalidate_transportfromVirtualMailManager.errorsimportVMMError,DomainErrorasDomErrfromVirtualMailManager.maillocationimportMailLocationfromVirtualMailManager.quotalimitimportQuotaLimitfromVirtualMailManager.servicesetimportServiceSetfromVirtualMailManager.transportimportTransportMAILDIR_CHARS='0123456789abcdefghijklmnopqrstuvwxyz'RE_DOMAIN=re.compile(r"^(?:[a-z0-9-]{1,63}\.){1,}[a-z]{2,6}$")_=lambdamsg:msgcfg_dget=lambdaoption:NoneclassDomain(object):"""Class to manage e-mail domains."""__slots__=('_directory','_gid','_name','_qlimit','_services','_transport','_note','_dbh','_new')def__init__(self,dbh,domainname):"""Creates a new Domain instance. Loads all relevant data from the database, if the domain could be found. To create a new domain call the methods set_directory() and set_transport() before save(). A DomainError will be thrown when the *domainname* is the name of an alias domain. Arguments: `dbh` : pyPgSQL.PgSQL.Connection a database connection for the database access `domainname` : basestring The name of the domain """self._name=check_domainname(domainname)self._dbh=dbhself._gid=0self._qlimit=Noneself._services=Noneself._transport=Noneself._directory=Noneself._note=Noneself._new=Trueself._load()def_load(self):"""Load information from the database and checks if the domain name is the primary one. Raises a DomainError if Domain._name isn't the primary name of the domain. """dbc=self._dbh.cursor()dbc.execute('SELECT dd.gid, qid, ssid, tid, domaindir, is_primary, ''note ''FROM domain_data dd, domain_name dn WHERE domainname = ''%s AND dn.gid = dd.gid',(self._name,))result=dbc.fetchone()dbc.close()ifresult:ifnotresult[5]:raiseDomErr(_("The domain '%s' is an alias domain.")%self._name,DOMAIN_ALIAS_EXISTS)self._gid,self._directory=result[0],result[4]self._qlimit=QuotaLimit(self._dbh,qid=result[1])self._services=ServiceSet(self._dbh,ssid=result[2])self._transport=Transport(self._dbh,tid=result[3])self._note=result[6]self._new=Falsedef_set_gid(self):"""Sets the ID of the domain - if not set yet."""assertself._gid==0dbc=self._dbh.cursor()dbc.execute("SELECT nextval('domain_gid')")self._gid=dbc.fetchone()[0]dbc.close()def_check_for_addresses(self):"""Checks dependencies for deletion. Raises a DomainError if there are accounts, aliases and/or relocated users. """dbc=self._dbh.cursor()dbc.execute('SELECT ''(SELECT count(gid) FROM users WHERE gid = %(gid)u)'' as account_count, ''(SELECT count(gid) FROM alias WHERE gid = %(gid)u)'' as alias_count, ''(SELECT count(gid) FROM relocated WHERE gid = %(gid)u)'' as relocated_count'%{'gid':self._gid})result=dbc.fetchall()dbc.close()result=result[0]ifany(result):keys=('account_count','alias_count','relocated_count')raiseDomErr(_('There are %(account_count)u accounts, ''%(alias_count)u aliases and %(relocated_count)u ''relocated users.')%dict(list(zip(keys,result))),ACCOUNT_AND_ALIAS_PRESENT)def_chk_state(self,must_exist=True):"""Checks the state of the Domain instance and will raise a VirtualMailManager.errors.DomainError: - if *must_exist* is `True` and the domain doesn't exist - or *must_exist* is `False` and the domain exists """ifmust_existandself._new:raiseDomErr(_("The domain '%s' does not exist.")%self._name,NO_SUCH_DOMAIN)elifnotmust_existandnotself._new:raiseDomErr(_("The domain '%s' already exists.")%self._name,DOMAIN_EXISTS)def_update_tables(self,column,value):"""Update table columns in the domain_data table."""dbc=self._dbh.cursor()dbc.execute('UPDATE domain_data SET %s = %%s WHERE gid = %%s'%column,(value,self._gid))ifdbc.rowcount>0:self._dbh.commit()dbc.close()def_update_tables_ref(self,column,value,force=False):"""Update various columns in the domain_data table. When *force* is `True`, the corresponding column in the users table will be reset to NULL. Arguments: `column` : basestring Name of the table column. Currently: qid, ssid and tid `value` : int The referenced key `force` : bool reset existing users. Default: `False` """ifcolumnnotin('qid','ssid','tid'):raiseValueError('Unknown column: %r'%column)self._update_tables(column,value)ifforce:dbc=self._dbh.cursor()dbc.execute('UPDATE users SET %s = NULL WHERE gid = %%s'%column,(self._gid,))ifdbc.rowcount>0:self._dbh.commit()dbc.close()@propertydefgid(self):"""The GID of the Domain."""returnself._gid@propertydefname(self):"""The Domain's name."""returnself._name@propertydefdirectory(self):"""The Domain's directory."""returnself._directory@propertydefquotalimit(self):"""The Domain's quota limit."""returnself._qlimit@propertydefserviceset(self):"""The Domain's serviceset."""returnself._services@propertydeftransport(self):"""The Domain's transport."""returnself._transport@propertydefnote(self):"""The Domain's note."""returnself._notedefset_directory(self,basedir):"""Set the path value of the Domain's directory, inside *basedir*. Argument: `basedir` : basestring The base directory of all domains """self._chk_state(False)assertself._directoryisNoneself._set_gid()self._directory=os.path.join(basedir,choice(MAILDIR_CHARS),str(self._gid))defset_quotalimit(self,quotalimit):"""Set the quota limit for the new Domain. Argument: `quotalimit` : VirtualMailManager.quotalimit.QuotaLimit The quota limit of the new Domain. """self._chk_state(False)assertisinstance(quotalimit,QuotaLimit)self._qlimit=quotalimitdefset_serviceset(self,serviceset):"""Set the services for the new Domain. Argument: `serviceset` : VirtualMailManager.serviceset.ServiceSet The service set for the new Domain. """self._chk_state(False)assertisinstance(serviceset,ServiceSet)self._services=servicesetdefset_transport(self,transport):"""Set the transport for the new Domain. Argument: `transport` : VirtualMailManager.Transport The transport of the new Domain """self._chk_state(False)assertisinstance(transport,Transport)validate_transport(transport,MailLocation(self._dbh,mbfmt=cfg_dget('mailbox.format'),directory=cfg_dget('mailbox.root')))self._transport=transportdefset_note(self,note):"""Set the domain's (optional) note. Argument: `note` : basestring or None The note, or None to remove """self._chk_state(False)assertnoteisNoneorisinstance(note,str)self._note=notedefsave(self):"""Stores the new domain in the database."""self._chk_state(False)assertall((self._directory,self._qlimit,self._services,self._transport))dbc=self._dbh.cursor()dbc.execute('INSERT INTO domain_data (gid, qid, ssid, tid, domaindir, ''note) ''VALUES (%s, %s, %s, %s, %s, %s)',(self._gid,self._qlimit.qid,self._services.ssid,self._transport.tid,self._directory,self._note))dbc.execute('INSERT INTO domain_name (domainname, gid, is_primary) ''VALUES (%s, %s, TRUE)',(self._name,self._gid))self._dbh.commit()dbc.close()self._new=Falsedefdelete(self,force=False):"""Deletes the domain. Arguments: `force` : bool force the deletion of all available accounts, aliases and relocated users. When *force* is `False` and there are accounts, aliases and/or relocated users a DomainError will be raised. Default `False` """ifnotisinstance(force,bool):raiseTypeError('force must be a bool')self._chk_state()ifnotforce:self._check_for_addresses()dbc=self._dbh.cursor()fortblin('alias','users','relocated','domain_name','domain_data'):dbc.execute("DELETE FROM %s WHERE gid = %u"%(tbl,self._gid))self._dbh.commit()dbc.close()self._gid=0self._directory=self._qlimit=self._transport=Noneself._services=Noneself._new=Truedefupdate_quotalimit(self,quotalimit,force=False):"""Update the quota limit of the Domain. If *force* is `True`, accounts-specific overrides will be reset for all existing accounts of the domain. Otherwise, the limit will only affect accounts that use the default. Arguments: `quotalimit` : VirtualMailManager.quotalimit.QuotaLimit the new quota limit of the domain. `force` : bool enforce new quota limit for all accounts, default `False` """self._chk_state()assertisinstance(quotalimit,QuotaLimit)ifnotforceandquotalimit==self._qlimit:returnself._update_tables_ref('qid',quotalimit.qid,force)self._qlimit=quotalimitdefupdate_serviceset(self,serviceset,force=False):"""Assign a different set of services to the Domain, If *force* is `True`, accounts-specific overrides will be reset for all existing accounts of the domain. Otherwise, the service set will only affect accounts that use the default. Arguments: `serviceset` : VirtualMailManager.serviceset.ServiceSet the new set of services `force` enforce the serviceset for all accounts, default `False` """self._chk_state()assertisinstance(serviceset,ServiceSet)ifnotforceandserviceset==self._services:returnself._update_tables_ref('ssid',serviceset.ssid,force)self._services=servicesetdefupdate_transport(self,transport,force=False):"""Sets a new transport for the Domain. If *force* is `True`, accounts-specific overrides will be reset for all existing accounts of the domain. Otherwise, the transport setting will only affect accounts that use the default. Arguments: `transport` : VirtualMailManager.Transport the new transport `force` : bool enforce new transport setting for all accounts, default `False` """self._chk_state()assertisinstance(transport,Transport)ifnotforceandtransport==self._transport:returnvalidate_transport(transport,MailLocation(self._dbh,mbfmt=cfg_dget('mailbox.format'),directory=cfg_dget('mailbox.root')))self._update_tables_ref('tid',transport.tid,force)self._transport=transportdefupdate_note(self,note):"""Sets a new note for the Domain. Arguments: `transport` : basestring or None the new note """self._chk_state()assertnoteisNoneorisinstance(note,str)ifnote==self._note:returnself._update_tables('note',note)self._note=notedefget_info(self):"""Returns a dictionary with information about the domain."""self._chk_state()dbc=self._dbh.cursor()dbc.execute('SELECT aliasdomains "alias domains", accounts, aliases, ''relocated, catchall "catch-all dests" ''FROM vmm_domain_info WHERE gid = %s',(self._gid,))info=dbc.fetchone()dbc.close()keys=('alias domains','accounts','aliases','relocated','catch-all dests')info=dict(list(zip(keys,info)))info['gid']=self._gidinfo['domain name']=self._nameinfo['transport']=self._transport.transportinfo['domain directory']=self._directoryinfo['bytes']=self._qlimit.bytesinfo['messages']=self._qlimit.messagesservices=self._services.servicesservices=[s.upper()forsinservicesifservices[s]]ifservices:services.sort()else:services.append('None')info['active services']=' '.join(services)info['note']=self._notereturninfodefget_accounts(self):"""Returns a list with all accounts of the domain."""self._chk_state()dbc=self._dbh.cursor()dbc.execute('SELECT local_part from users where gid = %s ORDER BY ''local_part',(self._gid,))users=dbc.fetchall()dbc.close()accounts=[]ifusers:addr='@'.join_dom=self._nameaccounts=[addr((account[0],_dom))foraccountinusers]returnaccountsdefget_aliases(self):"""Returns a list with all aliases e-mail addresses of the domain."""self._chk_state()dbc=self._dbh.cursor()dbc.execute('SELECT DISTINCT address FROM alias WHERE gid = %s ORDER ''BY address',(self._gid,))addresses=dbc.fetchall()dbc.close()aliases=[]ifaddresses:addr='@'.join_dom=self._namealiases=[addr((alias[0],_dom))foraliasinaddresses]returnaliasesdefget_relocated(self):"""Returns a list with all addresses of relocated users."""self._chk_state()dbc=self._dbh.cursor()dbc.execute('SELECT address FROM relocated WHERE gid = %s ORDER BY ''address',(self._gid,))addresses=dbc.fetchall()dbc.close()relocated=[]ifaddresses:addr='@'.join_dom=self._namerelocated=[addr((address[0],_dom))foraddressinaddresses]returnrelocateddefget_catchall(self):"""Returns a list with all catchall e-mail addresses of the domain."""self._chk_state()dbc=self._dbh.cursor()dbc.execute('SELECT DISTINCT destination FROM catchall WHERE gid = %s ''ORDER BY destination',(self._gid,))addresses=dbc.fetchall()dbc.close()returnaddressesdefget_aliase_names(self):"""Returns a list with all alias domain names of the domain."""self._chk_state()dbc=self._dbh.cursor()dbc.execute('SELECT domainname FROM domain_name WHERE gid = %s AND ''NOT is_primary ORDER BY domainname',(self._gid,))anames=dbc.fetchall()dbc.close()aliasdomains=[]ifanames:aliasdomains=[aname[0]foranameinanames]returnaliasdomainsdefcheck_domainname(domainname):"""Returns the validated domain name `domainname`. Throws an `DomainError`, if the domain name is too long or doesn't look like a valid domain name (label.label.label). """ifnotRE_DOMAIN.match(domainname):domainname=domainname.encode('idna').decode()iflen(domainname)>255:raiseDomErr(_('The domain name is too long'),DOMAIN_TOO_LONG)ifnotRE_DOMAIN.match(domainname):raiseDomErr(_("The domain name '%s' is invalid")%domainname,DOMAIN_INVALID)returndomainnamedefget_gid(dbh,domainname):"""Returns the group id of the domain *domainname*. If the domain couldn't be found in the database 0 will be returned. """domainname=check_domainname(domainname)dbc=dbh.cursor()dbc.execute('SELECT gid FROM domain_name WHERE domainname = %s',(domainname,))gid=dbc.fetchone()dbc.close()ifgid:returngid[0]return0defsearch(dbh,pattern=None,like=False):"""'Search' for domains by *pattern* in the database. *pattern* may be a domain name or a partial domain name - starting and/or ending with a '%' sign. When the *pattern* starts or ends with a '%' sign *like* has to be `True` to perform a wildcard search. To retrieve all available domains use the arguments' default values. This function returns a tuple with a list and a dict: (order, domains). The order list contains the domains' gid, alphabetical sorted by the primary domain name. The domains dict's keys are the gids of the domains. The value of item is a list. The first list element contains the primary domain name or `None`. The elements [1:] contains the names of alias domains. Arguments: `pattern` : basestring a (partial) domain name (starting and/or ending with a "%" sign) `like` : bool should be `True` when *pattern* starts/ends with a "%" sign """ifpatternandnotlike:pattern=check_domainname(pattern)sql='SELECT gid, domainname, is_primary FROM domain_name'ifpattern:iflike:sql+=" WHERE domainname LIKE '%s'"%patternelse:sql+=" WHERE domainname = '%s'"%patternsql+=' ORDER BY is_primary DESC, domainname'dbc=dbh.cursor()dbc.execute(sql)result=dbc.fetchall()dbc.close()gids=[domain[0]fordomaininresultifdomain[2]]domains={}forgid,domain,is_primaryinresult:ifis_primary:ifnotgidindomains:domains[gid]=[domain]else:domains[gid].insert(0,domain)else:ifgidingids:ifgidindomains:domains[gid].append(domain)else:domains[gid]=[domain]else:gids.append(gid)domains[gid]=[None,domain]returngids,domainsdel_,cfg_dget