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
131
132
133
134
135
136
137
138
|
# coding=utf-8
import json
import os
from flask import make_response, request
import requests
from pysite.base_route import APIView
from pysite.constants import ErrorCodes
ASANA_KEY = os.environ.get("ASANA_KEY")
ASANA_WEBHOOK = os.environ.get("ASANA_WEBHOOK")
BASE_URL = "https://app.asana.com/api/1.0"
STORY_URL = f"{BASE_URL}/stories"
TASK_URL = f"{BASE_URL}/tasks"
USER_URL = f"{BASE_URL}/users"
COLOUR_RED = 0xFF0000
COLOUR_GREEN = 0x00FF00
COLOUR_BLUE = 0x0000FF
class IndexView(APIView):
path = "/asana/<asana_key>"
name = "asana"
def post(self, asana_key):
if asana_key != ASANA_KEY:
return self.error(ErrorCodes.unauthorized)
if "X-Hook-Secret" in request.headers: # Confirm to Asana that we would like to make this hook
response = make_response() # type: flask.Response
response.headers["X-Hook-Secret"] = request.headers["X-Hook-Secret"]
self.send_webhook(title="Asana", description="Hook added", color=COLOUR_GREEN)
return response
events = request.get_json()["events"]
for event in events:
func_name = f"asana_{event['type']}"
if hasattr(self, func_name):
func = getattr(self, func_name)
else:
func = self.asana_unknown
try:
func(**event)
except Exception as e:
pretty_event = json.dumps(event, indent=4, sort_keys=True)
try:
self.send_webhook(
title="Error during webhook",
description=f"Failed to handle webhook: {e}\n\n```json\n{pretty_event}\n```",
color=COLOUR_RED
)
except Exception as e:
print(f"Fatal error sending webhook: {e}")
return "", 200 # Empty 200 response
def send_webhook(self, *, title, description, color=COLOUR_BLUE, url=None, author_name=None, author_icon=None):
session = requests.session()
embed = {
"title": title,
"description": description,
"color": color
}
if url:
embed["url"] = url
if author_name:
embed["author"] = {
"name": author_name,
"icon_url": author_icon
}
session.post(ASANA_WEBHOOK, json={"embeds": [embed]})
session.close()
def asana_story(self, *, resource, parent, created_at, user, action, type):
session = requests.session()
story = session.get(f"{STORY_URL}/{resource}").json()
if story.get("type", None) == "comment" and action == "added": # New comment!
task = session.get(f"{TASK_URL}/{parent}").json()
user = session.get(f"{USER_URL}/{user}").json()
project = task["projects"][0] # Just use the first project in the list
if user["photo"]:
photo = user["photo"]["image_128x128"]
else:
photo = None
self.send_webhook(
title=f"Comment: {project['name']}",
description=story["text"],
color=COLOUR_GREEN,
url=f"https://app.asana.com/0/{project['id']}/{parent}",
author_name=story["created_by"]["name"],
author_icon=photo
)
else:
pretty_story = json.dumps(
story,
indent=4,
sort_keys=True
)
self.send_webhook(
title=f"Unknown story action/type: {action}/{story['type']}",
description=f"```json\n{pretty_story}\n```"
)
session.close()
def asana_unknown(self, *, resource, parent, created_at, user, action, type):
pretty_event = json.dumps(
{
"resource": resource,
"parent": parent,
"created_at": created_at,
"user": user,
"action": action,
"type": type
},
indent=4,
sort_keys=True
)
self.send_webhook(
title="Unknown event",
description=f"```json\n{pretty_event}\n```"
)
|