aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--core/Controller/LanguageModelApiController.php96
-rw-r--r--core/Migrations/Version28000Date20230616104802.php3
-rw-r--r--core/routes.php4
-rw-r--r--lib/private/LanguageModel/Db/Task.php7
-rw-r--r--lib/private/LanguageModel/LanguageModelManager.php11
-rw-r--r--lib/private/LanguageModel/TaskBackgroundJob.php4
-rw-r--r--lib/private/Server.php4
-rw-r--r--lib/public/AppFramework/Bootstrap/IRegistrationContext.php11
-rw-r--r--lib/public/LanguageModel/AbstractLanguageModelTask.php21
-rw-r--r--lib/public/LanguageModel/Events/TaskSuccessfulEvent.php10
-rw-r--r--lib/public/LanguageModel/ILanguageModelManager.php12
11 files changed, 167 insertions, 16 deletions
diff --git a/core/Controller/LanguageModelApiController.php b/core/Controller/LanguageModelApiController.php
new file mode 100644
index 00000000000..5699dd75526
--- /dev/null
+++ b/core/Controller/LanguageModelApiController.php
@@ -0,0 +1,96 @@
+<?php
+
+declare(strict_types=1);
+
+/**
+ * @copyright Copyright (c) 2022 Julius Härtl <jus@bitgrid.net>
+ *
+ * @author Julius Härtl <jus@bitgrid.net>
+ *
+ * @license GNU AGPL version 3 or any later version
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Affero General Public License as
+ * published by the Free Software Foundation, either version 3 of the
+ * License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU Affero General Public License for more details.
+ *
+ * You should have received a copy of the GNU Affero General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+
+namespace OC\Core\Controller;
+
+use InvalidArgumentException;
+use OCP\AppFramework\Http;
+use OCP\AppFramework\Http\DataResponse;
+use OCP\IL10N;
+use OCP\IRequest;
+use OCP\LanguageModel\AbstractLanguageModelTask;
+use OCP\LanguageModel\ILanguageModelManager;
+use OCP\PreConditionNotMetException;
+
+class LanguageModelApiController extends \OCP\AppFramework\OCSController {
+ public function __construct(
+ string $appName,
+ IRequest $request,
+ private ILanguageModelManager $languageModelManager,
+ private IL10N $l,
+ private ?string $userId,
+ ) {
+ parent::__construct($appName, $request);
+ }
+
+ /**
+ * @PublicPage
+ */
+ public function tasks(): DataResponse {
+ return new DataResponse([
+ 'tasks' => $this->languageModelManager->getAvailableTaskTypes(),
+ ]);
+ }
+
+ /**
+ * @PublicPage
+ * @UserRateThrottle(limit=20, period=120)
+ * @AnonRateThrottle(limit=5, period=120)
+ */
+ public function schedule(string $text, string $type, ?string $appId): DataResponse {
+ try {
+ $task = AbstractLanguageModelTask::factory($type, $text, $this->userId, $appId);
+ } catch (InvalidArgumentException $e) {
+ return new DataResponse(['message' => $this->l->t('Requested task type does not exist')], Http::STATUS_BAD_REQUEST);
+ }
+ try {
+ $this->languageModelManager->scheduleTask($task);
+
+ return new DataResponse([
+ 'task' => $task,
+ ]);
+ } catch (PreConditionNotMetException) {
+ return new DataResponse(['message' => $this->l->t('Necessary language model provider is not available')], Http::STATUS_PRECONDITION_FAILED);
+ }
+ }
+
+ /**
+ * @PublicPage
+ */
+ public function getTask(int $id): DataResponse {
+ try {
+ $task = $this->languageModelManager->getTask($id);
+
+ return new DataResponse([
+ 'task' => $task,
+ ]);
+ } catch (\ValueError $e) {
+ return new DataResponse(['message' => $this->l->t('Task not found')], Http::STATUS_NOT_FOUND);
+ } catch (\RuntimeException $e) {
+ return new DataResponse(['message' => $this->l->t('Internal error')], Http::STATUS_INTERNAL_SERVER_ERROR);
+ }
+ }
+}
diff --git a/core/Migrations/Version28000Date20230616104802.php b/core/Migrations/Version28000Date20230616104802.php
index ffb2b38f57e..76d8173861f 100644
--- a/core/Migrations/Version28000Date20230616104802.php
+++ b/core/Migrations/Version28000Date20230616104802.php
@@ -61,6 +61,9 @@ class Version28000Date20230616104802 extends SimpleMigrationStep {
$table->addColumn('input', Types::TEXT, [
'notnull' => true,
]);
+ $table->addColumn('output', Types::TEXT, [
+ 'notnull' => false,
+ ]);
$table->addColumn('status', Types::INTEGER, [
'notnull' => false,
'length' => 6,
diff --git a/core/routes.php b/core/routes.php
index 0f9729e54eb..20be6ef63f4 100644
--- a/core/routes.php
+++ b/core/routes.php
@@ -145,6 +145,10 @@ $application->registerRoutes($this, [
['root' => '/translation', 'name' => 'TranslationApi#languages', 'url' => '/languages', 'verb' => 'GET'],
['root' => '/translation', 'name' => 'TranslationApi#translate', 'url' => '/translate', 'verb' => 'POST'],
+
+ ['root' => '/languagemodel', 'name' => 'LanguageModelApi#tasks', 'url' => '/tasks', 'verb' => 'GET'],
+ ['root' => '/languagemodel', 'name' => 'LanguageModelApi#schedule', 'url' => '/schedule', 'verb' => 'POST'],
+ ['root' => '/languagemodel', 'name' => 'LanguageModelApi#getTask', 'url' => '/task/{id}', 'verb' => 'GET'],
],
]);
diff --git a/lib/private/LanguageModel/Db/Task.php b/lib/private/LanguageModel/Db/Task.php
index cee6c2fd8b9..e5d2f79e453 100644
--- a/lib/private/LanguageModel/Db/Task.php
+++ b/lib/private/LanguageModel/Db/Task.php
@@ -28,12 +28,12 @@ class Task extends Entity {
/**
* @var string[]
*/
- public static array $columns = ['id', 'type', 'input', 'status', 'user_id', 'app_id'];
+ public static array $columns = ['id', 'type', 'input', 'output', 'status', 'user_id', 'app_id'];
/**
* @var string[]
*/
- public static array $fields = ['id', 'type', 'input', 'status', 'userId', 'appId'];
+ public static array $fields = ['id', 'type', 'input', 'output', 'status', 'userId', 'appId'];
public function __construct() {
@@ -49,8 +49,9 @@ class Task extends Entity {
public static function fromLanguageModelTask(ILanguageModelTask $task): Task {
return Task::fromParams([
'type' => $task->getType(),
- 'status' => ILanguageModelTask::STATUS_UNKNOWN,
+ 'status' => $task->getStatus(),
'input' => $task->getInput(),
+ 'output' => $task->getOutput(),
'userId' => $task->getUserId(),
'appId' => $task->getAppId(),
]);
diff --git a/lib/private/LanguageModel/LanguageModelManager.php b/lib/private/LanguageModel/LanguageModelManager.php
index b0e45f5812a..7db2e656a0a 100644
--- a/lib/private/LanguageModel/LanguageModelManager.php
+++ b/lib/private/LanguageModel/LanguageModelManager.php
@@ -86,6 +86,13 @@ class LanguageModelManager implements ILanguageModelManager {
return array_keys($tasks);
}
+ /**
+ * @inheritDoc
+ */
+ public function getAvailableTaskTypes(): array {
+ return array_map(fn ($taskClass) => $taskClass::TYPE, $this->getAvailableTasks());
+ }
+
public function canHandleTask(ILanguageModelTask $task): bool {
return !empty(array_filter($this->getAvailableTasks(), fn ($class) => $task instanceof $class));
}
@@ -104,10 +111,10 @@ class LanguageModelManager implements ILanguageModelManager {
try {
$task->setStatus(ILanguageModelTask::STATUS_RUNNING);
$this->taskMapper->update(Task::fromLanguageModelTask($task));
- $output = $task->visitProvider($provider);
+ $task->setOutput($task->visitProvider($provider));
$task->setStatus(ILanguageModelTask::STATUS_SUCCESSFUL);
$this->taskMapper->update(Task::fromLanguageModelTask($task));
- return $output;
+ return $task->getOutput();
} catch (\RuntimeException $e) {
$this->logger->info('LanguageModel call using provider ' . $provider->getName() . ' failed', ['exception' => $e]);
$task->setStatus(ILanguageModelTask::STATUS_FAILED);
diff --git a/lib/private/LanguageModel/TaskBackgroundJob.php b/lib/private/LanguageModel/TaskBackgroundJob.php
index 55413ba3714..3c18ff03102 100644
--- a/lib/private/LanguageModel/TaskBackgroundJob.php
+++ b/lib/private/LanguageModel/TaskBackgroundJob.php
@@ -61,8 +61,8 @@ class TaskBackgroundJob extends QueuedJob {
$taskId = $argument['taskId'];
$task = $this->languageModelManager->getTask($taskId);
try {
- $output = $this->languageModelManager->runTask($task);
- $event = new TaskSuccessfulEvent($task, $output);
+ $this->languageModelManager->runTask($task);
+ $event = new TaskSuccessfulEvent($task);
} catch (\RuntimeException|PreConditionNotMetException $e) {
$event = new TaskFailedEvent($task, $e->getMessage());
diff --git a/lib/private/Server.php b/lib/private/Server.php
index bb4e217efa3..d1f18a1235f 100644
--- a/lib/private/Server.php
+++ b/lib/private/Server.php
@@ -110,6 +110,7 @@ use OC\IntegrityCheck\Checker;
use OC\IntegrityCheck\Helpers\AppLocator;
use OC\IntegrityCheck\Helpers\EnvironmentHelper;
use OC\IntegrityCheck\Helpers\FileAccessHelper;
+use OC\LanguageModel\LanguageModelManager;
use OC\LDAP\NullLDAPProviderFactory;
use OC\KnownUser\KnownUserService;
use OC\Lock\DBLockingProvider;
@@ -228,6 +229,7 @@ use OCP\IURLGenerator;
use OCP\IUserManager;
use OCP\IUserSession;
use OCP\L10N\IFactory;
+use OCP\LanguageModel\ILanguageModelManager;
use OCP\LDAP\ILDAPProvider;
use OCP\LDAP\ILDAPProviderFactory;
use OCP\Lock\ILockingProvider;
@@ -1461,6 +1463,8 @@ class Server extends ServerContainer implements IServerContainer {
$this->registerAlias(ISpeechToTextManager::class, SpeechToTextManager::class);
+ $this->registerAlias(ILanguageModelManager::class, LanguageModelManager::class);
+
$this->connectDispatcher();
}
diff --git a/lib/public/AppFramework/Bootstrap/IRegistrationContext.php b/lib/public/AppFramework/Bootstrap/IRegistrationContext.php
index 66cf1ef2306..19ef6832a2c 100644
--- a/lib/public/AppFramework/Bootstrap/IRegistrationContext.php
+++ b/lib/public/AppFramework/Bootstrap/IRegistrationContext.php
@@ -37,6 +37,7 @@ use OCP\Collaboration\Reference\IReferenceProvider;
use OCP\EventDispatcher\IEventDispatcher;
use OCP\Files\Template\ICustomTemplateProvider;
use OCP\IContainer;
+use OCP\LanguageModel\ILanguageModelProvider;
use OCP\Notification\INotifier;
use OCP\Preview\IProviderV2;
use OCP\SpeechToText\ISpeechToTextProvider;
@@ -220,6 +221,16 @@ interface IRegistrationContext {
public function registerSpeechToTextProvider(string $providerClass): void;
/**
+ * Register a custom LanguageModel provider class that provides a promptable language model
+ * through the OCP\LanguageModel APIs
+ *
+ * @param string $providerClass
+ * @psalm-param class-string<ILanguageModelProvider> $providerClass
+ * @since 27.0.0
+ */
+ public function registerLanguageModelProvider(string $providerClass): void;
+
+ /**
* Register a custom template provider class that is able to inject custom templates
* in addition to the user defined ones
*
diff --git a/lib/public/LanguageModel/AbstractLanguageModelTask.php b/lib/public/LanguageModel/AbstractLanguageModelTask.php
index 12aedc95fe5..63b6396fb43 100644
--- a/lib/public/LanguageModel/AbstractLanguageModelTask.php
+++ b/lib/public/LanguageModel/AbstractLanguageModelTask.php
@@ -76,6 +76,19 @@ abstract class AbstractLanguageModelTask implements ILanguageModelTask {
return $this->userId;
}
+ public function jsonSerialize() {
+ return [
+ 'id' => $this->getId(),
+ 'type' => $this->getType(),
+ 'status' => $this->getStatus(),
+ 'userId' => $this->getUserId(),
+ 'appId' => $this->getAppId(),
+ 'input' => $this->getInput(),
+ 'output' => $this->getOutput(),
+ ];
+ }
+
+
public final static function fromTaskEntity(Task $taskEntity): ILanguageModelTask {
$task = self::factory($taskEntity->getType(), $taskEntity->getInput(), $taskEntity->getuserId(), $taskEntity->getAppId());
$task->setId($taskEntity->getId());
@@ -83,6 +96,14 @@ abstract class AbstractLanguageModelTask implements ILanguageModelTask {
return $task;
}
+ /**
+ * @param string $type
+ * @param string $input
+ * @param string|null $userId
+ * @param string $appId
+ * @return ILanguageModelTask
+ * @throws \InvalidArgumentException
+ */
public final static function factory(string $type, string $input, ?string $userId, string $appId): ILanguageModelTask {
if (!in_array($type, self::TYPES)) {
throw new \InvalidArgumentException('Unknown task type');
diff --git a/lib/public/LanguageModel/Events/TaskSuccessfulEvent.php b/lib/public/LanguageModel/Events/TaskSuccessfulEvent.php
index 156c5679e0b..61be3a20cd1 100644
--- a/lib/public/LanguageModel/Events/TaskSuccessfulEvent.php
+++ b/lib/public/LanguageModel/Events/TaskSuccessfulEvent.php
@@ -9,15 +9,7 @@ use OCP\LanguageModel\ILanguageModelTask;
*/
class TaskSuccessfulEvent extends AbstractLanguageModelEvent {
- public function __construct(ILanguageModelTask $task,
- private string $output) {
+ public function __construct(ILanguageModelTask $task) {
parent::__construct($task);
}
-
- /**
- * @return string
- */
- public function getErrorMessage(): string {
- return $this->output;
- }
}
diff --git a/lib/public/LanguageModel/ILanguageModelManager.php b/lib/public/LanguageModel/ILanguageModelManager.php
index a4d3079c180..439cfb76176 100644
--- a/lib/public/LanguageModel/ILanguageModelManager.php
+++ b/lib/public/LanguageModel/ILanguageModelManager.php
@@ -45,6 +45,12 @@ interface ILanguageModelManager {
public function getAvailableTasks(): array;
/**
+ * @return string[]
+ * @since 28.0.0
+ */
+ public function getAvailableTaskTypes(): array;
+
+ /**
* @throws PreConditionNotMetException If no or not the requested provider was registered but this method was still called
* @throws RuntimeException If something else failed
* @since 28.0.0
@@ -60,5 +66,11 @@ interface ILanguageModelManager {
*/
public function scheduleTask(ILanguageModelTask $task) : void;
+ /**
+ * @param int $id The id of the task
+ * @return ILanguageModelTask
+ * @throws RuntimeException If the query failed
+ * @throws \ValueError If the task could not be found
+ */
public function getTask(int $id): ILanguageModelTask;
}