Compare commits

...

2 Commits
1.4.2 ... 1.4.x

Author SHA1 Message Date
Tim Abbott
a063dd3b26 Update changelog for Zulip 1.4.3 release. 2017-01-29 15:09:02 -08:00
Tim Abbott
1cdd451d70 streams: Fix autosubscribe security bug (CVE-2017-0881).
A bug in Zulip's implementation of the "stream exists" endpoint meant
that any user of a Zulip server could subscribe to an invite-only
stream without needing to be invited by using the "autosubscribe"
argument.

Thanks to Rafid Aslam for discovering this issue.
2017-01-29 15:09:02 -08:00
3 changed files with 27 additions and 1 deletions

View File

@@ -4,6 +4,9 @@ All notable changes to the Zulip server are documented in this file.
### Unreleased ### Unreleased
### 1.4.3 - 2017-01-29
- CVE-2017-0881: Users could subscribe to invite-only streams.
### 1.4.2 - 2016-09-27 ### 1.4.2 - 2016-09-27
- Upgraded Django to version 1.8.15 (with the Zulip patches applied), - Upgraded Django to version 1.8.15 (with the Zulip patches applied),
fixing a CSRF vulnerability in Django (see fixing a CSRF vulnerability in Django (see

View File

@@ -1501,6 +1501,29 @@ class SubscriptionAPITest(ZulipTestCase):
self.assertIn("exists", json) self.assertIn("exists", json)
self.assertTrue(json["exists"]) self.assertTrue(json["exists"])
def test_existing_subscriptions_autosubscription_private_stream(self):
# type: () -> None
"""Call /json/subscriptions/exist on an existing private stream with
autosubscribe should fail.
"""
stream_name = "Saxony"
result = self.common_subscribe_to_streams("cordelia@zulip.com", [stream_name],
invite_only=True)
stream = get_stream(stream_name, self.realm)
result = self.client_post("/json/subscriptions/exists",
{"stream": stream_name, "autosubscribe": True})
self.assert_json_success(result)
json = ujson.loads(result.content)
self.assertIn("exists", json)
self.assertTrue(json["exists"])
self.assertIn("subscribed", json)
# Importantly, we are not now subscribed
self.assertFalse(json["subscribed"])
self.assertEqual(Subscription.objects.filter(
recipient__type=Recipient.STREAM,
recipient__type_id=stream.id).count(), 1)
def get_subscription(self, user_profile, stream_name): def get_subscription(self, user_profile, stream_name):
# type: (UserProfile, text_type) -> Subscription # type: (UserProfile, text_type) -> Subscription
stream = Stream.objects.get(realm=self.realm, name=stream_name) stream = Stream.objects.get(realm=self.realm, name=stream_name)

View File

@@ -447,7 +447,7 @@ def stream_exists_backend(request, user_profile, stream_name, autosubscribe):
result = {"exists": bool(stream)} result = {"exists": bool(stream)}
if stream is not None: if stream is not None:
recipient = get_recipient(Recipient.STREAM, stream.id) recipient = get_recipient(Recipient.STREAM, stream.id)
if autosubscribe: if not stream.invite_only and autosubscribe:
bulk_add_subscriptions([stream], [user_profile]) bulk_add_subscriptions([stream], [user_profile])
result["subscribed"] = Subscription.objects.filter(user_profile=user_profile, result["subscribed"] = Subscription.objects.filter(user_profile=user_profile,
recipient=recipient, recipient=recipient,