1
0
Fork 1
mirror of https://github.com/pixelfed/pixelfed.git synced 2025-01-03 05:34:14 +00:00

Update ApiV1Controller, improve notification filtering

This commit is contained in:
Daniel Supernault 2024-03-05 01:56:16 -07:00
parent 31e6487dc9
commit 01535a6cfe
No known key found for this signature in database
GPG key ID: 23740873EE6F76A1
4 changed files with 370 additions and 306 deletions

View file

@ -2247,7 +2247,8 @@ class ApiV1Controller extends Controller
'max_id' => 'nullable|integer|min:1|max:'.PHP_INT_MAX, 'max_id' => 'nullable|integer|min:1|max:'.PHP_INT_MAX,
'since_id' => 'nullable|integer|min:1|max:'.PHP_INT_MAX, 'since_id' => 'nullable|integer|min:1|max:'.PHP_INT_MAX,
'types[]' => 'sometimes|array', 'types[]' => 'sometimes|array',
'type' => 'sometimes|string|in:mention,reblog,follow,favourite' 'type' => 'sometimes|string|in:mention,reblog,follow,favourite',
'_pe' => 'sometimes',
]); ]);
$pid = $request->user()->profile_id; $pid = $request->user()->profile_id;
@ -2259,6 +2260,7 @@ class ApiV1Controller extends Controller
$since = $request->input('since_id'); $since = $request->input('since_id');
$min = $request->input('min_id'); $min = $request->input('min_id');
$max = $request->input('max_id'); $max = $request->input('max_id');
$pe = $request->filled('_pe');
if(!$since && !$min && !$max) { if(!$since && !$min && !$max) {
$min = 1; $min = 1;
@ -2298,12 +2300,39 @@ class ApiV1Controller extends Controller
$minId = null; $minId = null;
} }
$res = collect($res)->filter(function($n) { $res = collect($res)
->map(function($n) use($pe) {
if(!$pe) {
if($n['type'] == 'comment') {
$n['type'] = 'mention';
return $n;
}
return $n;
}
return $n;
})
->filter(function($n) use($pe) {
if(in_array($n['type'], ['mention', 'reblog', 'favourite'])) { if(in_array($n['type'], ['mention', 'reblog', 'favourite'])) {
return isset($n['status'], $n['status']['id']); return isset($n['status'], $n['status']['id']);
} }
if(!$pe) {
if(in_array($n['type'], [
'tagged',
'modlog',
'story:react',
'story:comment',
'group:comment',
'group:join:approved',
'group:join:rejected',
])) {
return false;
}
return isset($n['account'], $n['account']['id']); return isset($n['account'], $n['account']['id']);
}
return true;
})->values(); })->values();
if($maxId) { if($maxId) {

View file

@ -8,6 +8,12 @@ class MediaTag extends Model
{ {
protected $guarded = []; protected $guarded = [];
protected $visible = [
'status_id',
'profile_id',
'tagged_username',
];
public function status() public function status()
{ {
return $this->belongsTo(Status::class); return $this->belongsTo(Status::class);

View file

@ -2,23 +2,22 @@
namespace App\Services; namespace App\Services;
use App\Jobs\InternalPipeline\NotificationEpochUpdatePipeline;
use App\Notification;
use App\Transformer\Api\NotificationTransformer;
use Cache; use Cache;
use Illuminate\Support\Facades\Redis; use Illuminate\Support\Facades\Redis;
use App\{
Notification,
Profile
};
use App\Transformer\Api\NotificationTransformer;
use League\Fractal; use League\Fractal;
use League\Fractal\Serializer\ArraySerializer; use League\Fractal\Serializer\ArraySerializer;
use League\Fractal\Pagination\IlluminatePaginatorAdapter;
use App\Jobs\InternalPipeline\NotificationEpochUpdatePipeline;
class NotificationService {
class NotificationService
{
const CACHE_KEY = 'pf:services:notifications:ids:'; const CACHE_KEY = 'pf:services:notifications:ids:';
const EPOCH_CACHE_KEY = 'pf:services:notifications:epoch-id:by-months:'; const EPOCH_CACHE_KEY = 'pf:services:notifications:epoch-id:by-months:';
const ITEM_CACHE_TTL = 86400; const ITEM_CACHE_TTL = 86400;
const MASTODON_TYPES = [ const MASTODON_TYPES = [
'follow', 'follow',
'follow_request', 'follow_request',
@ -26,7 +25,7 @@ class NotificationService {
'reblog', 'reblog',
'favourite', 'favourite',
'poll', 'poll',
'status' 'status',
]; ];
public static function get($id, $start = 0, $stop = 400) public static function get($id, $start = 0, $stop = 400)
@ -44,6 +43,7 @@ class NotificationService {
$res->push($n); $res->push($n);
} }
} }
return $res; return $res;
} }
@ -52,8 +52,10 @@ class NotificationService {
$epoch = Cache::get(self::EPOCH_CACHE_KEY.$months); $epoch = Cache::get(self::EPOCH_CACHE_KEY.$months);
if (! $epoch) { if (! $epoch) {
NotificationEpochUpdatePipeline::dispatch(); NotificationEpochUpdatePipeline::dispatch();
return 1; return 1;
} }
return $epoch; return $epoch;
} }
@ -69,6 +71,7 @@ class NotificationService {
foreach ($ids as $key) { foreach ($ids as $key) {
self::set($id, $key); self::set($id, $key);
} }
return $ids; return $ids;
} }
@ -87,6 +90,7 @@ class NotificationService {
$res->push($n); $res->push($n);
} }
} }
return $res->toArray(); return $res->toArray();
} }
@ -105,10 +109,10 @@ class NotificationService {
$res->push($n); $res->push($n);
} }
} }
return $res->toArray(); return $res->toArray();
} }
public static function getMaxMastodon($id = false, $start = 0, $limit = 10) public static function getMaxMastodon($id = false, $start = 0, $limit = 10)
{ {
$ids = self::getRankedMaxId($id, $start, $limit); $ids = self::getRankedMaxId($id, $start, $limit);
@ -129,6 +133,11 @@ class NotificationService {
unset($n['relationship']); unset($n['relationship']);
} }
if ($n['type'] === 'mention' && isset($n['tagged'], $n['tagged']['status_id'])) {
$n['status'] = StatusService::getMastodon($n['tagged']['status_id'], false);
unset($n['tagged']);
}
if (isset($n['status'])) { if (isset($n['status'])) {
$n['status'] = StatusService::getMastodon($n['status']['id'], false); $n['status'] = StatusService::getMastodon($n['status']['id'], false);
} }
@ -136,6 +145,7 @@ class NotificationService {
$res->push($n); $res->push($n);
} }
} }
return $res->toArray(); return $res->toArray();
} }
@ -159,6 +169,11 @@ class NotificationService {
unset($n['relationship']); unset($n['relationship']);
} }
if ($n['type'] === 'mention' && isset($n['tagged'], $n['tagged']['status_id'])) {
$n['status'] = StatusService::getMastodon($n['tagged']['status_id'], false);
unset($n['tagged']);
}
if (isset($n['status'])) { if (isset($n['status'])) {
$n['status'] = StatusService::getMastodon($n['status']['id'], false); $n['status'] = StatusService::getMastodon($n['status']['id'], false);
} }
@ -166,6 +181,7 @@ class NotificationService {
$res->push($n); $res->push($n);
} }
} }
return $res->toArray(); return $res->toArray();
} }
@ -177,7 +193,7 @@ class NotificationService {
return array_keys(Redis::zrevrangebyscore(self::CACHE_KEY.$id, $start, '-inf', [ return array_keys(Redis::zrevrangebyscore(self::CACHE_KEY.$id, $start, '-inf', [
'withscores' => true, 'withscores' => true,
'limit' => [1, $limit] 'limit' => [1, $limit],
])); ]));
} }
@ -189,7 +205,7 @@ class NotificationService {
return array_keys(Redis::zrevrangebyscore(self::CACHE_KEY.$id, '+inf', $end, [ return array_keys(Redis::zrevrangebyscore(self::CACHE_KEY.$id, '+inf', $end, [
'withscores' => true, 'withscores' => true,
'limit' => [0, $limit] 'limit' => [0, $limit],
])); ]));
} }
@ -207,6 +223,10 @@ class NotificationService {
$notification['type'] = 'reblog'; $notification['type'] = 'reblog';
} }
if ($notification['type'] === 'tagged') {
$notification['type'] = 'mention';
}
return $notification; return $notification;
} }
@ -215,12 +235,14 @@ class NotificationService {
if (self::count($id) > 400) { if (self::count($id) > 400) {
Redis::zpopmin(self::CACHE_KEY.$id); Redis::zpopmin(self::CACHE_KEY.$id);
} }
return Redis::zadd(self::CACHE_KEY.$id, $val, $val); return Redis::zadd(self::CACHE_KEY.$id, $val, $val);
} }
public static function del($id, $val) public static function del($id, $val)
{ {
Cache::forget('service:notification:'.$val); Cache::forget('service:notification:'.$val);
return Redis::zrem(self::CACHE_KEY.$id, $val); return Redis::zrem(self::CACHE_KEY.$id, $val);
} }
@ -257,6 +279,7 @@ class NotificationService {
$fractal = new Fractal\Manager(); $fractal = new Fractal\Manager();
$fractal->setSerializer(new ArraySerializer()); $fractal->setSerializer(new ArraySerializer());
$resource = new Fractal\Resource\Item($n, new NotificationTransformer()); $resource = new Fractal\Resource\Item($n, new NotificationTransformer());
return $fractal->createData($resource)->toArray(); return $fractal->createData($resource)->toArray();
}); });
@ -277,6 +300,7 @@ class NotificationService {
$fractal = new Fractal\Manager(); $fractal = new Fractal\Manager();
$fractal->setSerializer(new ArraySerializer()); $fractal->setSerializer(new ArraySerializer());
$resource = new Fractal\Resource\Item($notification, new NotificationTransformer()); $resource = new Fractal\Resource\Item($notification, new NotificationTransformer());
return $fractal->createData($resource)->toArray(); return $fractal->createData($resource)->toArray();
}); });
} }
@ -292,8 +316,10 @@ class NotificationService {
foreach ($ids as $key) { foreach ($ids as $key) {
self::set($id, $key); self::set($id, $key);
} }
return 1; return 1;
} }
return 0; return 0;
} }
} }

View file

@ -4,7 +4,6 @@ namespace App\Transformer\Api;
use App\Notification; use App\Notification;
use App\Services\AccountService; use App\Services\AccountService;
use App\Services\HashidService;
use App\Services\RelationshipService; use App\Services\RelationshipService;
use App\Services\StatusService; use App\Services\StatusService;
use League\Fractal; use League\Fractal;
@ -37,7 +36,7 @@ class NotificationTransformer extends Fractal\TransformerAbstract
if ($ml && $ml->object_uid) { if ($ml && $ml->object_uid) {
$res['modlog'] = [ $res['modlog'] = [
'id' => $ml->object_uid, 'id' => $ml->object_uid,
'url' => url('/i/admin/users/modlogs/' . $ml->object_uid) 'url' => url('/i/admin/users/modlogs/'.$ml->object_uid),
]; ];
} }
} }
@ -45,12 +44,17 @@ class NotificationTransformer extends Fractal\TransformerAbstract
if ($n->item_id && $n->item_type == 'App\MediaTag') { if ($n->item_id && $n->item_type == 'App\MediaTag') {
$ml = $n->item; $ml = $n->item;
if ($ml && $ml->tagged_username) { if ($ml && $ml->tagged_username) {
$np = StatusService::get($ml->status_id, false);
if ($np && isset($np['id'])) {
$res['tagged'] = [ $res['tagged'] = [
'username' => $ml->tagged_username, 'username' => $ml->tagged_username,
'post_url' => '/p/'.HashidService::encode($ml->status_id) 'post_url' => $np['url'],
'status_id' => $ml->status_id,
'profile_id' => $ml->profile_id,
]; ];
} }
} }
}
return $res; return $res;
} }
@ -64,7 +68,6 @@ class NotificationTransformer extends Fractal\TransformerAbstract
'reblog' => 'share', 'reblog' => 'share',
'share' => 'share', 'share' => 'share',
'like' => 'favourite', 'like' => 'favourite',
'group:like' => 'favourite',
'comment' => 'comment', 'comment' => 'comment',
'admin.user.modlog.comment' => 'modlog', 'admin.user.modlog.comment' => 'modlog',
'tagged' => 'tagged', 'tagged' => 'tagged',