diff --git a/lib/Service/ApiService.php b/lib/Service/ApiService.php index 288171640e0..436ddc73eee 100644 --- a/lib/Service/ApiService.php +++ b/lib/Service/ApiService.php @@ -184,11 +184,8 @@ public function close(int $documentId, int $sessionId, string $sessionToken): Da /** * @throws NotFoundException - * @throws DoesNotExistException - * - * @param null|string $token */ - public function push(Session $session, Document $document, int $version, array $steps, string $awareness, string|null $token = null): DataResponse { + public function push(Session $session, Document $document, int $version, array $steps, string $awareness, ?string $token = null): DataResponse { try { $session = $this->sessionService->updateSessionAwareness($session, $awareness); } catch (DoesNotExistException $e) { @@ -198,16 +195,12 @@ public function push(Session $session, Document $document, int $version, array $ if (empty($steps)) { return new DataResponse([]); } - $file = $this->documentService->getFileForSession($session, $token); - if ($this->documentService->isReadOnly($file, $token)) { - return new DataResponse([], 403); - } try { - $result = $this->documentService->addStep($document, $session, $steps, $version); + $result = $this->documentService->addStep($document, $session, $steps, $version, $token); } catch (InvalidArgumentException $e) { return new DataResponse($e->getMessage(), 422); - } catch (DoesNotExistException $e) { - // Session was removed in the meantime. #3875 + } catch (DoesNotExistException|NotPermittedException) { + // Either no write access or session was removed in the meantime (#3875). return new DataResponse([], 403); } return new DataResponse($result); diff --git a/lib/Service/DocumentService.php b/lib/Service/DocumentService.php index d6c8edbe98b..42bf6df159c 100644 --- a/lib/Service/DocumentService.php +++ b/lib/Service/DocumentService.php @@ -206,10 +206,12 @@ public function writeDocumentState(int $documentId, string $content): void { } /** - * @throws DoesNotExistException * @throws InvalidArgumentException + * @throws NotFoundException + * @throws NotPermittedException + * @throws DoesNotExistException */ - public function addStep(Document $document, Session $session, array $steps, int $version): array { + public function addStep(Document $document, Session $session, array $steps, int $version, ?string $token): array { $documentId = $session->getDocumentId(); $stepsToInsert = []; $querySteps = []; @@ -224,6 +226,9 @@ public function addStep(Document $document, Session $session, array $steps, int } } if (count($stepsToInsert) > 0) { + if ($this->isReadOnly($this->getFileForSession($session, $token), $token)) { + throw new NotPermittedException('Read-only client tries to push steps with changes'); + } $newVersion = $this->insertSteps($document, $session, $stepsToInsert); } // If there were any queries in the steps send the entire history