-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtinyrestcallbackservice.py
106 lines (83 loc) · 3.87 KB
/
tinyrestcallbackservice.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
from utils import loadConfig, Database, PLACEHOLDER
import sys
import os
import signal
from flask import Flask, request, render_template
from markupsafe import escape
import time
import json
import requests
import string
import threading
from gevent.pywsgi import WSGIServer
# Constants
MAIN_HTML_FILE = "main.html"
# Variables
CONFIG = {}
service = Flask( __name__ )
database = None
server = None
# Load config
if len( sys.argv ) < 2:
sys.exit( "Config path as parameter is missing. Exiting." )
CONFIG = loadConfig( str( sys.argv[1] ) )
# Define our signal handler for gracefully ending the script
def signalHandler( signal, frame ):
print( "\nStopping all services..." )
server.stop()
# Init SIGINT-Signal to gracefully quit the Script
def initInterruptSignal():
signal.signal( signal.SIGINT, signalHandler )
print( "Press Ctrl+C to exit." )
# Calls the webhook after complementing the target URL & the body
def callWebhook( target, method, body, requestInfo ):
try:
target = complementString( target, requestInfo )
print( "--> Calling webhook URL \"" + target + "\"" )
if body:
body = complementString( body, requestInfo )
response = requests.request( method, target, data = body )
print( "--> Response code: " + str( response.status_code ) )
except Exception as e:
print( "--> Error while calling webhook URL \"" + target + "\": " + str( e ) )
# Replace all placeholders (defined in PLACEHOLDERS) in the string with the actual values from requestInfo
def complementString( string, requestInfo ):
requestInfo["timestamp"] = str( requestInfo["timestamp"] )
# Names of dict-entries in PLACEHOLDER and requestInfo must be in sync, or otherwise the items can not be matched
for entry in PLACEHOLDER:
string = string.replace( PLACEHOLDER[entry], requestInfo[entry] )
return string
# Endpoint processor
@service.route( CONFIG["PATH_PREFIX"] + "/<path:endpoint>", methods = ['GET', 'POST', 'PUT', 'PATCH', 'DELETE'] )
def endpoint( endpoint ):
requestInfo = \
{
"endpoint" : str( escape( endpoint ) ),
"method" : str( request.method ),
"host" : str( request.host ),
"requestUrl" : str( request.url ),
"remoteIp" : str( request.remote_addr ),
"userAgent" : str( request.user_agent ),
"timestamp" : int( time.time() )
}
print( "-> Endpoint \"" + requestInfo["endpoint"] + "\" called with method \"" + requestInfo["method"] + "\" from " + requestInfo["remoteIp"] )
endpointConfig = database.getEndpointConfig( requestInfo["endpoint"], requestInfo["method"] )
# In case no matching endpoint config is found, just return our default message
if endpointConfig is None:
return CONFIG["DEFAULT_MESSAGE"]
if endpointConfig["log"]:
database.logRequest( requestInfo )
if endpointConfig["webhookUrl"]:
# Calls the webhook asynchronous, so long answer times or failures don't block/interrupt the service request too long
webhookThread = threading.Thread( target = callWebhook, args = ( endpointConfig["webhookUrl"], endpointConfig["webhookMethod"], endpointConfig["webhookBody"], requestInfo ) )
webhookThread.start()
# Inject the HTML message, the redirect URL and the redirect waiting time into our template and return it
endpointConfig["message"] = complementString( endpointConfig["message"], requestInfo )
endpointConfig["redirectUrl"] = complementString( endpointConfig["redirectUrl"], requestInfo )
template = render_template( MAIN_HTML_FILE, message = endpointConfig["message"], redirectUrl = endpointConfig["redirectUrl"], redirectWait = endpointConfig["redirectWait"] )
return template
print( "Starting service." )
initInterruptSignal()
database = Database( CONFIG["DATABASE"], CONFIG["SCHEMA"] )
server = WSGIServer( ( CONFIG["HOST"], CONFIG["PORT"] ), service )
server.serve_forever()