mirror of
https://github.com/flarum/core.git
synced 2025-07-19 07:41:22 +02:00
It turns out that the idea of “sending” a notification is flawed. (What happens if the notification subject is deleted shortly after? The notified user would end up with a dud notification which would be confusing. What about if a post is edited to mention an extra user? If you sent out notifications, the users who’ve already been mentioned would get a duplicate notification.) Instead, I’ve introduced the idea of notification “syncing”. Whenever a change is made to a piece of data (e.g. a post is created, edited, or deleted), you make a common notification and “sync” it to a set of users. The users who’ve received this notification before won’t get it again. It will be sent out to new users, and hidden from users who’ve received it before but are no longer recipients (e.g. users who’ve been “unmentioned” in a post). To keep track of this, we use the existing notifications database table, with an added `is_deleted` column. The syncing/diffing is handled all behind the scenes; the API is extremely simple (see Core\Notifications\DiscussionRenamedNotification + Core\Events\Handlers\DiscussionRenamedNotifier)
229 lines
8.3 KiB
PHP
229 lines
8.3 KiB
PHP
<?php namespace Flarum\Core;
|
|
|
|
use Illuminate\Bus\Dispatcher as Bus;
|
|
use Illuminate\Contracts\Container\Container;
|
|
use Illuminate\Contracts\Events\Dispatcher;
|
|
use Flarum\Support\ServiceProvider;
|
|
use Flarum\Core\Formatter\FormatterManager;
|
|
use Flarum\Core\Models\CommentPost;
|
|
use Flarum\Core\Models\Post;
|
|
use Flarum\Core\Models\Model;
|
|
use Flarum\Core\Models\Forum;
|
|
use Flarum\Core\Models\User;
|
|
use Flarum\Core\Models\Discussion;
|
|
use Flarum\Core\Search\GambitManager;
|
|
use League\Flysystem\Adapter\Local;
|
|
use Flarum\Core\Events\RegisterDiscussionGambits;
|
|
use Flarum\Core\Events\RegisterUserGambits;
|
|
use Flarum\Extend\Permission;
|
|
use Flarum\Extend\NotificationType;
|
|
|
|
class CoreServiceProvider extends ServiceProvider
|
|
{
|
|
/**
|
|
* Bootstrap the application events.
|
|
*
|
|
* @return void
|
|
*/
|
|
public function boot(Dispatcher $events, Bus $bus)
|
|
{
|
|
$this->loadViewsFrom(__DIR__.'/../../views', 'flarum');
|
|
|
|
$this->registerEventHandlers($events);
|
|
$this->registerPostTypes();
|
|
$this->registerPermissions();
|
|
$this->registerGambits();
|
|
$this->setupModels();
|
|
|
|
$this->app['flarum.formatter']->add('linkify', 'Flarum\Core\Formatter\LinkifyFormatter');
|
|
|
|
$bus->mapUsing(function ($command) {
|
|
return Bus::simpleMapping(
|
|
$command,
|
|
'Flarum\Core\Commands',
|
|
'Flarum\Core\Handlers\Commands'
|
|
);
|
|
});
|
|
|
|
$events->subscribe('Flarum\Core\Handlers\Events\DiscussionRenamedNotifier');
|
|
$this->extend(
|
|
(new NotificationType('Flarum\Core\Notifications\DiscussionRenamedNotification', 'Flarum\Api\Serializers\DiscussionBasicSerializer'))
|
|
->enableByDefault('alert'),
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Register the service provider.
|
|
*
|
|
* @return void
|
|
*/
|
|
public function register()
|
|
{
|
|
// Register a singleton entity that represents this forum. This entity
|
|
// will be used to check for global forum permissions (like viewing the
|
|
// forum, registering, and starting discussions.)
|
|
$this->app->singleton('flarum.forum', 'Flarum\Core\Models\Forum');
|
|
|
|
$this->app->singleton('flarum.formatter', 'Flarum\Core\Formatter\FormatterManager');
|
|
|
|
$this->app->bind(
|
|
'Flarum\Core\Repositories\DiscussionRepositoryInterface',
|
|
'Flarum\Core\Repositories\EloquentDiscussionRepository'
|
|
);
|
|
$this->app->bind(
|
|
'Flarum\Core\Repositories\PostRepositoryInterface',
|
|
'Flarum\Core\Repositories\EloquentPostRepository'
|
|
);
|
|
$this->app->bind(
|
|
'Flarum\Core\Repositories\UserRepositoryInterface',
|
|
'Flarum\Core\Repositories\EloquentUserRepository'
|
|
);
|
|
$this->app->bind(
|
|
'Flarum\Core\Repositories\ActivityRepositoryInterface',
|
|
'Flarum\Core\Repositories\EloquentActivityRepository'
|
|
);
|
|
|
|
$avatarFilesystem = function (Container $app) {
|
|
return $app->make('Illuminate\Contracts\Filesystem\Factory')->disk('flarum-avatars')->getDriver();
|
|
};
|
|
|
|
$this->app->when('Flarum\Core\Handlers\Commands\UploadAvatarCommandHandler')
|
|
->needs('League\Flysystem\FilesystemInterface')
|
|
->give($avatarFilesystem);
|
|
|
|
$this->app->when('Flarum\Core\Handlers\Commands\DeleteAvatarCommandHandler')
|
|
->needs('League\Flysystem\FilesystemInterface')
|
|
->give($avatarFilesystem);
|
|
|
|
$this->app->bind(
|
|
'Flarum\Core\Repositories\NotificationRepositoryInterface',
|
|
'Flarum\Core\Repositories\EloquentNotificationRepository'
|
|
);
|
|
}
|
|
|
|
public function registerGambits()
|
|
{
|
|
$this->app->when('Flarum\Core\Search\Discussions\DiscussionSearcher')
|
|
->needs('Flarum\Core\Search\GambitManager')
|
|
->give(function () {
|
|
$gambits = new GambitManager($this->app);
|
|
$gambits->add('Flarum\Core\Search\Discussions\Gambits\AuthorGambit');
|
|
$gambits->add('Flarum\Core\Search\Discussions\Gambits\UnreadGambit');
|
|
$gambits->setFulltextGambit('Flarum\Core\Search\Discussions\Gambits\FulltextGambit');
|
|
|
|
event(new RegisterDiscussionGambits($gambits));
|
|
|
|
return $gambits;
|
|
});
|
|
|
|
$this->app->when('Flarum\Core\Search\Users\UserSearcher')
|
|
->needs('Flarum\Core\Search\GambitManager')
|
|
->give(function () {
|
|
$gambits = new GambitManager($this->app);
|
|
$gambits->setFulltextGambit('Flarum\Core\Search\Users\Gambits\FulltextGambit');
|
|
|
|
event(new RegisterUserGambits($gambits));
|
|
|
|
return $gambits;
|
|
});
|
|
}
|
|
|
|
public function registerPostTypes()
|
|
{
|
|
Post::addType('Flarum\Core\Models\CommentPost');
|
|
Post::addType('Flarum\Core\Models\DiscussionRenamedPost');
|
|
|
|
CommentPost::setFormatter($this->app['flarum.formatter']);
|
|
}
|
|
|
|
public function registerEventHandlers($events)
|
|
{
|
|
$events->subscribe('Flarum\Core\Handlers\Events\DiscussionMetadataUpdater');
|
|
$events->subscribe('Flarum\Core\Handlers\Events\UserMetadataUpdater');
|
|
$events->subscribe('Flarum\Core\Handlers\Events\EmailConfirmationMailer');
|
|
}
|
|
|
|
public function setupModels()
|
|
{
|
|
Model::setForum($this->app['flarum.forum']);
|
|
Model::setValidator($this->app['validator']);
|
|
|
|
User::setHasher($this->app['hash']);
|
|
User::setFormatter($this->app['flarum.formatter']);
|
|
|
|
User::registerPreference('discloseOnline', 'boolval', true);
|
|
User::registerPreference('indexProfile', 'boolval', true);
|
|
}
|
|
|
|
public function registerPermissions()
|
|
{
|
|
$this->extend(
|
|
new Permission('forum.view'),
|
|
new Permission('forum.startDiscussion'),
|
|
new Permission('discussion.rename'),
|
|
new Permission('discussion.delete'),
|
|
new Permission('discussion.reply'),
|
|
new Permission('post.edit'),
|
|
new Permission('post.delete')
|
|
);
|
|
|
|
Forum::grantPermission(function ($grant, $user, $permission) {
|
|
return $user->hasPermission('forum.'.$permission);
|
|
});
|
|
|
|
Post::grantPermission(function ($grant, $user, $permission) {
|
|
return $user->hasPermission('post'.$permission);
|
|
});
|
|
|
|
// Grant view access to a post only if the user can also view the
|
|
// discussion which the post is in. Also, the if the post is hidden,
|
|
// the user must have edit permissions too.
|
|
Post::grantPermission('view', function ($grant) {
|
|
$grant->whereCan('view', 'discussion');
|
|
});
|
|
|
|
Post::demandPermission('view', function ($demand) {
|
|
$demand->whereNull('hide_user_id')
|
|
->orWhereCan('edit');
|
|
});
|
|
|
|
// Allow a user to edit their own post, unless it has been hidden by
|
|
// someone else.
|
|
Post::grantPermission('edit', function ($grant, $user) {
|
|
$grant->where('user_id', $user->id)
|
|
->whereNull('hide_user_id')
|
|
->orWhere('hide_user_id', $user->id);
|
|
// @todo add limitations to time etc. according to a config setting
|
|
});
|
|
|
|
User::grantPermission(function ($grant, $user, $permission) {
|
|
return $user->hasPermission('user.'.$permission);
|
|
});
|
|
|
|
// Grant view access to a user if the user can view the forum.
|
|
User::grantPermission('view', function ($grant, $user) {
|
|
$grant->whereCan('view', 'forum');
|
|
});
|
|
|
|
// Allow a user to edit their own account.
|
|
User::grantPermission('edit', function ($grant, $user) {
|
|
$grant->where('id', $user->id);
|
|
});
|
|
|
|
Discussion::grantPermission(function ($grant, $user, $permission) {
|
|
return $user->hasPermission('discussion.'.$permission);
|
|
});
|
|
|
|
// Grant view access to a discussion if the user can view the forum.
|
|
Discussion::grantPermission('view', function ($grant, $user) {
|
|
$grant->whereCan('view', 'forum');
|
|
});
|
|
|
|
// Allow a user to rename their own discussion.
|
|
Discussion::grantPermission('rename', function ($grant, $user) {
|
|
$grant->where('start_user_id', $user->id);
|
|
// @todo add limitations to time etc. according to a config setting
|
|
});
|
|
}
|
|
}
|