Browse Source

Fix CSRF in Slack, Pushbullet and Discord callbacks

pull/109/head
Pēteris Caune 8 years ago
parent
commit
17bf0d109e
6 changed files with 113 additions and 18 deletions
  1. +22
    -1
      hc/front/tests/test_add_discord.py
  2. +22
    -1
      hc/front/tests/test_add_pushbullet.py
  3. +27
    -2
      hc/front/tests/test_add_slack_btn.py
  4. +37
    -11
      hc/front/views.py
  5. +4
    -2
      templates/front/welcome.html
  6. +1
    -1
      templates/integrations/add_slack.html

+ 22
- 1
hc/front/tests/test_add_discord.py View File

@ -16,6 +16,9 @@ class AddDiscordTestCase(BaseTestCase):
self.assertContains(r, "Connect Discord", status_code=200) self.assertContains(r, "Connect Discord", status_code=200)
self.assertContains(r, "discordapp.com/api/oauth2/authorize") self.assertContains(r, "discordapp.com/api/oauth2/authorize")
# There should now be a key in session
self.assertTrue("discord" in self.client.session)
@override_settings(DISCORD_CLIENT_ID=None) @override_settings(DISCORD_CLIENT_ID=None)
def test_it_requires_client_id(self): def test_it_requires_client_id(self):
self.client.login(username="[email protected]", password="password") self.client.login(username="[email protected]", password="password")
@ -24,6 +27,10 @@ class AddDiscordTestCase(BaseTestCase):
@patch("hc.front.views.requests.post") @patch("hc.front.views.requests.post")
def test_it_handles_oauth_response(self, mock_post): def test_it_handles_oauth_response(self, mock_post):
session = self.client.session
session["discord"] = "foo"
session.save()
oauth_response = { oauth_response = {
"access_token": "test-token", "access_token": "test-token",
"webhook": { "webhook": {
@ -35,7 +42,7 @@ class AddDiscordTestCase(BaseTestCase):
mock_post.return_value.text = json.dumps(oauth_response) mock_post.return_value.text = json.dumps(oauth_response)
mock_post.return_value.json.return_value = oauth_response mock_post.return_value.json.return_value = oauth_response
url = self.url + "?code=12345678"
url = self.url + "?code=12345678&state=foo"
self.client.login(username="[email protected]", password="password") self.client.login(username="[email protected]", password="password")
r = self.client.get(url, follow=True) r = self.client.get(url, follow=True)
@ -44,3 +51,17 @@ class AddDiscordTestCase(BaseTestCase):
ch = Channel.objects.get() ch = Channel.objects.get()
self.assertEqual(ch.discord_webhook_url, "foo") self.assertEqual(ch.discord_webhook_url, "foo")
# Session should now be clean
self.assertFalse("discord" in self.client.session)
def test_it_avoids_csrf(self):
session = self.client.session
session["discord"] = "foo"
session.save()
url = self.url + "?code=12345678&state=bar"
self.client.login(username="[email protected]", password="password")
r = self.client.get(url)
self.assertEqual(r.status_code, 400)

+ 22
- 1
hc/front/tests/test_add_pushbullet.py View File

@ -16,6 +16,9 @@ class AddPushbulletTestCase(BaseTestCase):
self.assertContains(r, "www.pushbullet.com/authorize", status_code=200) self.assertContains(r, "www.pushbullet.com/authorize", status_code=200)
self.assertContains(r, "Connect Pushbullet") self.assertContains(r, "Connect Pushbullet")
# There should now be a key in session
self.assertTrue("pushbullet" in self.client.session)
@override_settings(PUSHBULLET_CLIENT_ID=None) @override_settings(PUSHBULLET_CLIENT_ID=None)
def test_it_requires_client_id(self): def test_it_requires_client_id(self):
self.client.login(username="[email protected]", password="password") self.client.login(username="[email protected]", password="password")
@ -24,12 +27,16 @@ class AddPushbulletTestCase(BaseTestCase):
@patch("hc.front.views.requests.post") @patch("hc.front.views.requests.post")
def test_it_handles_oauth_response(self, mock_post): def test_it_handles_oauth_response(self, mock_post):
session = self.client.session
session["pushbullet"] = "foo"
session.save()
oauth_response = {"access_token": "test-token"} oauth_response = {"access_token": "test-token"}
mock_post.return_value.text = json.dumps(oauth_response) mock_post.return_value.text = json.dumps(oauth_response)
mock_post.return_value.json.return_value = oauth_response mock_post.return_value.json.return_value = oauth_response
url = self.url + "?code=12345678"
url = self.url + "?code=12345678&state=foo"
self.client.login(username="[email protected]", password="password") self.client.login(username="[email protected]", password="password")
r = self.client.get(url, follow=True) r = self.client.get(url, follow=True)
@ -38,3 +45,17 @@ class AddPushbulletTestCase(BaseTestCase):
ch = Channel.objects.get() ch = Channel.objects.get()
self.assertEqual(ch.value, "test-token") self.assertEqual(ch.value, "test-token")
# Session should now be clean
self.assertFalse("pushbullet" in self.client.session)
def test_it_avoids_csrf(self):
session = self.client.session
session["pushbullet"] = "foo"
session.save()
url = self.url + "?code=12345678&state=bar"
self.client.login(username="[email protected]", password="password")
r = self.client.get(url)
self.assertEqual(r.status_code, 400)

+ 27
- 2
hc/front/tests/test_add_slack_btn.py View File

@ -14,6 +14,9 @@ class AddSlackBtnTestCase(BaseTestCase):
self.assertContains(r, "Before adding Slack integration", self.assertContains(r, "Before adding Slack integration",
status_code=200) status_code=200)
# There should now be a key in session
self.assertTrue("slack" in self.client.session)
@override_settings(SLACK_CLIENT_ID="foo") @override_settings(SLACK_CLIENT_ID="foo")
def test_slack_button(self): def test_slack_button(self):
self.client.login(username="[email protected]", password="password") self.client.login(username="[email protected]", password="password")
@ -22,6 +25,10 @@ class AddSlackBtnTestCase(BaseTestCase):
@patch("hc.front.views.requests.post") @patch("hc.front.views.requests.post")
def test_it_handles_oauth_response(self, mock_post): def test_it_handles_oauth_response(self, mock_post):
session = self.client.session
session["slack"] = "foo"
session.save()
oauth_response = { oauth_response = {
"ok": True, "ok": True,
"team_name": "foo", "team_name": "foo",
@ -34,7 +41,7 @@ class AddSlackBtnTestCase(BaseTestCase):
mock_post.return_value.text = json.dumps(oauth_response) mock_post.return_value.text = json.dumps(oauth_response)
mock_post.return_value.json.return_value = oauth_response mock_post.return_value.json.return_value = oauth_response
url = "/integrations/add_slack_btn/?code=12345678"
url = "/integrations/add_slack_btn/?code=12345678&state=foo"
self.client.login(username="[email protected]", password="password") self.client.login(username="[email protected]", password="password")
r = self.client.get(url, follow=True) r = self.client.get(url, follow=True)
@ -46,8 +53,26 @@ class AddSlackBtnTestCase(BaseTestCase):
self.assertEqual(ch.slack_channel, "bar") self.assertEqual(ch.slack_channel, "bar")
self.assertEqual(ch.slack_webhook_url, "http://example.org") self.assertEqual(ch.slack_webhook_url, "http://example.org")
# Session should now be clean
self.assertFalse("slack" in self.client.session)
def test_it_avoids_csrf(self):
session = self.client.session
session["slack"] = "foo"
session.save()
url = "/integrations/add_slack_btn/?code=12345678&state=bar"
self.client.login(username="[email protected]", password="password")
r = self.client.get(url)
self.assertEqual(r.status_code, 400)
@patch("hc.front.views.requests.post") @patch("hc.front.views.requests.post")
def test_it_handles_oauth_error(self, mock_post): def test_it_handles_oauth_error(self, mock_post):
session = self.client.session
session["slack"] = "foo"
session.save()
oauth_response = { oauth_response = {
"ok": False, "ok": False,
"error": "something went wrong" "error": "something went wrong"
@ -56,7 +81,7 @@ class AddSlackBtnTestCase(BaseTestCase):
mock_post.return_value.text = json.dumps(oauth_response) mock_post.return_value.text = json.dumps(oauth_response)
mock_post.return_value.json.return_value = oauth_response mock_post.return_value.json.return_value = oauth_response
url = "/integrations/add_slack_btn/?code=12345678"
url = "/integrations/add_slack_btn/?code=12345678&state=foo"
self.client.login(username="[email protected]", password="password") self.client.login(username="[email protected]", password="password")
r = self.client.get(url, follow=True) r = self.client.get(url, follow=True)


+ 37
- 11
hc/front/views.py View File

@ -91,7 +91,9 @@ def index(request):
"page": "welcome", "page": "welcome",
"check": check, "check": check,
"ping_url": check.url(), "ping_url": check.url(),
"enable_pushover": settings.PUSHOVER_API_TOKEN is not None
"enable_pushbullet": settings.PUSHBULLET_CLIENT_ID is not None,
"enable_pushover": settings.PUSHOVER_API_TOKEN is not None,
"enable_discord": settings.DISCORD_CLIENT_ID is not None
} }
return render(request, "front/welcome.html", ctx) return render(request, "front/welcome.html", ctx)
@ -430,6 +432,24 @@ def add_pd(request):
return render(request, "integrations/add_pd.html", ctx) return render(request, "integrations/add_pd.html", ctx)
def _prepare_state(request, session_key):
state = get_random_string()
request.session[session_key] = state
return state
def _get_validated_code(request, session_key):
if session_key not in request.session:
return None
session_state = request.session.pop(session_key)
request_state = request.GET.get("state")
if session_state is None or session_state != request_state:
return None
return request.GET.get("code")
def add_slack(request): def add_slack(request):
if not settings.SLACK_CLIENT_ID and not request.user.is_authenticated: if not settings.SLACK_CLIENT_ID and not request.user.is_authenticated:
return redirect("hc-login") return redirect("hc-login")
@ -452,13 +472,16 @@ def add_slack(request):
"slack_client_id": settings.SLACK_CLIENT_ID "slack_client_id": settings.SLACK_CLIENT_ID
} }
if settings.SLACK_CLIENT_ID:
ctx["state"] = _prepare_state(request, "slack")
return render(request, "integrations/add_slack.html", ctx) return render(request, "integrations/add_slack.html", ctx)
@login_required @login_required
def add_slack_btn(request): def add_slack_btn(request):
code = request.GET.get("code", "")
if len(code) < 8:
code = _get_validated_code(request, "slack")
if code is None:
return HttpResponseBadRequest() return HttpResponseBadRequest()
result = requests.post("https://slack.com/api/oauth.access", { result = requests.post("https://slack.com/api/oauth.access", {
@ -507,8 +530,8 @@ def add_pushbullet(request):
raise Http404("pushbullet integration is not available") raise Http404("pushbullet integration is not available")
if "code" in request.GET: if "code" in request.GET:
code = request.GET.get("code", "")
if len(code) < 8:
code = _get_validated_code(request, "pushbullet")
if code is None:
return HttpResponseBadRequest() return HttpResponseBadRequest()
result = requests.post("https://api.pushbullet.com/oauth2/token", { result = requests.post("https://api.pushbullet.com/oauth2/token", {
@ -536,7 +559,8 @@ def add_pushbullet(request):
authorize_url = "https://www.pushbullet.com/authorize?" + urlencode({ authorize_url = "https://www.pushbullet.com/authorize?" + urlencode({
"client_id": settings.PUSHBULLET_CLIENT_ID, "client_id": settings.PUSHBULLET_CLIENT_ID,
"redirect_uri": redirect_uri, "redirect_uri": redirect_uri,
"response_type": "code"
"response_type": "code",
"state": _prepare_state(request, "pushbullet")
}) })
ctx = { ctx = {
@ -553,8 +577,8 @@ def add_discord(request):
redirect_uri = settings.SITE_ROOT + reverse("hc-add-discord") redirect_uri = settings.SITE_ROOT + reverse("hc-add-discord")
if "code" in request.GET: if "code" in request.GET:
code = request.GET.get("code", "")
if len(code) < 8:
code = _get_validated_code(request, "discord")
if code is None:
return HttpResponseBadRequest() return HttpResponseBadRequest()
result = requests.post("https://discordapp.com/api/oauth2/token", { result = requests.post("https://discordapp.com/api/oauth2/token", {
@ -579,17 +603,19 @@ def add_discord(request):
return redirect("hc-channels") return redirect("hc-channels")
authorize_url = "https://discordapp.com/api/oauth2/authorize?" + urlencode({
auth_url = "https://discordapp.com/api/oauth2/authorize?" + urlencode({
"client_id": settings.DISCORD_CLIENT_ID, "client_id": settings.DISCORD_CLIENT_ID,
"scope": "webhook.incoming", "scope": "webhook.incoming",
"redirect_uri": redirect_uri, "redirect_uri": redirect_uri,
"response_type": "code"
"response_type": "code",
"state": _prepare_state(request, "discord")
}) })
ctx = { ctx = {
"page": "channels", "page": "channels",
"authorize_url": authorize_url
"authorize_url": auth_url
} }
return render(request, "integrations/add_discord.html", ctx) return render(request, "integrations/add_discord.html", ctx)


+ 4
- 2
templates/front/welcome.html View File

@ -248,12 +248,14 @@
<td>Notifications in <a href="https://slack.com/">Slack</a> channel.</td> <td>Notifications in <a href="https://slack.com/">Slack</a> channel.</td>
</tr> </tr>
{% if enable_discord %}
<tr> <tr>
<td> <td>
<img width="22" height="22" alt="Discord icon" src="" /> <img width="22" height="22" alt="Discord icon" src="" />
</td> </td>
<td>Notifications in <a href="https://discordapp.com/">Discord</a> channel.</td> <td>Notifications in <a href="https://discordapp.com/">Discord</a> channel.</td>
</tr> </tr>
{% endif %}
<tr> <tr>
<td> <td>
@ -283,14 +285,14 @@
<td>Open and resolve incidents in <a href="https://victorops.com/">VictorOps</a>.</td> <td>Open and resolve incidents in <a href="https://victorops.com/">VictorOps</a>.</td>
</tr> </tr>
{% if enable_pushbullet %}
<tr> <tr>
<td> <td>
<img width="22" height="22" alt="Pushbullet icon" src="" /> <img width="22" height="22" alt="Pushbullet icon" src="" />
</td> </td>
<td>Instant push notifications with <a href="https://www.pushbullet.com/">Pushbullet</a>.</td> <td>Instant push notifications with <a href="https://www.pushbullet.com/">Pushbullet</a>.</td>
</tr> </tr>
{% endif %}
{% if enable_pushover %} {% if enable_pushover %}
<tr> <tr>


+ 1
- 1
templates/integrations/add_slack.html View File

@ -16,7 +16,7 @@
Slack channel.</p> Slack channel.</p>
<div class="text-center"> <div class="text-center">
<a href="https://slack.com/oauth/authorize?scope=incoming-webhook&client_id={{ slack_client_id }}">
<a href="https://slack.com/oauth/authorize?scope=incoming-webhook&client_id={{ slack_client_id }}&state={{ state }}">
<img alt="Add to Slack" height="40" width="139" src="https://platform.slack-edge.com/img/add_to_slack.png" srcset="https://platform.slack-edge.com/img/add_to_slack.png 1x, https://platform.slack-edge.com/img/[email protected] 2x" /> <img alt="Add to Slack" height="40" width="139" src="https://platform.slack-edge.com/img/add_to_slack.png" srcset="https://platform.slack-edge.com/img/add_to_slack.png 1x, https://platform.slack-edge.com/img/[email protected] 2x" />
</a> </a>
</div> </div>


Loading…
Cancel
Save