forked from mirror/pixelfed
256 lines
7.4 KiB
PHP
256 lines
7.4 KiB
PHP
<?php
|
|
|
|
namespace App\Http\Controllers\Admin;
|
|
|
|
use DB, Cache;
|
|
use App\{
|
|
AccountInterstitial,
|
|
DiscoverCategory,
|
|
DiscoverCategoryHashtag,
|
|
Hashtag,
|
|
Media,
|
|
Profile,
|
|
Status,
|
|
StatusHashtag,
|
|
User
|
|
};
|
|
use App\Models\ConfigCache;
|
|
use App\Models\AutospamCustomTokens;
|
|
use App\Services\AccountService;
|
|
use App\Services\ConfigCacheService;
|
|
use App\Services\StatusService;
|
|
use Carbon\Carbon;
|
|
use Illuminate\Http\Request;
|
|
use Illuminate\Validation\Rule;
|
|
use League\ISO3166\ISO3166;
|
|
use Illuminate\Support\Str;
|
|
use Illuminate\Support\Facades\Storage;
|
|
use Illuminate\Support\Facades\Validator;
|
|
use Illuminate\Support\Facades\Http;
|
|
use App\Http\Controllers\PixelfedDirectoryController;
|
|
use \DateInterval;
|
|
use \DatePeriod;
|
|
use App\Http\Resources\AdminSpamReport;
|
|
use App\Util\Lexer\Classifier;
|
|
use App\Jobs\AutospamPipeline\AutospamPretrainPipeline;
|
|
use App\Jobs\AutospamPipeline\AutospamPretrainNonSpamPipeline;
|
|
use App\Jobs\AutospamPipeline\AutospamUpdateCachedDataPipeline;
|
|
use Illuminate\Support\Facades\URL;
|
|
use App\Services\AutospamService;
|
|
|
|
trait AdminAutospamController
|
|
{
|
|
public function autospamHome(Request $request)
|
|
{
|
|
return view('admin.autospam.home');
|
|
}
|
|
|
|
public function getAutospamConfigApi(Request $request)
|
|
{
|
|
$open = Cache::remember('admin-dash:reports:spam-count', 3600, function() {
|
|
return AccountInterstitial::whereType('post.autospam')->whereNull('appeal_handled_at')->count();
|
|
});
|
|
|
|
$closed = Cache::remember('admin-dash:reports:spam-count-closed', 3600, function() {
|
|
return AccountInterstitial::whereType('post.autospam')->whereNotNull('appeal_handled_at')->count();
|
|
});
|
|
|
|
$thisWeek = Cache::remember('admin-dash:reports:spam-count-stats-this-week ', 86400, function() {
|
|
$sr = config('database.default') == 'pgsql' ? "to_char(created_at, 'MM-YYYY')" : "DATE_FORMAT(created_at, '%m-%Y')";
|
|
$gb = config('database.default') == 'pgsql' ? [DB::raw($sr)] : DB::raw($sr);
|
|
$s = AccountInterstitial::select(
|
|
DB::raw('count(id) as count'),
|
|
DB::raw($sr . " as month_year")
|
|
)
|
|
->where('created_at', '>=', now()->subWeeks(52))
|
|
->groupBy($gb)
|
|
->get()
|
|
->map(function($s) {
|
|
$dt = now()->parse('01-' . $s->month_year);
|
|
return [
|
|
'id' => $dt->format('Ym'),
|
|
'x' => $dt->format('M Y'),
|
|
'y' => $s->count
|
|
];
|
|
})
|
|
->sortBy('id')
|
|
->values()
|
|
->toArray();
|
|
return $s;
|
|
});
|
|
|
|
$files = [
|
|
'spam' => [
|
|
'exists' => Storage::exists(AutospamService::MODEL_SPAM_PATH),
|
|
'size' => 0
|
|
],
|
|
'ham' => [
|
|
'exists' => Storage::exists(AutospamService::MODEL_HAM_PATH),
|
|
'size' => 0
|
|
],
|
|
'combined' => [
|
|
'exists' => Storage::exists(AutospamService::MODEL_FILE_PATH),
|
|
'size' => 0
|
|
]
|
|
];
|
|
|
|
if($files['spam']['exists']) {
|
|
$files['spam']['size'] = Storage::size(AutospamService::MODEL_SPAM_PATH);
|
|
}
|
|
|
|
if($files['ham']['exists']) {
|
|
$files['ham']['size'] = Storage::size(AutospamService::MODEL_HAM_PATH);
|
|
}
|
|
|
|
if($files['combined']['exists']) {
|
|
$files['combined']['size'] = Storage::size(AutospamService::MODEL_FILE_PATH);
|
|
}
|
|
|
|
return [
|
|
'autospam_enabled' => (bool) config_cache('pixelfed.bouncer.enabled') ?? false,
|
|
'nlp_enabled' => (bool) AutospamService::active(),
|
|
'files' => $files,
|
|
'open' => $open,
|
|
'closed' => $closed,
|
|
'graph' => collect($thisWeek)->map(fn($s) => $s['y'])->values(),
|
|
'graphLabels' => collect($thisWeek)->map(fn($s) => $s['x'])->values()
|
|
];
|
|
}
|
|
|
|
public function getAutospamReportsClosedApi(Request $request)
|
|
{
|
|
$appeals = AdminSpamReport::collection(
|
|
AccountInterstitial::orderBy('id', 'desc')
|
|
->whereType('post.autospam')
|
|
->whereIsSpam(true)
|
|
->whereNotNull('appeal_handled_at')
|
|
->cursorPaginate(6)
|
|
->withQueryString()
|
|
);
|
|
|
|
return $appeals;
|
|
}
|
|
|
|
public function postAutospamTrainSpamApi(Request $request)
|
|
{
|
|
$aiCount = AccountInterstitial::whereItemType('App\Status')
|
|
->whereIsSpam(true)
|
|
->count();
|
|
abort_if($aiCount < 100, 422, 'You don\'t have enough data to pre-train against.');
|
|
|
|
$existing = Cache::get('pf:admin:autospam:pretrain:recent');
|
|
abort_if($existing, 422, 'You\'ve already run this recently, please wait 30 minutes before pre-training again');
|
|
AutospamPretrainPipeline::dispatch();
|
|
Cache::put('pf:admin:autospam:pretrain:recent', 1, 1440);
|
|
|
|
return [
|
|
'msg' => 'Success!'
|
|
];
|
|
}
|
|
|
|
public function postAutospamTrainNonSpamSearchApi(Request $request)
|
|
{
|
|
$this->validate($request, [
|
|
'q' => 'required|string|min:1'
|
|
]);
|
|
|
|
$q = $request->input('q');
|
|
|
|
$res = Profile::whereNull(['status', 'domain'])
|
|
->where('username', 'like', '%' . $q . '%')
|
|
->orderByDesc('followers_count')
|
|
->take(10)
|
|
->get()
|
|
->map(function($p) {
|
|
$acct = AccountService::get($p->id, true);
|
|
return [
|
|
'id' => (string) $p->id,
|
|
'avatar' => $acct['avatar'],
|
|
'username' => $p->username
|
|
];
|
|
})
|
|
->values();
|
|
return $res;
|
|
}
|
|
|
|
public function postAutospamTrainNonSpamSubmitApi(Request $request)
|
|
{
|
|
$this->validate($request, [
|
|
'accounts' => 'required|array|min:1|max:10'
|
|
]);
|
|
|
|
$accts = $request->input('accounts');
|
|
|
|
$accounts = Profile::whereNull(['domain', 'status'])->find(collect($accts)->map(function($a) { return $a['id'];}));
|
|
|
|
abort_if(!$accounts || !$accounts->count(), 422, 'One or more of the selected accounts are not valid');
|
|
|
|
AutospamPretrainNonSpamPipeline::dispatch($accounts);
|
|
return $accounts;
|
|
}
|
|
|
|
public function getAutospamCustomTokensApi(Request $request)
|
|
{
|
|
return AutospamCustomTokens::latest()->cursorPaginate(6);
|
|
}
|
|
|
|
public function saveNewAutospamCustomTokensApi(Request $request)
|
|
{
|
|
$this->validate($request, [
|
|
'token' => 'required|unique:autospam_custom_tokens,token',
|
|
]);
|
|
|
|
$ct = new AutospamCustomTokens;
|
|
$ct->token = $request->input('token');
|
|
$ct->weight = $request->input('weight');
|
|
$ct->category = $request->input('category') === 'spam' ? 'spam' : 'ham';
|
|
$ct->note = $request->input('note');
|
|
$ct->active = $request->input('active');
|
|
$ct->save();
|
|
|
|
AutospamUpdateCachedDataPipeline::dispatch();
|
|
return $ct;
|
|
}
|
|
|
|
public function updateAutospamCustomTokensApi(Request $request)
|
|
{
|
|
$this->validate($request, [
|
|
'id' => 'required',
|
|
'token' => 'required',
|
|
'category' => 'required|in:spam,ham',
|
|
'active' => 'required|boolean'
|
|
]);
|
|
|
|
$ct = AutospamCustomTokens::findOrFail($request->input('id'));
|
|
$ct->weight = $request->input('weight');
|
|
$ct->category = $request->input('category');
|
|
$ct->note = $request->input('note');
|
|
$ct->active = $request->input('active');
|
|
$ct->save();
|
|
|
|
AutospamUpdateCachedDataPipeline::dispatch();
|
|
|
|
return $ct;
|
|
}
|
|
|
|
public function exportAutospamCustomTokensApi(Request $request)
|
|
{
|
|
abort_if(!Storage::exists(AutospamService::MODEL_SPAM_PATH), 422, 'Autospam Dataset does not exist, please train spam before attempting to export');
|
|
return Storage::download(AutospamService::MODEL_SPAM_PATH);
|
|
}
|
|
|
|
public function enableAutospamApi(Request $request)
|
|
{
|
|
ConfigCacheService::put('autospam.nlp.enabled', true);
|
|
Cache::forget(AutospamService::CHCKD_CACHE_KEY);
|
|
return ['msg' => 'Success'];
|
|
}
|
|
|
|
public function disableAutospamApi(Request $request)
|
|
{
|
|
ConfigCacheService::put('autospam.nlp.enabled', false);
|
|
Cache::forget(AutospamService::CHCKD_CACHE_KEY);
|
|
return ['msg' => 'Success'];
|
|
}
|
|
}
|