不明なContentTypeや406はOGPにフォールバックする

This commit is contained in:
unarist 2019-02-11 01:43:07 +09:00
parent 5750eeb3a5
commit 1ba4999a83

View File

@ -2,6 +2,9 @@
namespace App\MetadataResolver; namespace App\MetadataResolver;
use GuzzleHttp\Exception\ClientException;
use GuzzleHttp\Exception\ServerException;
class MetadataResolver implements Resolver class MetadataResolver implements Resolver
{ {
public $rules = [ public $rules = [
@ -23,9 +26,12 @@ class MetadataResolver implements Resolver
public $mimeTypes = [ public $mimeTypes = [
'application/activity+json' => ActivityPubResolver::class, 'application/activity+json' => ActivityPubResolver::class,
'application/ld+json' => ActivityPubResolver::class, 'application/ld+json' => ActivityPubResolver::class,
'text/html' => OGPResolver::class 'text/html' => OGPResolver::class,
'*/*' => OGPResolver::class
]; ];
public $defaultResolver = OGPResolver::class;
public function resolve(string $url): Metadata public function resolve(string $url): Metadata
{ {
foreach ($this->rules as $pattern => $class) { foreach ($this->rules as $pattern => $class) {
@ -36,6 +42,22 @@ class MetadataResolver implements Resolver
} }
} }
$result = $this->resolveWithAcceptHeader($url);
if ($result !== null) {
return $result;
}
if (isset($this->defaultResolver)) {
$resolver = new $this->defaultResolver();
return $resolver->resolve($url);
}
throw new \UnexpectedValueException('URL not matched.');
}
public function resolveWithAcceptHeader(string $url): ?Metadata
{
try {
$client = new \GuzzleHttp\Client(); $client = new \GuzzleHttp\Client();
$res = $client->request('GET', $url, [ $res = $client->request('GET', $url, [
'headers' => [ 'headers' => [
@ -52,11 +74,26 @@ class MetadataResolver implements Resolver
$parser = new $class(); $parser = new $class();
return $parser->parse($res->getBody()); return $parser->parse($res->getBody());
} else { }
throw new \UnexpectedValueException('URL not matched.');
if (isset($this->mimeTypes['*/*'])) {
$class = $this->mimeTypes['*/*'];
$parser = new $class();
return $parser->parse($res->getBody());
} }
} else { } else {
throw new \RuntimeException("{$res->getStatusCode()}: $url"); // code < 400 && code !== 200 => fallback
} }
} catch (ClientException $e) {
// 406 Not Acceptable は多分Acceptが原因なので無視してフォールバック
if ($e->getResponse()->getStatusCode() !== 406) {
throw $e;
}
} catch (ServerException $e) {
// 5xx は変なAcceptが原因かもしれないので無視してフォールバック
}
return null;
} }
} }