forked from mirror/pixelfed
168 lines
5.9 KiB
PHP
168 lines
5.9 KiB
PHP
|
<?php
|
||
|
|
||
|
namespace App\Http\Controllers;
|
||
|
|
||
|
use Illuminate\Http\Request;
|
||
|
use App\Models\ConfigCache;
|
||
|
use Storage;
|
||
|
use App\Services\AccountService;
|
||
|
use App\Services\StatusService;
|
||
|
use Illuminate\Support\Str;
|
||
|
|
||
|
class PixelfedDirectoryController extends Controller
|
||
|
{
|
||
|
public function get(Request $request)
|
||
|
{
|
||
|
if(!$request->filled('sk')) {
|
||
|
abort(404);
|
||
|
}
|
||
|
|
||
|
if(!config_cache('pixelfed.directory.submission-key')) {
|
||
|
abort(404);
|
||
|
}
|
||
|
|
||
|
if(!hash_equals(config_cache('pixelfed.directory.submission-key'), $request->input('sk'))) {
|
||
|
abort(403);
|
||
|
}
|
||
|
|
||
|
$res = $this->buildListing();
|
||
|
return response()->json($res, 200, [], JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
|
||
|
}
|
||
|
|
||
|
public function buildListing()
|
||
|
{
|
||
|
$res = config_cache('pixelfed.directory');
|
||
|
if($res) {
|
||
|
$res = is_string($res) ? json_decode($res, true) : $res;
|
||
|
}
|
||
|
|
||
|
$res['_domain'] = config_cache('pixelfed.domain.app');
|
||
|
$res['_sk'] = config_cache('pixelfed.directory.submission-key');
|
||
|
$res['_ts'] = config_cache('pixelfed.directory.submission-ts');
|
||
|
$res['version'] = config_cache('pixelfed.version');
|
||
|
|
||
|
if(empty($res['summary'])) {
|
||
|
$summary = ConfigCache::whereK('app.short_description')->pluck('v');
|
||
|
$res['summary'] = $summary ? $summary[0] : null;
|
||
|
}
|
||
|
|
||
|
if(isset($res['admin'])) {
|
||
|
$res['admin'] = AccountService::get($res['admin'], true);
|
||
|
}
|
||
|
|
||
|
if(isset($res['banner_image']) && !empty($res['banner_image'])) {
|
||
|
$res['banner_image'] = url(Storage::url($res['banner_image']));
|
||
|
}
|
||
|
|
||
|
if(isset($res['favourite_posts'])) {
|
||
|
$res['favourite_posts'] = collect($res['favourite_posts'])->map(function($id) {
|
||
|
return StatusService::get($id);
|
||
|
})
|
||
|
->filter(function($post) {
|
||
|
return $post && isset($post['account']);
|
||
|
})
|
||
|
->map(function($post) {
|
||
|
return [
|
||
|
'avatar' => $post['account']['avatar'],
|
||
|
'display_name' => $post['account']['display_name'],
|
||
|
'username' => $post['account']['username'],
|
||
|
'media' => $post['media_attachments'][0]['url'],
|
||
|
'url' => $post['url']
|
||
|
];
|
||
|
})
|
||
|
->values();
|
||
|
}
|
||
|
|
||
|
$guidelines = ConfigCache::whereK('app.rules')->first();
|
||
|
if($guidelines) {
|
||
|
$res['community_guidelines'] = json_decode($guidelines->v, true);
|
||
|
}
|
||
|
|
||
|
$openRegistration = ConfigCache::whereK('pixelfed.open_registration')->first();
|
||
|
if($openRegistration) {
|
||
|
$res['open_registration'] = (bool) $openRegistration;
|
||
|
}
|
||
|
|
||
|
$oauthEnabled = ConfigCache::whereK('pixelfed.oauth_enabled')->first();
|
||
|
if($oauthEnabled) {
|
||
|
$keys = file_exists(storage_path('oauth-public.key')) && file_exists(storage_path('oauth-private.key'));
|
||
|
$res['oauth_enabled'] = (bool) $oauthEnabled && $keys;
|
||
|
}
|
||
|
|
||
|
$activityPubEnabled = ConfigCache::whereK('federation.activitypub.enabled')->first();
|
||
|
if($activityPubEnabled) {
|
||
|
$res['activitypub_enabled'] = (bool) $activityPubEnabled;
|
||
|
}
|
||
|
|
||
|
$res['feature_config'] = [
|
||
|
'media_types' => Str::of(config_cache('pixelfed.media_types'))->explode(','),
|
||
|
'image_quality' => config_cache('pixelfed.image_quality'),
|
||
|
'optimize_image' => config_cache('pixelfed.optimize_image'),
|
||
|
'max_photo_size' => config_cache('pixelfed.max_photo_size'),
|
||
|
'max_caption_length' => config_cache('pixelfed.max_caption_length'),
|
||
|
'max_altext_length' => config_cache('pixelfed.max_altext_length'),
|
||
|
'enforce_account_limit' => config_cache('pixelfed.enforce_account_limit'),
|
||
|
'max_account_size' => config_cache('pixelfed.max_account_size'),
|
||
|
'max_album_length' => config_cache('pixelfed.max_album_length'),
|
||
|
'account_deletion' => config_cache('pixelfed.account_deletion'),
|
||
|
];
|
||
|
|
||
|
$res['is_eligible'] = $this->validVal($res, 'admin') &&
|
||
|
$this->validVal($res, 'summary', null, 10) &&
|
||
|
$this->validVal($res, 'favourite_posts', 3) &&
|
||
|
$this->validVal($res, 'contact_email') &&
|
||
|
$this->validVal($res, 'privacy_pledge') &&
|
||
|
$this->validVal($res, 'location');
|
||
|
|
||
|
if(config_cache('pixelfed.directory.testimonials')) {
|
||
|
$res['testimonials'] = collect(json_decode(config_cache('pixelfed.directory.testimonials'), true))
|
||
|
->map(function($testimonial) {
|
||
|
$profile = AccountService::get($testimonial['profile_id']);
|
||
|
return [
|
||
|
'profile' => [
|
||
|
'username' => $profile['username'],
|
||
|
'display_name' => $profile['display_name'],
|
||
|
'avatar' => $profile['avatar'],
|
||
|
'created_at' => $profile['created_at']
|
||
|
],
|
||
|
'body' => $testimonial['body']
|
||
|
];
|
||
|
});
|
||
|
}
|
||
|
|
||
|
$res['features_enabled'] = [
|
||
|
'stories' => (bool) config_cache('instance.stories.enabled')
|
||
|
];
|
||
|
|
||
|
$res['stats'] = [
|
||
|
'user_count' => \App\User::count(),
|
||
|
'post_count' => \App\Status::whereNull('uri')->count(),
|
||
|
];
|
||
|
|
||
|
$res['primary_locale'] = config('app.locale');
|
||
|
$hash = hash('sha256', json_encode($res));
|
||
|
$res['_hash'] = $hash;
|
||
|
ksort($res);
|
||
|
|
||
|
return $res;
|
||
|
}
|
||
|
|
||
|
protected function validVal($res, $val, $count = false, $minLen = false)
|
||
|
{
|
||
|
if(!isset($res[$val])) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if($count) {
|
||
|
return count($res[$val]) >= $count;
|
||
|
}
|
||
|
|
||
|
if($minLen) {
|
||
|
return strlen($res[$val]) >= $minLen;
|
||
|
}
|
||
|
|
||
|
return $res[$val];
|
||
|
}
|
||
|
|
||
|
}
|