Headline
CVE-2023-22488: Merge pull request from GHSA-8gcg-vwmw-rxj4 · flarum/framework@d0a2b95
Flarum is a forum software for building communities. Using the notifications feature, one can read restricted/private content and bypass access checks that would be in place for such content. The notification-sending component does not check that the subject of the notification can be seen by the receiver, and proceeds to send notifications through their different channels. The alerts do not leak data despite this as they are listed based on a visibility check, however, emails are still sent out. This means that, for extensions which restrict access to posts, any actor can bypass the restriction by subscribing to the discussion if the Subscriptions extension is enabled. The attack allows the leaking of some posts in the forum database, including posts awaiting approval, posts in tags the user has no access to if they could subscribe to a discussion before it becomes private, and posts restricted by third-party extensions. All Flarum versions prior to v1.6.3 are affected. The vulnerability has been fixed and published as flarum/core v1.6.3. All communities running Flarum should upgrade as soon as possible to v1.6.3. As a workaround, disable the Flarum Subscriptions extension or disable email notifications altogether. There are no other supported workarounds for this issue for Flarum versions below 1.6.3.
@@ -0,0 +1,169 @@ <?php
/* * This file is part of Flarum. * * For detailed copyright and license information, please view the * LICENSE file that was distributed with this source code. */
namespace Flarum\Tests\integration\notification;
use Carbon\Carbon; use Flarum\Api\Serializer\BasicDiscussionSerializer; use Flarum\Api\Serializer\BasicPostSerializer; use Flarum\Database\AbstractModel; use Flarum\Discussion\Discussion; use Flarum\Extend; use Flarum\Notification\Blueprint\BlueprintInterface; use Flarum\Notification\Notification; use Flarum\Notification\NotificationSyncer; use Flarum\Post\Post; use Flarum\Testing\integration\RetrievesAuthorizedUsers; use Flarum\Testing\integration\TestCase; use Flarum\User\User;
class NotificationSyncerTest extends TestCase { use RetrievesAuthorizedUsers;
protected function setUp(): void { parent::setUp();
$this->prepareDatabase([ ‘users’ => [ $this->normalUser(), [‘id’ => 3, ‘username’ => 'Receiver’, ‘email’ => '[email protected]’, ‘is_email_confirmed’ => 1], ], ‘discussions’ => [ [‘id’ => 1, ‘title’ => 'Public discussion’, ‘created_at’ => Carbon::parse(‘2021-11-01 13:00:00’)->toDateTimeString(), ‘user_id’ => 2, ‘first_post_id’ => 1, ‘comment_count’ => 2, ‘is_private’ => 0, ‘last_post_number’ => 2],
[‘id’ => 2, ‘title’ => 'Private discussion’, ‘created_at’ => Carbon::parse(‘2021-11-01 13:00:00’)->toDateTimeString(), ‘user_id’ => 2, ‘first_post_id’ => 3, ‘comment_count’ => 2, ‘is_private’ => 1, ‘last_post_number’ => 2], ], ‘posts’ => [ [‘id’ => 1, ‘discussion_id’ => 1, ‘number’ => 1, ‘created_at’ => Carbon::parse(‘2021-11-01 13:00:00’)->toDateTimeString(), ‘user_id’ => 2, ‘type’ => 'comment’, ‘content’ => '<t></t>’, ‘is_private’ => 0], [‘id’ => 2, ‘discussion_id’ => 1, ‘number’ => 2, ‘created_at’ => Carbon::parse(‘2021-11-01 13:00:03’)->toDateTimeString(), ‘user_id’ => 2, ‘type’ => 'comment’, ‘content’ => '<t></t>’, ‘is_private’ => 1],
[‘id’ => 3, ‘discussion_id’ => 2, ‘number’ => 1, ‘created_at’ => Carbon::parse(‘2021-11-01 13:00:00’)->toDateTimeString(), ‘user_id’ => 2, ‘type’ => 'comment’, ‘content’ => '<t></t>’, ‘is_private’ => 0], ], ]); }
/** * @dataProvider visibleSubjectsProvider * @param class-string<AbstractModel> $subjectClass * @test */ public function can_receive_notification_for_visible_subjects(string $subjectClass, int $subjectId, string $serializer) { $this->expect_notification_count_from_sending_notification_type_with_subject( 2, $subjectClass, $subjectId, $serializer ); }
/** * @dataProvider invisibleSubjectsProvider * @test */ public function cannot_receive_notification_for_restricted_subjects(string $subjectClass, int $subjectId, string $serializer) { $this->expect_notification_count_from_sending_notification_type_with_subject( 0, $subjectClass, $subjectId, $serializer ); }
/** * @param class-string<AbstractModel> $subjectClass */ protected function expect_notification_count_from_sending_notification_type_with_subject(int $count, string $subjectClass, int $subjectId, string $serializer) { CustomNotificationType::$subjectModel = $subjectClass;
$this->extend( (new Extend\Notification()) ->type(CustomNotificationType::class, $serializer, [‘alert’]) );
/** @var NotificationSyncer $syncer */ $syncer = $this->app()->getContainer()->make(NotificationSyncer::class);
$subject = $subjectClass::query()->find($subjectId);
$syncer->sync( $blueprint = new CustomNotificationType($subject), User::query() ->whereIn('id’, [1, 3]) ->get() ->all() );
$this->assertEquals( $count, Notification::query() ->matchingBlueprint($blueprint) ->whereSubject($subject) ->count() ); }
protected function visibleSubjectsProvider() { return [ [Post::class, 1, BasicPostSerializer::class], [Discussion::class, 1, BasicDiscussionSerializer::class], ]; }
protected function invisibleSubjectsProvider() { return [ [Post::class, 2, BasicPostSerializer::class], [Discussion::class, 2, BasicDiscussionSerializer::class], [Post::class, 3, BasicPostSerializer::class], ]; } }
class CustomNotificationType implements BlueprintInterface { protected $subject; public static $subjectModel;
public function __construct($subject) { $this->subject = $subject; }
public function getFromUser() { return null; }
public function getSubject() { return $this->subject; }
public function getData() { return []; }
public static function getType() { return 'customNotificationType’; }
public static function getSubjectModel() { return self::$subjectModel; } }
Related news
Using the notifications feature, one can read restricted/private content and bypass access checks that would be in place for such content. The notification-sending component does not check that the subject of the notification can be seen by the receiver, and proceeds to send notifications through their different channels. The alerts do not leak data despite this as they are listed based on a visibility check, however, emails are still sent out. This means that, for extensions which restrict access to posts, any actor can bypass the restriction by subscribing to the discussion if the [*Subscriptions*](https://extiverse.com/extension/flarum/subscriptions) extension is enabled. ### Impact The attack allows the leaking of some posts in the forum database, including posts awaiting approval, posts in tags the user has no access to if they could subscribe to a discussion before it becomes private, and posts restricted by third-party extensions. Other leaks could also happen for different ...