2019-02-10 19:29:55 -05:00
|
|
|
#!/usr/bin/env python3
|
2019-02-10 16:22:09 -05:00
|
|
|
"""
|
2020-03-08 17:10:32 -04:00
|
|
|
Matrix Webhook.
|
|
|
|
|
2019-02-17 05:46:00 -05:00
|
|
|
Post a message to a matrix room with a simple HTTP POST
|
2020-02-16 07:47:00 -05:00
|
|
|
v1: matrix-client & http.server
|
|
|
|
v2: matrix-nio & aiohttp
|
2019-02-10 16:22:09 -05:00
|
|
|
"""
|
|
|
|
|
2020-02-14 13:18:51 -05:00
|
|
|
import asyncio
|
2019-02-10 19:00:53 -05:00
|
|
|
import json
|
2019-02-10 18:38:08 -05:00
|
|
|
import os
|
2020-03-08 10:54:50 -04:00
|
|
|
from signal import SIGINT, SIGTERM
|
2019-02-10 16:22:09 -05:00
|
|
|
|
2020-02-14 13:18:51 -05:00
|
|
|
from aiohttp import web
|
2020-02-16 07:47:00 -05:00
|
|
|
from nio import AsyncClient
|
2019-02-10 18:38:08 -05:00
|
|
|
|
2019-02-10 19:51:35 -05:00
|
|
|
SERVER_ADDRESS = ('', int(os.environ.get('PORT', 4785)))
|
|
|
|
MATRIX_URL = os.environ.get('MATRIX_URL', 'https://matrix.org')
|
2020-02-16 07:47:00 -05:00
|
|
|
MATRIX_ID = os.environ.get('MATRIX_ID', '@wwm:matrix.org')
|
2019-02-10 19:51:35 -05:00
|
|
|
MATRIX_PW = os.environ['MATRIX_PW']
|
|
|
|
API_KEY = os.environ['API_KEY']
|
2020-02-16 07:47:00 -05:00
|
|
|
CLIENT = AsyncClient(MATRIX_URL, MATRIX_ID)
|
2019-02-10 18:38:08 -05:00
|
|
|
|
2020-02-14 13:18:51 -05:00
|
|
|
|
|
|
|
async def handler(request):
|
2019-02-10 18:38:08 -05:00
|
|
|
"""
|
2020-02-14 13:18:51 -05:00
|
|
|
Coroutine given to the server, st. it knows what to do with an HTTP request.
|
2020-03-08 17:10:32 -04:00
|
|
|
|
2020-02-14 13:18:51 -05:00
|
|
|
This one handles a POST, checks its content, and forwards it to the matrix room.
|
2019-02-10 18:38:08 -05:00
|
|
|
"""
|
2020-02-16 07:47:00 -05:00
|
|
|
data = await request.read()
|
|
|
|
data = json.loads(data.decode())
|
2020-02-14 13:18:51 -05:00
|
|
|
status, ret = 400, 'I need a json dict with text & key'
|
|
|
|
if all(key in data for key in ['text', 'key']):
|
|
|
|
status, ret = 401, 'I need the good "key"'
|
|
|
|
if data['key'] == API_KEY:
|
2020-02-16 07:47:00 -05:00
|
|
|
status, ret = 200, 'OK'
|
|
|
|
await CLIENT.room_send(room_id=str(request.rel_url)[1:],
|
|
|
|
message_type="m.room.message",
|
|
|
|
content={
|
|
|
|
"msgtype": "m.text",
|
|
|
|
"body": data['text'],
|
|
|
|
})
|
2019-02-10 18:38:08 -05:00
|
|
|
|
2020-02-16 07:47:00 -05:00
|
|
|
return web.Response(text='{"status": %i, "ret": "%s"}' % (status, ret),
|
2020-02-14 13:18:51 -05:00
|
|
|
content_type='application/json',
|
|
|
|
status=status)
|
2019-02-10 16:22:09 -05:00
|
|
|
|
|
|
|
|
2020-03-08 10:54:50 -04:00
|
|
|
async def main(event):
|
2020-02-16 07:47:00 -05:00
|
|
|
"""
|
2020-03-08 17:10:32 -04:00
|
|
|
Launch main coroutine.
|
2020-02-16 07:47:00 -05:00
|
|
|
|
|
|
|
matrix client login & start web server
|
|
|
|
"""
|
|
|
|
await CLIENT.login(MATRIX_PW)
|
|
|
|
|
2020-02-14 13:18:51 -05:00
|
|
|
server = web.Server(handler)
|
|
|
|
runner = web.ServerRunner(server)
|
|
|
|
await runner.setup()
|
|
|
|
site = web.TCPSite(runner, *SERVER_ADDRESS)
|
|
|
|
await site.start()
|
|
|
|
|
2020-03-08 10:54:50 -04:00
|
|
|
# Run until we get a shutdown request
|
|
|
|
await event.wait()
|
|
|
|
|
|
|
|
# Cleanup
|
|
|
|
await runner.cleanup()
|
|
|
|
await CLIENT.close()
|
|
|
|
|
|
|
|
|
|
|
|
def terminate(event, signal):
|
2020-03-08 17:10:32 -04:00
|
|
|
"""Close handling stuff."""
|
2020-03-08 10:54:50 -04:00
|
|
|
event.set()
|
2020-03-11 05:06:53 -04:00
|
|
|
asyncio.get_event_loop().remove_signal_handler(signal)
|
2019-02-10 16:22:09 -05:00
|
|
|
|
|
|
|
|
2020-03-11 05:06:53 -04:00
|
|
|
def run():
|
|
|
|
"""Launch everything."""
|
2020-02-14 13:18:51 -05:00
|
|
|
loop = asyncio.get_event_loop()
|
2020-03-08 10:54:50 -04:00
|
|
|
event = asyncio.Event()
|
|
|
|
|
|
|
|
for sig in (SIGINT, SIGTERM):
|
|
|
|
loop.add_signal_handler(sig, terminate, event, sig)
|
|
|
|
|
|
|
|
loop.run_until_complete(main(event))
|
2020-02-14 13:18:51 -05:00
|
|
|
|
|
|
|
loop.close()
|
2020-03-11 05:06:53 -04:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
run()
|