-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathmain.py
130 lines (116 loc) · 4.77 KB
/
main.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
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
from flask import Flask, request, Response
import requests
import redis
import os
import stripe
from dotenv import load_dotenv
import logging
load_dotenv(verbose=True)
PYTHON_LOG_LEVEL = os.environ.get("PYTHON_LOG_LEVEL", "WARNING")
log = logging.getLogger(__name__)
log.setLevel(PYTHON_LOG_LEVEL)
log.addHandler(logging.StreamHandler())
REDIS_HOSTNAME = os.environ.get("REDIS_HOSTNAME")
REDIS_PORT = os.environ.get("REDIS_PORT")
REDIS_TIMEOUT_SECS = int(os.environ.get("REDIS_TIMEOUT_SECS"))
REDIS_PASSWORD = os.environ.get("REDIS_PASSWORD")
STRIPE_WEBHOOK_SECRET = os.environ.get("STRIPE_WEBHOOK_SECRET")
STRIPE_WEBHOOK_PATH = os.environ.get("STRIPE_WEBHOOK_PATH")
ENFORCE_HTTPS_WEBHOOK_POST = os.environ.get("ENFORCE_HTTPS_WEBHOOK_POST")
if ENFORCE_HTTPS_WEBHOOK_POST is not None:
if (
ENFORCE_HTTPS_WEBHOOK_POST.lower() == "true"
or ENFORCE_HTTPS_WEBHOOK_POST == "1"
):
ENFORCE_HTTPS_WEBHOOK_POST = True
else:
ENFORCE_HTTPS_WEBHOOK_POST = False
print(f"PYTHON_LOG_LEVEL is: {PYTHON_LOG_LEVEL}")
log.debug(f"ENFORCE_HTTPS_WEBHOOK_POST is set to {ENFORCE_HTTPS_WEBHOOK_POST}")
app = Flask(__name__)
@app.route("/", methods=["POST"])
def route_stripe_connect_webhook():
"""Recieve stripe connect webhook & route to shop based on account id
Looks up shop address via redis using account_id as key.
- Sends to webhook to the correct stop
- Returns status code & response body from shop
- (Only returns 200 if the shop returns 200, otherwise return 400)
"""
load_dotenv(verbose=True)
redisConn = redis.Redis(
host=REDIS_HOSTNAME,
port=REDIS_PORT,
password=REDIS_PASSWORD,
socket_timeout=REDIS_TIMEOUT_SECS, # noqa E501
)
try:
log.debug("Attempting to parse account from request")
stripe_connect_account_id = request.json["account"]
except KeyError as e:
log.error(e)
msg = f"Not a connect request.\
No 'account' property in payload\n\n{request.json}"
log.error(msg)
return msg, 422
try:
stripe_connect_account_id = request.json["account"]
# Get shop url from redis via stripe connect account id
site_url = redisConn.get(stripe_connect_account_id)
log.debug(f"Routing for account: {stripe_connect_account_id}")
log.debug(f"Will be routing webhook to: {site_url}")
if site_url is not None:
# Verify stripe signature header
# Proxy stripe webhook to site_url (the shop)
# and await response code
# then forward response code and body back to stripe
sig_header = request.headers.get("Stripe-Signature", None)
event = None
try:
event = stripe.Webhook.construct_event(
request.data, sig_header, STRIPE_WEBHOOK_SECRET
)
except ValueError as e:
msg = "ValueError when attempting to get Stripe-Signature header" # noqa: E501
log.error(msg)
log.error(e)
return e, 400
except stripe.error.SignatureVerificationError as e:
log.error("Stripe SignatureVerificationError")
log.error(e)
return "Stripe SignatureVerificationError", 400
post_url = site_url.decode("utf-8") + STRIPE_WEBHOOK_PATH
if ENFORCE_HTTPS_WEBHOOK_POST:
msg = "Enforcing https post since ENFORCE_HTTPS_WEBHOOK_POST"
log.debug(msg)
post_url = post_url.replace("http://", "https://")
else:
log.warning(
"ENFORCE_HTTPS_WEBHOOK_POST not set so posting to http"
) # noqa: E501
log.debug(f"Posting webhook to: {post_url}")
resp = requests.post(
post_url,
json=event,
)
# Return (proxying) whatever the site (shop) responds
# including the return code
log.debug(f"{resp.status_code}, {resp.text}")
return resp.text, resp.status_code
else:
msg = f'{{"msg": "No site_url for that account id", \
"account id": "{stripe_connect_account_id}"}}'
log.error(msg)
return Response(msg, status=500, mimetype="application/json")
except redis.exceptions.ResponseError as e:
log.error("Redis ResponseError")
log.error(e)
except redis.exceptions.ConnectionError as e:
log.error("Redis ConnectionError")
log.error(e)
except redis.exceptions.TimeoutError as e:
log.error("Redis timeout")
log.error(e)
except Exception as e:
log.error("Error processing stripe webhook request")
log.error(e)
return "Invalid request to route_stripe_connect_webhook", 400