summaryrefslogtreecommitdiffstats
path: root/lib/private/Translation
diff options
context:
space:
mode:
authorJulius Härtl <jus@bitgrid.net>2023-02-07 14:13:04 +0100
committerJulius Härtl <jus@bitgrid.net>2023-02-27 16:52:03 +0100
commit3e6329838158892e3b89c7e5116fa313c282a8a1 (patch)
tree0e55e55fc3053b1263cf3a39fdfa25665fe3388e /lib/private/Translation
parent0d67fc23f4077e7b06a01bc519957f3f13d95f10 (diff)
downloadnextcloud-server-3e6329838158892e3b89c7e5116fa313c282a8a1.tar.gz
nextcloud-server-3e6329838158892e3b89c7e5116fa313c282a8a1.zip
feat(translations): Add translation provider API
Signed-off-by: Julius Härtl <jus@bitgrid.net>
Diffstat (limited to 'lib/private/Translation')
-rw-r--r--lib/private/Translation/TranslationManager.php120
1 files changed, 120 insertions, 0 deletions
diff --git a/lib/private/Translation/TranslationManager.php b/lib/private/Translation/TranslationManager.php
new file mode 100644
index 00000000000..ec829e83255
--- /dev/null
+++ b/lib/private/Translation/TranslationManager.php
@@ -0,0 +1,120 @@
+<?php
+
+declare(strict_types=1);
+
+/**
+ * @copyright Copyright (c) 2023 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\Translation;
+
+use InvalidArgumentException;
+use OC\AppFramework\Bootstrap\Coordinator;
+use OCP\IServerContainer;
+use OCP\PreConditionNotMetException;
+use OCP\Translation\IDetectLanguageProvider;
+use OCP\Translation\ITranslationManager;
+use OCP\Translation\ITranslationProvider;
+use Psr\Container\ContainerExceptionInterface;
+use Psr\Container\NotFoundExceptionInterface;
+use Psr\Log\LoggerInterface;
+use RuntimeException;
+use Throwable;
+
+class TranslationManager implements ITranslationManager {
+ /** @var ?ITranslationProvider[] */
+ private ?array $providers = null;
+
+ public function __construct(
+ private IServerContainer $serverContainer,
+ private Coordinator $coordinator,
+ private LoggerInterface $logger,
+ ) {
+ }
+
+ public function getLanguages(): array {
+ $languages = [];
+ foreach ($this->getProviders() as $provider) {
+ $languages = array_merge($languages, $provider->getAvailableLanguages());
+ }
+ return $languages;
+ }
+
+ public function translate(string $text, ?string $fromLanguage, string $toLanguage): string {
+ if (!$this->hasProviders()) {
+ throw new PreConditionNotMetException('No translation providers available');
+ }
+
+ foreach ($this->getProviders() as $provider) {
+ if ($fromLanguage === null && $provider instanceof IDetectLanguageProvider) {
+ $fromLanguage = $provider->detectLanguage($text);
+ }
+
+ if ($fromLanguage === null) {
+ throw new InvalidArgumentException('Could not detect language');
+ }
+
+ try {
+ return $provider->translate($fromLanguage, $toLanguage, $text);
+ } catch (RuntimeException $e) {
+ $this->logger->warning("Failed to translate from {$fromLanguage} to {$toLanguage}", ['exception' => $e]);
+ }
+ }
+
+ throw new RuntimeException('Could not translate text');
+ }
+
+ public function getProviders(): array {
+ $context = $this->coordinator->getRegistrationContext();
+
+ if ($this->providers !== null) {
+ return $this->providers;
+ }
+
+ $this->providers = [];
+ foreach ($context->getTranslationProviders() as $providerRegistration) {
+ $class = $providerRegistration->getService();
+ try {
+ $this->providers[$class] = $this->serverContainer->get($class);
+ } catch (NotFoundExceptionInterface|ContainerExceptionInterface|Throwable $e) {
+ $this->logger->error('Failed to load translation provider ' . $class, [
+ 'exception' => $e
+ ]);
+ }
+ }
+
+ return $this->providers;
+ }
+
+ public function hasProviders(): bool {
+ $context = $this->coordinator->getRegistrationContext();
+ return !empty($context->getTranslationProviders());
+ }
+
+ public function canDetectLanguage(): bool {
+ foreach ($this->getProviders() as $provider) {
+ if ($provider instanceof IDetectLanguageProvider) {
+ return true;
+ }
+ }
+ return false;
+ }
+}