Init
This commit is contained in:
parent
cc4ef9b16b
commit
eda43f1d3c
Binary file not shown.
Binary file not shown.
36
config.py
36
config.py
@ -29,6 +29,7 @@
|
||||
###
|
||||
|
||||
from supybot import conf, registry
|
||||
from supybot.commands import private
|
||||
try:
|
||||
from supybot.i18n import PluginInternationalization
|
||||
_ = PluginInternationalization('Mailcow')
|
||||
@ -52,5 +53,40 @@ Mailcow = conf.registerPlugin('Mailcow')
|
||||
# conf.registerGlobalValue(Mailcow, 'someConfigVariableName',
|
||||
# registry.Boolean(False, _("""Help for someConfigVariableName.""")))
|
||||
|
||||
conf.registerGroup(Mailcow, 'api')
|
||||
|
||||
conf.registerGlobalValue(Mailcow.api, 'key',
|
||||
registry.String('',
|
||||
"""
|
||||
Your Mailcow API Key
|
||||
"""
|
||||
, private=True
|
||||
))
|
||||
|
||||
conf.registerGlobalValue(Mailcow.api, 'server',
|
||||
registry.String('',
|
||||
"""
|
||||
Your Mailcow server \(https://example.com\)
|
||||
"""
|
||||
, private=False
|
||||
))
|
||||
|
||||
conf.registerGroup(Mailcow, 'access')
|
||||
|
||||
conf.registerGlobalValue(Mailcow.access, 'read',
|
||||
registry.CommaSeparatedListOfStrings('',
|
||||
"""
|
||||
Nicknames to grant Read-Only access
|
||||
"""
|
||||
, private=True
|
||||
))
|
||||
|
||||
conf.registerGlobalValue(Mailcow.access, 'write',
|
||||
registry.CommaSeparatedListOfStrings('',
|
||||
"""
|
||||
Nicknames to grant Write access
|
||||
"""
|
||||
, private=True
|
||||
))
|
||||
|
||||
# vim:set shiftwidth=4 tabstop=4 expandtab textwidth=79:
|
||||
|
315
plugin.py
315
plugin.py
@ -33,6 +33,7 @@ import secrets
|
||||
import string
|
||||
from supybot import utils, plugins, ircutils, callbacks, ircdb
|
||||
from supybot.commands import *
|
||||
from supybot.ircmsgs import nick
|
||||
try:
|
||||
from supybot.i18n import PluginInternationalization
|
||||
_ = PluginInternationalization('Mailcow')
|
||||
@ -41,149 +42,231 @@ except ImportError:
|
||||
# without the i18n module
|
||||
_ = lambda x: x
|
||||
|
||||
server = 'https://zz0.email'
|
||||
api = '/api/v1'
|
||||
get = api + '/get'
|
||||
api_key = ""
|
||||
|
||||
capability = ircdb.checkCapability(hostmask='cranberry!~u@cranberry.juice', capability='owner', ignoreDefaultAllow=True)
|
||||
|
||||
class Mailcow(callbacks.Plugin):
|
||||
"""Mailcow API through IRC"""
|
||||
pass
|
||||
|
||||
|
||||
def maildomain(self, irc, msg, args, variant, id):
|
||||
"""<variant> <id>
|
||||
i.e. 'get liberta.casa' will print infos about the respective MX zone"""
|
||||
"""<option> <id>
|
||||
i.e. 'summary liberta.casa' will print infos about the respective zone"""
|
||||
|
||||
server = self.registryValue('api.server')
|
||||
api_key = self.registryValue('api.key')
|
||||
read = self.registryValue('access.read')
|
||||
write = self.registryValue('access.write')
|
||||
nick = msg.nick
|
||||
hostmask = irc.state.nickToHostmask(msg.nick)
|
||||
|
||||
#Read-Functions: Summary
|
||||
|
||||
if 'summary' in variant:
|
||||
URL = server + get + '/domain/' + id
|
||||
response = requests.get(
|
||||
URL,
|
||||
headers = {'accept': 'application/json', 'X-API-Key': api_key},
|
||||
)
|
||||
data = response.json()
|
||||
domain = data['domain_name']
|
||||
description = data['description']
|
||||
bytes_total = data['bytes_total']
|
||||
irc.reply(f"Domain: {domain} | Description: {description} | Bytes Total: {bytes_total} " + str(capability))
|
||||
if hostmask in read or write:
|
||||
URL = server + get + '/domain/' + id
|
||||
response = requests.get(
|
||||
URL,
|
||||
headers = {'accept': 'application/json', 'X-API-Key': api_key},
|
||||
)
|
||||
data = response.json()
|
||||
domain = data['domain_name']
|
||||
description = data['description']
|
||||
bytes_total = data['bytes_total']
|
||||
irc.reply(f"Domain: {domain} | Description: {description} | Bytes Total: {bytes_total} ")
|
||||
else:
|
||||
irc.reply("Thou shall not pass.")
|
||||
print("Intrusion attempt: " + hostmask)
|
||||
|
||||
#Write-Functions: Create/Delete
|
||||
elif 'create' in variant:
|
||||
URL = server + api + '/add/domain'
|
||||
#domain = id['domain']
|
||||
payload = {
|
||||
"active": "1",
|
||||
"aliases": "20",
|
||||
"backupmx": "0",
|
||||
"defquota": "1024",
|
||||
"description": id,
|
||||
"domain": id,
|
||||
"mailboxes": "10",
|
||||
"maxquota": "2048",
|
||||
"quota": "5120",
|
||||
"relay_all_recipients": "0",
|
||||
"rl_frame": "s",
|
||||
"rl_value": "10",
|
||||
"restart_sogo": "10"
|
||||
}
|
||||
response = requests.post(
|
||||
URL,
|
||||
headers = {'accept': 'application/json', 'X-API-Key': api_key, 'Content-Type': 'application/json'},
|
||||
json = payload,
|
||||
)
|
||||
data = response.json()
|
||||
print(data)
|
||||
status = data[0]['type']
|
||||
#object = data[0]['domain']
|
||||
#active = data[0]['active']
|
||||
msg = data[0]['msg']
|
||||
#max_aliases = data['aliases']
|
||||
#max_mailboxes = data['mailboxes']
|
||||
#default_quota = data['defquota']
|
||||
#max_mailbox_quota = data['maxquota']
|
||||
#max_domain_quota = data['quota']
|
||||
irc.reply(f"CREATION: {status} | {msg} | NOTE: SOGo is NOT being restarted automatically.")# Issue 'sogo restart' to make the object visible through Groupware. | Summary of object properties: MaxAliases: {aliases} | MaxMailboxes: {mailboxes} | DefaultQuota: {default_quota} | MaxMailBoxQuota: {max_mailbox_quota} | MaxDomainQuota: {max_domain_quota}")
|
||||
if hostmask in write:
|
||||
URL = server + api + '/add/domain'
|
||||
#domain = id['domain']
|
||||
payload = {
|
||||
"active": "1",
|
||||
"aliases": "20",
|
||||
"backupmx": "0",
|
||||
"defquota": "1024",
|
||||
"description": id,
|
||||
"domain": id,
|
||||
"mailboxes": "10",
|
||||
"maxquota": "2048",
|
||||
"quota": "5120",
|
||||
"relay_all_recipients": "0",
|
||||
"rl_frame": "s",
|
||||
"rl_value": "10",
|
||||
"restart_sogo": "10"
|
||||
}
|
||||
response = requests.post(
|
||||
URL,
|
||||
headers = {'accept': 'application/json', 'X-API-Key': api_key, 'Content-Type': 'application/json'},
|
||||
json = payload,
|
||||
)
|
||||
data = response.json()
|
||||
print(data)
|
||||
status = data[0]['type']
|
||||
#object = data[0]['domain']
|
||||
#active = data[0]['active']
|
||||
msg = data[0]['msg']
|
||||
#max_aliases = data['aliases']
|
||||
#max_mailboxes = data['mailboxes']
|
||||
#default_quota = data['defquota']
|
||||
#max_mailbox_quota = data['maxquota']
|
||||
#max_domain_quota = data['quota']
|
||||
irc.reply(f"CREATION: {status} | {msg} | NOTE: SOGo is NOT being restarted automatically.")# Issue 'sogo restart' to make the object visible through Groupware. | Summary of object properties: MaxAliases: {aliases} | MaxMailboxes: {mailboxes} | DefaultQuota: {default_quota} | MaxMailBoxQuota: {max_mailbox_quota} | MaxDomainQuota: {max_domain_quota}")
|
||||
else:
|
||||
irc.reply("Thou shall not create.")
|
||||
print("Intrusion attempt: " + hostmask)
|
||||
|
||||
elif 'delete' in variant:
|
||||
URL = server + api + '/delete/domain'
|
||||
payload = [
|
||||
id
|
||||
]
|
||||
response = requests.post(
|
||||
URL,
|
||||
headers = {'accept': 'application/json', 'X-API-Key': api_key, 'Content-Type': 'application/json'},
|
||||
json = payload,
|
||||
)
|
||||
data = response.json()
|
||||
status = data[0]['type']
|
||||
object = data[0]['msg']
|
||||
irc.reply(f"DELETION: {status} - {msg} - Hey, where's that backup again?")
|
||||
if hostmask in write:
|
||||
URL = server + api + '/delete/domain'
|
||||
payload = [
|
||||
id
|
||||
]
|
||||
response = requests.post(
|
||||
URL,
|
||||
headers = {'accept': 'application/json', 'X-API-Key': api_key, 'Content-Type': 'application/json'},
|
||||
json = payload,
|
||||
)
|
||||
data = response.json()
|
||||
status = data[0]['type']
|
||||
object = data[0]['msg']
|
||||
irc.reply(f"DELETION: {status} - {msg} - Hey, where's that backup again?")
|
||||
else:
|
||||
irc.reply("Thou shall not delete.")
|
||||
print("Intrusion attempt: " + hostmask)
|
||||
|
||||
else:
|
||||
irc.reply("Unknown option.")
|
||||
|
||||
|
||||
maildomain = wrap(maildomain, ['anything', 'anything'])
|
||||
|
||||
####################
|
||||
#FOR DEBUGGING / REMOVE BEFORE RELEASE
|
||||
####################
|
||||
def mcdebug(self, irc, msg, args, variant):
|
||||
"""Prints values."""
|
||||
|
||||
server = self.registryValue('api.server')
|
||||
api_key = self.registryValue('api.key')
|
||||
read = self.registryValue('access.read')
|
||||
write = self.registryValue('access.write')
|
||||
nick = msg.nick
|
||||
hostmask = irc.state.nickToHostmask(msg.nick)
|
||||
|
||||
|
||||
if hostmask in read or hostmask in write:
|
||||
|
||||
if 'key' in variant:
|
||||
irc.reply('> ' + api_key + ' <')
|
||||
|
||||
elif 'server' in variant:
|
||||
irc.reply('> ' + server + ' <')
|
||||
|
||||
elif 'user' in variant:
|
||||
u = ircdb.users.getUser(msg.prefix)
|
||||
nick = msg.nick
|
||||
hm = irc.state.nickToHostmask(nick)
|
||||
print(hm)
|
||||
irc.reply('> ' + u.name + ' ' + hm + ' <')
|
||||
|
||||
elif 'read' in variant:
|
||||
irc.reply('> ' + str(read) + ' <')
|
||||
|
||||
elif 'write' in variant:
|
||||
irc.reply('> ' + str(write) + ' <')
|
||||
|
||||
else:
|
||||
irc.reply('What?')
|
||||
else:
|
||||
irc.reply('Go home')
|
||||
|
||||
mcdebug = wrap(mcdebug, ['anything'])
|
||||
####################
|
||||
|
||||
def mailbox(self, irc, msg, args, variant, id):
|
||||
"""<variant> <id>
|
||||
Modifies Mailboxes."""
|
||||
"""<option> <id>
|
||||
i.e. 'summary cranberrry@liberta.casa' will print some details about his mailbox"""
|
||||
|
||||
server = self.registryValue('api.server')
|
||||
api_key = self.registryValue('api.key')
|
||||
read = self.registryValue('access.read')
|
||||
write = self.registryValue('access.write')
|
||||
nick = msg.nick
|
||||
hostmask = irc.state.nickToHostmask(msg.nick)
|
||||
|
||||
#Read-Functions: Summary
|
||||
if 'summary' in variant:
|
||||
URL = server + get + '/mailbox/' + id
|
||||
response = requests.get(
|
||||
URL,
|
||||
params={'q': 'requests'},
|
||||
headers={'accept': 'application/json', 'X-API-Key': api_key},
|
||||
)
|
||||
data = response.json()
|
||||
#return(print(data))
|
||||
irc.reply('Username: ' + data['username'] + ' | Quota: ' + str(data['quota']) + ' | Messages: ' + str(data['messages']) + ' | Mail Active: ' + str(data['active']) + ' | XMPP Active: ' + str(data['domain_xmpp']))
|
||||
if hostmask in read or write:
|
||||
URL = server + get + '/mailbox/' + id
|
||||
response = requests.get(
|
||||
URL,
|
||||
params={'q': 'requests'},
|
||||
headers={'accept': 'application/json', 'X-API-Key': api_key},
|
||||
)
|
||||
data = response.json()
|
||||
#return(print(data))
|
||||
irc.reply('Username: ' + data['username'] + ' | Quota: ' + str(data['quota']) + ' | Messages: ' + str(data['messages']) + ' | Mail Active: ' + str(data['active']) + ' | XMPP Active: ' + str(data['domain_xmpp']))
|
||||
else:
|
||||
irc.reply("Thou shall not pass.")
|
||||
print("Intrusion attempt: " + hostmask)
|
||||
|
||||
#Write-Functions: Create/Delete
|
||||
elif 'create' in variant:
|
||||
URL = server + api + '/add/mailbox/' + id
|
||||
user = id.split('@')[0]
|
||||
domain = id.split('@')[1]
|
||||
random = secrets.token_urlsafe(64)
|
||||
payload = {
|
||||
"active": "1",
|
||||
"domain": domain,
|
||||
"local_part": user,
|
||||
"name": user + '@' + domain,
|
||||
"password": random,
|
||||
"password2": random,
|
||||
"quota": "512",
|
||||
"force_pw_update": "1",
|
||||
"tls_enforce_in": "1",
|
||||
"tls_enforce_out": "1"
|
||||
}
|
||||
response = requests.post(
|
||||
URL,
|
||||
headers = {'accept': 'application/json', 'X-API-Key': api_key, 'Content-Type': 'application/json'},
|
||||
json = payload,
|
||||
)
|
||||
data = response.json()
|
||||
status = data[0]['type']
|
||||
msg = data[0]['msg']
|
||||
if hostmask in write:
|
||||
URL = server + api + '/add/mailbox/' + id
|
||||
user = id.split('@')[0]
|
||||
domain = id.split('@')[1]
|
||||
random = secrets.token_urlsafe(64)
|
||||
payload = {
|
||||
"active": "1",
|
||||
"domain": domain,
|
||||
"local_part": user,
|
||||
"name": user + '@' + domain,
|
||||
"password": random,
|
||||
"password2": random,
|
||||
"quota": "512",
|
||||
"force_pw_update": "1",
|
||||
"tls_enforce_in": "1",
|
||||
"tls_enforce_out": "1"
|
||||
}
|
||||
response = requests.post(
|
||||
URL,
|
||||
headers = {'accept': 'application/json', 'X-API-Key': api_key, 'Content-Type': 'application/json'},
|
||||
json = payload,
|
||||
)
|
||||
data = response.json()
|
||||
status = data[0]['type']
|
||||
msg = data[0]['msg']
|
||||
|
||||
irc.reply(f"CREATION: {status} | {msg} | " + str(random) + server)
|
||||
irc.reply(f"CREATION: {status} | {msg} | " + str(random) + server)
|
||||
else:
|
||||
irc.reply("Thou shall not create.")
|
||||
print("Intrusion attempt: " + hostmask)
|
||||
|
||||
elif 'delete' in variant:
|
||||
URL = server + api + '/delete/mailbox/' + id
|
||||
payload = [
|
||||
id
|
||||
]
|
||||
response = requests.post(
|
||||
URL,
|
||||
headers = {'accept': 'application/json', 'X-API-Key': api_key, 'Content-Type': 'application/json'},
|
||||
json = payload,
|
||||
)
|
||||
data = response.json()
|
||||
status = data[0]['type']
|
||||
object = data[0]['msg']
|
||||
|
||||
irc.reply(f"DELETION: {status} - {msg}")
|
||||
if hostmask in write:
|
||||
URL = server + api + '/delete/mailbox/' + id
|
||||
payload = [
|
||||
id
|
||||
]
|
||||
response = requests.post(
|
||||
URL,
|
||||
headers = {'accept': 'application/json', 'X-API-Key': api_key, 'Content-Type': 'application/json'},
|
||||
json = payload,
|
||||
)
|
||||
data = response.json()
|
||||
status = data[0]['type']
|
||||
object = data[0]['msg']
|
||||
|
||||
irc.reply(f"DELETION: {status} - {msg}")
|
||||
else:
|
||||
irc.reply("Thou shall not delete.")
|
||||
print("Intrusion attempt: " + hostmask)
|
||||
else:
|
||||
irc.reply('Unknown function.')
|
||||
irc.reply('Unknown function.')
|
||||
|
||||
mailbox = wrap(mailbox, ['anything', 'anything'])
|
||||
|
||||
|
@ -31,3 +31,14 @@ INFO Shutdown initiated.
|
||||
INFO Killing Driver objects.
|
||||
INFO Killing Irc objects.
|
||||
INFO Shutdown complete.
|
||||
ERROR Invalid user dictionary file, resetting to empty.
|
||||
ERROR Exact error: FileNotFoundError: [Errno 2] No such file or directory: '/home/georg/limnoria/Mailcow/test-conf/users.conf'
|
||||
ERROR Invalid channel database, resetting to empty.
|
||||
ERROR Exact error: FileNotFoundError: [Errno 2] No such file or directory: '/home/georg/limnoria/Mailcow/test-conf/channels.conf'
|
||||
ERROR Invalid network database, resetting to empty.
|
||||
ERROR Exact error: FileNotFoundError: [Errno 2] No such file or directory: '/home/georg/limnoria/Mailcow/test-conf/networks.conf'
|
||||
WARNING Couldn't open ignore database: [Errno 2] No such file or directory: '/home/georg/limnoria/Mailcow/test-conf/ignores.conf'
|
||||
INFO Shutdown initiated.
|
||||
INFO Killing Driver objects.
|
||||
INFO Killing Irc objects.
|
||||
INFO Shutdown complete.
|
||||
|
Loading…
Reference in New Issue
Block a user