mirror of
https://github.com/pixelfed/pixelfed.git
synced 2025-01-12 07:00:45 +00:00
Merge branch 'staging' of https://github.com/pixelfed/pixelfed into translation-coverage-extension
This commit is contained in:
commit
96d5eb184c
14 changed files with 585 additions and 409 deletions
|
@ -7,7 +7,7 @@ jobs:
|
||||||
build:
|
build:
|
||||||
docker:
|
docker:
|
||||||
# Specify the version you desire here
|
# Specify the version you desire here
|
||||||
- image: cimg/php:8.2.5
|
- image: cimg/php:8.3.8
|
||||||
|
|
||||||
# Specify service dependencies here if necessary
|
# Specify service dependencies here if necessary
|
||||||
# CircleCI maintains a library of pre-built images
|
# CircleCI maintains a library of pre-built images
|
||||||
|
|
|
@ -6,6 +6,9 @@
|
||||||
- Update ApiV1Controller, add support for notification filter types ([f61159a1](https://github.com/pixelfed/pixelfed/commit/f61159a1))
|
- Update ApiV1Controller, add support for notification filter types ([f61159a1](https://github.com/pixelfed/pixelfed/commit/f61159a1))
|
||||||
- Update ApiV1Dot1Controller, fix mutual api ([a8bb97b2](https://github.com/pixelfed/pixelfed/commit/a8bb97b2))
|
- Update ApiV1Dot1Controller, fix mutual api ([a8bb97b2](https://github.com/pixelfed/pixelfed/commit/a8bb97b2))
|
||||||
- Update ApiV1Controller, fix /api/v1/favourits pagination ([72f68160](https://github.com/pixelfed/pixelfed/commit/72f68160))
|
- Update ApiV1Controller, fix /api/v1/favourits pagination ([72f68160](https://github.com/pixelfed/pixelfed/commit/72f68160))
|
||||||
|
- Update RegisterController, update username constraints, require atleast one alpha char ([dd6e3cc2](https://github.com/pixelfed/pixelfed/commit/dd6e3cc2))
|
||||||
|
- Update AdminUser, fix entity casting ([cb5620d4](https://github.com/pixelfed/pixelfed/commit/cb5620d4))
|
||||||
|
- ([](https://github.com/pixelfed/pixelfed/commit/))
|
||||||
- ([](https://github.com/pixelfed/pixelfed/commit/))
|
- ([](https://github.com/pixelfed/pixelfed/commit/))
|
||||||
|
|
||||||
## [v0.12.3 (2024-07-01)](https://github.com/pixelfed/pixelfed/compare/v0.12.2...v0.12.3)
|
## [v0.12.3 (2024-07-01)](https://github.com/pixelfed/pixelfed/compare/v0.12.2...v0.12.3)
|
||||||
|
|
51
app/Console/Commands/DeleteRemoteProfile.php
Normal file
51
app/Console/Commands/DeleteRemoteProfile.php
Normal file
|
@ -0,0 +1,51 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Console\Commands;
|
||||||
|
|
||||||
|
use App\Jobs\DeletePipeline\DeleteRemoteProfilePipeline;
|
||||||
|
use App\Profile;
|
||||||
|
use Illuminate\Console\Command;
|
||||||
|
|
||||||
|
use function Laravel\Prompts\confirm;
|
||||||
|
use function Laravel\Prompts\search;
|
||||||
|
|
||||||
|
class DeleteRemoteProfile extends Command
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* The name and signature of the console command.
|
||||||
|
*
|
||||||
|
* @var string
|
||||||
|
*/
|
||||||
|
protected $signature = 'app:delete-remote-profile';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The console command description.
|
||||||
|
*
|
||||||
|
* @var string
|
||||||
|
*/
|
||||||
|
protected $description = 'Delete remote profile';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute the console command.
|
||||||
|
*/
|
||||||
|
public function handle()
|
||||||
|
{
|
||||||
|
$id = search(
|
||||||
|
'Search for the account',
|
||||||
|
fn (string $value) => strlen($value) > 2
|
||||||
|
? Profile::whereNotNull('domain')->where('username', 'like', $value.'%')->pluck('username', 'id')->all()
|
||||||
|
: []
|
||||||
|
);
|
||||||
|
$profile = Profile::whereNotNull('domain')->find($id);
|
||||||
|
|
||||||
|
if (! $profile) {
|
||||||
|
$this->error('Could not find profile.');
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
$confirmed = confirm('Are you sure you want to delete '.$profile->username.'\'s account? This action cannot be reversed.');
|
||||||
|
DeleteRemoteProfilePipeline::dispatch($profile)->onQueue('adelete');
|
||||||
|
$this->info('Dispatched delete job, it may take a few minutes...');
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
}
|
|
@ -2,39 +2,34 @@
|
||||||
|
|
||||||
namespace App\Http\Controllers\Api;
|
namespace App\Http\Controllers\Api;
|
||||||
|
|
||||||
use Illuminate\Http\Request;
|
use App\AccountInterstitial;
|
||||||
use App\Http\Controllers\Controller;
|
use App\Http\Controllers\Controller;
|
||||||
|
use App\Http\Resources\AdminInstance;
|
||||||
|
use App\Http\Resources\AdminUser;
|
||||||
|
use App\Instance;
|
||||||
|
use App\Jobs\DeletePipeline\DeleteAccountPipeline;
|
||||||
|
use App\Jobs\DeletePipeline\DeleteRemoteProfilePipeline;
|
||||||
use App\Jobs\StatusPipeline\StatusDelete;
|
use App\Jobs\StatusPipeline\StatusDelete;
|
||||||
use Auth, Cache, DB;
|
|
||||||
use Carbon\Carbon;
|
|
||||||
use App\{
|
|
||||||
AccountInterstitial,
|
|
||||||
Instance,
|
|
||||||
Like,
|
|
||||||
Notification,
|
|
||||||
Media,
|
|
||||||
Profile,
|
|
||||||
Report,
|
|
||||||
Status,
|
|
||||||
User
|
|
||||||
};
|
|
||||||
use App\Models\Conversation;
|
use App\Models\Conversation;
|
||||||
use App\Models\RemoteReport;
|
use App\Models\RemoteReport;
|
||||||
|
use App\Notification;
|
||||||
|
use App\Profile;
|
||||||
|
use App\Report;
|
||||||
use App\Services\AccountService;
|
use App\Services\AccountService;
|
||||||
use App\Services\AdminStatsService;
|
use App\Services\AdminStatsService;
|
||||||
use App\Services\ConfigCacheService;
|
use App\Services\ConfigCacheService;
|
||||||
use App\Services\InstanceService;
|
use App\Services\InstanceService;
|
||||||
use App\Services\ModLogService;
|
use App\Services\ModLogService;
|
||||||
use App\Services\SnowflakeService;
|
|
||||||
use App\Services\StatusService;
|
|
||||||
use App\Services\PublicTimelineService;
|
|
||||||
use App\Services\NetworkTimelineService;
|
use App\Services\NetworkTimelineService;
|
||||||
use App\Services\NotificationService;
|
use App\Services\NotificationService;
|
||||||
use App\Http\Resources\AdminInstance;
|
use App\Services\PublicTimelineService;
|
||||||
use App\Http\Resources\AdminUser;
|
use App\Services\SnowflakeService;
|
||||||
use App\Jobs\DeletePipeline\DeleteAccountPipeline;
|
use App\Services\StatusService;
|
||||||
use App\Jobs\DeletePipeline\DeleteRemoteProfilePipeline;
|
use App\Status;
|
||||||
use App\Jobs\DeletePipeline\DeleteRemoteStatusPipeline;
|
use App\User;
|
||||||
|
use Cache;
|
||||||
|
use DB;
|
||||||
|
use Illuminate\Http\Request;
|
||||||
|
|
||||||
class AdminApiController extends Controller
|
class AdminApiController extends Controller
|
||||||
{
|
{
|
||||||
|
@ -59,6 +54,7 @@ class AdminApiController extends Controller
|
||||||
$res['autospam_count'] = AccountInterstitial::whereType('post.autospam')
|
$res['autospam_count'] = AccountInterstitial::whereType('post.autospam')
|
||||||
->whereNull('appeal_handled_at')
|
->whereNull('appeal_handled_at')
|
||||||
->count();
|
->count();
|
||||||
|
|
||||||
return $res;
|
return $res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -79,7 +75,7 @@ class AdminApiController extends Controller
|
||||||
'type' => $report->type,
|
'type' => $report->type,
|
||||||
'item_id' => $report->item_id,
|
'item_id' => $report->item_id,
|
||||||
'item_type' => $report->item_type,
|
'item_type' => $report->item_type,
|
||||||
'created_at' => $report->created_at
|
'created_at' => $report->created_at,
|
||||||
];
|
];
|
||||||
if ($report->item_type === 'App\\Status') {
|
if ($report->item_type === 'App\\Status') {
|
||||||
$status = StatusService::get($report->item_id, false);
|
$status = StatusService::get($report->item_id, false);
|
||||||
|
@ -93,6 +89,7 @@ class AdminApiController extends Controller
|
||||||
$r['parent'] = StatusService::get($status['in_reply_to_id'], false);
|
$r['parent'] = StatusService::get($status['in_reply_to_id'], false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return $r;
|
return $r;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -108,7 +105,7 @@ class AdminApiController extends Controller
|
||||||
|
|
||||||
$this->validate($request, [
|
$this->validate($request, [
|
||||||
'action' => 'required|in:dismiss,approve,dismiss-all,approve-all,delete-post,delete-account',
|
'action' => 'required|in:dismiss,approve,dismiss-all,approve-all,delete-post,delete-account',
|
||||||
'id' => 'required'
|
'id' => 'required',
|
||||||
]);
|
]);
|
||||||
|
|
||||||
$action = $request->input('action');
|
$action = $request->input('action');
|
||||||
|
@ -130,6 +127,7 @@ class AdminApiController extends Controller
|
||||||
Cache::forget('pf:bouncer_v0:exemption_by_pid:'.$profile->id);
|
Cache::forget('pf:bouncer_v0:exemption_by_pid:'.$profile->id);
|
||||||
Cache::forget('pf:bouncer_v0:recent_by_pid:'.$profile->id);
|
Cache::forget('pf:bouncer_v0:recent_by_pid:'.$profile->id);
|
||||||
Cache::forget('admin-dash:reports:spam-count');
|
Cache::forget('admin-dash:reports:spam-count');
|
||||||
|
|
||||||
return $res;
|
return $res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -148,6 +146,7 @@ class AdminApiController extends Controller
|
||||||
PublicTimelineService::deleteByProfileId($profile->id);
|
PublicTimelineService::deleteByProfileId($profile->id);
|
||||||
StatusDelete::dispatch($appeal->status)->onQueue('high');
|
StatusDelete::dispatch($appeal->status)->onQueue('high');
|
||||||
Cache::forget('admin-dash:reports:spam-count');
|
Cache::forget('admin-dash:reports:spam-count');
|
||||||
|
|
||||||
return $res;
|
return $res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -167,6 +166,7 @@ class AdminApiController extends Controller
|
||||||
PublicTimelineService::deleteByProfileId($profile->id);
|
PublicTimelineService::deleteByProfileId($profile->id);
|
||||||
DeleteAccountPipeline::dispatch($appeal->user)->onQueue('high');
|
DeleteAccountPipeline::dispatch($appeal->user)->onQueue('high');
|
||||||
Cache::forget('admin-dash:reports:spam-count');
|
Cache::forget('admin-dash:reports:spam-count');
|
||||||
|
|
||||||
return $res;
|
return $res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -179,6 +179,7 @@ class AdminApiController extends Controller
|
||||||
Cache::forget('pf:bouncer_v0:exemption_by_pid:'.$appeal->user->profile_id);
|
Cache::forget('pf:bouncer_v0:exemption_by_pid:'.$appeal->user->profile_id);
|
||||||
Cache::forget('pf:bouncer_v0:recent_by_pid:'.$appeal->user->profile_id);
|
Cache::forget('pf:bouncer_v0:recent_by_pid:'.$appeal->user->profile_id);
|
||||||
Cache::forget('admin-dash:reports:spam-count');
|
Cache::forget('admin-dash:reports:spam-count');
|
||||||
|
|
||||||
return $res;
|
return $res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -206,6 +207,7 @@ class AdminApiController extends Controller
|
||||||
Cache::forget('pf:bouncer_v0:exemption_by_pid:'.$appeal->user->profile_id);
|
Cache::forget('pf:bouncer_v0:exemption_by_pid:'.$appeal->user->profile_id);
|
||||||
Cache::forget('pf:bouncer_v0:recent_by_pid:'.$appeal->user->profile_id);
|
Cache::forget('pf:bouncer_v0:recent_by_pid:'.$appeal->user->profile_id);
|
||||||
Cache::forget('admin-dash:reports:spam-count');
|
Cache::forget('admin-dash:reports:spam-count');
|
||||||
|
|
||||||
return $res;
|
return $res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -239,6 +241,7 @@ class AdminApiController extends Controller
|
||||||
Cache::forget('pf:bouncer_v0:exemption_by_pid:'.$appeal->user->profile_id);
|
Cache::forget('pf:bouncer_v0:exemption_by_pid:'.$appeal->user->profile_id);
|
||||||
Cache::forget('pf:bouncer_v0:recent_by_pid:'.$appeal->user->profile_id);
|
Cache::forget('pf:bouncer_v0:recent_by_pid:'.$appeal->user->profile_id);
|
||||||
Cache::forget('admin-dash:reports:spam-count');
|
Cache::forget('admin-dash:reports:spam-count');
|
||||||
|
|
||||||
return $res;
|
return $res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -262,7 +265,7 @@ class AdminApiController extends Controller
|
||||||
'message' => $report->message,
|
'message' => $report->message,
|
||||||
'object_id' => $report->object_id,
|
'object_id' => $report->object_id,
|
||||||
'object_type' => $report->object_type,
|
'object_type' => $report->object_type,
|
||||||
'created_at' => $report->created_at
|
'created_at' => $report->created_at,
|
||||||
];
|
];
|
||||||
|
|
||||||
if ($report->profile_id) {
|
if ($report->profile_id) {
|
||||||
|
@ -277,14 +280,18 @@ class AdminApiController extends Controller
|
||||||
|
|
||||||
$r['status'] = $status;
|
$r['status'] = $status;
|
||||||
|
|
||||||
if($status['in_reply_to_id']) {
|
if (isset($status['in_reply_to_id'])) {
|
||||||
$r['parent'] = StatusService::get($status['in_reply_to_id'], false);
|
$r['parent'] = StatusService::get($status['in_reply_to_id'], false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($report->object_type === 'App\\Profile') {
|
if ($report->object_type === 'App\\Profile') {
|
||||||
$r['account'] = AccountService::get($report->object_id, false);
|
$acct = AccountService::get($report->object_id, true);
|
||||||
|
if ($acct) {
|
||||||
|
$r['account'] = $acct;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return $r;
|
return $r;
|
||||||
})
|
})
|
||||||
->filter()
|
->filter()
|
||||||
|
@ -302,7 +309,7 @@ class AdminApiController extends Controller
|
||||||
|
|
||||||
$this->validate($request, [
|
$this->validate($request, [
|
||||||
'action' => 'required|string',
|
'action' => 'required|string',
|
||||||
'id' => 'required'
|
'id' => 'required',
|
||||||
]);
|
]);
|
||||||
|
|
||||||
$action = $request->input('action');
|
$action = $request->input('action');
|
||||||
|
@ -311,7 +318,7 @@ class AdminApiController extends Controller
|
||||||
$actions = [
|
$actions = [
|
||||||
'ignore',
|
'ignore',
|
||||||
'cw',
|
'cw',
|
||||||
'unlist'
|
'unlist',
|
||||||
];
|
];
|
||||||
|
|
||||||
if (! in_array($action, $actions)) {
|
if (! in_array($action, $actions)) {
|
||||||
|
@ -366,35 +373,36 @@ class AdminApiController extends Controller
|
||||||
[
|
[
|
||||||
'name' => 'ActivityPub Federation',
|
'name' => 'ActivityPub Federation',
|
||||||
'description' => 'Enable activitypub federation support, compatible with Pixelfed, Mastodon and other platforms.',
|
'description' => 'Enable activitypub federation support, compatible with Pixelfed, Mastodon and other platforms.',
|
||||||
'key' => 'federation.activitypub.enabled'
|
'key' => 'federation.activitypub.enabled',
|
||||||
],
|
],
|
||||||
|
|
||||||
[
|
[
|
||||||
'name' => 'Open Registration',
|
'name' => 'Open Registration',
|
||||||
'description' => 'Allow new account registrations.',
|
'description' => 'Allow new account registrations.',
|
||||||
'key' => 'pixelfed.open_registration'
|
'key' => 'pixelfed.open_registration',
|
||||||
],
|
],
|
||||||
|
|
||||||
[
|
[
|
||||||
'name' => 'Stories',
|
'name' => 'Stories',
|
||||||
'description' => 'Enable the ephemeral Stories feature.',
|
'description' => 'Enable the ephemeral Stories feature.',
|
||||||
'key' => 'instance.stories.enabled'
|
'key' => 'instance.stories.enabled',
|
||||||
],
|
],
|
||||||
|
|
||||||
[
|
[
|
||||||
'name' => 'Require Email Verification',
|
'name' => 'Require Email Verification',
|
||||||
'description' => 'Require new accounts to verify their email address.',
|
'description' => 'Require new accounts to verify their email address.',
|
||||||
'key' => 'pixelfed.enforce_email_verification'
|
'key' => 'pixelfed.enforce_email_verification',
|
||||||
],
|
],
|
||||||
|
|
||||||
[
|
[
|
||||||
'name' => 'AutoSpam Detection',
|
'name' => 'AutoSpam Detection',
|
||||||
'description' => 'Detect and remove spam from public timelines.',
|
'description' => 'Detect and remove spam from public timelines.',
|
||||||
'key' => 'pixelfed.bouncer.enabled'
|
'key' => 'pixelfed.bouncer.enabled',
|
||||||
],
|
],
|
||||||
])
|
])
|
||||||
->map(function ($s) {
|
->map(function ($s) {
|
||||||
$s['state'] = (bool) config_cache($s['key']);
|
$s['state'] = (bool) config_cache($s['key']);
|
||||||
|
|
||||||
return $s;
|
return $s;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -410,7 +418,7 @@ class AdminApiController extends Controller
|
||||||
|
|
||||||
$this->validate($request, [
|
$this->validate($request, [
|
||||||
'key' => 'required',
|
'key' => 'required',
|
||||||
'value' => 'required'
|
'value' => 'required',
|
||||||
]);
|
]);
|
||||||
|
|
||||||
$allowedKeys = [
|
$allowedKeys = [
|
||||||
|
@ -431,35 +439,36 @@ class AdminApiController extends Controller
|
||||||
[
|
[
|
||||||
'name' => 'ActivityPub Federation',
|
'name' => 'ActivityPub Federation',
|
||||||
'description' => 'Enable activitypub federation support, compatible with Pixelfed, Mastodon and other platforms.',
|
'description' => 'Enable activitypub federation support, compatible with Pixelfed, Mastodon and other platforms.',
|
||||||
'key' => 'federation.activitypub.enabled'
|
'key' => 'federation.activitypub.enabled',
|
||||||
],
|
],
|
||||||
|
|
||||||
[
|
[
|
||||||
'name' => 'Open Registration',
|
'name' => 'Open Registration',
|
||||||
'description' => 'Allow new account registrations.',
|
'description' => 'Allow new account registrations.',
|
||||||
'key' => 'pixelfed.open_registration'
|
'key' => 'pixelfed.open_registration',
|
||||||
],
|
],
|
||||||
|
|
||||||
[
|
[
|
||||||
'name' => 'Stories',
|
'name' => 'Stories',
|
||||||
'description' => 'Enable the ephemeral Stories feature.',
|
'description' => 'Enable the ephemeral Stories feature.',
|
||||||
'key' => 'instance.stories.enabled'
|
'key' => 'instance.stories.enabled',
|
||||||
],
|
],
|
||||||
|
|
||||||
[
|
[
|
||||||
'name' => 'Require Email Verification',
|
'name' => 'Require Email Verification',
|
||||||
'description' => 'Require new accounts to verify their email address.',
|
'description' => 'Require new accounts to verify their email address.',
|
||||||
'key' => 'pixelfed.enforce_email_verification'
|
'key' => 'pixelfed.enforce_email_verification',
|
||||||
],
|
],
|
||||||
|
|
||||||
[
|
[
|
||||||
'name' => 'AutoSpam Detection',
|
'name' => 'AutoSpam Detection',
|
||||||
'description' => 'Detect and remove spam from public timelines.',
|
'description' => 'Detect and remove spam from public timelines.',
|
||||||
'key' => 'pixelfed.bouncer.enabled'
|
'key' => 'pixelfed.bouncer.enabled',
|
||||||
],
|
],
|
||||||
])
|
])
|
||||||
->map(function ($s) {
|
->map(function ($s) {
|
||||||
$s['state'] = (bool) config_cache($s['key']);
|
$s['state'] = (bool) config_cache($s['key']);
|
||||||
|
|
||||||
return $s;
|
return $s;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -482,6 +491,7 @@ class AdminApiController extends Controller
|
||||||
})
|
})
|
||||||
->orderBy('id', $sort)
|
->orderBy('id', $sort)
|
||||||
->cursorPaginate(10);
|
->cursorPaginate(10);
|
||||||
|
|
||||||
return AdminUser::collection($res);
|
return AdminUser::collection($res);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -497,6 +507,7 @@ class AdminApiController extends Controller
|
||||||
if ($request->has('refresh')) {
|
if ($request->has('refresh')) {
|
||||||
Cache::forget($key);
|
Cache::forget($key);
|
||||||
}
|
}
|
||||||
|
|
||||||
return Cache::remember($key, 86400, function () use ($id) {
|
return Cache::remember($key, 86400, function () use ($id) {
|
||||||
$user = User::findOrFail($id);
|
$user = User::findOrFail($id);
|
||||||
$profile = $user->profile;
|
$profile = $user->profile;
|
||||||
|
@ -510,8 +521,8 @@ class AdminApiController extends Controller
|
||||||
'moderation' => [
|
'moderation' => [
|
||||||
'unlisted' => (bool) $profile->unlisted,
|
'unlisted' => (bool) $profile->unlisted,
|
||||||
'cw' => (bool) $profile->cw,
|
'cw' => (bool) $profile->cw,
|
||||||
'no_autolink' => (bool) $profile->no_autolink
|
'no_autolink' => (bool) $profile->no_autolink,
|
||||||
]
|
],
|
||||||
]]);
|
]]);
|
||||||
|
|
||||||
return $res;
|
return $res;
|
||||||
|
@ -528,7 +539,7 @@ class AdminApiController extends Controller
|
||||||
$this->validate($request, [
|
$this->validate($request, [
|
||||||
'id' => 'required',
|
'id' => 'required',
|
||||||
'action' => 'required|in:unlisted,cw,no_autolink,refresh_stats,verify_email,delete',
|
'action' => 'required|in:unlisted,cw,no_autolink,refresh_stats,verify_email,delete',
|
||||||
'value' => 'sometimes'
|
'value' => 'sometimes',
|
||||||
]);
|
]);
|
||||||
|
|
||||||
$id = $request->input('id');
|
$id = $request->input('id');
|
||||||
|
@ -586,6 +597,7 @@ class AdminApiController extends Controller
|
||||||
AccountService::del($profile->id);
|
AccountService::del($profile->id);
|
||||||
DeleteRemoteProfilePipeline::dispatch($profile)->onQueue('high');
|
DeleteRemoteProfilePipeline::dispatch($profile)->onQueue('high');
|
||||||
}
|
}
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'status' => 200,
|
'status' => 200,
|
||||||
'msg' => 'deleted',
|
'msg' => 'deleted',
|
||||||
|
@ -612,7 +624,7 @@ class AdminApiController extends Controller
|
||||||
->action('admin.user.moderate')
|
->action('admin.user.moderate')
|
||||||
->metadata([
|
->metadata([
|
||||||
'action' => 'Manually verified email address',
|
'action' => 'Manually verified email address',
|
||||||
'message' => 'Success!'
|
'message' => 'Success!',
|
||||||
])
|
])
|
||||||
->accessLevel('admin')
|
->accessLevel('admin')
|
||||||
->save();
|
->save();
|
||||||
|
@ -625,7 +637,7 @@ class AdminApiController extends Controller
|
||||||
->action('admin.user.moderate')
|
->action('admin.user.moderate')
|
||||||
->metadata([
|
->metadata([
|
||||||
'action' => $action,
|
'action' => $action,
|
||||||
'message' => 'Success!'
|
'message' => 'Success!',
|
||||||
])
|
])
|
||||||
->accessLevel('admin')
|
->accessLevel('admin')
|
||||||
->save();
|
->save();
|
||||||
|
@ -640,7 +652,7 @@ class AdminApiController extends Controller
|
||||||
->action('admin.user.moderate')
|
->action('admin.user.moderate')
|
||||||
->metadata([
|
->metadata([
|
||||||
'action' => $action,
|
'action' => $action,
|
||||||
'message' => 'Success!'
|
'message' => 'Success!',
|
||||||
])
|
])
|
||||||
->accessLevel('admin')
|
->accessLevel('admin')
|
||||||
->save();
|
->save();
|
||||||
|
@ -655,7 +667,7 @@ class AdminApiController extends Controller
|
||||||
->action('admin.user.moderate')
|
->action('admin.user.moderate')
|
||||||
->metadata([
|
->metadata([
|
||||||
'action' => $action,
|
'action' => $action,
|
||||||
'message' => 'Success!'
|
'message' => 'Success!',
|
||||||
])
|
])
|
||||||
->accessLevel('admin')
|
->accessLevel('admin')
|
||||||
->save();
|
->save();
|
||||||
|
@ -673,7 +685,7 @@ class AdminApiController extends Controller
|
||||||
->action('admin.user.moderate')
|
->action('admin.user.moderate')
|
||||||
->metadata([
|
->metadata([
|
||||||
'action' => $action,
|
'action' => $action,
|
||||||
'message' => 'Success!'
|
'message' => 'Success!',
|
||||||
])
|
])
|
||||||
->accessLevel('admin')
|
->accessLevel('admin')
|
||||||
->save();
|
->save();
|
||||||
|
@ -687,8 +699,8 @@ class AdminApiController extends Controller
|
||||||
'moderation' => [
|
'moderation' => [
|
||||||
'unlisted' => (bool) $profile->unlisted,
|
'unlisted' => (bool) $profile->unlisted,
|
||||||
'cw' => (bool) $profile->cw,
|
'cw' => (bool) $profile->cw,
|
||||||
'no_autolink' => (bool) $profile->no_autolink
|
'no_autolink' => (bool) $profile->no_autolink,
|
||||||
]
|
],
|
||||||
]]);
|
]]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -755,7 +767,7 @@ class AdminApiController extends Controller
|
||||||
$this->validate($request, [
|
$this->validate($request, [
|
||||||
'id' => 'required',
|
'id' => 'required',
|
||||||
'key' => 'required|in:unlisted,auto_cw,banned',
|
'key' => 'required|in:unlisted,auto_cw,banned',
|
||||||
'value' => 'required'
|
'value' => 'required',
|
||||||
]);
|
]);
|
||||||
|
|
||||||
$id = $request->input('id');
|
$id = $request->input('id');
|
||||||
|
@ -816,28 +828,28 @@ class AdminApiController extends Controller
|
||||||
'date' => now()->subDays($day)->format('M j Y'),
|
'date' => now()->subDays($day)->format('M j Y'),
|
||||||
'label_full' => $label,
|
'label_full' => $label,
|
||||||
'label' => $labelShort,
|
'label' => $labelShort,
|
||||||
'count' => User::whereDate('created_at', now()->subDays($day))->count()
|
'count' => User::whereDate('created_at', now()->subDays($day))->count(),
|
||||||
];
|
];
|
||||||
|
|
||||||
$res['posts']['days'][] = [
|
$res['posts']['days'][] = [
|
||||||
'date' => now()->subDays($day)->format('M j Y'),
|
'date' => now()->subDays($day)->format('M j Y'),
|
||||||
'label_full' => $label,
|
'label_full' => $label,
|
||||||
'label' => $labelShort,
|
'label' => $labelShort,
|
||||||
'count' => Status::whereNull('uri')->where('id', '>', $minStatusId)->whereDate('created_at', now()->subDays($day))->count()
|
'count' => Status::whereNull('uri')->where('id', '>', $minStatusId)->whereDate('created_at', now()->subDays($day))->count(),
|
||||||
];
|
];
|
||||||
|
|
||||||
$res['instances']['days'][] = [
|
$res['instances']['days'][] = [
|
||||||
'date' => now()->subDays($day)->format('M j Y'),
|
'date' => now()->subDays($day)->format('M j Y'),
|
||||||
'label_full' => $label,
|
'label_full' => $label,
|
||||||
'label' => $labelShort,
|
'label' => $labelShort,
|
||||||
'count' => Instance::whereDate('created_at', now()->subDays($day))->count()
|
'count' => Instance::whereDate('created_at', now()->subDays($day))->count(),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
$res['users']['total'] = DB::table('users')->count();
|
$res['users']['total'] = DB::table('users')->count();
|
||||||
$res['users']['min'] = collect($res['users']['days'])->min('count');
|
$res['users']['min'] = collect($res['users']['days'])->min('count');
|
||||||
$res['users']['max'] = collect($res['users']['days'])->max('count');
|
$res['users']['max'] = collect($res['users']['days'])->max('count');
|
||||||
$res['users']['change'] = collect($res['users']['days'])->sum('count');;
|
$res['users']['change'] = collect($res['users']['days'])->sum('count');
|
||||||
$res['posts']['total'] = DB::table('statuses')->whereNull('uri')->count();
|
$res['posts']['total'] = DB::table('statuses')->whereNull('uri')->count();
|
||||||
$res['posts']['min'] = collect($res['posts']['days'])->min('count');
|
$res['posts']['min'] = collect($res['posts']['days'])->min('count');
|
||||||
$res['posts']['max'] = collect($res['posts']['days'])->max('count');
|
$res['posts']['max'] = collect($res['posts']['days'])->max('count');
|
||||||
|
|
|
@ -3805,11 +3805,11 @@ class ApiV1Controller extends Controller
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ($i['visibility'] === 'private') {
|
// if ($i['visibility'] === 'private') {
|
||||||
if ((int) $i['account']['id'] !== $pid) {
|
// if ((int) $i['account']['id'] !== $pid) {
|
||||||
return FollowerService::follows($pid, $i['account']['id'], true);
|
// return FollowerService::follows($pid, $i['account']['id'], true);
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
if ($onlyMedia == true) {
|
if ($onlyMedia == true) {
|
||||||
if (! isset($i['media_attachments']) || ! count($i['media_attachments'])) {
|
if (! isset($i['media_attachments']) || ! count($i['media_attachments'])) {
|
||||||
return false;
|
return false;
|
||||||
|
|
|
@ -2,8 +2,8 @@
|
||||||
|
|
||||||
namespace App\Http\Resources;
|
namespace App\Http\Resources;
|
||||||
|
|
||||||
use Illuminate\Http\Resources\Json\JsonResource;
|
|
||||||
use App\Services\AccountService;
|
use App\Services\AccountService;
|
||||||
|
use Illuminate\Http\Resources\Json\JsonResource;
|
||||||
|
|
||||||
class AdminUser extends JsonResource
|
class AdminUser extends JsonResource
|
||||||
{
|
{
|
||||||
|
@ -18,8 +18,8 @@ class AdminUser extends JsonResource
|
||||||
$account = AccountService::get($this->profile_id, true);
|
$account = AccountService::get($this->profile_id, true);
|
||||||
|
|
||||||
$res = [
|
$res = [
|
||||||
'id' => $this->id,
|
'id' => (string) $this->id,
|
||||||
'profile_id' => $this->profile_id,
|
'profile_id' => (string) $this->profile_id,
|
||||||
'name' => $this->name,
|
'name' => $this->name,
|
||||||
'username' => $this->username,
|
'username' => $this->username,
|
||||||
'is_admin' => (bool) $this->is_admin,
|
'is_admin' => (bool) $this->is_admin,
|
||||||
|
@ -28,6 +28,7 @@ class AdminUser extends JsonResource
|
||||||
'two_factor_enabled' => (bool) $this->{'2fa_enabled'},
|
'two_factor_enabled' => (bool) $this->{'2fa_enabled'},
|
||||||
'register_source' => $this->register_source,
|
'register_source' => $this->register_source,
|
||||||
'app_register_ip' => $this->app_register_ip,
|
'app_register_ip' => $this->app_register_ip,
|
||||||
|
'has_interstitial' => (bool) $this->has_interstitial,
|
||||||
'last_active_at' => $this->last_active_at,
|
'last_active_at' => $this->last_active_at,
|
||||||
'created_at' => $this->created_at,
|
'created_at' => $this->created_at,
|
||||||
];
|
];
|
||||||
|
@ -35,10 +36,10 @@ class AdminUser extends JsonResource
|
||||||
if ($account) {
|
if ($account) {
|
||||||
$res['avatar'] = $account['avatar'];
|
$res['avatar'] = $account['avatar'];
|
||||||
$res['bio'] = $account['note_text'];
|
$res['bio'] = $account['note_text'];
|
||||||
$res['statuses_count'] = $account['statuses_count'];
|
$res['statuses_count'] = (int) $account['statuses_count'];
|
||||||
$res['following_count'] = $account['following_count'];
|
$res['following_count'] = (int) $account['following_count'];
|
||||||
$res['followers_count'] = $account['followers_count'];
|
$res['followers_count'] = (int) $account['followers_count'];
|
||||||
$res['is_private'] = $account['locked'];
|
$res['is_private'] = (bool) $account['locked'];
|
||||||
}
|
}
|
||||||
|
|
||||||
return $res;
|
return $res;
|
||||||
|
|
|
@ -109,7 +109,7 @@ class RemoteStatusDelete implements ShouldQueue, ShouldBeUniqueUntilProcessing
|
||||||
}
|
}
|
||||||
|
|
||||||
StatusService::del($status->id, true);
|
StatusService::del($status->id, true);
|
||||||
AccountStatService::decrementPostCount($status->profile_id);
|
// AccountStatService::decrementPostCount($status->profile_id);
|
||||||
return $this->unlinkRemoveMedia($status);
|
return $this->unlinkRemoveMedia($status);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -176,11 +176,11 @@ class RemoteStatusDelete implements ShouldQueue, ShouldBeUniqueUntilProcessing
|
||||||
StatusView::whereStatusId($status->id)->delete();
|
StatusView::whereStatusId($status->id)->delete();
|
||||||
Status::whereInReplyToId($status->id)->update(['in_reply_to_id' => null]);
|
Status::whereInReplyToId($status->id)->update(['in_reply_to_id' => null]);
|
||||||
|
|
||||||
$status->delete();
|
|
||||||
|
|
||||||
StatusService::del($status->id, true);
|
StatusService::del($status->id, true);
|
||||||
AccountService::del($status->profile_id);
|
AccountService::del($status->profile_id);
|
||||||
|
|
||||||
|
$status->forceDelete();
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
10
app/User.php
10
app/User.php
|
@ -8,10 +8,13 @@ use Illuminate\Database\Eloquent\SoftDeletes;
|
||||||
use Illuminate\Foundation\Auth\User as Authenticatable;
|
use Illuminate\Foundation\Auth\User as Authenticatable;
|
||||||
use App\Util\RateLimit\User as UserRateLimit;
|
use App\Util\RateLimit\User as UserRateLimit;
|
||||||
use App\Services\AvatarService;
|
use App\Services\AvatarService;
|
||||||
|
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
||||||
|
use NotificationChannels\WebPush\HasPushSubscriptions;
|
||||||
|
use NotificationChannels\Expo\ExpoPushToken;
|
||||||
|
|
||||||
class User extends Authenticatable
|
class User extends Authenticatable
|
||||||
{
|
{
|
||||||
use Notifiable, SoftDeletes, HasApiTokens, UserRateLimit;
|
use Notifiable, SoftDeletes, HasApiTokens, UserRateLimit, HasFactory, HasPushSubscriptions;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The attributes that should be mutated to dates.
|
* The attributes that should be mutated to dates.
|
||||||
|
@ -23,6 +26,7 @@ class User extends Authenticatable
|
||||||
'email_verified_at' => 'datetime',
|
'email_verified_at' => 'datetime',
|
||||||
'2fa_setup_at' => 'datetime',
|
'2fa_setup_at' => 'datetime',
|
||||||
'last_active_at' => 'datetime',
|
'last_active_at' => 'datetime',
|
||||||
|
'expo_token' => ExpoPushToken::class
|
||||||
];
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -115,4 +119,8 @@ class User extends Authenticatable
|
||||||
return AvatarService::get($this->profile_id);
|
return AvatarService::get($this->profile_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function routeNotificationForExpo(): ?ExpoPushToken
|
||||||
|
{
|
||||||
|
return $this->expo_token;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,6 +19,7 @@
|
||||||
"doctrine/dbal": "^3.0",
|
"doctrine/dbal": "^3.0",
|
||||||
"intervention/image": "^2.4",
|
"intervention/image": "^2.4",
|
||||||
"jenssegers/agent": "^2.6",
|
"jenssegers/agent": "^2.6",
|
||||||
|
"laravel-notification-channels/expo": "^1.3.0|^2.0",
|
||||||
"laravel-notification-channels/webpush": "^8.0",
|
"laravel-notification-channels/webpush": "^8.0",
|
||||||
"laravel/framework": "^11.0",
|
"laravel/framework": "^11.0",
|
||||||
"laravel/helpers": "^1.1",
|
"laravel/helpers": "^1.1",
|
||||||
|
|
562
composer.lock
generated
562
composer.lock
generated
File diff suppressed because it is too large
Load diff
|
@ -90,6 +90,7 @@ return [
|
||||||
'redis:story' => 30,
|
'redis:story' => 30,
|
||||||
'redis:mmo' => 30,
|
'redis:mmo' => 30,
|
||||||
'redis:intbg' => 30,
|
'redis:intbg' => 30,
|
||||||
|
'redis:adelete' => 30,
|
||||||
'redis:groups' => 30,
|
'redis:groups' => 30,
|
||||||
],
|
],
|
||||||
|
|
||||||
|
@ -174,7 +175,7 @@ return [
|
||||||
'production' => [
|
'production' => [
|
||||||
'supervisor-1' => [
|
'supervisor-1' => [
|
||||||
'connection' => 'redis',
|
'connection' => 'redis',
|
||||||
'queue' => ['high', 'default', 'follow', 'shared', 'inbox', 'feed', 'low', 'story', 'delete', 'mmo', 'intbg', 'groups'],
|
'queue' => ['high', 'default', 'follow', 'shared', 'inbox', 'feed', 'low', 'story', 'delete', 'mmo', 'intbg', 'groups', 'adelete'],
|
||||||
'balance' => env('HORIZON_BALANCE_STRATEGY', 'auto'),
|
'balance' => env('HORIZON_BALANCE_STRATEGY', 'auto'),
|
||||||
'minProcesses' => env('HORIZON_MIN_PROCESSES', 1),
|
'minProcesses' => env('HORIZON_MIN_PROCESSES', 1),
|
||||||
'maxProcesses' => env('HORIZON_MAX_PROCESSES', 20),
|
'maxProcesses' => env('HORIZON_MAX_PROCESSES', 20),
|
||||||
|
@ -188,7 +189,7 @@ return [
|
||||||
'local' => [
|
'local' => [
|
||||||
'supervisor-1' => [
|
'supervisor-1' => [
|
||||||
'connection' => 'redis',
|
'connection' => 'redis',
|
||||||
'queue' => ['high', 'default', 'follow', 'shared', 'inbox', 'feed', 'low', 'story', 'delete', 'mmo', 'intbg', 'groups'],
|
'queue' => ['high', 'default', 'follow', 'shared', 'inbox', 'feed', 'low', 'story', 'delete', 'mmo', 'intbg', 'groups', 'adelete'],
|
||||||
'balance' => 'auto',
|
'balance' => 'auto',
|
||||||
'minProcesses' => 1,
|
'minProcesses' => 1,
|
||||||
'maxProcesses' => 20,
|
'maxProcesses' => 20,
|
||||||
|
|
|
@ -36,7 +36,7 @@ return [
|
||||||
'network' => [
|
'network' => [
|
||||||
'cached' => env('PF_NETWORK_TIMELINE') ? env('INSTANCE_NETWORK_TIMELINE_CACHED', false) : false,
|
'cached' => env('PF_NETWORK_TIMELINE') ? env('INSTANCE_NETWORK_TIMELINE_CACHED', false) : false,
|
||||||
'cache_dropoff' => env('INSTANCE_NETWORK_TIMELINE_CACHE_DROPOFF', 100),
|
'cache_dropoff' => env('INSTANCE_NETWORK_TIMELINE_CACHE_DROPOFF', 100),
|
||||||
'max_hours_old' => env('INSTANCE_NETWORK_TIMELINE_CACHE_MAX_HOUR_INGEST', 6),
|
'max_hours_old' => env('INSTANCE_NETWORK_TIMELINE_CACHE_MAX_HOUR_INGEST', 2160),
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
|
|
||||||
|
|
|
@ -35,4 +35,7 @@ return [
|
||||||
'secret' => env('STRIPE_SECRET'),
|
'secret' => env('STRIPE_SECRET'),
|
||||||
],
|
],
|
||||||
|
|
||||||
|
'expo' => [
|
||||||
|
'access_token' => env('EXPO_ACCESS_TOKEN'),
|
||||||
|
],
|
||||||
];
|
];
|
||||||
|
|
|
@ -0,0 +1,36 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
use Illuminate\Database\Migrations\Migration;
|
||||||
|
use Illuminate\Database\Schema\Blueprint;
|
||||||
|
use Illuminate\Support\Facades\Schema;
|
||||||
|
|
||||||
|
return new class extends Migration
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Run the migrations.
|
||||||
|
*/
|
||||||
|
public function up(): void
|
||||||
|
{
|
||||||
|
Schema::table('users', function (Blueprint $table) {
|
||||||
|
$table->string('expo_token')->nullable();
|
||||||
|
$table->boolean('notify_like')->default(true);
|
||||||
|
$table->boolean('notify_follow')->default(true);
|
||||||
|
$table->boolean('notify_mention')->default(true);
|
||||||
|
$table->boolean('notify_comment')->default(true);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reverse the migrations.
|
||||||
|
*/
|
||||||
|
public function down(): void
|
||||||
|
{
|
||||||
|
Schema::table('users', function (Blueprint $table) {
|
||||||
|
$table->dropColumn('expo_token');
|
||||||
|
$table->dropColumn('notify_like');
|
||||||
|
$table->dropColumn('notify_follow');
|
||||||
|
$table->dropColumn('notify_mention');
|
||||||
|
$table->dropColumn('notify_comment');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
Loading…
Reference in a new issue