Work on webmwtion code refactoring
This commit is contained in:
parent
84c7969a4e
commit
a9f089098c
8 changed files with 395 additions and 248 deletions
|
@ -7,6 +7,8 @@ use Twitter;
|
||||||
use App\Tag;
|
use App\Tag;
|
||||||
use App\Note;
|
use App\Note;
|
||||||
use Jonnybarnes\IndieWeb\Numbers;
|
use Jonnybarnes\IndieWeb\Numbers;
|
||||||
|
use Illuminate\Filesystem\Filesystem;
|
||||||
|
use Jonnybarnes\WebmentionsParser\Authorship;
|
||||||
|
|
||||||
// Need to sort out Twitter and webmentions!
|
// Need to sort out Twitter and webmentions!
|
||||||
|
|
||||||
|
@ -23,8 +25,8 @@ class NotesController extends Controller
|
||||||
foreach ($notes as $note) {
|
foreach ($notes as $note) {
|
||||||
$replies = 0;
|
$replies = 0;
|
||||||
foreach ($note->webmentions as $webmention) {
|
foreach ($note->webmentions as $webmention) {
|
||||||
if ($webmention->type == 'reply') {
|
if ($webmention->type == 'in-reply-to') {
|
||||||
$replies = $replies + 1;
|
$replies++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$note->replies = $replies;
|
$note->replies = $replies;
|
||||||
|
@ -67,31 +69,51 @@ class NotesController extends Controller
|
||||||
public function singleNote($urlId)
|
public function singleNote($urlId)
|
||||||
{
|
{
|
||||||
$numbers = new Numbers();
|
$numbers = new Numbers();
|
||||||
|
$authorship = new Authorship();
|
||||||
$realId = $numbers->b60tonum($urlId);
|
$realId = $numbers->b60tonum($urlId);
|
||||||
$note = Note::find($realId);
|
$note = Note::find($realId);
|
||||||
$replies = [];
|
$replies = [];
|
||||||
$reposts = [];
|
$reposts = [];
|
||||||
$likes = [];
|
$likes = [];
|
||||||
foreach ($note->webmentions as $webmention) {
|
foreach ($note->webmentions as $webmention) {
|
||||||
|
/*
|
||||||
|
reply->url |
|
||||||
|
reply->photo | Author
|
||||||
|
reply->name |
|
||||||
|
reply->source
|
||||||
|
reply->date
|
||||||
|
reply->reply
|
||||||
|
|
||||||
|
repost->url |
|
||||||
|
repost->photo | Author
|
||||||
|
repost->name |
|
||||||
|
repost->date
|
||||||
|
repost->source
|
||||||
|
|
||||||
|
like->url |
|
||||||
|
like->photo | Author
|
||||||
|
like->name |
|
||||||
|
*/
|
||||||
|
$microformats = json_decode($webmention->mf2);
|
||||||
|
$authorHCard = $authorship->findAuthor($microformats);
|
||||||
|
$content['url'] = $authorHCard['properties']['url'][0];
|
||||||
|
$content['photo'] = $this->createPhotoLink($authorHCard['properties']['photo'][0]);
|
||||||
|
$content['name'] = $authorHCard['properties']['name'][0];
|
||||||
switch ($webmention->type) {
|
switch ($webmention->type) {
|
||||||
case 'reply':
|
case 'in-reply-to':
|
||||||
$content = unserialize($webmention->content);
|
$content['source'] = $webmention->source;
|
||||||
$content['source'] = $this->bridgyReply($webmention->source);
|
|
||||||
$content['photo'] = $this->createPhotoLink($content['photo']);
|
|
||||||
$content['date'] = $carbon->parse($content['date'])->toDayDateTimeString();
|
$content['date'] = $carbon->parse($content['date'])->toDayDateTimeString();
|
||||||
|
$content['reply'] = $microformats['items'][0]['properties']['content'][0]['html_purified'];
|
||||||
$replies[] = $content;
|
$replies[] = $content;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'repost':
|
case 'repost-of':
|
||||||
$content = unserialize($webmention->content);
|
|
||||||
$content['photo'] = $this->createPhotoLink($content['photo']);
|
|
||||||
$content['date'] = $carbon->parse($content['date'])->toDayDateTimeString();
|
$content['date'] = $carbon->parse($content['date'])->toDayDateTimeString();
|
||||||
|
$content['source'] = $webmention->source;
|
||||||
$reposts[] = $content;
|
$reposts[] = $content;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'like':
|
case 'like-of':
|
||||||
$content = unserialize($webmention->content);
|
|
||||||
$content['photo'] = $this->createPhotoLink($content['photo']);
|
|
||||||
$likes[] = $content;
|
$likes[] = $content;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -164,41 +186,43 @@ class NotesController extends Controller
|
||||||
return view('taggednotes', ['notes' => $notes, 'tag' => $tag]);
|
return view('taggednotes', ['notes' => $notes, 'tag' => $tag]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Swap a brid.gy URL shim-ing a twitter reply to a real twitter link.
|
|
||||||
*
|
|
||||||
* @param string
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function bridgyReply($source)
|
|
||||||
{
|
|
||||||
$url = $source;
|
|
||||||
if (mb_substr($source, 0, 28, 'UTF-8') == 'https://brid-gy.appspot.com/') {
|
|
||||||
$parts = explode('/', $source);
|
|
||||||
$tweetId = array_pop($parts);
|
|
||||||
if ($tweetId) {
|
|
||||||
$url = 'https://twitter.com/_/status/' . $tweetId;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $url;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create the photo link.
|
* Create the photo link.
|
||||||
*
|
*
|
||||||
|
* We shall leave twitter.com and twimg.com links as they are. Then we shall
|
||||||
|
* check for local copies, if that fails leave the link as is.
|
||||||
|
*
|
||||||
* @param string
|
* @param string
|
||||||
* @return string
|
* @return string
|
||||||
*/
|
*/
|
||||||
public function createPhotoLink($url)
|
public function createPhotoLink($url)
|
||||||
{
|
{
|
||||||
$host = parse_url($url)['host'];
|
$host = parse_url($url, PHP_URL_HOST);
|
||||||
if ($host != 'twitter.com' && $host != 'pbs.twimg.com') {
|
if ($host == 'pbs.twimg.com') {
|
||||||
return '/assets/profile-images/' . $host . '/image';
|
//make sure we use HTTPS, we know twitter supports it
|
||||||
}
|
|
||||||
if (mb_substr($url, 0, 20) == 'http://pbs.twimg.com') {
|
|
||||||
return str_replace('http://', 'https://', $url);
|
return str_replace('http://', 'https://', $url);
|
||||||
}
|
}
|
||||||
|
if ($host == 'twitter.com') {
|
||||||
|
if (Cache::has($url)) {
|
||||||
|
return Cache::get($url);
|
||||||
|
}
|
||||||
|
$username = parse_url($url, PHP_URL_PATH);
|
||||||
|
try {
|
||||||
|
$info = Twitter::getUsers(['screen_name' => $username]);
|
||||||
|
$profile_image = $info->profile_image_url_https;
|
||||||
|
Cache::put($url, $profile_image, 10080); //1 week
|
||||||
|
} catch (Exception $e) {
|
||||||
|
return $url; //not sure here
|
||||||
|
}
|
||||||
|
|
||||||
|
return $profile_image;
|
||||||
|
}
|
||||||
|
$filesystem = new Filesystem();
|
||||||
|
if ($filesystem->exists(public_path() . '/assets/profile-images/' . $host . '/image')) {
|
||||||
|
return '/assets/profile-images/' . $host . '/image';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $url;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -13,13 +13,16 @@ use Illuminate\Queue\InteractsWithQueue;
|
||||||
use Jonnybarnes\WebmentionsParser\Parser;
|
use Jonnybarnes\WebmentionsParser\Parser;
|
||||||
use GuzzleHttp\Exception\RequestException;
|
use GuzzleHttp\Exception\RequestException;
|
||||||
use Illuminate\Contracts\Queue\ShouldQueue;
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||||
|
use Illuminate\Foundation\Bus\DispatchesJobs;
|
||||||
|
use App\Exceptions\RemoteContentNotFoundException;
|
||||||
|
|
||||||
class ProcessWebMention extends Job implements ShouldQueue
|
class ProcessWebMention extends Job implements ShouldQueue
|
||||||
{
|
{
|
||||||
use InteractsWithQueue, SerializesModels;
|
use InteractsWithQueue, SerializesModels, DispatchesJobs;
|
||||||
|
|
||||||
protected $note;
|
protected $note;
|
||||||
protected $source;
|
protected $source;
|
||||||
|
protected $guzzle;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new job instance.
|
* Create a new job instance.
|
||||||
|
@ -28,10 +31,11 @@ class ProcessWebMention extends Job implements ShouldQueue
|
||||||
* @param string $source
|
* @param string $source
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function __construct(Note $note, $source)
|
public function __construct(Note $note, $source, Client $guzzle = null)
|
||||||
{
|
{
|
||||||
$this->note = $note;
|
$this->note = $note;
|
||||||
$this->source = $source;
|
$this->source = $source;
|
||||||
|
$this->guzzle = $guzzle ?? new Client();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -46,100 +50,60 @@ class ProcessWebMention extends Job implements ShouldQueue
|
||||||
$baseURL = $sourceURL['scheme'] . '://' . $sourceURL['host'];
|
$baseURL = $sourceURL['scheme'] . '://' . $sourceURL['host'];
|
||||||
$remoteContent = $this->getRemoteContent($this->source);
|
$remoteContent = $this->getRemoteContent($this->source);
|
||||||
if ($remoteContent === null) {
|
if ($remoteContent === null) {
|
||||||
return false;
|
throw new RemoteContentNotFoundException;
|
||||||
}
|
}
|
||||||
$mf2Parser = new Mf2\Parser($remoteContent, $baseURL, true);
|
$microformats = Mf2\parse($remoteContent, $baseURL);
|
||||||
$microformats = $mf2Parser->parse();
|
$webmentions = WebMention::where('source', $this->source)->get();
|
||||||
$count = WebMention::where('source', '=', $this->source)->count();
|
|
||||||
if ($count > 0) {
|
|
||||||
//we already have a webmention from this source
|
|
||||||
$webmentions = WebMention::where('source', '=', $this->source)->get();
|
|
||||||
foreach ($webmentions as $webmention) {
|
foreach ($webmentions as $webmention) {
|
||||||
//now check it still 'mentions' this target
|
//check webmention still references target
|
||||||
//we switch for each type of mention (reply/like/repost)
|
//we try each type of mention (reply/like/repost)
|
||||||
switch ($webmention->type) {
|
if ($webmention->type == 'in-reply-to') {
|
||||||
case 'reply':
|
if ($parser->checkInReplyTo($microformats, $this->note->longurl) == false) {
|
||||||
if ($parser->checkInReplyTo($microformats, $note->longurl) == false) {
|
|
||||||
//it doesn't so delete
|
//it doesn't so delete
|
||||||
$webmention->delete();
|
$webmention->delete();
|
||||||
|
|
||||||
return true;
|
return;
|
||||||
}
|
}
|
||||||
//webmenion is still a reply, so update content
|
//webmenion is still a reply, so update content
|
||||||
$content = $parser->replyContent($microformats);
|
$microformats = $this->filterHTML($microformats);
|
||||||
$this->saveImage($content);
|
$this->dispatch(new SaveProfileImage($microformats));
|
||||||
$content['reply'] = $this->filterHTML($content['reply']);
|
$webmention->mf2 = json_encode($microformats);
|
||||||
$content = serialize($content);
|
|
||||||
$webmention->content = $content;
|
|
||||||
$webmention->save();
|
$webmention->save();
|
||||||
|
|
||||||
return true;
|
return;
|
||||||
break;
|
}
|
||||||
case 'like':
|
if ($webmention->type == 'like-of') {
|
||||||
if ($parser->checkLikeOf($microformats, $note->longurl) == false) {
|
if ($parser->checkLikeOf($microformats, $note->longurl) == false) {
|
||||||
//it doesn't so delete
|
//it doesn't so delete
|
||||||
$webmention->delete();
|
$webmention->delete();
|
||||||
|
|
||||||
return true;
|
return;
|
||||||
} //note we don't need to do anything if it still is a like
|
} //note we don't need to do anything if it still is a like
|
||||||
break;
|
}
|
||||||
case 'repost':
|
if ($webmention->type == 'repost-of') {
|
||||||
if ($parser->checkRepostOf($microformats, $note->longurl) == false) {
|
if ($parser->checkRepostOf($microformats, $note->longurl) == false) {
|
||||||
//it doesn't so delete
|
//it doesn't so delete
|
||||||
$webmention->delete();
|
$webmention->delete();
|
||||||
|
|
||||||
return true;
|
return;
|
||||||
} //again, we don't need to do anything if it still is a repost
|
} //again, we don't need to do anything if it still is a repost
|
||||||
break;
|
}
|
||||||
}//switch
|
|
||||||
}//foreach
|
}//foreach
|
||||||
}//if
|
|
||||||
//no wemention in db so create new one
|
//no wemention in db so create new one
|
||||||
$webmention = new WebMention();
|
$webmention = new WebMention();
|
||||||
//check it is in fact a reply
|
$type = $parser->getMentionType($microformats); //throw error here?
|
||||||
if ($parser->checkInReplyTo($microformats, $note->longurl)) {
|
$this->dispatch(new SaveProfileImage($microformats));
|
||||||
$content = $parser->replyContent($microformats);
|
$microformats = $this->filterHTML($microformats);
|
||||||
$this->saveImage($content);
|
|
||||||
$content['reply'] = $this->filterHTML($content['reply']);
|
|
||||||
$content = serialize($content);
|
|
||||||
$webmention->source = $this->source;
|
$webmention->source = $this->source;
|
||||||
$webmention->target = $note->longurl;
|
$webmention->target = $this->note->longurl;
|
||||||
$webmention->commentable_id = $this->note->id;
|
$webmention->commentable_id = $this->note->id;
|
||||||
$webmention->commentable_type = 'App\Note';
|
$webmention->commentable_type = 'App\Note';
|
||||||
$webmention->type = 'reply';
|
$webmention->type = $type;
|
||||||
$webmention->content = $content;
|
$webmention->mf2 = json_encode($microformats);
|
||||||
$webmention->save();
|
$webmention->save();
|
||||||
|
|
||||||
return true;
|
return;
|
||||||
} elseif ($parser->checkLikeOf($microformats, $note->longurl)) {
|
|
||||||
//it is a like
|
|
||||||
$content = $parser->likeContent($microformats);
|
|
||||||
$this->saveImage($content);
|
|
||||||
$content = serialize($content);
|
|
||||||
$webmention->source = $this->source;
|
|
||||||
$webmention->target = $note->longurl;
|
|
||||||
$webmention->commentable_id = $this->note->id;
|
|
||||||
$webmention->commentable_type = 'App\Note';
|
|
||||||
$webmention->type = 'like';
|
|
||||||
$webmention->content = $content;
|
|
||||||
$webmention->save();
|
|
||||||
|
|
||||||
return true;
|
|
||||||
} elseif ($parser->checkRepostOf($microformats, $note->longurl)) {
|
|
||||||
//it is a repost
|
|
||||||
$content = $parser->repostContent($microformats);
|
|
||||||
$this->saveImage($content);
|
|
||||||
$content = serialize($content);
|
|
||||||
$webmention->source = $this->source;
|
|
||||||
$webmention->target = $note->longurl;
|
|
||||||
$webmention->commentable_id = $this->note->id;
|
|
||||||
$webmention->commentable_type = 'App\Note';
|
|
||||||
$webmention->type = 'repost';
|
|
||||||
$webmention->content = $content;
|
|
||||||
$webmention->save();
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -150,16 +114,20 @@ class ProcessWebMention extends Job implements ShouldQueue
|
||||||
*/
|
*/
|
||||||
private function getRemoteContent($url)
|
private function getRemoteContent($url)
|
||||||
{
|
{
|
||||||
$client = new Client();
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$response = $client->request('GET', $url);
|
$response = $this->guzzle->request('GET', $url);
|
||||||
} catch (RequestException $e) {
|
} catch (RequestException $e) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
$html = (string) $response->getBody();
|
$html = (string) $response->getBody();
|
||||||
$path = storage_path() . '/HTML/' . $this->createFilenameFromURL($url);
|
$path = storage_path() . '/HTML/' . $this->createFilenameFromURL($url);
|
||||||
$this->fileForceContents($path, $html);
|
$parts = explode('/', $path);
|
||||||
|
$name = array_pop($parts);
|
||||||
|
$dir = implode('/', $parts);
|
||||||
|
if (! is_dir($dir)) {
|
||||||
|
mkdir($dir, 0755, true);
|
||||||
|
}
|
||||||
|
file_put_contents("$dir/$name", $html);
|
||||||
|
|
||||||
return $html;
|
return $html;
|
||||||
}
|
}
|
||||||
|
@ -182,65 +150,29 @@ class ProcessWebMention extends Job implements ShouldQueue
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Save a file, and create any necessary folders.
|
* Filter the HTML in a reply webmention.
|
||||||
*
|
*
|
||||||
* @param string The directory to save to
|
* @param array The unfiltered microformats
|
||||||
* @param binary The file to save
|
* @return array The filtered microformats
|
||||||
*/
|
*/
|
||||||
private function fileForceContents($dir, $contents)
|
private function filterHTML($microformats)
|
||||||
{
|
{
|
||||||
$parts = explode('/', $dir);
|
if (isset($microformats['items'][0]['properties']['content'][0]['html'])) {
|
||||||
$name = array_pop($parts);
|
$microformats['items'][0]['properties']['content'][0]['html_purified'] = $this->useHTMLPurifier(
|
||||||
$dir = implode('/', $parts);
|
$microformats['items'][0]['properties']['content'][0]['html']
|
||||||
if (! is_dir($dir)) {
|
);
|
||||||
mkdir($dir, 0755, true);
|
|
||||||
}
|
}
|
||||||
file_put_contents("$dir/$name", $contents);
|
|
||||||
|
return $microformats;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Save a profile image to the local cache.
|
* Set up and use HTMLPurifer on some HTML.
|
||||||
*
|
|
||||||
* @param array source content
|
|
||||||
* @return bool wether image was saved or not (we don’t save twitter profiles)
|
|
||||||
*/
|
|
||||||
public function saveImage(array $content)
|
|
||||||
{
|
|
||||||
$photo = $content['photo'];
|
|
||||||
$home = $content['url'];
|
|
||||||
//dont save pbs.twimg.com links
|
|
||||||
if (parse_url($photo)['host'] != 'pbs.twimg.com'
|
|
||||||
&& parse_url($photo)['host'] != 'twitter.com') {
|
|
||||||
$client = new Client();
|
|
||||||
try {
|
|
||||||
$response = $client->get($photo);
|
|
||||||
$image = $response->getBody(true);
|
|
||||||
$path = public_path() . '/assets/profile-images/' . parse_url($home)['host'] . '/image';
|
|
||||||
$this->fileForceContents($path, $image);
|
|
||||||
} catch (Exception $e) {
|
|
||||||
// we are openning and reading the default image so that
|
|
||||||
// fileForceContent work
|
|
||||||
$default = public_path() . '/assets/profile-images/default-image';
|
|
||||||
$handle = fopen($default, 'rb');
|
|
||||||
$image = fread($handle, filesize($default));
|
|
||||||
fclose($handle);
|
|
||||||
$path = public_path() . '/assets/profile-images/' . parse_url($home)['host'] . '/image';
|
|
||||||
$this->fileForceContents($path, $image);
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Purify HTML received from a webmention.
|
|
||||||
*
|
*
|
||||||
* @param string The HTML to be processed
|
* @param string The HTML to be processed
|
||||||
* @return string The processed HTML
|
* @return string The processed HTML
|
||||||
*/
|
*/
|
||||||
public function filterHTML($html)
|
private function useHTMLPurifier($html)
|
||||||
{
|
{
|
||||||
$config = HTMLPurifier_Config::createDefault();
|
$config = HTMLPurifier_Config::createDefault();
|
||||||
$config->set('Cache.SerializerPath', storage_path() . '/HTMLPurifier');
|
$config->set('Cache.SerializerPath', storage_path() . '/HTMLPurifier');
|
||||||
|
|
67
app/Jobs/SaveProfileImage.php
Normal file
67
app/Jobs/SaveProfileImage.php
Normal file
|
@ -0,0 +1,67 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Jobs\Job;
|
||||||
|
use Illuminate\Queue\SerializesModels;
|
||||||
|
use Illuminate\Queue\InteractsWithQueue;
|
||||||
|
use GuzzleHttp\Exception\RequestException;
|
||||||
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||||
|
use Jonnybarnes\WebmentionsParser\Authorship;
|
||||||
|
use Jonnybarnes\WebmentionsParser\Exceptions\AuthorshipParserException;
|
||||||
|
|
||||||
|
class SaveProfileImage extends Job implements ShouldQueue
|
||||||
|
{
|
||||||
|
use InteractsWithQueue, SerializesModels;
|
||||||
|
|
||||||
|
protected $microformats;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a new job instance.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function __construct($microformats)
|
||||||
|
{
|
||||||
|
$this->microformats = $microformats;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute the job.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function handle(Authorship $authorship)
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
$author = $authorship->findAuthor($microformats);
|
||||||
|
} catch (AuthorshipParserException $e) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
$photo = $author['properties'][0]['photo'][0];
|
||||||
|
$home = $author['properties'][0]['url'][0];
|
||||||
|
//dont save pbs.twimg.com links
|
||||||
|
if (parse_url($photo, PHP_URL_HOST) != 'pbs.twimg.com'
|
||||||
|
&& parse_url($photo, PHP_URL_HOST) != 'twitter.com') {
|
||||||
|
$client = new Client();
|
||||||
|
try {
|
||||||
|
$response = $client->get($photo);
|
||||||
|
$image = $response->getBody(true);
|
||||||
|
} catch (RequestException $e) {
|
||||||
|
// we are openning and reading the default image so that
|
||||||
|
$default = public_path() . '/assets/profile-images/default-image';
|
||||||
|
$handle = fopen($default, 'rb');
|
||||||
|
$image = fread($handle, filesize($default));
|
||||||
|
fclose($handle);
|
||||||
|
}
|
||||||
|
$path = public_path() . '/assets/profile-images/' . parse_url($home, PHP_URL_HOST) . '/image';
|
||||||
|
$parts = explode('/', $path);
|
||||||
|
$name = array_pop($parts);
|
||||||
|
$dir = implode('/', $parts);
|
||||||
|
if (! is_dir($dir)) {
|
||||||
|
mkdir($dir, 0755, true);
|
||||||
|
}
|
||||||
|
file_put_contents("$dir/$name", $image);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
116
composer.lock
generated
116
composer.lock
generated
|
@ -1672,16 +1672,16 @@
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "monolog/monolog",
|
"name": "monolog/monolog",
|
||||||
"version": "1.20.0",
|
"version": "1.21.0",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/Seldaek/monolog.git",
|
"url": "https://github.com/Seldaek/monolog.git",
|
||||||
"reference": "55841909e2bcde01b5318c35f2b74f8ecc86e037"
|
"reference": "f42fbdfd53e306bda545845e4dbfd3e72edb4952"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/Seldaek/monolog/zipball/55841909e2bcde01b5318c35f2b74f8ecc86e037",
|
"url": "https://api.github.com/repos/Seldaek/monolog/zipball/f42fbdfd53e306bda545845e4dbfd3e72edb4952",
|
||||||
"reference": "55841909e2bcde01b5318c35f2b74f8ecc86e037",
|
"reference": "f42fbdfd53e306bda545845e4dbfd3e72edb4952",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
|
@ -1746,7 +1746,7 @@
|
||||||
"logging",
|
"logging",
|
||||||
"psr-3"
|
"psr-3"
|
||||||
],
|
],
|
||||||
"time": "2016-07-02 14:02:10"
|
"time": "2016-07-29 03:23:52"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "mtdowling/cron-expression",
|
"name": "mtdowling/cron-expression",
|
||||||
|
@ -2652,16 +2652,16 @@
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/console",
|
"name": "symfony/console",
|
||||||
"version": "v3.0.8",
|
"version": "v3.0.9",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/console.git",
|
"url": "https://github.com/symfony/console.git",
|
||||||
"reference": "a7abb7153f6d1da47f87ec50274844e246b09d9f"
|
"reference": "926061e74229e935d3c5b4e9ba87237316c6693f"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/symfony/console/zipball/a7abb7153f6d1da47f87ec50274844e246b09d9f",
|
"url": "https://api.github.com/repos/symfony/console/zipball/926061e74229e935d3c5b4e9ba87237316c6693f",
|
||||||
"reference": "a7abb7153f6d1da47f87ec50274844e246b09d9f",
|
"reference": "926061e74229e935d3c5b4e9ba87237316c6693f",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
|
@ -2708,20 +2708,20 @@
|
||||||
],
|
],
|
||||||
"description": "Symfony Console Component",
|
"description": "Symfony Console Component",
|
||||||
"homepage": "https://symfony.com",
|
"homepage": "https://symfony.com",
|
||||||
"time": "2016-06-29 07:02:21"
|
"time": "2016-07-30 07:22:48"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/debug",
|
"name": "symfony/debug",
|
||||||
"version": "v3.0.8",
|
"version": "v3.0.9",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/debug.git",
|
"url": "https://github.com/symfony/debug.git",
|
||||||
"reference": "c54bc3539c3b87e86799533801e8ae0e971d78c2"
|
"reference": "697c527acd9ea1b2d3efac34d9806bf255278b0a"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/symfony/debug/zipball/c54bc3539c3b87e86799533801e8ae0e971d78c2",
|
"url": "https://api.github.com/repos/symfony/debug/zipball/697c527acd9ea1b2d3efac34d9806bf255278b0a",
|
||||||
"reference": "c54bc3539c3b87e86799533801e8ae0e971d78c2",
|
"reference": "697c527acd9ea1b2d3efac34d9806bf255278b0a",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
|
@ -2765,20 +2765,20 @@
|
||||||
],
|
],
|
||||||
"description": "Symfony Debug Component",
|
"description": "Symfony Debug Component",
|
||||||
"homepage": "https://symfony.com",
|
"homepage": "https://symfony.com",
|
||||||
"time": "2016-06-29 05:40:00"
|
"time": "2016-07-30 07:22:48"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/event-dispatcher",
|
"name": "symfony/event-dispatcher",
|
||||||
"version": "v3.1.2",
|
"version": "v3.1.3",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/event-dispatcher.git",
|
"url": "https://github.com/symfony/event-dispatcher.git",
|
||||||
"reference": "7f9839ede2070f53e7e2f0849b9bd14748c434c5"
|
"reference": "c0c00c80b3a69132c4e55c3e7db32b4a387615e5"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/symfony/event-dispatcher/zipball/7f9839ede2070f53e7e2f0849b9bd14748c434c5",
|
"url": "https://api.github.com/repos/symfony/event-dispatcher/zipball/c0c00c80b3a69132c4e55c3e7db32b4a387615e5",
|
||||||
"reference": "7f9839ede2070f53e7e2f0849b9bd14748c434c5",
|
"reference": "c0c00c80b3a69132c4e55c3e7db32b4a387615e5",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
|
@ -2825,11 +2825,11 @@
|
||||||
],
|
],
|
||||||
"description": "Symfony EventDispatcher Component",
|
"description": "Symfony EventDispatcher Component",
|
||||||
"homepage": "https://symfony.com",
|
"homepage": "https://symfony.com",
|
||||||
"time": "2016-06-29 05:41:56"
|
"time": "2016-07-19 10:45:57"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/finder",
|
"name": "symfony/finder",
|
||||||
"version": "v3.0.8",
|
"version": "v3.0.9",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/finder.git",
|
"url": "https://github.com/symfony/finder.git",
|
||||||
|
@ -2878,16 +2878,16 @@
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/http-foundation",
|
"name": "symfony/http-foundation",
|
||||||
"version": "v3.0.8",
|
"version": "v3.0.9",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/http-foundation.git",
|
"url": "https://github.com/symfony/http-foundation.git",
|
||||||
"reference": "1341139f906d295baa4f4abd55293d07e25a065a"
|
"reference": "49ba00f8ede742169cb6b70abe33243f4d673f82"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/symfony/http-foundation/zipball/1341139f906d295baa4f4abd55293d07e25a065a",
|
"url": "https://api.github.com/repos/symfony/http-foundation/zipball/49ba00f8ede742169cb6b70abe33243f4d673f82",
|
||||||
"reference": "1341139f906d295baa4f4abd55293d07e25a065a",
|
"reference": "49ba00f8ede742169cb6b70abe33243f4d673f82",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
|
@ -2927,20 +2927,20 @@
|
||||||
],
|
],
|
||||||
"description": "Symfony HttpFoundation Component",
|
"description": "Symfony HttpFoundation Component",
|
||||||
"homepage": "https://symfony.com",
|
"homepage": "https://symfony.com",
|
||||||
"time": "2016-06-29 07:02:21"
|
"time": "2016-07-17 13:54:30"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/http-kernel",
|
"name": "symfony/http-kernel",
|
||||||
"version": "v3.0.8",
|
"version": "v3.0.9",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/http-kernel.git",
|
"url": "https://github.com/symfony/http-kernel.git",
|
||||||
"reference": "177b63b2d50b63fa6d82ea41359ed9928cc7a1fb"
|
"reference": "d97ba4425e36e79c794e7d14ff36f00f081b37b3"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/symfony/http-kernel/zipball/177b63b2d50b63fa6d82ea41359ed9928cc7a1fb",
|
"url": "https://api.github.com/repos/symfony/http-kernel/zipball/d97ba4425e36e79c794e7d14ff36f00f081b37b3",
|
||||||
"reference": "177b63b2d50b63fa6d82ea41359ed9928cc7a1fb",
|
"reference": "d97ba4425e36e79c794e7d14ff36f00f081b37b3",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
|
@ -3009,7 +3009,7 @@
|
||||||
],
|
],
|
||||||
"description": "Symfony HttpKernel Component",
|
"description": "Symfony HttpKernel Component",
|
||||||
"homepage": "https://symfony.com",
|
"homepage": "https://symfony.com",
|
||||||
"time": "2016-06-30 16:30:17"
|
"time": "2016-07-30 09:10:37"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/polyfill-mbstring",
|
"name": "symfony/polyfill-mbstring",
|
||||||
|
@ -3180,16 +3180,16 @@
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/process",
|
"name": "symfony/process",
|
||||||
"version": "v3.0.8",
|
"version": "v3.0.9",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/process.git",
|
"url": "https://github.com/symfony/process.git",
|
||||||
"reference": "d7cde1f9d94d87060204f863779389b61c382eeb"
|
"reference": "768debc5996f599c4372b322d9061dba2a4bf505"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/symfony/process/zipball/d7cde1f9d94d87060204f863779389b61c382eeb",
|
"url": "https://api.github.com/repos/symfony/process/zipball/768debc5996f599c4372b322d9061dba2a4bf505",
|
||||||
"reference": "d7cde1f9d94d87060204f863779389b61c382eeb",
|
"reference": "768debc5996f599c4372b322d9061dba2a4bf505",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
|
@ -3225,11 +3225,11 @@
|
||||||
],
|
],
|
||||||
"description": "Symfony Process Component",
|
"description": "Symfony Process Component",
|
||||||
"homepage": "https://symfony.com",
|
"homepage": "https://symfony.com",
|
||||||
"time": "2016-06-29 05:40:00"
|
"time": "2016-07-28 11:13:34"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/routing",
|
"name": "symfony/routing",
|
||||||
"version": "v3.0.8",
|
"version": "v3.0.9",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/routing.git",
|
"url": "https://github.com/symfony/routing.git",
|
||||||
|
@ -3304,16 +3304,16 @@
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/translation",
|
"name": "symfony/translation",
|
||||||
"version": "v3.0.8",
|
"version": "v3.0.9",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/translation.git",
|
"url": "https://github.com/symfony/translation.git",
|
||||||
"reference": "6bf844e1ee3c820c012386c10427a5c67bbefec8"
|
"reference": "eee6c664853fd0576f21ae25725cfffeafe83f26"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/symfony/translation/zipball/6bf844e1ee3c820c012386c10427a5c67bbefec8",
|
"url": "https://api.github.com/repos/symfony/translation/zipball/eee6c664853fd0576f21ae25725cfffeafe83f26",
|
||||||
"reference": "6bf844e1ee3c820c012386c10427a5c67bbefec8",
|
"reference": "eee6c664853fd0576f21ae25725cfffeafe83f26",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
|
@ -3364,20 +3364,20 @@
|
||||||
],
|
],
|
||||||
"description": "Symfony Translation Component",
|
"description": "Symfony Translation Component",
|
||||||
"homepage": "https://symfony.com",
|
"homepage": "https://symfony.com",
|
||||||
"time": "2016-06-29 05:40:00"
|
"time": "2016-07-30 07:22:48"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/var-dumper",
|
"name": "symfony/var-dumper",
|
||||||
"version": "v3.0.8",
|
"version": "v3.0.9",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/var-dumper.git",
|
"url": "https://github.com/symfony/var-dumper.git",
|
||||||
"reference": "2f046e9a9d571f22cc8b26783564876713b06579"
|
"reference": "1f7e071aafc6676fcb6e3f0497f87c2397247377"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/symfony/var-dumper/zipball/2f046e9a9d571f22cc8b26783564876713b06579",
|
"url": "https://api.github.com/repos/symfony/var-dumper/zipball/1f7e071aafc6676fcb6e3f0497f87c2397247377",
|
||||||
"reference": "2f046e9a9d571f22cc8b26783564876713b06579",
|
"reference": "1f7e071aafc6676fcb6e3f0497f87c2397247377",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
|
@ -3427,7 +3427,7 @@
|
||||||
"debug",
|
"debug",
|
||||||
"dump"
|
"dump"
|
||||||
],
|
],
|
||||||
"time": "2016-06-29 05:40:00"
|
"time": "2016-07-26 08:03:56"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "themattharris/tmhoauth",
|
"name": "themattharris/tmhoauth",
|
||||||
|
@ -5100,7 +5100,7 @@
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/css-selector",
|
"name": "symfony/css-selector",
|
||||||
"version": "v3.0.8",
|
"version": "v3.0.9",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/css-selector.git",
|
"url": "https://github.com/symfony/css-selector.git",
|
||||||
|
@ -5153,16 +5153,16 @@
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/dom-crawler",
|
"name": "symfony/dom-crawler",
|
||||||
"version": "v3.0.8",
|
"version": "v3.0.9",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/dom-crawler.git",
|
"url": "https://github.com/symfony/dom-crawler.git",
|
||||||
"reference": "62769e3409006b937bb333b29da8df9a8b262975"
|
"reference": "dff8fecf1f56990d88058e3a1885c2a5f1b8e970"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/symfony/dom-crawler/zipball/62769e3409006b937bb333b29da8df9a8b262975",
|
"url": "https://api.github.com/repos/symfony/dom-crawler/zipball/dff8fecf1f56990d88058e3a1885c2a5f1b8e970",
|
||||||
"reference": "62769e3409006b937bb333b29da8df9a8b262975",
|
"reference": "dff8fecf1f56990d88058e3a1885c2a5f1b8e970",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
|
@ -5205,20 +5205,20 @@
|
||||||
],
|
],
|
||||||
"description": "Symfony DomCrawler Component",
|
"description": "Symfony DomCrawler Component",
|
||||||
"homepage": "https://symfony.com",
|
"homepage": "https://symfony.com",
|
||||||
"time": "2016-06-29 05:40:00"
|
"time": "2016-07-30 07:22:48"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/yaml",
|
"name": "symfony/yaml",
|
||||||
"version": "v3.1.2",
|
"version": "v3.1.3",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/yaml.git",
|
"url": "https://github.com/symfony/yaml.git",
|
||||||
"reference": "2884c26ce4c1d61aebf423a8b912950fe7c764de"
|
"reference": "1819adf2066880c7967df7180f4f662b6f0567ac"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/symfony/yaml/zipball/2884c26ce4c1d61aebf423a8b912950fe7c764de",
|
"url": "https://api.github.com/repos/symfony/yaml/zipball/1819adf2066880c7967df7180f4f662b6f0567ac",
|
||||||
"reference": "2884c26ce4c1d61aebf423a8b912950fe7c764de",
|
"reference": "1819adf2066880c7967df7180f4f662b6f0567ac",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
|
@ -5254,7 +5254,7 @@
|
||||||
],
|
],
|
||||||
"description": "Symfony Yaml Component",
|
"description": "Symfony Yaml Component",
|
||||||
"homepage": "https://symfony.com",
|
"homepage": "https://symfony.com",
|
||||||
"time": "2016-06-29 05:41:56"
|
"time": "2016-07-17 14:02:08"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "webmozart/assert",
|
"name": "webmozart/assert",
|
||||||
|
|
|
@ -20,13 +20,13 @@
|
||||||
</div>
|
</div>
|
||||||
@if(count($likes) > 0)<h1 class="notes-subtitle">Likes</h1>@endif
|
@if(count($likes) > 0)<h1 class="notes-subtitle">Likes</h1>@endif
|
||||||
@foreach($likes as $like)
|
@foreach($likes as $like)
|
||||||
<a href="{{ $like['url'] }}"><img src="{{ $like['photo'] }}" alt="" class="like-photo"></a>
|
<a href="{{ $like['url'] }}"><img src="{{ $like['photo'] }}" alt="profile picture of {{ $like['name'] }}" class="like-photo"></a>
|
||||||
@endforeach
|
@endforeach
|
||||||
@if(count($reposts) > 0)<h1 class="notes-subtitle">Reposts</h1>@endif
|
@if(count($reposts) > 0)<h1 class="notes-subtitle">Reposts</h1>@endif
|
||||||
@foreach($reposts as $repost)
|
@foreach($reposts as $repost)
|
||||||
<p><a class="h-card vcard mini-h-card p-author" href="{{ $repost['url'] }}">
|
<p><a class="h-card vcard mini-h-card p-author" href="{{ $repost['url'] }}">
|
||||||
<img src="{{ $repost['photo'] }}" alt="profile picture of {{ $repost['name'] }}" class="photo u-photo logo"> <span class="fn">{{ $repost['name'] }}</span>
|
<img src="{{ $repost['photo'] }}" alt="profile picture of {{ $repost['name'] }}" class="photo u-photo logo"> <span class="fn">{{ $repost['name'] }}</span>
|
||||||
</a> reposted this at <a href="{{ $repost['repost'] }}">{{ $repost['date'] }}</a>.</p>
|
</a> reposted this at <a href="{{ $repost['source'] }}">{{ $repost['date'] }}</a>.</p>
|
||||||
@endforeach
|
@endforeach
|
||||||
@stop
|
@stop
|
||||||
|
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
namespace App\Tests;
|
namespace App\Tests;
|
||||||
|
|
||||||
|
use Cache;
|
||||||
use TestCase;
|
use TestCase;
|
||||||
use Illuminate\Foundation\Testing\WithoutMiddleware;
|
use Illuminate\Foundation\Testing\WithoutMiddleware;
|
||||||
use Illuminate\Foundation\Testing\DatabaseMigrations;
|
use Illuminate\Foundation\Testing\DatabaseMigrations;
|
||||||
|
@ -131,15 +132,15 @@ class NotesTest extends TestCase
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test the bridgy url shim method.
|
* Test a correct profile link is formed from a generic URL.
|
||||||
*
|
*
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function testBridgy()
|
public function testCreatePhotoLinkWithNonCachedImage()
|
||||||
{
|
{
|
||||||
$url = 'https://brid-gy.appspot.com/comment/twitter/jonnybarnes/497778866816299008/497781260937203712';
|
$homepage = 'https://example.org/profile.png';
|
||||||
$expected = 'https://twitter.com/_/status/497781260937203712';
|
$expected = 'https://example.org/profile.png';
|
||||||
$this->assertEquals($expected, $this->notesController->bridgyReply($url));
|
$this->assertEquals($expected, $this->notesController->createPhotoLink($homepage));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -147,10 +148,10 @@ class NotesTest extends TestCase
|
||||||
*
|
*
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function testCreatePhotoLinkWithGenericURL()
|
public function testCreatePhotoLinkWithCachedImage()
|
||||||
{
|
{
|
||||||
$homepage = 'https://example.org';
|
$homepage = 'https://aaronparecki.com/profile.png';
|
||||||
$expected = '/assets/profile-images/example.org/image';
|
$expected = '/assets/profile-images/aaronparecki.com/image';
|
||||||
$this->assertEquals($expected, $this->notesController->createPhotoLink($homepage));
|
$this->assertEquals($expected, $this->notesController->createPhotoLink($homepage));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -159,7 +160,7 @@ class NotesTest extends TestCase
|
||||||
*
|
*
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function testCreatePhotoLinkWithTwitterProfileImageURL()
|
public function testCreatePhotoLinkWithTwimgProfileImageURL()
|
||||||
{
|
{
|
||||||
$twitterProfileImage = 'http://pbs.twimg.com/1234';
|
$twitterProfileImage = 'http://pbs.twimg.com/1234';
|
||||||
$expected = 'https://pbs.twimg.com/1234';
|
$expected = 'https://pbs.twimg.com/1234';
|
||||||
|
@ -171,9 +172,11 @@ class NotesTest extends TestCase
|
||||||
*
|
*
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function testCreatePhotoLinkWithTwitterURL()
|
public function testCreatePhotoLinkWithCachedTwitterURL()
|
||||||
{
|
{
|
||||||
$twitterURL = 'https://twitter.com/example';
|
$twitterURL = 'https://twitter.com/example';
|
||||||
$this->assertNull($this->notesController->createPhotoLink($twitterURL));
|
$expected = 'https://pbs.twimg.com/static_profile_link.jpg';
|
||||||
|
Cache::put($twitterURL, $expected, 1);
|
||||||
|
$this->assertEquals($expected, $this->notesController->createPhotoLink($twitterURL));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
29
tests/ProcessWebMentionTest.php
Normal file
29
tests/ProcessWebMentionTest.php
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Tests;
|
||||||
|
|
||||||
|
use TestCase;
|
||||||
|
use Illuminate\Foundation\Testing\WithoutMiddleware;
|
||||||
|
use Illuminate\Foundation\Testing\DatabaseMigrations;
|
||||||
|
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
||||||
|
|
||||||
|
class ProcessWebMentionTest extends TestCase
|
||||||
|
{
|
||||||
|
protected $appurl;
|
||||||
|
|
||||||
|
public function setUp()
|
||||||
|
{
|
||||||
|
parent::setUp();
|
||||||
|
$this->appurl = config('app.url');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A basic test.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testExample()
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
92
tests/WebMentionsTest.php
Normal file
92
tests/WebMentionsTest.php
Normal file
|
@ -0,0 +1,92 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Tests;
|
||||||
|
|
||||||
|
use TestCase;
|
||||||
|
use Illuminate\Foundation\Testing\WithoutMiddleware;
|
||||||
|
use Illuminate\Foundation\Testing\DatabaseMigrations;
|
||||||
|
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
||||||
|
|
||||||
|
class WebMentionsTest extends TestCase
|
||||||
|
{
|
||||||
|
protected $appurl;
|
||||||
|
|
||||||
|
public function setUp()
|
||||||
|
{
|
||||||
|
parent::setUp();
|
||||||
|
$this->appurl = config('app.url');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test webmentions without source and target are rejected.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testWebmentionsWithoutSourceAndTargetAreRejected()
|
||||||
|
{
|
||||||
|
$this->call('POST', $this->appurl . '/webmention', ['source' => 'https://example.org/post/123']);
|
||||||
|
$this->assertResponseStatus(400)
|
||||||
|
->see('You need both the target and source parameters');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test invalid target gets a 400 response.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testInvalidTargetReturns400Response()
|
||||||
|
{
|
||||||
|
$this->call('POST', $this->appurl . '/webmention', [
|
||||||
|
'source' => 'https://example.org/post/123',
|
||||||
|
'target' => $this->appurl . '/invalid/target'
|
||||||
|
]);
|
||||||
|
$this->assertResponseStatus(400)
|
||||||
|
->see('Invalid request');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test blog target gets a 501 response.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testBlogpostTargetReturns501Response()
|
||||||
|
{
|
||||||
|
$this->call('POST', $this->appurl . '/webmention', [
|
||||||
|
'source' => 'https://example.org/post/123',
|
||||||
|
'target' => $this->appurl . '/blog/target'
|
||||||
|
]);
|
||||||
|
$this->assertResponseStatus(501)
|
||||||
|
->see('I don’t accept webmentions for blog posts yet.');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test that a non-existant note gives a 400 response.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testNonexistantNoteReturns400Response()
|
||||||
|
{
|
||||||
|
$this->call('POST', $this->appurl . '/webmention', [
|
||||||
|
'source' => 'https://example.org/post/123',
|
||||||
|
'target' => $this->appurl . '/notes/ZZZZZ'
|
||||||
|
]);
|
||||||
|
$this->assertResponseStatus(400)
|
||||||
|
->see('This note doesn’t exist.');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test a legit webmention triggers the ProcessWebMention job.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testLegitimateWebmnetionTriggersProcessWebMentionJob()
|
||||||
|
{
|
||||||
|
$this->expectsJobs(\App\Jobs\ProcessWebMention::class);
|
||||||
|
$this->call('POST', $this->appurl . '/webmention', [
|
||||||
|
'source' => 'https://example.org/post/123',
|
||||||
|
'target' => $this->appurl . '/notes/B'
|
||||||
|
]);
|
||||||
|
$this->assertResponseStatus(202)
|
||||||
|
->see('Webmention received, it will be processed shortly');
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue