mirror of
https://github.com/BookStackApp/BookStack.git
synced 2024-11-22 14:10:48 +08:00
Added testing for avatar fetching systems & config
Abstracts imageservice http interaction. Closes #1193
This commit is contained in:
parent
866187830a
commit
68017e2553
5
app/Exceptions/HttpFetchException.php
Normal file
5
app/Exceptions/HttpFetchException.php
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
<?php namespace BookStack\Exceptions;
|
||||||
|
|
||||||
|
use Exception;
|
||||||
|
|
||||||
|
class HttpFetchException extends Exception {}
|
|
@ -9,6 +9,7 @@ use BookStack\Actions\ViewService;
|
||||||
use BookStack\Auth\Permissions\PermissionService;
|
use BookStack\Auth\Permissions\PermissionService;
|
||||||
use BookStack\Settings\Setting;
|
use BookStack\Settings\Setting;
|
||||||
use BookStack\Settings\SettingService;
|
use BookStack\Settings\SettingService;
|
||||||
|
use BookStack\Uploads\HttpFetcher;
|
||||||
use BookStack\Uploads\Image;
|
use BookStack\Uploads\Image;
|
||||||
use BookStack\Uploads\ImageService;
|
use BookStack\Uploads\ImageService;
|
||||||
use Illuminate\Contracts\Cache\Repository;
|
use Illuminate\Contracts\Cache\Repository;
|
||||||
|
@ -61,7 +62,8 @@ class CustomFacadeProvider extends ServiceProvider
|
||||||
$this->app->make(Image::class),
|
$this->app->make(Image::class),
|
||||||
$this->app->make(ImageManager::class),
|
$this->app->make(ImageManager::class),
|
||||||
$this->app->make(Factory::class),
|
$this->app->make(Factory::class),
|
||||||
$this->app->make(Repository::class)
|
$this->app->make(Repository::class),
|
||||||
|
$this->app->make(HttpFetcher::class)
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
34
app/Uploads/HttpFetcher.php
Normal file
34
app/Uploads/HttpFetcher.php
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
<?php namespace BookStack\Uploads;
|
||||||
|
|
||||||
|
use BookStack\Exceptions\HttpFetchException;
|
||||||
|
|
||||||
|
class HttpFetcher
|
||||||
|
{
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fetch content from an external URI.
|
||||||
|
* @param string $uri
|
||||||
|
* @return bool|string
|
||||||
|
* @throws HttpFetchException
|
||||||
|
*/
|
||||||
|
public function fetch(string $uri)
|
||||||
|
{
|
||||||
|
$ch = curl_init();
|
||||||
|
curl_setopt_array($ch, [
|
||||||
|
CURLOPT_URL => $uri,
|
||||||
|
CURLOPT_RETURNTRANSFER => 1,
|
||||||
|
CURLOPT_CONNECTTIMEOUT => 5
|
||||||
|
]);
|
||||||
|
|
||||||
|
$data = curl_exec($ch);
|
||||||
|
$err = curl_error($ch);
|
||||||
|
curl_close($ch);
|
||||||
|
|
||||||
|
if ($err) {
|
||||||
|
throw new HttpFetchException($err);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $data;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -1,6 +1,7 @@
|
||||||
<?php namespace BookStack\Uploads;
|
<?php namespace BookStack\Uploads;
|
||||||
|
|
||||||
use BookStack\Auth\User;
|
use BookStack\Auth\User;
|
||||||
|
use BookStack\Exceptions\HttpFetchException;
|
||||||
use BookStack\Exceptions\ImageUploadException;
|
use BookStack\Exceptions\ImageUploadException;
|
||||||
use DB;
|
use DB;
|
||||||
use Exception;
|
use Exception;
|
||||||
|
@ -17,6 +18,7 @@ class ImageService extends UploadService
|
||||||
protected $cache;
|
protected $cache;
|
||||||
protected $storageUrl;
|
protected $storageUrl;
|
||||||
protected $image;
|
protected $image;
|
||||||
|
protected $http;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ImageService constructor.
|
* ImageService constructor.
|
||||||
|
@ -24,12 +26,14 @@ class ImageService extends UploadService
|
||||||
* @param ImageManager $imageTool
|
* @param ImageManager $imageTool
|
||||||
* @param FileSystem $fileSystem
|
* @param FileSystem $fileSystem
|
||||||
* @param Cache $cache
|
* @param Cache $cache
|
||||||
|
* @param HttpFetcher $http
|
||||||
*/
|
*/
|
||||||
public function __construct(Image $image, ImageManager $imageTool, FileSystem $fileSystem, Cache $cache)
|
public function __construct(Image $image, ImageManager $imageTool, FileSystem $fileSystem, Cache $cache, HttpFetcher $http)
|
||||||
{
|
{
|
||||||
$this->image = $image;
|
$this->image = $image;
|
||||||
$this->imageTool = $imageTool;
|
$this->imageTool = $imageTool;
|
||||||
$this->cache = $cache;
|
$this->cache = $cache;
|
||||||
|
$this->http = $http;
|
||||||
parent::__construct($fileSystem);
|
parent::__construct($fileSystem);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -95,8 +99,9 @@ class ImageService extends UploadService
|
||||||
private function saveNewFromUrl($url, $type, $imageName = false)
|
private function saveNewFromUrl($url, $type, $imageName = false)
|
||||||
{
|
{
|
||||||
$imageName = $imageName ? $imageName : basename($url);
|
$imageName = $imageName ? $imageName : basename($url);
|
||||||
$imageData = file_get_contents($url);
|
try {
|
||||||
if ($imageData === false) {
|
$imageData = $this->http->fetch($url);
|
||||||
|
} catch (HttpFetchException $exception) {
|
||||||
throw new \Exception(trans('errors.cannot_get_image_from_url', ['url' => $url]));
|
throw new \Exception(trans('errors.cannot_get_image_from_url', ['url' => $url]));
|
||||||
}
|
}
|
||||||
return $this->saveNew($imageName, $imageData, $type);
|
return $this->saveNew($imageName, $imageData, $type);
|
||||||
|
@ -322,7 +327,13 @@ class ImageService extends UploadService
|
||||||
*/
|
*/
|
||||||
protected function getAvatarUrl()
|
protected function getAvatarUrl()
|
||||||
{
|
{
|
||||||
return trim(config('services.avatar_url'));
|
$url = trim(config('services.avatar_url'));
|
||||||
|
|
||||||
|
if (empty($url) && !config('services.disable_services')) {
|
||||||
|
$url = 'https://www.gravatar.com/avatar/${hash}?s=${size}&d=identicon';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $url;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -392,14 +403,7 @@ class ImageService extends UploadService
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
try {
|
try {
|
||||||
$ch = curl_init();
|
$imageData = $this->http->fetch($uri);
|
||||||
curl_setopt_array($ch, [CURLOPT_URL => $uri, CURLOPT_RETURNTRANSFER => 1, CURLOPT_CONNECTTIMEOUT => 5]);
|
|
||||||
$imageData = curl_exec($ch);
|
|
||||||
$err = curl_error($ch);
|
|
||||||
curl_close($ch);
|
|
||||||
if ($err) {
|
|
||||||
throw new \Exception("Image fetch failed, Received error: " . $err);
|
|
||||||
}
|
|
||||||
} catch (\Exception $e) {
|
} catch (\Exception $e) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,6 +12,7 @@
|
||||||
"ext-xml": "*",
|
"ext-xml": "*",
|
||||||
"ext-mbstring": "*",
|
"ext-mbstring": "*",
|
||||||
"ext-gd": "*",
|
"ext-gd": "*",
|
||||||
|
"ext-curl": "*",
|
||||||
"laravel/framework": "~5.5.44",
|
"laravel/framework": "~5.5.44",
|
||||||
"fideloper/proxy": "~3.3",
|
"fideloper/proxy": "~3.3",
|
||||||
"intervention/image": "^2.4",
|
"intervention/image": "^2.4",
|
||||||
|
|
|
@ -21,9 +21,7 @@ return [
|
||||||
'drawio' => env('DRAWIO', !env('DISABLE_EXTERNAL_SERVICES', false)),
|
'drawio' => env('DRAWIO', !env('DISABLE_EXTERNAL_SERVICES', false)),
|
||||||
|
|
||||||
// URL for fetching avatars
|
// URL for fetching avatars
|
||||||
'avatar_url' => env('AVATAR_URL',
|
'avatar_url' => env('AVATAR_URL', ''),
|
||||||
env('DISABLE_EXTERNAL_SERVICES', false) ? false : 'https://www.gravatar.com/avatar/${hash}?s=${size}&d=identicon'
|
|
||||||
),
|
|
||||||
|
|
||||||
// Callback URL for social authentication methods
|
// Callback URL for social authentication methods
|
||||||
'callback_url' => env('APP_URL', false),
|
'callback_url' => env('APP_URL', false),
|
||||||
|
|
|
@ -31,6 +31,7 @@
|
||||||
<env name="MAIL_DRIVER" value="log"/>
|
<env name="MAIL_DRIVER" value="log"/>
|
||||||
<env name="AUTH_METHOD" value="standard"/>
|
<env name="AUTH_METHOD" value="standard"/>
|
||||||
<env name="DISABLE_EXTERNAL_SERVICES" value="true"/>
|
<env name="DISABLE_EXTERNAL_SERVICES" value="true"/>
|
||||||
|
<env name="AVATAR_URL" value=""/>
|
||||||
<env name="LDAP_VERSION" value="3"/>
|
<env name="LDAP_VERSION" value="3"/>
|
||||||
<env name="STORAGE_TYPE" value="local"/>
|
<env name="STORAGE_TYPE" value="local"/>
|
||||||
<env name="GITHUB_APP_ID" value="aaaaaaaaaaaaaa"/>
|
<env name="GITHUB_APP_ID" value="aaaaaaaaaaaaaa"/>
|
||||||
|
|
84
tests/Uploads/AvatarTest.php
Normal file
84
tests/Uploads/AvatarTest.php
Normal file
|
@ -0,0 +1,84 @@
|
||||||
|
<?php namespace Tests\Uploads;
|
||||||
|
|
||||||
|
use BookStack\Auth\User;
|
||||||
|
use BookStack\Uploads\HttpFetcher;
|
||||||
|
use Tests\TestCase;
|
||||||
|
|
||||||
|
class AvatarTest extends TestCase
|
||||||
|
{
|
||||||
|
use UsesImages;
|
||||||
|
|
||||||
|
|
||||||
|
protected function createUserRequest($user)
|
||||||
|
{
|
||||||
|
$resp = $this->asAdmin()->post('/settings/users/create', [
|
||||||
|
'name' => $user->name,
|
||||||
|
'email' => $user->email,
|
||||||
|
'password' => 'testing',
|
||||||
|
'password-confirm' => 'testing',
|
||||||
|
]);
|
||||||
|
return User::where('email', '=', $user->email)->first();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function assertImageFetchFrom(string $url)
|
||||||
|
{
|
||||||
|
$http = \Mockery::mock(HttpFetcher::class);
|
||||||
|
$this->app->instance(HttpFetcher::class, $http);
|
||||||
|
|
||||||
|
$http->shouldReceive('fetch')
|
||||||
|
->once()->with($url)
|
||||||
|
->andReturn($this->getTestImageContent());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function deleteUserImage(User $user)
|
||||||
|
{
|
||||||
|
$this->deleteImage($user->avatar->path);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function test_gravatar_fetched_on_user_create()
|
||||||
|
{
|
||||||
|
config()->set([
|
||||||
|
'services.disable_services' => false,
|
||||||
|
]);
|
||||||
|
$user = factory(User::class)->make();
|
||||||
|
$this->assertImageFetchFrom('https://www.gravatar.com/avatar/'.md5(strtolower($user->email)).'?s=500&d=identicon');
|
||||||
|
|
||||||
|
$user = $this->createUserRequest($user);
|
||||||
|
$this->assertDatabaseHas('images', [
|
||||||
|
'type' => 'user',
|
||||||
|
'created_by' => $user->id
|
||||||
|
]);
|
||||||
|
$this->deleteUserImage($user);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public function test_custom_url_used_if_set()
|
||||||
|
{
|
||||||
|
config()->set([
|
||||||
|
'services.avatar_url' => 'https://example.com/${email}/${hash}/${size}',
|
||||||
|
]);
|
||||||
|
|
||||||
|
$user = factory(User::class)->make();
|
||||||
|
$url = 'https://example.com/'. urlencode(strtolower($user->email)) .'/'. md5(strtolower($user->email)).'/500';
|
||||||
|
$this->assertImageFetchFrom($url);
|
||||||
|
|
||||||
|
$user = $this->createUserRequest($user);
|
||||||
|
$this->deleteUserImage($user);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function test_avatar_not_fetched_if_no_custom_url_and_services_disabled()
|
||||||
|
{
|
||||||
|
config()->set([
|
||||||
|
'services.disable_services' => true,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$user = factory(User::class)->make();
|
||||||
|
|
||||||
|
$http = \Mockery::mock(HttpFetcher::class);
|
||||||
|
$this->app->instance(HttpFetcher::class, $http);
|
||||||
|
$http->shouldNotReceive('fetch');
|
||||||
|
|
||||||
|
$this->createUserRequest($user);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -1,67 +1,15 @@
|
||||||
<?php namespace Tests;
|
<?php namespace Tests\Uploads;
|
||||||
|
|
||||||
use BookStack\Entities\Repos\PageRepo;
|
use BookStack\Entities\Repos\PageRepo;
|
||||||
use BookStack\Uploads\Image;
|
use BookStack\Uploads\Image;
|
||||||
use BookStack\Entities\Page;
|
use BookStack\Entities\Page;
|
||||||
use BookStack\Entities\Repos\EntityRepo;
|
|
||||||
use BookStack\Uploads\ImageService;
|
use BookStack\Uploads\ImageService;
|
||||||
|
use Tests\TestCase;
|
||||||
|
|
||||||
class ImageTest extends TestCase
|
class ImageTest extends TestCase
|
||||||
{
|
{
|
||||||
/**
|
|
||||||
* Get the path to our basic test image.
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
protected function getTestImageFilePath()
|
|
||||||
{
|
|
||||||
return base_path('tests/test-data/test-image.png');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get a test image that can be uploaded
|
|
||||||
* @param $fileName
|
|
||||||
* @return \Illuminate\Http\UploadedFile
|
|
||||||
*/
|
|
||||||
protected function getTestImage($fileName)
|
|
||||||
{
|
|
||||||
return new \Illuminate\Http\UploadedFile($this->getTestImageFilePath(), $fileName, 'image/png', 5238);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the path for a test image.
|
|
||||||
* @param $type
|
|
||||||
* @param $fileName
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
protected function getTestImagePath($type, $fileName)
|
|
||||||
{
|
|
||||||
return '/uploads/images/' . $type . '/' . Date('Y-m-M') . '/' . $fileName;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Uploads an image with the given name.
|
|
||||||
* @param $name
|
|
||||||
* @param int $uploadedTo
|
|
||||||
* @return \Illuminate\Foundation\Testing\TestResponse
|
|
||||||
*/
|
|
||||||
protected function uploadImage($name, $uploadedTo = 0)
|
|
||||||
{
|
|
||||||
$file = $this->getTestImage($name);
|
|
||||||
return $this->call('POST', '/images/gallery/upload', ['uploaded_to' => $uploadedTo], [], ['file' => $file], []);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Delete an uploaded image.
|
|
||||||
* @param $relPath
|
|
||||||
*/
|
|
||||||
protected function deleteImage($relPath)
|
|
||||||
{
|
|
||||||
$path = public_path($relPath);
|
|
||||||
if (file_exists($path)) {
|
|
||||||
unlink($path);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
use UsesImages;
|
||||||
|
|
||||||
public function test_image_upload()
|
public function test_image_upload()
|
||||||
{
|
{
|
69
tests/Uploads/UsesImages.php
Normal file
69
tests/Uploads/UsesImages.php
Normal file
|
@ -0,0 +1,69 @@
|
||||||
|
<?php namespace Tests\Uploads;
|
||||||
|
|
||||||
|
|
||||||
|
trait UsesImages
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Get the path to our basic test image.
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
protected function getTestImageFilePath()
|
||||||
|
{
|
||||||
|
return base_path('tests/test-data/test-image.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get a test image that can be uploaded
|
||||||
|
* @param $fileName
|
||||||
|
* @return \Illuminate\Http\UploadedFile
|
||||||
|
*/
|
||||||
|
protected function getTestImage($fileName)
|
||||||
|
{
|
||||||
|
return new \Illuminate\Http\UploadedFile($this->getTestImageFilePath(), $fileName, 'image/png', 5238);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the raw file data for the test image.
|
||||||
|
* @return false|string
|
||||||
|
*/
|
||||||
|
protected function getTestImageContent()
|
||||||
|
{
|
||||||
|
return file_get_contents($this->getTestImageFilePath());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the path for a test image.
|
||||||
|
* @param $type
|
||||||
|
* @param $fileName
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
protected function getTestImagePath($type, $fileName)
|
||||||
|
{
|
||||||
|
return '/uploads/images/' . $type . '/' . Date('Y-m-M') . '/' . $fileName;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Uploads an image with the given name.
|
||||||
|
* @param $name
|
||||||
|
* @param int $uploadedTo
|
||||||
|
* @return \Illuminate\Foundation\Testing\TestResponse
|
||||||
|
*/
|
||||||
|
protected function uploadImage($name, $uploadedTo = 0)
|
||||||
|
{
|
||||||
|
$file = $this->getTestImage($name);
|
||||||
|
return $this->call('POST', '/images/gallery/upload', ['uploaded_to' => $uploadedTo], [], ['file' => $file], []);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete an uploaded image.
|
||||||
|
* @param $relPath
|
||||||
|
*/
|
||||||
|
protected function deleteImage($relPath)
|
||||||
|
{
|
||||||
|
$path = public_path($relPath);
|
||||||
|
if (file_exists($path)) {
|
||||||
|
unlink($path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user