2017-11-24 18:54:30 +08:00
|
|
|
<?php
|
|
|
|
|
|
|
|
use App\Models\User;
|
|
|
|
use App\Models\Closet;
|
|
|
|
use App\Models\Player;
|
|
|
|
use App\Models\Texture;
|
|
|
|
use App\Services\Utils;
|
|
|
|
use org\bovigo\vfs\vfsStream;
|
|
|
|
use Illuminate\Http\UploadedFile;
|
|
|
|
use Illuminate\Foundation\Testing\WithoutMiddleware;
|
|
|
|
use Illuminate\Foundation\Testing\DatabaseMigrations;
|
|
|
|
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
|
|
|
|
|
|
|
class SkinlibControllerTest extends TestCase
|
|
|
|
{
|
|
|
|
use DatabaseTransactions;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var \org\bovigo\vfs\vfsStreamDirectory
|
|
|
|
*/
|
|
|
|
private $vfs_root;
|
|
|
|
|
|
|
|
protected function setUp()
|
|
|
|
{
|
|
|
|
parent::setUp();
|
|
|
|
$this->vfs_root = vfsStream::setup();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function serializeTextures($textures) {
|
|
|
|
return $textures
|
|
|
|
->map(function ($texture) {
|
|
|
|
return [
|
|
|
|
'tid' => $texture->tid,
|
|
|
|
'name' => $texture->name,
|
|
|
|
'type' => $texture->type,
|
|
|
|
'likes' => $texture->likes,
|
|
|
|
'hash' => $texture->hash,
|
|
|
|
'size' => $texture->size,
|
|
|
|
'uploader' => $texture->uploader,
|
2018-07-16 11:10:01 +08:00
|
|
|
'public' => $texture->public,
|
2017-11-24 18:54:30 +08:00
|
|
|
'upload_at' => $texture->upload_at->format('Y-m-d H:i:s')
|
|
|
|
];
|
|
|
|
})
|
|
|
|
->all();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testIndex()
|
|
|
|
{
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->get('/skinlib')->assertViewHas('user');
|
2017-11-24 18:54:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetSkinlibFiltered()
|
|
|
|
{
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => [],
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 0
|
|
|
|
]);
|
|
|
|
|
|
|
|
$steves = factory(Texture::class)->times(5)->create();
|
|
|
|
$alexs = factory(Texture::class, 'alex')->times(5)->create();
|
|
|
|
$skins = $steves->merge($alexs);
|
|
|
|
$capes = factory(Texture::class, 'cape')->times(5)->create();
|
|
|
|
|
|
|
|
// Default arguments
|
|
|
|
$expected = $skins
|
|
|
|
->sortByDesc('upload_at')
|
|
|
|
->values(); // WTF! DO NOT FORGET IT!!
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $this->serializeTextures($expected),
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 1
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Only steve
|
2018-07-13 19:02:16 +08:00
|
|
|
$expected = $steves
|
2017-11-24 18:54:30 +08:00
|
|
|
->sortByDesc('upload_at')
|
|
|
|
->values();
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data?filter=steve')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $this->serializeTextures($expected),
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 1
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Invalid type
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data?filter=what')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => [],
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 0
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Only capes
|
|
|
|
$expected = $capes
|
|
|
|
->sortByDesc('upload_at')
|
|
|
|
->values();
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data?filter=cape')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $this->serializeTextures($expected),
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 1
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Only specified uploader
|
|
|
|
$uid = $skins->random()->uploader;
|
|
|
|
$expected = $skins
|
|
|
|
->filter(function ($texture) use ($uid) {
|
|
|
|
return $texture->uploader == $uid;
|
|
|
|
})
|
|
|
|
->sortByDesc('upload_at')
|
|
|
|
->values();
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data?uploader='.$uid)
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $this->serializeTextures($expected),
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 1
|
|
|
|
]);
|
|
|
|
|
2018-07-15 18:34:38 +08:00
|
|
|
// Sort by `tid`
|
|
|
|
$this->getJson('/skinlib/data?sort=tid')
|
|
|
|
->assertJson([
|
|
|
|
'items' => $this->serializeTextures($skins->sortByDesc('tid')->values()),
|
2017-11-24 18:54:30 +08:00
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 1
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Search
|
|
|
|
$keyword = str_limit($skins->random()->name, 1, '');
|
|
|
|
$expected = $skins
|
|
|
|
->filter(function ($texture) use ($keyword) {
|
|
|
|
return str_contains($texture->name, $keyword) ||
|
|
|
|
str_contains($texture->name, strtolower($keyword));
|
|
|
|
})
|
|
|
|
->sortByDesc('upload_at')
|
|
|
|
->values();
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data?keyword='.$keyword)
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $this->serializeTextures($expected),
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 1
|
|
|
|
]);
|
|
|
|
|
|
|
|
// More than one argument
|
|
|
|
$keyword = str_limit($skins->random()->name, 1, '');
|
|
|
|
$expected = $skins
|
|
|
|
->filter(function ($texture) use ($keyword) {
|
|
|
|
return str_contains($texture->name, $keyword) ||
|
|
|
|
str_contains($texture->name, strtolower($keyword));
|
|
|
|
})
|
|
|
|
->sortByDesc('likes')
|
|
|
|
->values();
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data?sort=likes&keyword='.$keyword)
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $this->serializeTextures($expected),
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 1
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Pagination
|
|
|
|
$steves = factory(Texture::class)
|
|
|
|
->times(15)
|
|
|
|
->create()
|
|
|
|
->merge($steves);
|
|
|
|
$skins = $steves->merge($alexs);
|
|
|
|
$expected = $skins
|
|
|
|
->sortByDesc('upload_at')
|
|
|
|
->values()
|
|
|
|
->forPage(1, 20);
|
|
|
|
$expected = $this->serializeTextures($expected);
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $expected,
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 2
|
|
|
|
]);
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data?page=-5')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $expected,
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 2
|
|
|
|
]);
|
|
|
|
$expected = $skins
|
|
|
|
->sortByDesc('upload_at')
|
|
|
|
->values()
|
|
|
|
->forPage(2, 20)
|
|
|
|
->values();
|
|
|
|
$expected = $this->serializeTextures($expected);
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data?page=2')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $expected,
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 2
|
|
|
|
]);
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data?page=8')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => [],
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 2
|
|
|
|
]);
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data?items_per_page=-6&page=2')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $expected,
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 2
|
|
|
|
]);
|
|
|
|
$expected = $skins
|
|
|
|
->sortByDesc('upload_at')
|
|
|
|
->values()
|
|
|
|
->forPage(3, 8)
|
|
|
|
->values();
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data?page=3&items_per_page=8')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $this->serializeTextures($expected),
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 4
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Add some private textures
|
|
|
|
$uploader = factory(User::class)->create();
|
|
|
|
$otherUser = factory(User::class)->create();
|
|
|
|
$private = factory(Texture::class)
|
|
|
|
->times(5)
|
|
|
|
->create(['public' => false, 'uploader' => $uploader->uid]);
|
|
|
|
|
|
|
|
// If not logged in, private textures should not be shown
|
|
|
|
$expected = $skins
|
|
|
|
->sortByDesc('upload_at')
|
|
|
|
->values()
|
|
|
|
->forPage(1, 20);
|
|
|
|
$expected = $this->serializeTextures($expected);
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $expected,
|
|
|
|
'anonymous' => true,
|
|
|
|
'total_pages' => 2
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Other users should not see someone's private textures
|
2018-07-16 16:01:58 +08:00
|
|
|
for ($i = 0, $length = count($expected); $i < $length; $i++) {
|
2017-11-24 18:54:30 +08:00
|
|
|
$expected[$i]['liked'] = false;
|
|
|
|
}
|
|
|
|
$this->actAs($otherUser)
|
2018-07-13 19:02:16 +08:00
|
|
|
->getJson('/skinlib/data')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $expected,
|
|
|
|
'anonymous' => false,
|
|
|
|
'total_pages' => 2
|
|
|
|
]);
|
|
|
|
|
|
|
|
// A user has added a texture from skin library to his closet
|
|
|
|
$texture = $skins
|
|
|
|
->sortByDesc('upload_at')
|
|
|
|
->values()
|
|
|
|
->first();
|
|
|
|
$closet = new Closet($otherUser->uid);
|
|
|
|
$closet->add($texture->tid, $texture->name);
|
|
|
|
$closet->save();
|
2018-07-16 16:01:58 +08:00
|
|
|
for ($i = 0, $length = count($expected); $i < $length; $i++) {
|
2017-11-24 18:54:30 +08:00
|
|
|
if ($expected[$i]['tid'] == $texture->tid) {
|
|
|
|
$expected[$i]['liked'] = true;
|
|
|
|
} else {
|
|
|
|
$expected[$i]['liked'] = false;
|
|
|
|
}
|
|
|
|
}
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->getJson('/skinlib/data')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $expected,
|
|
|
|
'anonymous' => false,
|
|
|
|
'total_pages' => 2
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Uploader can see his private textures
|
|
|
|
$expected = $skins
|
|
|
|
->merge($private)
|
|
|
|
->sortByDesc('upload_at')
|
|
|
|
->values()
|
|
|
|
->forPage(1, 20);
|
|
|
|
$expected = $this->serializeTextures($expected);
|
2018-07-16 16:01:58 +08:00
|
|
|
for ($i = 0, $length = count($expected); $i < $length; $i++) {
|
2017-11-24 18:54:30 +08:00
|
|
|
// The reason we use `false` here is that some textures just were
|
|
|
|
// uploaded by this user, but these textures are not in his closet.
|
|
|
|
// By default(not in testing like now), when you uploaded a texture,
|
|
|
|
// that texture will be added to your closet.
|
|
|
|
// So here, we can assume that a user upload some textures, but he
|
|
|
|
// has deleted them from his closet.
|
|
|
|
$expected[$i]['liked'] = false;
|
|
|
|
}
|
|
|
|
$this->actAs($uploader)
|
2018-07-13 19:02:16 +08:00
|
|
|
->getJson('/skinlib/data')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $expected,
|
|
|
|
'anonymous' => false,
|
|
|
|
'total_pages' => 2
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Administrators can see private textures
|
|
|
|
$admin = factory(User::class, 'admin')->create();
|
|
|
|
$this->actAs($admin)
|
2018-07-13 19:02:16 +08:00
|
|
|
->getJson('/skinlib/data')
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'items' => $expected,
|
|
|
|
'anonymous' => false,
|
|
|
|
'total_pages' => 2
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testShow()
|
|
|
|
{
|
|
|
|
// Cannot find texture
|
|
|
|
$this->get('/skinlib/show/1')
|
2018-07-13 19:02:16 +08:00
|
|
|
->assertSee(trans('skinlib.show.deleted'));
|
2017-11-24 18:54:30 +08:00
|
|
|
|
|
|
|
// Invalid texture
|
|
|
|
option(['auto_del_invalid_texture' => false]);
|
|
|
|
$texture = factory(Texture::class)->create();
|
|
|
|
$this->get('/skinlib/show/'.$texture->tid)
|
2018-07-13 19:02:16 +08:00
|
|
|
->assertSee(trans('skinlib.show.deleted').trans('skinlib.show.contact-admin'));
|
2017-11-24 18:54:30 +08:00
|
|
|
$this->assertNotNull(Texture::find($texture->tid));
|
|
|
|
|
|
|
|
option(['auto_del_invalid_texture' => true]);
|
|
|
|
$this->get('/skinlib/show/'.$texture->tid)
|
2018-07-13 19:02:16 +08:00
|
|
|
->assertSee(trans('skinlib.show.deleted'));
|
2017-11-24 18:54:30 +08:00
|
|
|
$this->assertNull(Texture::find($texture->tid));
|
|
|
|
|
|
|
|
// Show a texture
|
|
|
|
$texture = factory(Texture::class)->create();
|
|
|
|
Storage::disk('textures')->put($texture->hash, '');
|
|
|
|
$this->get('/skinlib/show/'.$texture->tid)
|
|
|
|
->assertViewHas('texture')
|
|
|
|
->assertViewHas('with_out_filter', true)
|
|
|
|
->assertViewHas('user');
|
|
|
|
|
|
|
|
// Guest should not see private texture
|
|
|
|
$uploader = factory(User::class)->create();
|
|
|
|
$texture = factory(Texture::class)->create([
|
|
|
|
'uploader' => $uploader->uid,
|
|
|
|
'public' => false
|
|
|
|
]);
|
|
|
|
Storage::disk('textures')->put($texture->hash, '');
|
|
|
|
$this->get('/skinlib/show/'.$texture->tid)
|
2018-07-13 19:02:16 +08:00
|
|
|
->assertSee(trans('skinlib.show.private'));
|
2017-11-24 18:54:30 +08:00
|
|
|
|
|
|
|
// Other user should not see private texture
|
|
|
|
$this->actAs('normal')
|
|
|
|
->get('/skinlib/show/'.$texture->tid)
|
2018-07-13 19:02:16 +08:00
|
|
|
->assertSee(trans('skinlib.show.private'));
|
2017-11-24 18:54:30 +08:00
|
|
|
|
|
|
|
// Administrators should be able to see private textures
|
|
|
|
$this->actAs('admin')
|
|
|
|
->get('/skinlib/show/'.$texture->tid)
|
|
|
|
->assertViewHas('texture');
|
|
|
|
|
|
|
|
// Uploader should be able to see private textures
|
|
|
|
$this->actAs($uploader)
|
|
|
|
->get('/skinlib/show/'.$texture->tid)
|
|
|
|
->assertViewHas('texture');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testInfo()
|
|
|
|
{
|
|
|
|
// Non-existed texture
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->get('/skinlib/info/1')->assertJson([]);
|
2017-11-24 18:54:30 +08:00
|
|
|
|
|
|
|
$texture = factory(Texture::class)->create();
|
|
|
|
$this->get('/skinlib/info/'.$texture->tid)
|
2018-07-13 19:02:16 +08:00
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'tid' => $texture->tid,
|
|
|
|
'name' => $texture->name,
|
|
|
|
'type' => $texture->type,
|
|
|
|
'likes' => $texture->likes,
|
|
|
|
'hash' => $texture->hash,
|
|
|
|
'size' => $texture->size,
|
|
|
|
'uploader' => $texture->uploader,
|
2018-07-16 11:10:01 +08:00
|
|
|
'public' => $texture->public,
|
2017-11-24 18:54:30 +08:00
|
|
|
'upload_at' => $texture->upload_at->format('Y-m-d H:i:s')
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testUpload()
|
|
|
|
{
|
|
|
|
$this->actAs('normal')
|
2018-07-13 19:02:16 +08:00
|
|
|
->get('/skinlib/upload')
|
2017-11-24 18:54:30 +08:00
|
|
|
->assertViewHas('user')
|
|
|
|
->assertViewHas('with_out_filter', true);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testHandleUpload()
|
|
|
|
{
|
2018-07-13 19:02:16 +08:00
|
|
|
Storage::fake('textures');
|
|
|
|
|
2017-11-24 18:54:30 +08:00
|
|
|
// Some error occurred when uploading file
|
|
|
|
$file = vfsStream::newFile('test.png')
|
|
|
|
->at($this->vfs_root);
|
|
|
|
$upload = new UploadedFile(
|
|
|
|
$file->url(),
|
|
|
|
$file->getName(),
|
|
|
|
'image/png',
|
|
|
|
50,
|
|
|
|
UPLOAD_ERR_NO_TMP_DIR,
|
|
|
|
true
|
|
|
|
);
|
|
|
|
$this->actAs('normal')
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson(
|
2017-11-24 18:54:30 +08:00
|
|
|
'/skinlib/upload',
|
|
|
|
['file' => $upload]
|
2018-07-13 19:02:16 +08:00
|
|
|
)->assertJson([
|
|
|
|
'errno' => UPLOAD_ERR_NO_TMP_DIR,
|
|
|
|
'msg' => Utils::convertUploadFileError(UPLOAD_ERR_NO_TMP_DIR)
|
|
|
|
]);
|
2017-11-24 18:54:30 +08:00
|
|
|
|
|
|
|
// Without `name` field
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson('/skinlib/upload', [], [
|
2017-11-24 18:54:30 +08:00
|
|
|
'X-Requested-With' => 'XMLHttpRequest'
|
2018-07-13 19:02:16 +08:00
|
|
|
])->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('validation.required', ['attribute' => 'Name'])
|
|
|
|
]);
|
|
|
|
|
|
|
|
// With some special chars
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson('/skinlib/upload', [
|
2017-11-24 18:54:30 +08:00
|
|
|
'name' => '\\'
|
|
|
|
], [
|
|
|
|
'X-Requested-With' => 'XMLHttpRequest'
|
2018-07-13 19:02:16 +08:00
|
|
|
])->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('validation.no_special_chars', ['attribute' => 'Name'])
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Without file
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson('/skinlib/upload', [
|
2017-11-24 18:54:30 +08:00
|
|
|
'name' => 'texture'
|
|
|
|
], [
|
|
|
|
'X-Requested-With' => 'XMLHttpRequest'
|
2018-07-13 19:02:16 +08:00
|
|
|
])->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('validation.required', ['attribute' => 'File'])
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Too large file
|
|
|
|
option(['max_upload_file_size' => 2]);
|
2018-07-13 19:02:16 +08:00
|
|
|
$upload = UploadedFile::fake()->create('large.png', 5);
|
|
|
|
$this->postJson('/skinlib/upload', [
|
2017-11-24 18:54:30 +08:00
|
|
|
'name' => 'texture',
|
|
|
|
'file' => $upload
|
|
|
|
], [
|
|
|
|
'X-Requested-With' => 'XMLHttpRequest'
|
2018-07-13 19:02:16 +08:00
|
|
|
])->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('validation.max.file', ['attribute' => 'File', 'max' => '2'])
|
|
|
|
]);
|
|
|
|
option(['max_upload_file_size' => 1024]);
|
|
|
|
|
|
|
|
// Without `public` field
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson('/skinlib/upload', [
|
2017-11-24 18:54:30 +08:00
|
|
|
'name' => 'texture',
|
|
|
|
'file' => 'content' // Though it is not a file, it is OK
|
|
|
|
], [
|
|
|
|
'X-Requested-With' => 'XMLHttpRequest'
|
2018-07-13 19:02:16 +08:00
|
|
|
])->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('validation.required', ['attribute' => 'public'])
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Not a PNG image
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson(
|
2017-11-24 18:54:30 +08:00
|
|
|
'/skinlib/upload',
|
|
|
|
[
|
|
|
|
'name' => 'texture',
|
2018-07-13 19:02:16 +08:00
|
|
|
'public' => 'true',
|
|
|
|
'file' => UploadedFile::fake()->create('fake', 5)
|
|
|
|
]
|
|
|
|
)->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('skinlib.upload.type-error')
|
|
|
|
]);
|
|
|
|
|
|
|
|
// No texture type is specified
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson(
|
2017-11-24 18:54:30 +08:00
|
|
|
'/skinlib/upload',
|
|
|
|
[
|
|
|
|
'name' => 'texture',
|
2018-07-13 19:02:16 +08:00
|
|
|
'public' => 'true',
|
|
|
|
'file' => UploadedFile::fake()->image('texture.png', 64, 32)
|
|
|
|
]
|
|
|
|
)->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('general.illegal-parameters')
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Invalid skin size
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson(
|
2017-11-24 18:54:30 +08:00
|
|
|
'/skinlib/upload',
|
|
|
|
[
|
|
|
|
'name' => 'texture',
|
|
|
|
'public' => 'true',
|
2018-07-13 19:02:16 +08:00
|
|
|
'type' => 'steve',
|
|
|
|
'file' => UploadedFile::fake()->image('texture.png', 64, 30)
|
|
|
|
]
|
|
|
|
)->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans(
|
|
|
|
'skinlib.upload.invalid-size',
|
|
|
|
[
|
|
|
|
'type' => trans('general.skin'),
|
|
|
|
'width' => 64,
|
|
|
|
'height' => 30
|
|
|
|
]
|
|
|
|
)
|
|
|
|
]);
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson(
|
2017-11-24 18:54:30 +08:00
|
|
|
'/skinlib/upload',
|
|
|
|
[
|
|
|
|
'name' => 'texture',
|
|
|
|
'public' => 'true',
|
2018-07-13 19:02:16 +08:00
|
|
|
'type' => 'alex',
|
|
|
|
'file' => UploadedFile::fake()->image('texture.png', 100, 50)
|
|
|
|
]
|
|
|
|
)->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans(
|
|
|
|
'skinlib.upload.invalid-hd-skin',
|
|
|
|
[
|
|
|
|
'type' => trans('general.skin'),
|
|
|
|
'width' => 100,
|
|
|
|
'height' => 50
|
|
|
|
]
|
|
|
|
)
|
|
|
|
]);
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson(
|
2017-11-24 18:54:30 +08:00
|
|
|
'/skinlib/upload',
|
|
|
|
[
|
|
|
|
'name' => 'texture',
|
|
|
|
'public' => 'true',
|
2018-07-13 19:02:16 +08:00
|
|
|
'type' => 'cape',
|
|
|
|
'file' => UploadedFile::fake()->image('texture.png', 64, 30)
|
|
|
|
]
|
|
|
|
)->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans(
|
|
|
|
'skinlib.upload.invalid-size',
|
|
|
|
[
|
|
|
|
'type' => trans('general.cape'),
|
|
|
|
'width' => 64,
|
|
|
|
'height' => 30
|
|
|
|
]
|
|
|
|
)
|
|
|
|
]);
|
|
|
|
|
2018-07-13 19:02:16 +08:00
|
|
|
$upload = UploadedFile::fake()->image('texture.png', 64, 32);
|
2017-11-24 18:54:30 +08:00
|
|
|
|
|
|
|
// Score is not enough
|
|
|
|
$user = factory(User::class)->create(['score' => 0]);
|
|
|
|
$this->actAs($user)
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson(
|
2017-11-24 18:54:30 +08:00
|
|
|
'/skinlib/upload',
|
|
|
|
[
|
|
|
|
'name' => 'texture',
|
2018-07-13 19:02:16 +08:00
|
|
|
'public' => 'true',
|
|
|
|
'type' => 'steve',
|
|
|
|
'file' => $upload
|
|
|
|
]
|
|
|
|
)
|
|
|
|
->assertJson([
|
|
|
|
'errno' => 7,
|
|
|
|
'msg' => trans('skinlib.upload.lack-score')
|
|
|
|
]);
|
|
|
|
|
|
|
|
$user = factory(User::class)->create([
|
|
|
|
'score' => (int) option('score_per_closet_item') + (int) option('score_per_storage')
|
|
|
|
]);
|
|
|
|
$this->actAs($user)->postJson(
|
|
|
|
'/skinlib/upload',
|
|
|
|
[
|
|
|
|
'name' => 'texture',
|
|
|
|
'public' => 'false', // Private texture cost more scores
|
|
|
|
'type' => 'steve',
|
|
|
|
'file' => $upload
|
|
|
|
]
|
|
|
|
)->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 7,
|
|
|
|
'msg' => trans('skinlib.upload.lack-score')
|
|
|
|
]);
|
2018-07-13 19:02:16 +08:00
|
|
|
$response = $this->postJson(
|
2017-11-24 18:54:30 +08:00
|
|
|
'/skinlib/upload',
|
|
|
|
[
|
|
|
|
'name' => 'texture',
|
|
|
|
'public' => 'true', // Public texture
|
2018-07-13 19:02:16 +08:00
|
|
|
'type' => 'steve',
|
|
|
|
'file' => $upload
|
|
|
|
]
|
2017-11-24 18:54:30 +08:00
|
|
|
);
|
2018-07-13 19:02:16 +08:00
|
|
|
$t = Texture::where('name', 'texture')->first();
|
|
|
|
$response->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 0,
|
|
|
|
'msg' => trans('skinlib.upload.success', ['name' => 'texture']),
|
2018-07-13 19:02:16 +08:00
|
|
|
'tid' => $t->tid
|
2017-11-24 18:54:30 +08:00
|
|
|
]);
|
2018-07-13 19:02:16 +08:00
|
|
|
Storage::disk('textures')->assertExists($t->hash);
|
|
|
|
$user = User::find($user->uid);
|
|
|
|
$this->assertEquals(0, $user->score);
|
|
|
|
$this->assertEquals('texture', $t->name);
|
|
|
|
$this->assertEquals('steve', $t->type);
|
|
|
|
$this->assertEquals(1, $t->likes);
|
|
|
|
$this->assertEquals(1, $t->size);
|
|
|
|
$this->assertEquals($user->uid, $t->uploader);
|
2018-07-16 11:10:01 +08:00
|
|
|
$this->assertTrue($t->public);
|
2017-11-24 18:54:30 +08:00
|
|
|
|
|
|
|
// Upload a duplicated texture
|
2018-07-13 19:02:16 +08:00
|
|
|
$user = factory(User::class)->create();
|
|
|
|
$this->actAs($user)
|
|
|
|
->postJson(
|
|
|
|
'/skinlib/upload',
|
|
|
|
[
|
|
|
|
'name' => 'texture',
|
|
|
|
'public' => 'true',
|
|
|
|
'type' => 'steve',
|
|
|
|
'file' => $upload
|
|
|
|
]
|
|
|
|
)->assertJson([
|
|
|
|
'errno' => 0,
|
|
|
|
'msg' => trans('skinlib.upload.repeated'),
|
|
|
|
'tid' => $t->tid
|
|
|
|
]);
|
|
|
|
|
|
|
|
unlink(storage_path('framework/testing/disks/textures/'.$t->hash));
|
2017-11-24 18:54:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testDelete()
|
|
|
|
{
|
|
|
|
$uploader = factory(User::class)->create();
|
|
|
|
$other = factory(User::class)->create();
|
|
|
|
$texture = factory(Texture::class)->create(['uploader' => $uploader->uid]);
|
|
|
|
option(['return_score' => false]);
|
|
|
|
|
|
|
|
// Non-existed texture
|
|
|
|
$this->actAs($uploader)
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson('/skinlib/delete', ['tid' => -1])
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('skinlib.non-existent')
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Other user should not be able to delete
|
|
|
|
$this->actAs($other)
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson('/skinlib/delete', ['tid' => $texture->tid])
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('skinlib.no-permission')
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Administrators can delete it
|
|
|
|
$this->actAs('admin')
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson('/skinlib/delete', ['tid' => $texture->tid])
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 0,
|
|
|
|
'msg' => trans('skinlib.delete.success')
|
|
|
|
]);
|
|
|
|
$this->assertNull(Texture::find($texture->tid));
|
|
|
|
|
|
|
|
$texture = factory(Texture::class)->create();
|
|
|
|
factory(Texture::class)->create(['hash' => $texture->hash]);
|
|
|
|
Storage::disk('textures')->put($texture->hash, '');
|
|
|
|
|
|
|
|
// When file is occupied, the file should not be deleted
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson('/skinlib/delete', ['tid' => $texture->tid])
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 0,
|
|
|
|
'msg' => trans('skinlib.delete.success')
|
|
|
|
]);
|
|
|
|
$this->assertNull(Texture::find($texture->tid));
|
|
|
|
$this->assertTrue(Storage::disk('textures')->exists($texture->hash));
|
|
|
|
|
|
|
|
$texture = factory(Texture::class)->create();
|
|
|
|
factory(Texture::class)->create(['hash' => $texture->hash]);
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson('/skinlib/delete', ['tid' => $texture->tid])
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 0,
|
|
|
|
'msg' => trans('skinlib.delete.success')
|
|
|
|
]);
|
|
|
|
$this->assertNull(Texture::find($texture->tid));
|
|
|
|
$this->assertFalse(Storage::disk('textures')->exists($texture->hash));
|
|
|
|
|
|
|
|
// Return score
|
|
|
|
option(['return_score' => true]);
|
|
|
|
$texture = factory(Texture::class)->create(['uploader' => $uploader->uid]);
|
|
|
|
$this->actAs($uploader)
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson('/skinlib/delete', ['tid' => $texture->tid])
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 0,
|
|
|
|
'msg' => trans('skinlib.delete.success')
|
|
|
|
]);
|
|
|
|
$this->assertEquals(
|
|
|
|
$uploader->score + $texture->size * option('score_per_storage'),
|
|
|
|
User::find($uploader->uid)->score
|
|
|
|
);
|
|
|
|
|
|
|
|
$uploader = User::find($uploader->uid);
|
|
|
|
$texture = factory(Texture::class)->create([
|
|
|
|
'uploader' => $uploader->uid,
|
|
|
|
'public' => false
|
|
|
|
]);
|
|
|
|
$this->actAs($uploader)
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson('/skinlib/delete', ['tid' => $texture->tid])
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 0,
|
|
|
|
'msg' => trans('skinlib.delete.success')
|
|
|
|
]);
|
|
|
|
$this->assertEquals(
|
|
|
|
$uploader->score + $texture->size * option('private_score_per_storage'),
|
|
|
|
User::find($uploader->uid)->score
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testPrivacy()
|
|
|
|
{
|
|
|
|
$uploader = factory(User::class)->create();
|
|
|
|
$other = factory(User::class)->create();
|
|
|
|
$texture = factory(Texture::class)->create(['uploader' => $uploader->uid]);
|
|
|
|
|
|
|
|
// Non-existed texture
|
|
|
|
$this->actAs($uploader)
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson('/skinlib/privacy', ['tid' => -1])
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('skinlib.non-existent')
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Other user should not be able to set privacy
|
|
|
|
$this->actAs($other)
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson('/skinlib/privacy', ['tid' => $texture->tid])
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('skinlib.no-permission')
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Administrators can change it
|
|
|
|
$uploader->score += $texture->size * option('private_score_per_storage');
|
|
|
|
$uploader->save();
|
|
|
|
$this->actAs('admin')
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson('/skinlib/privacy', ['tid' => $texture->tid])
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 0,
|
|
|
|
'msg' => trans('skinlib.privacy.success', ['privacy' => trans('general.private')]),
|
2018-07-16 11:10:01 +08:00
|
|
|
'public' => false
|
2017-11-24 18:54:30 +08:00
|
|
|
]);
|
|
|
|
$this->assertEquals(0, Texture::find($texture->tid)->public);
|
|
|
|
|
|
|
|
// Setting a texture to be private needs more scores
|
|
|
|
$texture = factory(Texture::class)->create(['uploader' => $uploader->uid]);
|
|
|
|
$uploader->score = 0;
|
|
|
|
$uploader->save();
|
|
|
|
$this->actAs($uploader)
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson('/skinlib/privacy', ['tid' => $texture->tid])
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('skinlib.upload.lack-score')
|
|
|
|
]);
|
|
|
|
$this->assertEquals(1, Texture::find($texture->tid)->public);
|
|
|
|
|
|
|
|
$texture->public = true;
|
|
|
|
$texture->save();
|
|
|
|
$uploader->score = $texture->size *
|
|
|
|
(option('private_score_per_storage') - option('score_per_storage'));
|
|
|
|
$uploader->save();
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson('/skinlib/privacy', ['tid' => $texture->tid])
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 0,
|
|
|
|
'msg' => trans('skinlib.privacy.success', ['privacy' => trans('general.private')]),
|
2018-07-16 11:10:01 +08:00
|
|
|
'public' => false
|
2017-11-24 18:54:30 +08:00
|
|
|
]);
|
|
|
|
$this->assertEquals(0, User::find($uploader->uid)->score);
|
|
|
|
|
|
|
|
// When setting a texture to be private,
|
|
|
|
// other players should not be able to use it.
|
2018-07-14 08:27:15 +08:00
|
|
|
$texture = factory(Texture::class)->create(['uploader' => $uploader->uid]);
|
2017-11-24 18:54:30 +08:00
|
|
|
$uploader->score += $texture->size * option('private_score_per_storage');
|
|
|
|
$uploader->save();
|
|
|
|
$player = factory(Player::class)->create(['tid_steve' => $texture->tid]);
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson('/skinlib/privacy', ['tid' => $texture->tid])
|
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 0,
|
|
|
|
'msg' => trans('skinlib.privacy.success', ['privacy' => trans('general.private')]),
|
2018-07-16 11:10:01 +08:00
|
|
|
'public' => false
|
2017-11-24 18:54:30 +08:00
|
|
|
]);
|
|
|
|
$this->assertEquals(0, Player::find($player->pid)->tid_steve);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testRename()
|
|
|
|
{
|
|
|
|
$uploader = factory(User::class)->create();
|
|
|
|
$other = factory(User::class)->create();
|
|
|
|
$texture = factory(Texture::class)->create(['uploader' => $uploader->uid]);
|
|
|
|
|
|
|
|
// Without `tid` field
|
|
|
|
$this->actAs($uploader)
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson('/skinlib/rename', [], [
|
2017-11-24 18:54:30 +08:00
|
|
|
'X-Requested-With' => 'XMLHttpRequest'
|
|
|
|
])
|
2018-07-13 19:02:16 +08:00
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('validation.required', ['attribute' => 'tid'])
|
|
|
|
]);
|
|
|
|
|
|
|
|
// `tid` is not a integer
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson('/skinlib/rename', [
|
2017-11-24 18:54:30 +08:00
|
|
|
'tid' => 'str'
|
|
|
|
], [
|
|
|
|
'X-Requested-With' => 'XMLHttpRequest'
|
|
|
|
])
|
2018-07-13 19:02:16 +08:00
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('validation.integer', ['attribute' => 'tid'])
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Without `new_name` field
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson('/skinlib/rename', [
|
2017-11-24 18:54:30 +08:00
|
|
|
'tid' => $texture->tid
|
|
|
|
], [
|
|
|
|
'X-Requested-With' => 'XMLHttpRequest'
|
|
|
|
])
|
2018-07-13 19:02:16 +08:00
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('validation.required', ['attribute' => 'new name'])
|
|
|
|
]);
|
|
|
|
|
|
|
|
// `new_name` has special chars
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson('/skinlib/rename', [
|
2017-11-24 18:54:30 +08:00
|
|
|
'tid' => $texture->tid,
|
|
|
|
'new_name' => '\\'
|
|
|
|
], [
|
|
|
|
'X-Requested-With' => 'XMLHttpRequest'
|
|
|
|
])
|
2018-07-13 19:02:16 +08:00
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('validation.no_special_chars', ['attribute' => 'new name'])
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Non-existed texture
|
2018-07-13 19:02:16 +08:00
|
|
|
$this->postJson('/skinlib/rename', [
|
2017-11-24 18:54:30 +08:00
|
|
|
'tid' => -1,
|
|
|
|
'new_name' => 'name'
|
|
|
|
])
|
2018-07-13 19:02:16 +08:00
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('skinlib.non-existent')
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Other user should not be able to rename
|
|
|
|
$this->actAs($other)
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson('/skinlib/rename', [
|
2017-11-24 18:54:30 +08:00
|
|
|
'tid' => $texture->tid,
|
|
|
|
'new_name' => 'name'
|
|
|
|
])
|
2018-07-13 19:02:16 +08:00
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 1,
|
|
|
|
'msg' => trans('skinlib.no-permission')
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Administrators should be able to rename
|
|
|
|
$this->actAs('admin')
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson('/skinlib/rename', [
|
2017-11-24 18:54:30 +08:00
|
|
|
'tid' => $texture->tid,
|
|
|
|
'new_name' => 'name'
|
|
|
|
])
|
2018-07-13 19:02:16 +08:00
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 0,
|
|
|
|
'msg' => trans('skinlib.rename.success', ['name' => 'name'])
|
|
|
|
]);
|
|
|
|
$this->assertEquals('name', Texture::find($texture->tid)->name);
|
|
|
|
|
|
|
|
// Uploader should be able to rename
|
|
|
|
$this->actAs($uploader)
|
2018-07-13 19:02:16 +08:00
|
|
|
->postJson('/skinlib/rename', [
|
2017-11-24 18:54:30 +08:00
|
|
|
'tid' => $texture->tid,
|
|
|
|
'new_name' => 'new_name'
|
|
|
|
])
|
2018-07-13 19:02:16 +08:00
|
|
|
->assertJson([
|
2017-11-24 18:54:30 +08:00
|
|
|
'errno' => 0,
|
|
|
|
'msg' => trans('skinlib.rename.success', ['name' => 'new_name'])
|
|
|
|
]);
|
|
|
|
$this->assertEquals('new_name', Texture::find($texture->tid)->name);
|
|
|
|
}
|
|
|
|
}
|