mirror of
https://github.com/chenasraf/nextcloud-pantry.git
synced 2026-05-18 01:28:57 +00:00
188 lines
6.1 KiB
PHP
188 lines
6.1 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
// SPDX-FileCopyrightText: Chen Asraf <contact@casraf.dev>
|
|
// SPDX-License-Identifier: AGPL-3.0-or-later
|
|
|
|
namespace OCA\Pantry\Controller;
|
|
|
|
use OCA\Pantry\Exception\ForbiddenException;
|
|
use OCA\Pantry\Exception\NotFoundException;
|
|
use OCA\Pantry\ResponseDefinitions;
|
|
use OCA\Pantry\Service\HouseAuthService;
|
|
use OCA\Pantry\Service\NoteService;
|
|
use OCA\Pantry\Service\NotificationService;
|
|
use OCP\AppFramework\Http;
|
|
use OCP\AppFramework\Http\Attribute\ApiRoute;
|
|
use OCP\AppFramework\Http\Attribute\NoAdminRequired;
|
|
use OCP\AppFramework\Http\DataResponse;
|
|
use OCP\AppFramework\OCSController;
|
|
use OCP\IRequest;
|
|
use OCP\IUserSession;
|
|
|
|
/**
|
|
* @psalm-import-type PantryNote from ResponseDefinitions
|
|
* @psalm-import-type PantrySuccess from ResponseDefinitions
|
|
*/
|
|
final class NoteController extends OCSController {
|
|
use TranslatesDomainExceptions;
|
|
|
|
public function __construct(
|
|
string $appName,
|
|
IRequest $request,
|
|
private NoteService $notes,
|
|
private HouseAuthService $auth,
|
|
private NotificationService $notifications,
|
|
private IUserSession $userSession,
|
|
) {
|
|
parent::__construct($appName, $request);
|
|
}
|
|
|
|
/**
|
|
* List all notes in a house
|
|
*
|
|
* @param int $houseId House id.
|
|
* @param int<1, 500> $limit Maximum number of notes to return.
|
|
* @param int<0, max> $offset Number of notes to skip.
|
|
*
|
|
* @return DataResponse<Http::STATUS_OK, list<PantryNote>, array{}>
|
|
*
|
|
* 200: Notes returned
|
|
*/
|
|
#[ApiRoute(verb: 'GET', url: '/api/houses/{houseId}/notes')]
|
|
#[NoAdminRequired]
|
|
public function indexNotes(int $houseId, int $limit = 100, int $offset = 0): DataResponse {
|
|
return $this->runAction(function () use ($houseId, $limit, $offset): DataResponse {
|
|
$this->auth->requireMember($houseId, $this->requireUid());
|
|
$all = $this->notes->listNotes($houseId);
|
|
$sliced = array_slice($all, max(0, $offset), max(0, $limit));
|
|
return new DataResponse(array_map(fn ($n) => $n->jsonSerialize(), $sliced));
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Create a note
|
|
*
|
|
* @param int $houseId House id.
|
|
* @param string $title Note title.
|
|
* @param string|null $content Markdown content.
|
|
* @param string|null $color Hex color (#RRGGBB).
|
|
*
|
|
* @return DataResponse<Http::STATUS_OK, PantryNote, array{}>
|
|
*
|
|
* 200: Note created
|
|
*/
|
|
#[ApiRoute(verb: 'POST', url: '/api/houses/{houseId}/notes')]
|
|
#[NoAdminRequired]
|
|
public function createNote(int $houseId, string $title, ?string $content = null, ?string $color = null): DataResponse {
|
|
return $this->runAction(function () use ($houseId, $title, $content, $color): DataResponse {
|
|
$uid = $this->requireUid();
|
|
$this->auth->requireMember($houseId, $uid);
|
|
$note = $this->notes->createNote($houseId, $uid, $title, $content, $color);
|
|
$this->notifications->notifyNoteCreated($houseId, $uid, (int)$note->getId(), $note->getTitle());
|
|
return new DataResponse($note->jsonSerialize());
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Update a note
|
|
*
|
|
* @param int $houseId House id.
|
|
* @param int $noteId Note id.
|
|
* @param string|null $title New title.
|
|
* @param string|null $content New content (empty string clears).
|
|
* @param string|null $color New color (empty string clears).
|
|
* @param int|null $sortOrder New sort order.
|
|
*
|
|
* @return DataResponse<Http::STATUS_OK, PantryNote, array{}>
|
|
*
|
|
* 200: Note updated
|
|
*/
|
|
#[ApiRoute(verb: 'PATCH', url: '/api/houses/{houseId}/notes/{noteId}')]
|
|
#[NoAdminRequired]
|
|
public function updateNote(int $houseId, int $noteId, ?string $title = null, ?string $content = null, ?string $color = null, ?int $sortOrder = null): DataResponse {
|
|
return $this->runAction(function () use ($houseId, $noteId, $title, $content, $color, $sortOrder): DataResponse {
|
|
$uid = $this->requireUid();
|
|
$this->auth->requireMember($houseId, $uid);
|
|
$existing = $this->notes->getNote($noteId);
|
|
$this->assertInHouse($existing->getHouseId(), $houseId);
|
|
$patch = [];
|
|
if ($title !== null) {
|
|
$patch['title'] = $title;
|
|
}
|
|
if ($content !== null) {
|
|
$patch['content'] = $content;
|
|
}
|
|
if ($color !== null) {
|
|
$patch['color'] = $color;
|
|
}
|
|
if ($sortOrder !== null) {
|
|
$patch['sortOrder'] = $sortOrder;
|
|
}
|
|
$note = $this->notes->updateNote($noteId, $patch);
|
|
// Only notify for content/title changes, not color/sort-order-only changes
|
|
if ($title !== null || $content !== null) {
|
|
$this->notifications->notifyNoteEdited($houseId, $uid, (int)$note->getId(), $note->getTitle());
|
|
}
|
|
return new DataResponse($note->jsonSerialize());
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Delete a note
|
|
*
|
|
* @param int $houseId House id.
|
|
* @param int $noteId Note id.
|
|
*
|
|
* @return DataResponse<Http::STATUS_OK, PantrySuccess, array{}>
|
|
*
|
|
* 200: Note deleted
|
|
*/
|
|
#[ApiRoute(verb: 'DELETE', url: '/api/houses/{houseId}/notes/{noteId}')]
|
|
#[NoAdminRequired]
|
|
public function deleteNote(int $houseId, int $noteId): DataResponse {
|
|
return $this->runAction(function () use ($houseId, $noteId): DataResponse {
|
|
$this->auth->requireMember($houseId, $this->requireUid());
|
|
$existing = $this->notes->getNote($noteId);
|
|
$this->assertInHouse($existing->getHouseId(), $houseId);
|
|
$this->notes->deleteNote($noteId);
|
|
return new DataResponse(['success' => true]);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Batch reorder notes
|
|
*
|
|
* @param int $houseId House id.
|
|
* @param list<array{id: int, sortOrder: int}> $items Reorder entries.
|
|
*
|
|
* @return DataResponse<Http::STATUS_OK, PantrySuccess, array{}>
|
|
*
|
|
* 200: Notes reordered
|
|
*/
|
|
#[ApiRoute(verb: 'POST', url: '/api/houses/{houseId}/notes/reorder')]
|
|
#[NoAdminRequired]
|
|
public function reorderNotes(int $houseId, array $items = []): DataResponse {
|
|
return $this->runAction(function () use ($houseId, $items): DataResponse {
|
|
$this->auth->requireMember($houseId, $this->requireUid());
|
|
$this->notes->reorderNotes($houseId, $items);
|
|
return new DataResponse(['success' => true]);
|
|
});
|
|
}
|
|
|
|
private function requireUid(): string {
|
|
$user = $this->userSession->getUser();
|
|
if ($user === null) {
|
|
throw new ForbiddenException('Not authenticated');
|
|
}
|
|
return $user->getUID();
|
|
}
|
|
|
|
private function assertInHouse(int $entityHouseId, int $routeHouseId): void {
|
|
if ($entityHouseId !== $routeHouseId) {
|
|
throw new NotFoundException('Note does not belong to this house');
|
|
}
|
|
}
|
|
}
|