# -*- coding: UTF-8 -*-# Copyright (c) 2012 martin f. krafft# See COPYING for distribution information.""" VirtualMailManager.catchall ~~~~~~~~~~~~~~~~~~~~~~~~~~~ Virtual Mail Manager's CatchallAlias class to manage domain catch-all aliases. This is heavily based on (more or less a copy of) the Alias class, because fundamentally, catchall aliases are aliases, but without a localpart. While Alias could potentially derive from CatchallAlias to reuse some of the functionality, it's probably not worth it. I found no sensible way to derive CatchallAlias from Alias, or at least none that would harness the powers of polymorphism. Yet, we reuse the AliasError exception class, which makes sense."""fromVirtualMailManager.domainimportget_gidfromVirtualMailManager.emailaddressimport \EmailAddress,DestinationEmailAddressasDestAddrfromVirtualMailManager.errorsimportAliasErrorasAErrfromVirtualMailManager.ext.postconfimportPostconffromVirtualMailManager.constantsimport \ALIAS_EXCEEDS_EXPANSION_LIMIT,NO_SUCH_ALIAS,NO_SUCH_DOMAIN_=lambdamsg:msgcfg_dget=lambdaoption:NoneclassCatchallAlias(object):"""Class to manage domain catch-all aliases."""__slots__=('_domain','_dests','_gid','_dbh')def__init__(self,dbh,domain):self._domain=domainself._dbh=dbhself._gid=get_gid(self._dbh,self.domain)ifnotself._gid:raiseAErr(_("The domain '%s' does not exist.")%self.domain,NO_SUCH_DOMAIN)self._dests=[]self._load_dests()def_load_dests(self):"""Loads all known destination addresses into the _dests list."""dbc=self._dbh.cursor()dbc.execute('SELECT destination FROM catchall WHERE gid = %s',(self._gid,))dests=dbc.fetchall()ifdbc.rowcount>0:self._dests.extend(DestAddr(dest[0],self._dbh)fordestindests)dbc.close()def_check_expansion(self,count_new):"""Checks the current expansion limit of the alias."""postconf=Postconf(cfg_dget('bin.postconf'))limit=int(postconf.read('virtual_alias_expansion_limit'))dcount=len(self._dests)failed=Falseifdcount==limitordcount+count_new>limit:failed=Trueerrmsg=_("""Cannot add %(count_new)i new destination(s) to catch-all alias fordomain '%(domain)s'. Currently this alias expands into %(count)i/%(limit)irecipients. %(count_new)i additional destination(s) will render this aliasunusable.Hint: Increase Postfix' virtual_alias_expansion_limit""")elifdcount>limit:failed=Trueerrmsg=_("""Cannot add %(count_new)i new destination(s) to catch-all alias for domain'%(domain)s'. This alias already exceeds its expansion limit \(%(count)i/%(limit)i).So its unusable, all messages addressed to this alias will be bounced.Hint: Delete some destination addresses.""")iffailed:raiseAErr(errmsg%{'domain':self._domain,'count':dcount,'limit':limit,'count_new':count_new},ALIAS_EXCEEDS_EXPANSION_LIMIT)def_delete(self,destinations=None):"""Delete one ore multiple destinations from the catchall alias, if ``destinations`` is not ``None``. If ``destinations`` is None, the catchall alias with all its destination addresses will be deleted. """dbc=self._dbh.cursor()ifnotdestinations:dbc.execute('DELETE FROM catchall WHERE gid = %s',(self._gid,))else:dbc.executemany('DELETE FROM catchall WHERE gid = %d AND ''destination = %%s'%self._gid,((str(dest),)fordestindestinations))ifdbc.rowcount>0:self._dbh.commit()dbc.close()def__len__(self):"""Returns the number of destinations of the catchall alias."""returnlen(self._dests)@propertydefdomain(self):"""The Alias' domain."""returnself._domaindefadd_destinations(self,destinations,warnings=None):"""Adds the `EmailAddress`es from *destinations* list to the destinations of the catchall alias. Destinations, that are already assigned to the alias, will be removed from *destinations*. When done, this method will return a set with all destinations, that were saved in the database. """destinations=set(destinations)assertdestinationsand \all(isinstance(dest,EmailAddress)fordestindestinations)ifnotwarningsisNone:assertisinstance(warnings,list)duplicates=destinations.intersection(set(self._dests))ifduplicates:destinations.difference_update(set(self._dests))ifnotwarningsisNone:warnings.extend(duplicates)ifnotdestinations:returndestinationsself._check_expansion(len(destinations))dbc=self._dbh.cursor()dbc.executemany("INSERT INTO catchall (gid, destination) ""VALUES (%d, %%s)"%self._gid,((str(destination),)fordestinationindestinations))self._dbh.commit()dbc.close()self._dests.extend(destinations)returndestinationsdefdel_destinations(self,destinations,warnings=None):"""Deletes the specified ``destinations`` from the catchall alias."""destinations=set(destinations)assertdestinationsand \all(isinstance(dest,EmailAddress)fordestindestinations)ifnotwarningsisNone:assertisinstance(warnings,list)ifnotself._dests:raiseAErr(_("There are no catch-all aliases defined for ""domain '%s'.")%self._domain,NO_SUCH_ALIAS)unknown=destinations.difference(set(self._dests))ifunknown:destinations.intersection_update(set(self._dests))ifnotwarningsisNone:warnings.extend(unknown)ifnotdestinations:raiseAErr(_("No suitable destinations left to remove from the ""catch-all alias of domain '%s'.")%self._domain,NO_SUCH_ALIAS)self._delete(destinations)fordestinationindestinations:self._dests.remove(destination)defget_destinations(self):"""Returns an iterator for all destinations of the catchall alias."""ifnotself._dests:raiseAErr(_("There are no catch-all aliases defined for ""domain '%s'.")%self._domain,NO_SUCH_ALIAS)returniter(self._dests)defdelete(self):"""Deletes all catchall destinations for the domain."""ifnotself._dests:raiseAErr(_("There are no catch-all aliases defined for ""domain '%s'.")%self._domain,NO_SUCH_ALIAS)self._delete()delself._dests[:]del_,cfg_dget