Files
nextcloud-forum/lib/Controller/FileController.php

123 lines
4.2 KiB
PHP

<?php
declare(strict_types=1);
// SPDX-FileCopyrightText: Chen Asraf <contact@casraf.dev>
// SPDX-License-Identifier: AGPL-3.0-or-later
namespace OCA\Forum\Controller;
use OCA\Forum\Db\PostMapper;
use OCP\AppFramework\Controller;
use OCP\AppFramework\Db\DoesNotExistException;
use OCP\AppFramework\Http;
use OCP\AppFramework\Http\Attribute\NoAdminRequired;
use OCP\AppFramework\Http\Attribute\NoCSRFRequired;
use OCP\AppFramework\Http\Attribute\Route;
use OCP\AppFramework\Http\DataResponse;
use OCP\AppFramework\Http\FileDisplayResponse;
use OCP\Files\IRootFolder;
use OCP\Files\NotFoundException;
use OCP\IRequest;
use Psr\Log\LoggerInterface;
class FileController extends Controller {
public function __construct(
string $appName,
IRequest $request,
private PostMapper $postMapper,
private IRootFolder $rootFolder,
private LoggerInterface $logger,
private ?string $userId,
) {
parent::__construct($appName, $request);
}
/**
* Download a BBCode attachment file
*/
#[NoAdminRequired]
#[NoCSRFRequired]
#[Route(type: Route::TYPE_FRONTPAGE, verb: 'GET', url: '/api/posts/{postId}/files')]
public function download(int $postId, string $filePath): FileDisplayResponse|DataResponse {
try {
$post = $this->postMapper->find($postId);
// Verify the file path is in the post content
$attachmentTag = '[attachment]' . $filePath . '[/attachment]';
if (strpos($post->getContent(), $attachmentTag) === false) {
return new DataResponse(['error' => 'File not attached to this post'], Http::STATUS_FORBIDDEN);
}
// Get the author's folder
$authorId = $post->getAuthorId();
$userFolder = $this->rootFolder->getUserFolder($authorId);
// Get the file
$file = $userFolder->get($filePath);
if (!($file instanceof \OCP\Files\File)) {
return new DataResponse(['error' => 'Invalid file'], Http::STATUS_BAD_REQUEST);
}
$response = new FileDisplayResponse($file, Http::STATUS_OK, ['Content-Type' => $file->getMimeType()]);
$response->addHeader('Cache-Control', 'public, max-age=3600');
return $response;
} catch (DoesNotExistException $e) {
return new DataResponse(['error' => 'Post not found'], Http::STATUS_NOT_FOUND);
} catch (NotFoundException $e) {
return new DataResponse(['error' => 'File not found'], Http::STATUS_NOT_FOUND);
} catch (\Exception $e) {
$this->logger->error('Error serving attachment: ' . $e->getMessage());
return new DataResponse(['error' => 'Error loading file'], Http::STATUS_INTERNAL_SERVER_ERROR);
}
}
/**
* Get preview for a BBCode attachment
*/
#[NoAdminRequired]
#[NoCSRFRequired]
#[Route(type: Route::TYPE_FRONTPAGE, verb: 'GET', url: '/api/posts/{postId}/preview')]
public function preview(int $postId, string $filePath, int $x = 1920, int $y = 1080): FileDisplayResponse|DataResponse {
try {
$post = $this->postMapper->find($postId);
// Verify the file path is in the post content
$attachmentTag = '[attachment]' . $filePath . '[/attachment]';
if (strpos($post->getContent(), $attachmentTag) === false) {
return new DataResponse(['error' => 'File not attached to this post'], Http::STATUS_FORBIDDEN);
}
// Get the author's folder
$authorId = $post->getAuthorId();
$userFolder = $this->rootFolder->getUserFolder($authorId);
// Get the file
$file = $userFolder->get($filePath);
if (!($file instanceof \OCP\Files\File)) {
return new DataResponse(['error' => 'Invalid file'], Http::STATUS_BAD_REQUEST);
}
// Only for images
if (!str_starts_with($file->getMimeType(), 'image/')) {
return $this->download($postId, $filePath);
}
$response = new FileDisplayResponse($file, Http::STATUS_OK, ['Content-Type' => $file->getMimeType()]);
$response->addHeader('Cache-Control', 'public, max-age=86400');
return $response;
} catch (DoesNotExistException $e) {
return new DataResponse(['error' => 'Post not found'], Http::STATUS_NOT_FOUND);
} catch (NotFoundException $e) {
return new DataResponse(['error' => 'File not found'], Http::STATUS_NOT_FOUND);
} catch (\Exception $e) {
$this->logger->error('Error serving preview: ' . $e->getMessage());
return new DataResponse(['error' => 'Error loading preview'], Http::STATUS_INTERNAL_SERVER_ERROR);
}
}
}