mirror of
https://github.com/Mailu/Mailu.git
synced 2024-12-14 10:53:30 +02:00
added dump option to dump dns data of domains
This commit is contained in:
parent
2a5c46c890
commit
adc9c70c3e
@ -301,11 +301,13 @@ def config_update(verbose=False, delete_objects=False, dry_run=False, file=None)
|
|||||||
@mailu.command()
|
@mailu.command()
|
||||||
@click.option('-f', '--full', is_flag=True, help='Include default attributes')
|
@click.option('-f', '--full', is_flag=True, help='Include default attributes')
|
||||||
@click.option('-s', '--secrets', is_flag=True, help='Include secrets (dkim-key, plain-text / not hashed)')
|
@click.option('-s', '--secrets', is_flag=True, help='Include secrets (dkim-key, plain-text / not hashed)')
|
||||||
|
@click.option('-d', '--dns', is_flag=True, help='Include dns records')
|
||||||
@click.argument('sections', nargs=-1)
|
@click.argument('sections', nargs=-1)
|
||||||
@flask_cli.with_appcontext
|
@flask_cli.with_appcontext
|
||||||
def config_dump(full=False, secrets=False, sections=None):
|
def config_dump(full=False, secrets=False, dns=False, sections=None):
|
||||||
"""dump configuration as YAML-formatted data to stdout
|
"""dump configuration as YAML-formatted data to stdout
|
||||||
valid SECTIONS are: domains, relays, users, aliases
|
|
||||||
|
SECTIONS can be: domains, relays, users, aliases
|
||||||
"""
|
"""
|
||||||
|
|
||||||
class spacedDumper(yaml.Dumper):
|
class spacedDumper(yaml.Dumper):
|
||||||
@ -325,10 +327,14 @@ def config_dump(full=False, secrets=False, sections=None):
|
|||||||
print(f'[ERROR] Invalid section: {section}')
|
print(f'[ERROR] Invalid section: {section}')
|
||||||
return 1
|
return 1
|
||||||
|
|
||||||
|
extra = []
|
||||||
|
if dns:
|
||||||
|
extra.append('dns')
|
||||||
|
|
||||||
config = {}
|
config = {}
|
||||||
for section, model in yaml_sections:
|
for section, model in yaml_sections:
|
||||||
if not sections or section in sections:
|
if not sections or section in sections:
|
||||||
dump = [item.to_dict(full, secrets) for item in model.query.all()]
|
dump = [item.to_dict(full, secrets, extra) for item in model.query.all()]
|
||||||
if len(dump):
|
if len(dump):
|
||||||
config[section] = dump
|
config[section] = dump
|
||||||
|
|
||||||
|
@ -119,7 +119,7 @@ class Base(db.Model):
|
|||||||
def _dict_pval(self):
|
def _dict_pval(self):
|
||||||
return getattr(self, self._dict_pkey())
|
return getattr(self, self._dict_pkey())
|
||||||
|
|
||||||
def to_dict(self, full=False, include_secrets=False, recursed=False, hide=None):
|
def to_dict(self, full=False, include_secrets=False, include_extra=None, recursed=False, hide=None):
|
||||||
""" Return a dictionary representation of this model.
|
""" Return a dictionary representation of this model.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@ -136,9 +136,17 @@ class Base(db.Model):
|
|||||||
|
|
||||||
convert = getattr(self, '_dict_output', {})
|
convert = getattr(self, '_dict_output', {})
|
||||||
|
|
||||||
|
extra_keys = getattr(self, '_dict_extra', {})
|
||||||
|
if include_extra is None:
|
||||||
|
include_extra = []
|
||||||
|
|
||||||
res = {}
|
res = {}
|
||||||
|
|
||||||
for key in itertools.chain(self.__table__.columns.keys(), getattr(self, '_dict_show', [])):
|
for key in itertools.chain(
|
||||||
|
self.__table__.columns.keys(),
|
||||||
|
getattr(self, '_dict_show', []),
|
||||||
|
*[extra_keys.get(extra, []) for extra in include_extra]
|
||||||
|
):
|
||||||
if key in hide:
|
if key in hide:
|
||||||
continue
|
continue
|
||||||
if key in self.__table__.columns:
|
if key in self.__table__.columns:
|
||||||
@ -167,14 +175,14 @@ class Base(db.Model):
|
|||||||
if key in secret:
|
if key in secret:
|
||||||
res[key] = '<hidden>'
|
res[key] = '<hidden>'
|
||||||
else:
|
else:
|
||||||
res[key] = [item.to_dict(full, include_secrets, True) for item in items]
|
res[key] = [item.to_dict(full, include_secrets, include_extra, True) for item in items]
|
||||||
else:
|
else:
|
||||||
value = getattr(self, key)
|
value = getattr(self, key)
|
||||||
if full or value is not None:
|
if full or value is not None:
|
||||||
if key in secret:
|
if key in secret:
|
||||||
res[key] = '<hidden>'
|
res[key] = '<hidden>'
|
||||||
else:
|
else:
|
||||||
res[key] = value.to_dict(full, include_secrets, True)
|
res[key] = value.to_dict(full, include_secrets, include_extra, True)
|
||||||
|
|
||||||
return res
|
return res
|
||||||
|
|
||||||
@ -219,7 +227,7 @@ class Base(db.Model):
|
|||||||
if rel is None:
|
if rel is None:
|
||||||
itype = getattr(model, '_dict_types', {}).get(key)
|
itype = getattr(model, '_dict_types', {}).get(key)
|
||||||
if itype is not None:
|
if itype is not None:
|
||||||
if not itype: # empty tuple => ignore value
|
if itype is False: # ignore value
|
||||||
del data[key]
|
del data[key]
|
||||||
continue
|
continue
|
||||||
elif not isinstance(value, itype):
|
elif not isinstance(value, itype):
|
||||||
@ -291,11 +299,11 @@ class Base(db.Model):
|
|||||||
|
|
||||||
# delete referenced items missing in yaml
|
# delete referenced items missing in yaml
|
||||||
rel_pkey = rel_model._dict_pkey()
|
rel_pkey = rel_model._dict_pkey()
|
||||||
new_data = list([i.to_dict(True, True, True, [rel_pkey]) for i in new])
|
new_data = list([i.to_dict(True, True, None, True, [rel_pkey]) for i in new])
|
||||||
for rel_item in old:
|
for rel_item in old:
|
||||||
if rel_item not in new:
|
if rel_item not in new:
|
||||||
# check if item with same data exists to stabilze import without primary key
|
# check if item with same data exists to stabilze import without primary key
|
||||||
rel_data = rel_item.to_dict(True, True, True, [rel_pkey])
|
rel_data = rel_item.to_dict(True, True, None, True, [rel_pkey])
|
||||||
try:
|
try:
|
||||||
same_idx = new_data.index(rel_data)
|
same_idx = new_data.index(rel_data)
|
||||||
except ValueError:
|
except ValueError:
|
||||||
@ -367,10 +375,18 @@ class Domain(Base):
|
|||||||
__tablename__ = "domain"
|
__tablename__ = "domain"
|
||||||
|
|
||||||
_dict_hide = {'users', 'managers', 'aliases'}
|
_dict_hide = {'users', 'managers', 'aliases'}
|
||||||
_dict_show = {'dkim_key', 'dkim_publickey'}
|
_dict_show = {'dkim_key'}
|
||||||
|
_dict_extra = {'dns':{'dkim_publickey', 'dns_mx', 'dns_spf', 'dns_dkim', 'dns_dmarc'}}
|
||||||
_dict_secret = {'dkim_key'}
|
_dict_secret = {'dkim_key'}
|
||||||
_dict_types = {'dkim_key': (bytes, type(None)), 'dkim_publickey': tuple()}
|
_dict_types = {
|
||||||
_dict_output = {'dkim_key': lambda v: v.decode('utf-8').strip().split('\n')[1:-1]}
|
'dkim_key': (bytes, type(None)),
|
||||||
|
'dkim_publickey': False,
|
||||||
|
'dns_mx': False,
|
||||||
|
'dns_spf': False,
|
||||||
|
'dns_dkim': False,
|
||||||
|
'dns_dmarc': False,
|
||||||
|
}
|
||||||
|
_dict_output = {'dkim_key': lambda key: key.decode('utf-8').strip().split('\n')[1:-1]}
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def _dict_input(data):
|
def _dict_input(data):
|
||||||
if 'dkim_key' in data:
|
if 'dkim_key' in data:
|
||||||
@ -408,18 +424,46 @@ class Domain(Base):
|
|||||||
max_quota_bytes = db.Column(db.BigInteger(), nullable=False, default=0)
|
max_quota_bytes = db.Column(db.BigInteger(), nullable=False, default=0)
|
||||||
signup_enabled = db.Column(db.Boolean(), nullable=False, default=False)
|
signup_enabled = db.Column(db.Boolean(), nullable=False, default=False)
|
||||||
|
|
||||||
|
def _dkim_file(self):
|
||||||
|
return app.config["DKIM_PATH"].format(
|
||||||
|
domain=self.name, selector=app.config["DKIM_SELECTOR"])
|
||||||
|
|
||||||
|
@property
|
||||||
|
def dns_mx(self):
|
||||||
|
hostname = app.config['HOSTNAMES'].split(',')[0]
|
||||||
|
return f'{self.name}. 600 IN MX 10 {hostname}.'
|
||||||
|
|
||||||
|
@property
|
||||||
|
def dns_spf(self):
|
||||||
|
hostname = app.config['HOSTNAMES'].split(',')[0]
|
||||||
|
return f'{self.name}. 600 IN TXT "v=spf1 mx a:{hostname} ~all"'
|
||||||
|
|
||||||
|
@property
|
||||||
|
def dns_dkim(self):
|
||||||
|
if os.path.exists(self._dkim_file()):
|
||||||
|
selector = app.config['DKIM_SELECTOR']
|
||||||
|
return f'{selector}._domainkey.{self.name}. 600 IN TXT "v=DKIM1; k=rsa; p={self.dkim_publickey}"'
|
||||||
|
|
||||||
|
@property
|
||||||
|
def dns_dmarc(self):
|
||||||
|
if os.path.exists(self._dkim_file()):
|
||||||
|
domain = app.config['DOMAIN']
|
||||||
|
rua = app.config['DMARC_RUA']
|
||||||
|
rua = f' rua=mailto:{rua}@{domain};' if rua else ''
|
||||||
|
ruf = app.config['DMARC_RUF']
|
||||||
|
ruf = f' ruf=mailto:{ruf}@{domain};' if ruf else ''
|
||||||
|
return f'_dmarc.{self.name}. 600 IN TXT "v=DMARC1; p=reject;{rua}{ruf} adkim=s; aspf=s"'
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def dkim_key(self):
|
def dkim_key(self):
|
||||||
file_path = app.config["DKIM_PATH"].format(
|
file_path = self._dkim_file()
|
||||||
domain=self.name, selector=app.config["DKIM_SELECTOR"])
|
|
||||||
if os.path.exists(file_path):
|
if os.path.exists(file_path):
|
||||||
with open(file_path, "rb") as handle:
|
with open(file_path, "rb") as handle:
|
||||||
return handle.read()
|
return handle.read()
|
||||||
|
|
||||||
@dkim_key.setter
|
@dkim_key.setter
|
||||||
def dkim_key(self, value):
|
def dkim_key(self, value):
|
||||||
file_path = app.config["DKIM_PATH"].format(
|
file_path = self._dkim_file()
|
||||||
domain=self.name, selector=app.config["DKIM_SELECTOR"])
|
|
||||||
if value is None:
|
if value is None:
|
||||||
if os.path.exists(file_path):
|
if os.path.exists(file_path):
|
||||||
os.unlink(file_path)
|
os.unlink(file_path)
|
||||||
|
Loading…
Reference in New Issue
Block a user