blessing-skin-server/app/Http/Controllers/SkinlibController.php

436 lines
14 KiB
PHP
Raw Normal View History

2016-07-21 22:01:57 +08:00
<?php
2016-08-28 10:05:21 +08:00
namespace App\Http\Controllers;
2016-07-21 22:01:57 +08:00
2019-09-03 18:44:21 +08:00
use Auth;
use View;
use Option;
use Session;
use Storage;
2016-07-21 22:01:57 +08:00
use App\Models\User;
use App\Models\Player;
2016-07-21 22:01:57 +08:00
use App\Models\Texture;
use App\Services\Filter;
use Illuminate\Http\Request;
use Illuminate\Http\JsonResponse;
2016-07-21 22:01:57 +08:00
class SkinlibController extends Controller
2016-07-21 22:01:57 +08:00
{
2018-08-17 22:54:26 +08:00
/**
* Map error code of file uploading to human-readable text.
*
* @see http://php.net/manual/en/features.file-upload.errors.php
* @var array
*/
public static $phpFileUploadErrors = [
0 => 'There is no error, the file uploaded with success',
1 => 'The uploaded file exceeds the upload_max_filesize directive in php.ini',
2 => 'The uploaded file exceeds the MAX_FILE_SIZE directive that was specified in the HTML form',
3 => 'The uploaded file was only partially uploaded',
4 => 'No file was uploaded',
6 => 'Missing a temporary folder',
7 => 'Failed to write file to disk.',
8 => 'A PHP extension stopped the file upload.',
];
/**
* Get skin library data filtered.
* Available Query String: filter, uploader, page, sort, keyword, items_per_page.
*
* @param Request $request [description]
2017-11-24 18:54:30 +08:00
* @return JsonResponse
*/
public function getSkinlibFiltered(Request $request)
{
2019-03-14 23:55:49 +08:00
$user = Auth::user();
2016-07-26 13:36:24 +08:00
// Available filters: skin, steve, alex, cape
$filter = $request->input('filter', 'skin');
2016-07-21 22:01:57 +08:00
// Filter result by uploader's uid
$uploader = intval($request->input('uploader', 0));
2017-05-05 12:43:53 +08:00
// Current page
$page = $request->input('page', 1);
$currentPage = ($page <= 0) ? 1 : $page;
2016-08-13 22:31:24 +08:00
// How many items to show in one page
$itemsPerPage = $request->input('items_per_page', 20);
2017-11-24 18:54:30 +08:00
$itemsPerPage = $itemsPerPage <= 0 ? 20 : $itemsPerPage;
2016-07-24 11:12:52 +08:00
// Keyword to search
$keyword = $request->input('keyword', '');
2016-07-21 22:01:57 +08:00
if ($filter == 'skin') {
$query = Texture::where(function ($innerQuery) {
// Nested condition, DO NOT MODIFY
2019-03-14 23:55:49 +08:00
$innerQuery->where('type', 'steve')->orWhere('type', 'alex');
});
} else {
$query = Texture::where('type', $filter);
}
2016-07-21 22:01:57 +08:00
if ($keyword !== '') {
$query = $query->like('name', $keyword);
}
2016-07-21 22:01:57 +08:00
if ($uploader !== 0) {
$query = $query->where('uploader', $uploader);
}
2019-03-14 23:55:49 +08:00
if (! $user) {
// Show public textures only to anonymous visitors
2018-07-16 11:10:01 +08:00
$query = $query->where('public', true);
2016-07-21 22:01:57 +08:00
} else {
// Show private textures when show uploaded textures of current user
2019-03-14 23:55:49 +08:00
if ($uploader != $user->uid && ! $user->isAdmin()) {
$query = $query->where(function ($innerQuery) use ($user) {
$innerQuery->where('public', true)->orWhere('uploader', '=', $user->uid);
});
}
2016-07-21 22:01:57 +08:00
}
2017-11-24 18:54:30 +08:00
$totalPages = ceil($query->count() / $itemsPerPage);
2019-03-19 15:19:33 +08:00
$sort = $request->input('sort', 'time');
$sortBy = $sort == 'time' ? 'upload_at' : $sort;
2019-05-04 16:02:46 +08:00
$query = $query->orderBy($sortBy, 'desc');
2019-05-04 16:02:46 +08:00
$textures = $query->skip(($currentPage - 1) * $itemsPerPage)->take($itemsPerPage)->get();
2019-03-19 15:19:33 +08:00
2019-03-14 23:55:49 +08:00
if ($user) {
$closet = $user->closet()->get();
foreach ($textures as $item) {
2019-03-14 23:55:49 +08:00
$item->liked = $closet->contains('tid', $item->tid);
}
}
2019-04-23 19:14:41 +08:00
return json('', 0, [
'items' => $textures,
2019-03-14 23:55:49 +08:00
'current_uid' => $user ? $user->uid : 0,
'total_pages' => $totalPages,
]);
2016-07-21 22:01:57 +08:00
}
public function show(Filter $filter, $tid)
2016-07-21 22:01:57 +08:00
{
$texture = Texture::find($tid);
$user = Auth::user();
if (! $texture || $texture && ! Storage::disk('textures')->has($texture->hash)) {
if (option('auto_del_invalid_texture')) {
if ($texture) {
2016-08-28 20:33:35 +08:00
$texture->delete();
}
2016-08-28 20:33:35 +08:00
2016-09-24 22:49:20 +08:00
abort(404, trans('skinlib.show.deleted'));
}
2016-09-24 22:49:20 +08:00
abort(404, trans('skinlib.show.deleted').trans('skinlib.show.contact-admin'));
2016-08-16 22:58:21 +08:00
}
if (! $texture->public) {
if (! Auth::check() || ($user->uid != $texture->uploader && ! $user->isAdmin())) {
abort(option('status_code_for_private'), trans('skinlib.show.private'));
}
2016-07-21 22:01:57 +08:00
}
$badges = [];
$uploader = User::find($texture->uploader);
if ($uploader) {
if ($uploader->isAdmin()) {
$badges[] = ['text' => 'STAFF', 'color' => 'primary'];
}
$badges = $filter->apply('user_badges', $badges, [$uploader]);
}
2019-09-18 23:06:48 +08:00
$commentScript = get_string_replaced(
option('comment_script'),
[
'{tid}' => $texture->tid,
'{name}' => $texture->name,
'{url}' => request()->url(),
]
);
return view('skinlib.show')
->with('texture', $texture)
2019-09-18 23:06:48 +08:00
->with('comment_script', $commentScript)
2019-03-23 19:52:14 +08:00
->with('extra', [
'download' => option('allow_downloading_texture'),
'currentUid' => $user ? $user->uid : 0,
'admin' => $user && $user->isAdmin(),
'inCloset' => $user && $user->closet()->where('tid', $texture->tid)->count() > 0,
2019-08-24 10:22:26 +08:00
'nickname' => ($up = User::find($texture->uploader)) ? $up->nickname : null,
'report' => intval(option('reporter_score_modification', 0)),
'badges' => $badges,
2019-03-23 19:52:14 +08:00
]);
2016-07-21 22:01:57 +08:00
}
public function info($tid)
{
2016-12-10 19:49:45 +08:00
if ($t = Texture::find($tid)) {
2019-04-23 19:14:41 +08:00
return json('', 0, $t->toArray());
2016-12-10 19:49:45 +08:00
} else {
2019-04-23 19:14:41 +08:00
return abort(404);
2016-12-10 19:49:45 +08:00
}
2016-07-21 22:01:57 +08:00
}
public function upload()
{
return view('skinlib.upload')
2019-03-23 19:52:14 +08:00
->with('extra', [
'rule' => ($regexp = option('texture_name_regexp'))
? trans('skinlib.upload.name-rule-regexp', compact('regexp'))
: trans('skinlib.upload.name-rule'),
'privacyNotice' => trans(
'skinlib.upload.private-score-notice',
['score' => option('private_score_per_storage')]
),
'scorePublic' => intval(option('score_per_storage')),
'scorePrivate' => intval(option('private_score_per_storage')),
'award' => intval(option('score_award_per_texture')),
'contentPolicy' => app('parsedown')->text(option_localized('content_policy')),
2019-09-18 23:06:48 +08:00
]);
2016-07-21 22:01:57 +08:00
}
public function handleUpload(Request $request)
2016-07-21 22:01:57 +08:00
{
$user = Auth::user();
2018-07-13 19:02:16 +08:00
if (($response = $this->checkUpload($request)) instanceof JsonResponse) {
return $response;
}
2016-07-21 22:01:57 +08:00
2019-04-04 11:16:04 +08:00
$file = $request->file('file');
$responses = event(new \App\Events\HashingFile($file));
if (isset($responses[0]) && is_string($responses[0])) {
return $responses[0]; // @codeCoverageIgnore
}
$t = new Texture();
$t->name = $request->input('name');
$t->type = $request->input('type');
2019-04-04 11:16:04 +08:00
$t->hash = hash_file('sha256', $file);
$t->size = ceil($request->file('file')->getSize() / 1024);
$t->public = $request->input('public') == 'true';
$t->uploader = $user->uid;
2016-07-21 22:01:57 +08:00
2018-07-16 11:10:01 +08:00
$cost = $t->size * ($t->public ? Option::get('score_per_storage') : Option::get('private_score_per_storage'));
2017-04-22 23:35:25 +08:00
$cost += option('score_per_closet_item');
2019-03-20 23:28:04 +08:00
$cost -= option('score_award_per_texture', 0);
2019-03-23 00:20:28 +08:00
if ($user->score < $cost) {
2016-09-24 22:49:20 +08:00
return json(trans('skinlib.upload.lack-score'), 7);
}
2016-07-21 22:01:57 +08:00
$results = Texture::where('hash', $t->hash)->get();
2018-02-16 17:31:04 +08:00
if (! $results->isEmpty()) {
2016-07-21 22:01:57 +08:00
foreach ($results as $result) {
2017-11-24 18:54:30 +08:00
// if the texture already uploaded was set to private,
// then allow to re-upload it.
2018-07-16 11:10:01 +08:00
if ($result->type == $t->type && $result->public) {
2019-04-23 19:14:41 +08:00
return json(trans('skinlib.upload.repeated'), 0, ['tid' => $result->tid]);
2016-07-21 22:01:57 +08:00
}
}
}
2019-12-03 15:41:54 +08:00
if (Storage::disk('textures')->missing($t->hash)) {
Storage::disk('textures')->put($t->hash, file_get_contents($request->file('file')));
}
$t->likes++;
2016-07-21 22:01:57 +08:00
$t->save();
2019-07-30 14:29:02 +08:00
$user->score -= $cost;
2019-03-14 23:55:49 +08:00
$user->closet()->attach($t->tid, ['item_name' => $t->name]);
2019-07-30 14:29:02 +08:00
$user->save();
2019-03-14 23:55:49 +08:00
return json(trans('skinlib.upload.success', ['name' => $request->input('name')]), 0, [
'tid' => $t->tid,
]);
}
// @codeCoverageIgnore
2016-07-21 22:01:57 +08:00
2019-03-23 11:06:36 +08:00
public function delete(Request $request)
2016-07-21 22:01:57 +08:00
{
$texture = Texture::find($request->tid);
$user = Auth::user();
2016-07-21 22:01:57 +08:00
if (! $texture) {
2016-09-24 22:49:20 +08:00
return json(trans('skinlib.non-existent'), 1);
}
2016-07-21 22:01:57 +08:00
if ($texture->uploader != $user->uid && ! $user->isAdmin()) {
2016-09-24 22:49:20 +08:00
return json(trans('skinlib.no-permission'), 1);
}
2016-07-21 22:01:57 +08:00
2016-07-24 15:56:23 +08:00
// check if file occupied
if (Texture::where('hash', $texture->hash)->count() == 1) {
Storage::disk('textures')->delete($texture->hash);
}
2016-07-21 22:01:57 +08:00
2019-05-07 15:16:53 +08:00
$texture->delete();
2019-05-19 13:49:44 +08:00
2019-05-07 15:16:53 +08:00
return json(trans('skinlib.delete.success'), 0);
}
2019-09-07 11:15:23 +08:00
public function privacy(Request $request)
2016-07-21 22:01:57 +08:00
{
$t = Texture::find($request->input('tid'));
$user = $request->user();
2016-07-24 15:56:23 +08:00
if (! $t) {
2016-09-24 22:49:20 +08:00
return json(trans('skinlib.non-existent'), 1);
}
2016-07-24 15:56:23 +08:00
if ($t->uploader != $user->uid && ! $user->isAdmin()) {
2016-09-24 22:49:20 +08:00
return json(trans('skinlib.no-permission'), 1);
}
2016-07-24 15:56:23 +08:00
2019-08-24 10:22:26 +08:00
$uploader = User::find($t->uploader);
2018-07-16 11:10:01 +08:00
$score_diff = $t->size * (option('private_score_per_storage') - option('score_per_storage')) * ($t->public ? -1 : 1);
2019-03-20 23:28:04 +08:00
if ($t->public && option('take_back_scores_after_deletion', true)) {
$score_diff -= option('score_award_per_texture', 0);
}
2019-03-23 11:06:36 +08:00
if ($uploader->score + $score_diff < 0) {
2017-09-18 19:28:38 +08:00
return json(trans('skinlib.upload.lack-score'), 1);
}
$type = $t->type == 'cape' ? 'cape' : 'skin';
2017-11-24 18:54:30 +08:00
Player::where("tid_$type", $t->tid)
->where('uid', '<>', session('uid'))
2019-03-14 23:55:49 +08:00
->update(["tid_$type" => 0]);
$t->likers()->get()->each(function ($user) use ($t) {
$user->closet()->detach($t->tid);
if (option('return_score')) {
2019-07-30 14:29:02 +08:00
$user->score += option('score_per_closet_item');
$user->save();
2019-03-14 23:55:49 +08:00
}
$t->likes--;
2019-03-14 23:55:49 +08:00
});
2019-07-30 14:29:02 +08:00
$uploader->score += $score_diff;
$uploader->save();
2017-04-21 18:44:11 +08:00
2019-03-03 00:38:44 +08:00
$t->public = ! $t->public;
2019-03-02 23:58:35 +08:00
$t->save();
2019-03-03 00:38:44 +08:00
2019-04-23 19:14:41 +08:00
return json(
trans('skinlib.privacy.success', ['privacy' => (! $t->public ? trans('general.private') : trans('general.public'))]),
0
);
}
public function rename(Request $request)
{
$this->validate($request, [
'tid' => 'required|integer',
'new_name' => 'required|no_special_chars',
]);
$user = $request->user();
$t = Texture::find($request->input('tid'));
2016-07-24 15:56:23 +08:00
if (! $t) {
2016-09-24 22:49:20 +08:00
return json(trans('skinlib.non-existent'), 1);
}
2016-07-24 15:56:23 +08:00
if ($t->uploader != $user->uid && ! $user->isAdmin()) {
2016-09-24 22:49:20 +08:00
return json(trans('skinlib.no-permission'), 1);
}
2016-07-24 15:56:23 +08:00
$t->name = $request->input('new_name');
2016-07-24 15:56:23 +08:00
if ($t->save()) {
2016-09-24 22:49:20 +08:00
return json(trans('skinlib.rename.success', ['name' => $request->input('new_name')]), 0);
2016-07-24 15:56:23 +08:00
}
}
// @codeCoverageIgnore
2016-07-24 15:56:23 +08:00
public function model(Request $request)
{
$user = $request->user();
$data = $this->validate($request, [
'tid' => 'required|integer',
'model' => 'required|in:steve,alex,cape',
]);
$t = Texture::find($request->input('tid'));
if (! $t) {
return json(trans('skinlib.non-existent'), 1);
}
if ($t->uploader != $user->uid && ! $user->isAdmin()) {
return json(trans('skinlib.no-permission'), 1);
}
$duplicate = Texture::where('hash', $t->hash)
->where('type', $request->input('model'))
->where('tid', '<>', $t->tid)
->first();
if ($duplicate && $duplicate->public) {
return json(trans('skinlib.model.duplicate', ['name' => $duplicate->name]), 1);
}
$t->type = $request->input('model');
$t->save();
return json(trans('skinlib.model.success', ['model' => $data['model']]), 0);
}
/**
* Check Uploaded Files.
*
* @param Request $request
2017-11-24 18:54:30 +08:00
* @return JsonResponse
*/
2018-02-16 17:31:04 +08:00
protected function checkUpload(Request $request)
2016-07-21 22:01:57 +08:00
{
if ($file = $request->files->get('file')) {
if ($file->getError() !== UPLOAD_ERR_OK) {
2018-08-17 22:54:26 +08:00
return json(static::$phpFileUploadErrors[$file->getError()], $file->getError());
}
}
$this->validate($request, [
'name' => [
'required',
option('texture_name_regexp') ? 'regex:'.option('texture_name_regexp') : 'no_special_chars',
],
'file' => 'required|max:'.option('max_upload_file_size'),
'public' => 'required',
]);
2018-07-20 17:32:27 +08:00
$mime = $request->file('file')->getMimeType();
if ($mime != 'image/png' && $mime != 'image/x-png') {
2016-09-24 22:49:20 +08:00
return json(trans('skinlib.upload.type-error'), 1);
}
$type = $request->input('type');
$size = getimagesize($request->file('file'));
$ratio = $size[0] / $size[1];
if ($type == 'steve' || $type == 'alex') {
if ($ratio != 2 && $ratio != 1) {
2016-09-24 22:49:20 +08:00
return json(trans('skinlib.upload.invalid-size', ['type' => trans('general.skin'), 'width' => $size[0], 'height' => $size[1]]), 1);
}
if ($size[0] % 64 != 0 || $size[1] % 32 != 0) {
2017-01-25 15:34:43 +08:00
return json(trans('skinlib.upload.invalid-hd-skin', ['type' => trans('general.skin'), 'width' => $size[0], 'height' => $size[1]]), 1);
}
} elseif ($type == 'cape') {
if ($ratio != 2) {
2016-09-24 22:49:20 +08:00
return json(trans('skinlib.upload.invalid-size', ['type' => trans('general.cape'), 'width' => $size[0], 'height' => $size[1]]), 1);
}
2016-07-21 22:01:57 +08:00
} else {
2016-09-24 22:49:20 +08:00
return json(trans('general.illegal-parameters'), 1);
2016-07-21 22:01:57 +08:00
}
}
2016-07-21 22:01:57 +08:00
// @codeCoverageIgnore
2016-07-21 22:01:57 +08:00
}