Update SearchController, add WebfingerService support

This commit is contained in:
Daniel Supernault 2020-03-24 22:54:19 -06:00
parent 801502382b
commit 869b4ff727
No known key found for this signature in database
GPG key ID: 0DEF1C662C9033F7

View file

@ -15,9 +15,15 @@ use App\Transformer\Api\{
HashtagTransformer, HashtagTransformer,
StatusTransformer, StatusTransformer,
}; };
use App\Services\WebfingerService;
class SearchController extends Controller class SearchController extends Controller
{ {
public $tokens = [];
public $term = '';
public $hash = '';
public $cacheKey = 'api:search:tag:';
public function __construct() public function __construct()
{ {
$this->middleware('auth'); $this->middleware('auth');
@ -28,50 +34,98 @@ class SearchController extends Controller
$this->validate($request, [ $this->validate($request, [
'q' => 'required|string|min:3|max:120', 'q' => 'required|string|min:3|max:120',
'src' => 'required|string|in:metro', 'src' => 'required|string|in:metro',
'v' => 'required|integer|in:1' 'v' => 'required|integer|in:1',
'scope' => 'required|in:all,hashtag,profile,remote,webfinger'
]); ]);
$tag = $request->input('q');
$tag = e(urldecode($tag));
$scope = $request->input('scope') ?? 'all';
$this->term = e(urldecode($request->input('q')));
$this->hash = hash('sha256', $this->term);
switch ($scope) {
case 'all':
$this->getHashtags();
$this->getPosts();
$this->getProfiles();
break;
case 'hashtag':
$this->getHashtags();
break;
case 'profile':
$this->getProfiles();
break;
case 'webfinger':
$this->webfingerSearch();
break;
default:
break;
}
return response()->json($this->tokens, 200, [], JSON_PRETTY_PRINT);
}
protected function getPosts()
{
$tag = $this->term;
$hash = hash('sha256', $tag); $hash = hash('sha256', $tag);
$tokens = Cache::remember('api:search:tag:'.$hash, now()->addMinutes(5), function () use ($tag) { if( Helpers::validateUrl($tag) != false &&
$tokens = []; Helpers::validateLocalUrl($tag) != true &&
if(Helpers::validateUrl($tag) != false && config('federation.activitypub.enabled') == true && config('federation.activitypub.remoteFollow') == true) { config('federation.activitypub.enabled') == true &&
abort_if(Helpers::validateLocalUrl($tag), 404); config('federation.activitypub.remoteFollow') == true
$remote = Helpers::fetchFromUrl($tag); ) {
if(isset($remote['type']) && in_array($remote['type'], ['Note', 'Person']) == true) { $remote = Helpers::fetchFromUrl($tag);
$type = $remote['type']; if( isset($remote['type']) &&
if($type == 'Person') { $remote['type'] == 'Note') {
$item = Helpers::profileFirstOrNew($tag); $item = Helpers::statusFetch($tag);
$tokens['profiles'] = [[ $this->tokens['posts'] = [[
'count' => 1, 'count' => 0,
'url' => $item->url(), 'url' => $item->url(),
'type' => 'profile', 'type' => 'status',
'value' => $item->username, 'value' => "by {$item->profile->username} <span class='float-right'>{$item->created_at->diffForHumans(null, true, true)}</span>",
'tokens' => [$item->username], 'tokens' => [$item->caption],
'name' => $item->name, 'name' => $item->caption,
'entity' => [ 'thumb' => $item->thumb(),
'id' => (string) $item->id, ]];
'following' => $item->followedBy(Auth::user()->profile),
'follow_request' => $item->hasFollowRequestById(Auth::user()->profile_id),
'thumb' => $item->avatarUrl(),
'local' => (bool) !$item->domain
]
]];
} else if ($type == 'Note') {
$item = Helpers::statusFetch($tag);
$tokens['posts'] = [[
'count' => 0,
'url' => $item->url(),
'type' => 'status',
'value' => "by {$item->profile->username} <span class='float-right'>{$item->created_at->diffForHumans(null, true, true)}</span>",
'tokens' => [$item->caption],
'name' => $item->caption,
'thumb' => $item->thumb(),
]];
}
}
} }
} else {
$posts = Status::select('id', 'profile_id', 'caption', 'created_at')
->whereHas('media')
->whereNull('in_reply_to_id')
->whereNull('reblog_of_id')
->whereProfileId(Auth::user()->profile_id)
->where('caption', 'like', '%'.$tag.'%')
->latest()
->limit(10)
->get();
if($posts->count() > 0) {
$posts = $posts->map(function($item, $key) {
return [
'count' => 0,
'url' => $item->url(),
'type' => 'status',
'value' => "by {$item->profile->username} <span class='float-right'>{$item->created_at->diffForHumans(null, true, true)}</span>",
'tokens' => [$item->caption],
'name' => $item->caption,
'thumb' => $item->thumb(),
'filter' => $item->firstMedia()->filter_class
];
});
$this->tokens['posts'] = $posts;
}
}
}
protected function getHashtags()
{
$tag = $this->term;
$key = $this->cacheKey . 'hashtags:' . $this->hash;
$ttl = now()->addMinutes(1);
$tokens = Cache::remember($key, $ttl, function() use($tag) {
$htag = Str::startsWith($tag, '#') == true ? mb_substr($tag, 1) : $tag; $htag = Str::startsWith($tag, '#') == true ? mb_substr($tag, 1) : $tag;
$hashtags = Hashtag::select('id', 'name', 'slug') $hashtags = Hashtag::select('id', 'name', 'slug')
->where('slug', 'like', '%'.$htag.'%') ->where('slug', 'like', '%'.$htag.'%')
@ -89,72 +143,93 @@ class SearchController extends Controller
'name' => null, 'name' => null,
]; ];
}); });
$tokens['hashtags'] = $tags; return $tags;
} }
return $tokens;
}); });
$users = Profile::select('domain', 'username', 'name', 'id') $this->tokens['hashtags'] = $tokens;
->whereNull('status') }
->whereNull('domain')
->where('id', '!=', Auth::user()->profile->id)
->where('username', 'like', '%'.$tag.'%')
//->orWhere('remote_url', $tag)
->limit(20)
->get();
if($users->count() > 0) { protected function getProfiles()
$profiles = $users->map(function ($item, $key) { {
return [ $tag = $this->term;
'count' => 0, $remoteKey = $this->cacheKey . 'profiles:remote:' . $this->hash;
'url' => $item->url(), $key = $this->cacheKey . 'profiles:' . $this->hash;
'type' => 'profile', $remoteTtl = now()->addMinutes(15);
'value' => $item->username, $ttl = now()->addHours(2);
'tokens' => [$item->username], if( Helpers::validateUrl($tag) != false &&
'name' => $item->name, Helpers::validateLocalUrl($tag) != true &&
'avatar' => $item->avatarUrl(), config('federation.activitypub.enabled') == true &&
'id' => $item->id, config('federation.activitypub.remoteFollow') == true
'entity' => [ ) {
'id' => (string) $item->id, $remote = Helpers::fetchFromUrl($tag);
'following' => $item->followedBy(Auth::user()->profile), if( isset($remote['type']) &&
'follow_request' => $item->hasFollowRequestById(Auth::user()->profile_id), $remote['type'] == 'Person'
'thumb' => $item->avatarUrl(), ) {
'local' => (bool) !$item->domain $this->tokens['profiles'] = Cache::remember($remoteKey, $remoteTtl, function() use($tag) {
] $item = Helpers::profileFirstOrNew($tag);
]; $tokens = [[
}); 'count' => 1,
if(isset($tokens['profiles'])) { 'url' => $item->url(),
array_push($tokens['profiles'], $profiles); 'type' => 'profile',
} else { 'value' => $item->username,
$tokens['profiles'] = $profiles; 'tokens' => [$item->username],
'name' => $item->name,
'entity' => [
'id' => (string) $item->id,
'following' => $item->followedBy(Auth::user()->profile),
'follow_request' => $item->hasFollowRequestById(Auth::user()->profile_id),
'thumb' => $item->avatarUrl(),
'local' => (bool) !$item->domain
]
]];
return $tokens;
});
} }
} }
$posts = Status::select('id', 'profile_id', 'caption', 'created_at') // elseif( Str::containsAll($tag, ['@', '.'])
->whereHas('media') // config('federation.activitypub.enabled') == true &&
->whereNull('in_reply_to_id') // config('federation.activitypub.remoteFollow') == true
->whereNull('reblog_of_id') // ) {
->whereProfileId(Auth::user()->profile->id) // if(substr_count($tag, '@') == 2) {
->where('caption', 'like', '%'.$tag.'%') // $domain = last(explode('@', sub_str($u, 1)));
->latest() // } else {
->limit(10) // $domain = last(explode('@', $u));
// }
// }
else {
$this->tokens['profiles'] = Cache::remember($key, $ttl, function() use($tag) {
$users = Profile::select('domain', 'username', 'name', 'id')
->whereNull('status')
->where('id', '!=', Auth::user()->profile->id)
->where('username', 'like', '%'.$tag.'%')
->limit(20)
->get(); ->get();
if($posts->count() > 0) { if($users->count() > 0) {
$posts = $posts->map(function($item, $key) { return $users->map(function ($item, $key) {
return [ return [
'count' => 0, 'count' => 0,
'url' => $item->url(), 'url' => $item->url(),
'type' => 'status', 'type' => 'profile',
'value' => "by {$item->profile->username} <span class='float-right'>{$item->created_at->diffForHumans(null, true, true)}</span>", 'value' => $item->username,
'tokens' => [$item->caption], 'tokens' => [$item->username],
'name' => $item->caption, 'name' => $item->name,
'thumb' => $item->thumb(), 'avatar' => $item->avatarUrl(),
'filter' => $item->firstMedia()->filter_class 'id' => (string) $item->id,
]; 'entity' => [
'id' => (string) $item->id,
'following' => $item->followedBy(Auth::user()->profile),
'follow_request' => $item->hasFollowRequestById(Auth::user()->profile_id),
'thumb' => $item->avatarUrl(),
'local' => (bool) !$item->domain,
'post_count' => $item->statuses()->count()
]
];
});
}
}); });
$tokens['posts'] = $posts;
} }
return response()->json($tokens);
} }
public function results(Request $request) public function results(Request $request)
@ -166,4 +241,31 @@ class SearchController extends Controller
return view('search.results'); return view('search.results');
} }
} protected function webfingerSearch()
{
$wfs = WebfingerService::lookup($this->term);
if(empty($wfs)) {
return;
}
$this->tokens['profiles'] = [
[
'count' => 1,
'url' => $wfs['url'],
'type' => 'profile',
'value' => $wfs['username'],
'tokens' => [$wfs['username']],
'name' => $wfs['display_name'],
'entity' => [
'id' => (string) $wfs['id'],
'following' => null,
'follow_request' => null,
'thumb' => $wfs['avatar'],
'local' => (bool) $wfs['local']
]
]
];
return;
}
}