aboutsummaryrefslogtreecommitdiffstats
path: root/apps
diff options
context:
space:
mode:
Diffstat (limited to 'apps')
-rw-r--r--apps/dav/l10n/ar.js2
-rw-r--r--apps/dav/l10n/ar.json2
-rw-r--r--apps/encryption/composer/composer/autoload_classmap.php5
-rw-r--r--apps/encryption/composer/composer/autoload_static.php5
-rw-r--r--apps/encryption/l10n/da.js6
-rw-r--r--apps/encryption/l10n/da.json6
-rw-r--r--apps/encryption/lib/AppInfo/Application.php72
-rw-r--r--apps/encryption/lib/Crypto/Crypt.php2
-rw-r--r--apps/encryption/lib/HookManager.php43
-rw-r--r--apps/encryption/lib/Hooks/Contracts/IHook.php17
-rw-r--r--apps/encryption/lib/Hooks/UserHooks.php266
-rw-r--r--apps/encryption/lib/KeyManager.php8
-rw-r--r--apps/encryption/lib/Listeners/UserEventsListener.php143
-rw-r--r--apps/encryption/lib/Services/PassphraseService.php142
-rw-r--r--apps/encryption/lib/Users/Setup.php12
-rw-r--r--apps/encryption/lib/Util.php5
-rw-r--r--apps/encryption/tests/HookManagerTest.php52
-rw-r--r--apps/encryption/tests/Hooks/UserHooksTest.php370
-rw-r--r--apps/encryption/tests/Listeners/UserEventsListenersTest.php258
-rw-r--r--apps/encryption/tests/PassphraseServiceTest.php196
-rw-r--r--apps/files/l10n/ar.js6
-rw-r--r--apps/files/l10n/ar.json6
-rw-r--r--apps/files_sharing/l10n/ar.js6
-rw-r--r--apps/files_sharing/l10n/ar.json6
-rw-r--r--apps/files_sharing/l10n/cs.js2
-rw-r--r--apps/files_sharing/l10n/cs.json2
-rw-r--r--apps/files_sharing/l10n/da.js1
-rw-r--r--apps/files_sharing/l10n/da.json1
-rw-r--r--apps/files_sharing/l10n/de_DE.js2
-rw-r--r--apps/files_sharing/l10n/de_DE.json2
-rw-r--r--apps/files_sharing/l10n/en_GB.js4
-rw-r--r--apps/files_sharing/l10n/en_GB.json4
-rw-r--r--apps/files_sharing/l10n/gl.js2
-rw-r--r--apps/files_sharing/l10n/gl.json2
-rw-r--r--apps/files_sharing/l10n/lt_LT.js15
-rw-r--r--apps/files_sharing/l10n/lt_LT.json15
-rw-r--r--apps/files_sharing/l10n/sr.js2
-rw-r--r--apps/files_sharing/l10n/sr.json2
-rw-r--r--apps/files_sharing/l10n/sv.js2
-rw-r--r--apps/files_sharing/l10n/sv.json2
-rw-r--r--apps/files_sharing/l10n/zh_HK.js4
-rw-r--r--apps/files_sharing/l10n/zh_HK.json4
-rw-r--r--apps/files_sharing/l10n/zh_TW.js2
-rw-r--r--apps/files_sharing/l10n/zh_TW.json2
-rw-r--r--apps/files_sharing/src/mixins/SharesMixin.js65
-rw-r--r--apps/files_versions/l10n/ar.js2
-rw-r--r--apps/files_versions/l10n/ar.json2
-rw-r--r--apps/settings/l10n/ar.js24
-rw-r--r--apps/settings/l10n/ar.json24
-rw-r--r--apps/settings/l10n/cs.js28
-rw-r--r--apps/settings/l10n/cs.json28
-rw-r--r--apps/settings/l10n/lt_LT.js1
-rw-r--r--apps/settings/l10n/lt_LT.json1
-rw-r--r--apps/systemtags/l10n/ar.js6
-rw-r--r--apps/systemtags/l10n/ar.json6
-rw-r--r--apps/user_ldap/l10n/ar.js8
-rw-r--r--apps/user_ldap/l10n/ar.json8
-rw-r--r--apps/user_ldap/l10n/gl.js4
-rw-r--r--apps/user_ldap/l10n/gl.json4
-rw-r--r--apps/weather_status/l10n/nl.js4
-rw-r--r--apps/weather_status/l10n/nl.json4
61 files changed, 1042 insertions, 885 deletions
diff --git a/apps/dav/l10n/ar.js b/apps/dav/l10n/ar.js
index c88147cdbee..ea0a3dd0a81 100644
--- a/apps/dav/l10n/ar.js
+++ b/apps/dav/l10n/ar.js
@@ -264,7 +264,7 @@ OC.L10N.register(
"Availability" : "أوقات التواجد ",
"If you configure your working hours, other people will see when you are out of office when they book a meeting." : "إذا قمت بضبط ساعات عملك، سيرى الآخرون متى تكون خارج المكتب عندما يقومون بحجز اجتماع معك.",
"Absence" : "غياب",
- "Configure your next absence period." : "قٌم بتهيئة فترة غيابك القادمة.",
+ "Configure your next absence period." : "تهيئة فترة غيابك القادمة.",
"Calendar server" : "خادم التقويم",
"Send invitations to attendees" : "إرسال دعوات للمطلوب حضورهم",
"Automatically generate a birthday calendar" : "تجاهل تقويم أعياد الميلاد تلقائيّاً",
diff --git a/apps/dav/l10n/ar.json b/apps/dav/l10n/ar.json
index 348138a6e41..d217d3d5c56 100644
--- a/apps/dav/l10n/ar.json
+++ b/apps/dav/l10n/ar.json
@@ -262,7 +262,7 @@
"Availability" : "أوقات التواجد ",
"If you configure your working hours, other people will see when you are out of office when they book a meeting." : "إذا قمت بضبط ساعات عملك، سيرى الآخرون متى تكون خارج المكتب عندما يقومون بحجز اجتماع معك.",
"Absence" : "غياب",
- "Configure your next absence period." : "قٌم بتهيئة فترة غيابك القادمة.",
+ "Configure your next absence period." : "تهيئة فترة غيابك القادمة.",
"Calendar server" : "خادم التقويم",
"Send invitations to attendees" : "إرسال دعوات للمطلوب حضورهم",
"Automatically generate a birthday calendar" : "تجاهل تقويم أعياد الميلاد تلقائيّاً",
diff --git a/apps/encryption/composer/composer/autoload_classmap.php b/apps/encryption/composer/composer/autoload_classmap.php
index 059296338b4..814f39653e9 100644
--- a/apps/encryption/composer/composer/autoload_classmap.php
+++ b/apps/encryption/composer/composer/autoload_classmap.php
@@ -26,12 +26,11 @@ return array(
'OCA\\Encryption\\Exceptions\\MultiKeyEncryptException' => $baseDir . '/../lib/Exceptions/MultiKeyEncryptException.php',
'OCA\\Encryption\\Exceptions\\PrivateKeyMissingException' => $baseDir . '/../lib/Exceptions/PrivateKeyMissingException.php',
'OCA\\Encryption\\Exceptions\\PublicKeyMissingException' => $baseDir . '/../lib/Exceptions/PublicKeyMissingException.php',
- 'OCA\\Encryption\\HookManager' => $baseDir . '/../lib/HookManager.php',
- 'OCA\\Encryption\\Hooks\\Contracts\\IHook' => $baseDir . '/../lib/Hooks/Contracts/IHook.php',
- 'OCA\\Encryption\\Hooks\\UserHooks' => $baseDir . '/../lib/Hooks/UserHooks.php',
'OCA\\Encryption\\KeyManager' => $baseDir . '/../lib/KeyManager.php',
+ 'OCA\\Encryption\\Listeners\\UserEventsListener' => $baseDir . '/../lib/Listeners/UserEventsListener.php',
'OCA\\Encryption\\Migration\\SetMasterKeyStatus' => $baseDir . '/../lib/Migration/SetMasterKeyStatus.php',
'OCA\\Encryption\\Recovery' => $baseDir . '/../lib/Recovery.php',
+ 'OCA\\Encryption\\Services\\PassphraseService' => $baseDir . '/../lib/Services/PassphraseService.php',
'OCA\\Encryption\\Session' => $baseDir . '/../lib/Session.php',
'OCA\\Encryption\\Settings\\Admin' => $baseDir . '/../lib/Settings/Admin.php',
'OCA\\Encryption\\Settings\\Personal' => $baseDir . '/../lib/Settings/Personal.php',
diff --git a/apps/encryption/composer/composer/autoload_static.php b/apps/encryption/composer/composer/autoload_static.php
index 6c458eabddd..af5e5192520 100644
--- a/apps/encryption/composer/composer/autoload_static.php
+++ b/apps/encryption/composer/composer/autoload_static.php
@@ -41,12 +41,11 @@ class ComposerStaticInitEncryption
'OCA\\Encryption\\Exceptions\\MultiKeyEncryptException' => __DIR__ . '/..' . '/../lib/Exceptions/MultiKeyEncryptException.php',
'OCA\\Encryption\\Exceptions\\PrivateKeyMissingException' => __DIR__ . '/..' . '/../lib/Exceptions/PrivateKeyMissingException.php',
'OCA\\Encryption\\Exceptions\\PublicKeyMissingException' => __DIR__ . '/..' . '/../lib/Exceptions/PublicKeyMissingException.php',
- 'OCA\\Encryption\\HookManager' => __DIR__ . '/..' . '/../lib/HookManager.php',
- 'OCA\\Encryption\\Hooks\\Contracts\\IHook' => __DIR__ . '/..' . '/../lib/Hooks/Contracts/IHook.php',
- 'OCA\\Encryption\\Hooks\\UserHooks' => __DIR__ . '/..' . '/../lib/Hooks/UserHooks.php',
'OCA\\Encryption\\KeyManager' => __DIR__ . '/..' . '/../lib/KeyManager.php',
+ 'OCA\\Encryption\\Listeners\\UserEventsListener' => __DIR__ . '/..' . '/../lib/Listeners/UserEventsListener.php',
'OCA\\Encryption\\Migration\\SetMasterKeyStatus' => __DIR__ . '/..' . '/../lib/Migration/SetMasterKeyStatus.php',
'OCA\\Encryption\\Recovery' => __DIR__ . '/..' . '/../lib/Recovery.php',
+ 'OCA\\Encryption\\Services\\PassphraseService' => __DIR__ . '/..' . '/../lib/Services/PassphraseService.php',
'OCA\\Encryption\\Session' => __DIR__ . '/..' . '/../lib/Session.php',
'OCA\\Encryption\\Settings\\Admin' => __DIR__ . '/..' . '/../lib/Settings/Admin.php',
'OCA\\Encryption\\Settings\\Personal' => __DIR__ . '/..' . '/../lib/Settings/Personal.php',
diff --git a/apps/encryption/l10n/da.js b/apps/encryption/l10n/da.js
index a9c1c8188c2..e877a150ea0 100644
--- a/apps/encryption/l10n/da.js
+++ b/apps/encryption/l10n/da.js
@@ -28,15 +28,21 @@ OC.L10N.register(
"Bad Signature" : "Ugyldig signatur",
"Missing Signature" : "Signatur mangler",
"one-time password for server-side-encryption" : "Engangs password for kryptering på serverdelen",
+ "Encryption password" : "Krypterings password",
+ "The administration enabled server-side-encryption. Your files were encrypted using the password <strong>%s</strong>." : "Administrationen aktiverede server-side-kryptering. Dine filer blev krypteret med adgangskoden 1%s1.",
+ "The administration enabled server-side-encryption. Your files were encrypted using the password \"%s\"." : "Administrationen aktiverede server-side-kryptering. Dine filer blev krypteret med adgangskoden \"%s\".",
+ "Please login to the web interface, go to the \"Security\" section of your personal settings and update your encryption password by entering this password into the \"Old login password\" field and your current login password." : "Log venligst ind på webgrænsefladen, gå til afsnittet \"Sikkerhed\" i dine personlige indstillinger og opdater din krypteringsadgangskode ved at indtaste denne adgangskode i feltet \"Gammel loginadgangskode\" og din nuværende loginadgangskode.",
"Cannot decrypt this file, probably this is a shared file. Please ask the file owner to reshare the file with you." : "Kan ikke dekryptere denne fil, sandsynligvis er dette en delt fil. Bed filejeren om at videredele filen med dig.",
"Cannot read this file, probably this is a shared file. Please ask the file owner to reshare the file with you." : "Kan ikke læse denne fil, sandsynligvis er dette en delt fil. Bed filejeren om at videredele filen med dig.",
"Default encryption module" : "Standard krypterings modul",
"Default encryption module for server-side encryption" : "Standard krypteringsmodul til kryptering på serveren",
+ "In order to use this encryption module you need to enable server-side encryption in the admin settings. Once enabled this module will encrypt all your files transparently. The encryption is based on AES 256 keys.\nThe module will not touch existing files, only new files will be encrypted after server-side encryption was enabled. It is also not possible to disable the encryption again and switch back to an unencrypted system.\nPlease read the documentation to know all implications before you decide to enable server-side encryption." : "For at bruge dette krypteringsmodul skal du aktivere serversidekryptering i admin-indstillingerne. Når det er aktiveret, vil dette modul kryptere alle dine filer gennemsigtigt. Krypteringen er baseret på AES 256 nøgler.\nModulet vil ikke røre ved eksisterende filer, kun nye filer vil blive krypteret efter server-side kryptering blev aktiveret. Det er heller ikke muligt at deaktivere krypteringen igen og skifte tilbage til et ukrypteret system.\nLæs venligst dokumentationen for at kende alle implikationer, før du beslutter dig for at aktivere server-side-kryptering.",
"Encryption app is enabled but your keys are not initialized, please log-out and log-in again" : "Krypteringsprogrammet er aktiveret men dine nøgler er ikke indlæst, log venligst ud og ind igen",
"Encrypt the home storage" : "Krypter hjemmelageret",
"Enabling this option encrypts all files stored on the main storage, otherwise only files on external storage will be encrypted" : "Ved at slå denne valgmulighed til krypteres alle filer i hovedlageret, ellers vil kun filer på eksternt lager blive krypteret",
"Enable recovery key" : "Aktivér gendannelsesnøgle",
"Disable recovery key" : "Deaktivér gendannelsesnøgle",
+ "The recovery key is an additional encryption key used to encrypt files. It is used to recover files from an account if the password is forgotten." : "Gendannelsesnøglen er en ekstra krypteringsnøgle, der bruges til at kryptere filer. Det bruges til at gendanne filer fra en konto, hvis adgangskoden er glemt.",
"Recovery key password" : "Gendannelsesnøgle kodeord",
"Repeat recovery key password" : "Gentag adgangskode for gendannelsesnøgle",
"Change recovery key password:" : "Skift gendannelsesnøgle kodeord:",
diff --git a/apps/encryption/l10n/da.json b/apps/encryption/l10n/da.json
index f4dc54d66ed..eac38b01e82 100644
--- a/apps/encryption/l10n/da.json
+++ b/apps/encryption/l10n/da.json
@@ -26,15 +26,21 @@
"Bad Signature" : "Ugyldig signatur",
"Missing Signature" : "Signatur mangler",
"one-time password for server-side-encryption" : "Engangs password for kryptering på serverdelen",
+ "Encryption password" : "Krypterings password",
+ "The administration enabled server-side-encryption. Your files were encrypted using the password <strong>%s</strong>." : "Administrationen aktiverede server-side-kryptering. Dine filer blev krypteret med adgangskoden 1%s1.",
+ "The administration enabled server-side-encryption. Your files were encrypted using the password \"%s\"." : "Administrationen aktiverede server-side-kryptering. Dine filer blev krypteret med adgangskoden \"%s\".",
+ "Please login to the web interface, go to the \"Security\" section of your personal settings and update your encryption password by entering this password into the \"Old login password\" field and your current login password." : "Log venligst ind på webgrænsefladen, gå til afsnittet \"Sikkerhed\" i dine personlige indstillinger og opdater din krypteringsadgangskode ved at indtaste denne adgangskode i feltet \"Gammel loginadgangskode\" og din nuværende loginadgangskode.",
"Cannot decrypt this file, probably this is a shared file. Please ask the file owner to reshare the file with you." : "Kan ikke dekryptere denne fil, sandsynligvis er dette en delt fil. Bed filejeren om at videredele filen med dig.",
"Cannot read this file, probably this is a shared file. Please ask the file owner to reshare the file with you." : "Kan ikke læse denne fil, sandsynligvis er dette en delt fil. Bed filejeren om at videredele filen med dig.",
"Default encryption module" : "Standard krypterings modul",
"Default encryption module for server-side encryption" : "Standard krypteringsmodul til kryptering på serveren",
+ "In order to use this encryption module you need to enable server-side encryption in the admin settings. Once enabled this module will encrypt all your files transparently. The encryption is based on AES 256 keys.\nThe module will not touch existing files, only new files will be encrypted after server-side encryption was enabled. It is also not possible to disable the encryption again and switch back to an unencrypted system.\nPlease read the documentation to know all implications before you decide to enable server-side encryption." : "For at bruge dette krypteringsmodul skal du aktivere serversidekryptering i admin-indstillingerne. Når det er aktiveret, vil dette modul kryptere alle dine filer gennemsigtigt. Krypteringen er baseret på AES 256 nøgler.\nModulet vil ikke røre ved eksisterende filer, kun nye filer vil blive krypteret efter server-side kryptering blev aktiveret. Det er heller ikke muligt at deaktivere krypteringen igen og skifte tilbage til et ukrypteret system.\nLæs venligst dokumentationen for at kende alle implikationer, før du beslutter dig for at aktivere server-side-kryptering.",
"Encryption app is enabled but your keys are not initialized, please log-out and log-in again" : "Krypteringsprogrammet er aktiveret men dine nøgler er ikke indlæst, log venligst ud og ind igen",
"Encrypt the home storage" : "Krypter hjemmelageret",
"Enabling this option encrypts all files stored on the main storage, otherwise only files on external storage will be encrypted" : "Ved at slå denne valgmulighed til krypteres alle filer i hovedlageret, ellers vil kun filer på eksternt lager blive krypteret",
"Enable recovery key" : "Aktivér gendannelsesnøgle",
"Disable recovery key" : "Deaktivér gendannelsesnøgle",
+ "The recovery key is an additional encryption key used to encrypt files. It is used to recover files from an account if the password is forgotten." : "Gendannelsesnøglen er en ekstra krypteringsnøgle, der bruges til at kryptere filer. Det bruges til at gendanne filer fra en konto, hvis adgangskoden er glemt.",
"Recovery key password" : "Gendannelsesnøgle kodeord",
"Repeat recovery key password" : "Gentag adgangskode for gendannelsesnøgle",
"Change recovery key password:" : "Skift gendannelsesnøgle kodeord:",
diff --git a/apps/encryption/lib/AppInfo/Application.php b/apps/encryption/lib/AppInfo/Application.php
index d683c82286a..600bc300797 100644
--- a/apps/encryption/lib/AppInfo/Application.php
+++ b/apps/encryption/lib/AppInfo/Application.php
@@ -7,14 +7,14 @@
*/
namespace OCA\Encryption\AppInfo;
+use OC\Core\Events\BeforePasswordResetEvent;
+use OC\Core\Events\PasswordResetEvent;
use OCA\Encryption\Crypto\Crypt;
use OCA\Encryption\Crypto\DecryptAll;
use OCA\Encryption\Crypto\EncryptAll;
use OCA\Encryption\Crypto\Encryption;
-use OCA\Encryption\HookManager;
-use OCA\Encryption\Hooks\UserHooks;
use OCA\Encryption\KeyManager;
-use OCA\Encryption\Recovery;
+use OCA\Encryption\Listeners\UserEventsListener;
use OCA\Encryption\Session;
use OCA\Encryption\Users\Setup;
use OCA\Encryption\Util;
@@ -23,7 +23,14 @@ use OCP\AppFramework\Bootstrap\IBootContext;
use OCP\AppFramework\Bootstrap\IBootstrap;
use OCP\AppFramework\Bootstrap\IRegistrationContext;
use OCP\Encryption\IManager;
+use OCP\EventDispatcher\IEventDispatcher;
use OCP\IConfig;
+use OCP\IL10N;
+use OCP\IUserSession;
+use OCP\User\Events\BeforePasswordUpdatedEvent;
+use OCP\User\Events\PasswordUpdatedEvent;
+use OCP\User\Events\UserCreatedEvent;
+use OCP\User\Events\UserDeletedEvent;
use Psr\Log\LoggerInterface;
class Application extends App implements IBootstrap {
@@ -49,7 +56,7 @@ class Application extends App implements IBootstrap {
}
$context->injectFn($this->registerEncryptionModule(...));
- $context->injectFn($this->registerHooks(...));
+ $context->injectFn($this->registerEventListeners(...));
$context->injectFn($this->setUp(...));
});
}
@@ -57,38 +64,29 @@ class Application extends App implements IBootstrap {
public function setUp(IManager $encryptionManager) {
if ($encryptionManager->isEnabled()) {
/** @var Setup $setup */
- $setup = $this->getContainer()->query(Setup::class);
+ $setup = $this->getContainer()->get(Setup::class);
$setup->setupSystem();
}
}
- /**
- * register hooks
- */
- public function registerHooks(IConfig $config) {
- if (!$config->getSystemValueBool('maintenance')) {
- $container = $this->getContainer();
- $server = $container->getServer();
- // Register our hooks and fire them.
- $hookManager = new HookManager();
-
- $hookManager->registerHook([
- new UserHooks($container->query(KeyManager::class),
- $server->getUserManager(),
- $server->get(LoggerInterface::class),
- $container->query(Setup::class),
- $server->getUserSession(),
- $container->query(Util::class),
- $container->query(Session::class),
- $container->query(Crypt::class),
- $container->query(Recovery::class))
- ]);
+ public function registerEventListeners(IConfig $config, IEventDispatcher $eventDispatcher, IManager $encryptionManager): void {
+ if (!$encryptionManager->isEnabled()) {
+ return;
+ }
- $hookManager->fireHooks();
- } else {
+ if ($config->getSystemValueBool('maintenance')) {
// Logout user if we are in maintenance to force re-login
- $this->getContainer()->getServer()->getUserSession()->logout();
+ $this->getContainer()->get(IUserSession::class)->logout();
+ return;
}
+
+ // No maintenance so register all events
+ $eventDispatcher->addServiceListener(UserCreatedEvent::class, UserEventsListener::class);
+ $eventDispatcher->addServiceListener(UserDeletedEvent::class, UserEventsListener::class);
+ $eventDispatcher->addServiceListener(BeforePasswordUpdatedEvent::class, UserEventsListener::class);
+ $eventDispatcher->addServiceListener(PasswordUpdatedEvent::class, UserEventsListener::class);
+ $eventDispatcher->addServiceListener(BeforePasswordResetEvent::class, UserEventsListener::class);
+ $eventDispatcher->addServiceListener(PasswordResetEvent::class, UserEventsListener::class);
}
public function registerEncryptionModule(IManager $encryptionManager) {
@@ -99,14 +97,14 @@ class Application extends App implements IBootstrap {
Encryption::DISPLAY_NAME,
function () use ($container) {
return new Encryption(
- $container->query(Crypt::class),
- $container->query(KeyManager::class),
- $container->query(Util::class),
- $container->query(Session::class),
- $container->query(EncryptAll::class),
- $container->query(DecryptAll::class),
- $container->getServer()->get(LoggerInterface::class),
- $container->getServer()->getL10N($container->getAppName())
+ $container->get(Crypt::class),
+ $container->get(KeyManager::class),
+ $container->get(Util::class),
+ $container->get(Session::class),
+ $container->get(EncryptAll::class),
+ $container->get(DecryptAll::class),
+ $container->get(LoggerInterface::class),
+ $container->get(IL10N::class),
);
});
}
diff --git a/apps/encryption/lib/Crypto/Crypt.php b/apps/encryption/lib/Crypto/Crypt.php
index b38734dd061..463ca4e22bb 100644
--- a/apps/encryption/lib/Crypto/Crypt.php
+++ b/apps/encryption/lib/Crypto/Crypt.php
@@ -83,7 +83,7 @@ class Crypt {
/**
* create new private/public key-pair for user
*
- * @return array|bool
+ * @return array{publicKey: string, privateKey: string}|false
*/
public function createKeyPair() {
$res = $this->getOpenSSLPKey();
diff --git a/apps/encryption/lib/HookManager.php b/apps/encryption/lib/HookManager.php
deleted file mode 100644
index 6ad56ebad78..00000000000
--- a/apps/encryption/lib/HookManager.php
+++ /dev/null
@@ -1,43 +0,0 @@
-<?php
-
-/**
- * SPDX-FileCopyrightText: 2019-2024 Nextcloud GmbH and Nextcloud contributors
- * SPDX-FileCopyrightText: 2016 ownCloud, Inc.
- * SPDX-License-Identifier: AGPL-3.0-only
- */
-namespace OCA\Encryption;
-
-use OCA\Encryption\Hooks\Contracts\IHook;
-
-class HookManager {
- /** @var IHook[] */
- private $hookInstances = [];
-
- /**
- * @param array|IHook $instances
- * - This accepts either a single instance of IHook or an array of instances of IHook
- * @return bool
- */
- public function registerHook($instances) {
- if (is_array($instances)) {
- foreach ($instances as $instance) {
- if (!$instance instanceof IHook) {
- return false;
- }
- $this->hookInstances[] = $instance;
- }
- } elseif ($instances instanceof IHook) {
- $this->hookInstances[] = $instances;
- }
- return true;
- }
-
- public function fireHooks() {
- foreach ($this->hookInstances as $instance) {
- /**
- * Fire off the add hooks method of each instance stored in cache
- */
- $instance->addHooks();
- }
- }
-}
diff --git a/apps/encryption/lib/Hooks/Contracts/IHook.php b/apps/encryption/lib/Hooks/Contracts/IHook.php
deleted file mode 100644
index 5bb8046e230..00000000000
--- a/apps/encryption/lib/Hooks/Contracts/IHook.php
+++ /dev/null
@@ -1,17 +0,0 @@
-<?php
-
-/**
- * SPDX-FileCopyrightText: 2019-2024 Nextcloud GmbH and Nextcloud contributors
- * SPDX-FileCopyrightText: 2016 ownCloud, Inc.
- * SPDX-License-Identifier: AGPL-3.0-only
- */
-namespace OCA\Encryption\Hooks\Contracts;
-
-interface IHook {
- /**
- * Connects Hooks
- *
- * @return null
- */
- public function addHooks();
-}
diff --git a/apps/encryption/lib/Hooks/UserHooks.php b/apps/encryption/lib/Hooks/UserHooks.php
deleted file mode 100644
index e75a1e545d1..00000000000
--- a/apps/encryption/lib/Hooks/UserHooks.php
+++ /dev/null
@@ -1,266 +0,0 @@
-<?php
-
-/**
- * SPDX-FileCopyrightText: 2017-2024 Nextcloud GmbH and Nextcloud contributors
- * SPDX-FileCopyrightText: 2016 ownCloud, Inc.
- * SPDX-License-Identifier: AGPL-3.0-only
- */
-namespace OCA\Encryption\Hooks;
-
-use OC\Files\Filesystem;
-use OCA\Encryption\Crypto\Crypt;
-use OCA\Encryption\Hooks\Contracts\IHook;
-use OCA\Encryption\KeyManager;
-use OCA\Encryption\Recovery;
-use OCA\Encryption\Session;
-use OCA\Encryption\Users\Setup;
-use OCA\Encryption\Util;
-use OCP\Encryption\Exceptions\GenericEncryptionException;
-use OCP\IUserManager;
-use OCP\IUserSession;
-use OCP\Util as OCUtil;
-use Psr\Log\LoggerInterface;
-
-class UserHooks implements IHook {
- /**
- * list of user for which we perform a password reset
- * @var array<string, true>
- */
- protected static array $passwordResetUsers = [];
-
- public function __construct(
- private KeyManager $keyManager,
- private IUserManager $userManager,
- private LoggerInterface $logger,
- private Setup $userSetup,
- private IUserSession $userSession,
- private Util $util,
- private Session $session,
- private Crypt $crypt,
- private Recovery $recovery,
- ) {
- }
-
- /**
- * Connects Hooks
- *
- * @return null
- */
- public function addHooks() {
- OCUtil::connectHook('OC_User', 'post_login', $this, 'login');
- OCUtil::connectHook('OC_User', 'logout', $this, 'logout');
-
- // this hooks only make sense if no master key is used
- if ($this->util->isMasterKeyEnabled() === false) {
- OCUtil::connectHook('OC_User',
- 'post_setPassword',
- $this,
- 'setPassphrase');
-
- OCUtil::connectHook('OC_User',
- 'pre_setPassword',
- $this,
- 'preSetPassphrase');
-
- OCUtil::connectHook('\OC\Core\LostPassword\Controller\LostController',
- 'post_passwordReset',
- $this,
- 'postPasswordReset');
-
- OCUtil::connectHook('\OC\Core\LostPassword\Controller\LostController',
- 'pre_passwordReset',
- $this,
- 'prePasswordReset');
-
- OCUtil::connectHook('OC_User',
- 'post_createUser',
- $this,
- 'postCreateUser');
-
- OCUtil::connectHook('OC_User',
- 'post_deleteUser',
- $this,
- 'postDeleteUser');
- }
- }
-
-
- /**
- * Startup encryption backend upon user login
- *
- * @note This method should never be called for users using client side encryption
- * @param array $params
- * @return boolean|null
- */
- public function login($params) {
- // ensure filesystem is loaded
- if (!Filesystem::$loaded) {
- $this->setupFS($params['uid']);
- }
- if ($this->util->isMasterKeyEnabled() === false) {
- $this->userSetup->setupUser($params['uid'], $params['password']);
- }
-
- $this->keyManager->init($params['uid'], $params['password']);
- }
-
- /**
- * remove keys from session during logout
- */
- public function logout() {
- $this->session->clear();
- }
-
- /**
- * setup encryption backend upon user created
- *
- * @note This method should never be called for users using client side encryption
- * @param array $params
- */
- public function postCreateUser($params) {
- $this->userSetup->setupUser($params['uid'], $params['password']);
- }
-
- /**
- * cleanup encryption backend upon user deleted
- *
- * @param array $params : uid, password
- * @note This method should never be called for users using client side encryption
- */
- public function postDeleteUser($params) {
- $this->keyManager->deletePublicKey($params['uid']);
- }
-
- public function prePasswordReset($params) {
- $user = $params['uid'];
- self::$passwordResetUsers[$user] = true;
- }
-
- public function postPasswordReset($params) {
- $uid = $params['uid'];
- $password = $params['password'];
- $this->keyManager->backupUserKeys('passwordReset', $uid);
- $this->keyManager->deleteUserKeys($uid);
- $this->userSetup->setupUser($uid, $password);
- unset(self::$passwordResetUsers[$uid]);
- }
-
- /**
- * If the password can't be changed within Nextcloud, than update the key password in advance.
- *
- * @param array $params : uid, password
- * @return boolean|null
- */
- public function preSetPassphrase($params) {
- $user = $this->userManager->get($params['uid']);
-
- if ($user && !$user->canChangePassword()) {
- $this->setPassphrase($params);
- }
- }
-
- /**
- * Change a user's encryption passphrase
- *
- * @param array $params keys: uid, password
- * @return boolean|null
- */
- public function setPassphrase($params) {
- // if we are in the process to resetting a user password, we have nothing
- // to do here
- if (isset(self::$passwordResetUsers[$params['uid']])) {
- return true;
- }
-
- // Get existing decrypted private key
- $user = $this->userSession->getUser();
-
- // current logged in user changes their own password
- if ($user && $params['uid'] === $user->getUID()) {
- $privateKey = $this->session->getPrivateKey();
-
- // Encrypt private key with new user pwd as passphrase
- $encryptedPrivateKey = $this->crypt->encryptPrivateKey($privateKey, $params['password'], $params['uid']);
-
- // Save private key
- if ($encryptedPrivateKey) {
- $this->keyManager->setPrivateKey($user->getUID(),
- $this->crypt->generateHeader() . $encryptedPrivateKey);
- } else {
- $this->logger->error('Encryption could not update users encryption password');
- }
-
- // NOTE: Session does not need to be updated as the
- // private key has not changed, only the passphrase
- // used to decrypt it has changed
- } else { // admin changed the password for a different user, create new keys and re-encrypt file keys
- $userId = $params['uid'];
- $this->initMountPoints($userId);
- $recoveryPassword = $params['recoveryPassword'] ?? null;
-
- $recoveryKeyId = $this->keyManager->getRecoveryKeyId();
- $recoveryKey = $this->keyManager->getSystemPrivateKey($recoveryKeyId);
- try {
- $decryptedRecoveryKey = $this->crypt->decryptPrivateKey($recoveryKey, $recoveryPassword);
- } catch (\Exception $e) {
- $decryptedRecoveryKey = false;
- }
- if ($decryptedRecoveryKey === false) {
- $message = 'Can not decrypt the recovery key. Maybe you provided the wrong password. Try again.';
- throw new GenericEncryptionException($message, $message);
- }
-
- // we generate new keys if...
- // ...we have a recovery password and the user enabled the recovery key
- // ...encryption was activated for the first time (no keys exists)
- // ...the user doesn't have any files
- if (
- ($this->recovery->isRecoveryEnabledForUser($userId) && $recoveryPassword)
- || !$this->keyManager->userHasKeys($userId)
- || !$this->util->userHasFiles($userId)
- ) {
- // backup old keys
- //$this->backupAllKeys('recovery');
-
- $newUserPassword = $params['password'];
-
- $keyPair = $this->crypt->createKeyPair();
-
- // Save public key
- $this->keyManager->setPublicKey($userId, $keyPair['publicKey']);
-
- // Encrypt private key with new password
- $encryptedKey = $this->crypt->encryptPrivateKey($keyPair['privateKey'], $newUserPassword, $userId);
-
- if ($encryptedKey) {
- $this->keyManager->setPrivateKey($userId, $this->crypt->generateHeader() . $encryptedKey);
-
- if ($recoveryPassword) { // if recovery key is set we can re-encrypt the key files
- $this->recovery->recoverUsersFiles($recoveryPassword, $userId);
- }
- } else {
- $this->logger->error('Encryption Could not update users encryption password');
- }
- }
- }
- }
-
- /**
- * init mount points for given user
- *
- * @param string $user
- * @throws \OC\User\NoUserException
- */
- protected function initMountPoints($user) {
- Filesystem::initMountPoints($user);
- }
-
- /**
- * setup file system for user
- *
- * @param string $uid user id
- */
- protected function setupFS($uid) {
- \OC_Util::setupFS($uid);
- }
-}
diff --git a/apps/encryption/lib/KeyManager.php b/apps/encryption/lib/KeyManager.php
index 0c9c02760a8..f694e6550f1 100644
--- a/apps/encryption/lib/KeyManager.php
+++ b/apps/encryption/lib/KeyManager.php
@@ -287,11 +287,9 @@ class KeyManager {
/**
* Decrypt private key and store it
*
- * @param string $uid user id
- * @param string $passPhrase users password
* @return boolean
*/
- public function init($uid, $passPhrase) {
+ public function init(string $uid, ?string $passPhrase) {
$this->session->setStatus(Session::INIT_EXECUTED);
try {
@@ -300,6 +298,10 @@ class KeyManager {
$passPhrase = $this->getMasterKeyPassword();
$privateKey = $this->getSystemPrivateKey($uid);
} else {
+ if ($passPhrase === null) {
+ $this->logger->warning('Master key is disabled but not passphrase provided.');
+ return false;
+ }
$privateKey = $this->getPrivateKey($uid);
}
$privateKey = $this->crypt->decryptPrivateKey($privateKey, $passPhrase, $uid);
diff --git a/apps/encryption/lib/Listeners/UserEventsListener.php b/apps/encryption/lib/Listeners/UserEventsListener.php
new file mode 100644
index 00000000000..694640a0103
--- /dev/null
+++ b/apps/encryption/lib/Listeners/UserEventsListener.php
@@ -0,0 +1,143 @@
+<?php
+
+declare(strict_types=1);
+/**
+ * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
+ * SPDX-License-Identifier: AGPL-3.0-or-later
+ */
+
+namespace OCA\Encryption\Listeners;
+
+use OC\Core\Events\BeforePasswordResetEvent;
+use OC\Core\Events\PasswordResetEvent;
+use OC\Files\SetupManager;
+use OCA\Encryption\KeyManager;
+use OCA\Encryption\Services\PassphraseService;
+use OCA\Encryption\Session;
+use OCA\Encryption\Users\Setup;
+use OCA\Encryption\Util;
+use OCP\EventDispatcher\Event;
+use OCP\EventDispatcher\IEventListener;
+use OCP\IUser;
+use OCP\IUserManager;
+use OCP\IUserSession;
+use OCP\User\Events\BeforePasswordUpdatedEvent;
+use OCP\User\Events\PasswordUpdatedEvent;
+use OCP\User\Events\UserCreatedEvent;
+use OCP\User\Events\UserDeletedEvent;
+use OCP\User\Events\UserLoggedInEvent;
+use OCP\User\Events\UserLoggedOutEvent;
+
+/**
+ * @template-implements IEventListener<UserCreatedEvent|UserDeletedEvent|UserLoggedInEvent|UserLoggedOutEvent|BeforePasswordUpdatedEvent|PasswordUpdatedEvent|BeforePasswordResetEvent|PasswordResetEvent>
+ */
+class UserEventsListener implements IEventListener {
+
+ public function __construct(
+ private Util $util,
+ private Setup $userSetup,
+ private Session $session,
+ private KeyManager $keyManager,
+ private IUserManager $userManager,
+ private IUserSession $userSession,
+ private SetupManager $setupManager,
+ private PassphraseService $passphraseService,
+ ) {
+ }
+
+ public function handle(Event $event): void {
+ if ($event instanceof UserCreatedEvent) {
+ $this->onUserCreated($event->getUid(), $event->getPassword());
+ } elseif ($event instanceof UserDeletedEvent) {
+ $this->onUserDeleted($event->getUid());
+ } elseif ($event instanceof UserLoggedInEvent) {
+ $this->onUserLogin($event->getUser(), $event->getPassword());
+ } elseif ($event instanceof UserLoggedOutEvent) {
+ $this->onUserLogout();
+ } elseif ($event instanceof BeforePasswordUpdatedEvent) {
+ $this->onBeforePasswordUpdated($event->getUser(), $event->getPassword(), $event->getRecoveryPassword());
+ } elseif ($event instanceof PasswordUpdatedEvent) {
+ $this->onPasswordUpdated($event->getUid(), $event->getPassword(), $event->getRecoveryPassword());
+ } elseif ($event instanceof BeforePasswordResetEvent) {
+ $this->onBeforePasswordReset($event->getUid());
+ } elseif ($event instanceof PasswordResetEvent) {
+ $this->onPasswordReset($event->getUid(), $event->getPassword());
+ }
+ }
+
+ /**
+ * Startup encryption backend upon user login
+ */
+ private function onUserLogin(IUser $user, ?string $password): void {
+ // ensure filesystem is loaded
+ $this->setupManager->setupForUser($user);
+ if ($this->util->isMasterKeyEnabled() === false) {
+ // Skip if no master key and the password is not provided
+ if ($password === null) {
+ return;
+ }
+
+ $this->userSetup->setupUser($user->getUID(), $password);
+ }
+
+ $this->keyManager->init($user->getUID(), $password);
+ }
+
+ /**
+ * Remove keys from session during logout
+ */
+ private function onUserLogout(): void {
+ $this->session->clear();
+ }
+
+ /**
+ * Setup encryption backend upon user created
+ *
+ * This method should never be called for users using client side encryption
+ */
+ protected function onUserCreated(string $userId, string $password): void {
+ $this->userSetup->setupUser($userId, $password);
+ }
+
+ /**
+ * Cleanup encryption backend upon user deleted
+ *
+ * This method should never be called for users using client side encryption
+ */
+ protected function onUserDeleted(string $userId): void {
+ $this->keyManager->deletePublicKey($userId);
+ }
+
+ /**
+ * If the password can't be changed within Nextcloud, than update the key password in advance.
+ */
+ public function onBeforePasswordUpdated(IUser $user, string $password, ?string $recoveryPassword = null): void {
+ if (!$user->canChangePassword()) {
+ $this->passphraseService->setPassphraseForUser($user->getUID(), $password, $recoveryPassword);
+ }
+ }
+
+ /**
+ * Change a user's encryption passphrase
+ */
+ public function onPasswordUpdated(string $userId, string $password, ?string $recoveryPassword): void {
+ $this->passphraseService->setPassphraseForUser($userId, $password, $recoveryPassword);
+ }
+
+ /**
+ * Set user password resetting state to allow ignoring "reset"-requests on password update
+ */
+ public function onBeforePasswordReset(string $userId): void {
+ $this->passphraseService->setProcessingReset($userId);
+ }
+
+ /**
+ * Create new encryption keys on password reset and backup the old one
+ */
+ public function onPasswordReset(string $userId, string $password): void {
+ $this->keyManager->backupUserKeys('passwordReset', $userId);
+ $this->keyManager->deleteUserKeys($userId);
+ $this->userSetup->setupUser($userId, $password);
+ $this->passphraseService->setProcessingReset($userId, false);
+ }
+}
diff --git a/apps/encryption/lib/Services/PassphraseService.php b/apps/encryption/lib/Services/PassphraseService.php
new file mode 100644
index 00000000000..4b3e83375f5
--- /dev/null
+++ b/apps/encryption/lib/Services/PassphraseService.php
@@ -0,0 +1,142 @@
+<?php
+
+declare(strict_types=1);
+/**
+ * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
+ * SPDX-License-Identifier: AGPL-3.0-or-later
+ */
+
+namespace OCA\Encryption\Services;
+
+use OCA\Encryption\Crypto\Crypt;
+use OCA\Encryption\KeyManager;
+use OCA\Encryption\Recovery;
+use OCA\Encryption\Session;
+use OCA\Encryption\Util;
+use OCP\Encryption\Exceptions\GenericEncryptionException;
+use OCP\IUser;
+use OCP\IUserManager;
+use OCP\IUserSession;
+use Psr\Log\LoggerInterface;
+
+class PassphraseService {
+
+ /** @var array<string, bool> */
+ private static array $passwordResetUsers = [];
+
+ public function __construct(
+ private Util $util,
+ private Crypt $crypt,
+ private Session $session,
+ private Recovery $recovery,
+ private KeyManager $keyManager,
+ private LoggerInterface $logger,
+ private IUserManager $userManager,
+ private IUserSession $userSession,
+ ) {
+ }
+
+ public function setProcessingReset(string $uid, bool $processing = true): void {
+ if ($processing) {
+ self::$passwordResetUsers[$uid] = true;
+ } else {
+ unset(self::$passwordResetUsers[$uid]);
+ }
+ }
+
+ /**
+ * Change a user's encryption passphrase
+ */
+ public function setPassphraseForUser(string $userId, string $password, ?string $recoveryPassword = null): bool {
+ // if we are in the process to resetting a user password, we have nothing
+ // to do here
+ if (isset(self::$passwordResetUsers[$userId])) {
+ return true;
+ }
+
+ // Check user exists on backend
+ $user = $this->userManager->get($userId);
+ if ($user === null) {
+ return false;
+ }
+
+ // Get existing decrypted private key
+ $currentUser = $this->userSession->getUser();
+
+ // current logged in user changes his own password
+ if ($currentUser !== null && $userId === $currentUser->getUID()) {
+ $privateKey = $this->session->getPrivateKey();
+
+ // Encrypt private key with new user pwd as passphrase
+ $encryptedPrivateKey = $this->crypt->encryptPrivateKey($privateKey, $password, $userId);
+
+ // Save private key
+ if ($encryptedPrivateKey !== false) {
+ $key = $this->crypt->generateHeader() . $encryptedPrivateKey;
+ $this->keyManager->setPrivateKey($userId, $key);
+ return true;
+ }
+
+ $this->logger->error('Encryption could not update users encryption password');
+
+ // NOTE: Session does not need to be updated as the
+ // private key has not changed, only the passphrase
+ // used to decrypt it has changed
+ } else {
+ // admin changed the password for a different user, create new keys and re-encrypt file keys
+ $recoveryPassword = $recoveryPassword ?? '';
+ $this->initMountPoints($user);
+
+ $recoveryKeyId = $this->keyManager->getRecoveryKeyId();
+ $recoveryKey = $this->keyManager->getSystemPrivateKey($recoveryKeyId);
+ try {
+ $this->crypt->decryptPrivateKey($recoveryKey, $recoveryPassword);
+ } catch (\Exception) {
+ $message = 'Can not decrypt the recovery key. Maybe you provided the wrong password. Try again.';
+ throw new GenericEncryptionException($message, $message);
+ }
+
+ // we generate new keys if...
+ // ...we have a recovery password and the user enabled the recovery key
+ // ...encryption was activated for the first time (no keys exists)
+ // ...the user doesn't have any files
+ if (
+ ($this->recovery->isRecoveryEnabledForUser($userId) && $recoveryPassword !== '')
+ || !$this->keyManager->userHasKeys($userId)
+ || !$this->util->userHasFiles($userId)
+ ) {
+ $keyPair = $this->crypt->createKeyPair();
+ if ($keyPair === false) {
+ $this->logger->error('Could not create new private key-pair for user.');
+ return false;
+ }
+
+ // Save public key
+ $this->keyManager->setPublicKey($userId, $keyPair['publicKey']);
+
+ // Encrypt private key with new password
+ $encryptedKey = $this->crypt->encryptPrivateKey($keyPair['privateKey'], $password, $userId);
+ if ($encryptedKey === false) {
+ $this->logger->error('Encryption could not update users encryption password');
+ return false;
+ }
+
+ $this->keyManager->setPrivateKey($userId, $this->crypt->generateHeader() . $encryptedKey);
+
+ if ($recoveryPassword !== '') {
+ // if recovery key is set we can re-encrypt the key files
+ $this->recovery->recoverUsersFiles($recoveryPassword, $userId);
+ }
+ return true;
+ }
+ }
+ return false;
+ }
+
+ /**
+ * Init mount points for given user
+ */
+ private function initMountPoints(IUser $user): void {
+ \OC\Files\Filesystem::initMountPoints($user);
+ }
+}
diff --git a/apps/encryption/lib/Users/Setup.php b/apps/encryption/lib/Users/Setup.php
index 30e7c5461cc..f2189d6dab2 100644
--- a/apps/encryption/lib/Users/Setup.php
+++ b/apps/encryption/lib/Users/Setup.php
@@ -11,15 +11,11 @@ use OCA\Encryption\Crypto\Crypt;
use OCA\Encryption\KeyManager;
class Setup {
- /** @var Crypt */
- private $crypt;
- /** @var KeyManager */
- private $keyManager;
- public function __construct(Crypt $crypt,
- KeyManager $keyManager) {
- $this->crypt = $crypt;
- $this->keyManager = $keyManager;
+ public function __construct(
+ private Crypt $crypt,
+ private KeyManager $keyManager,
+ ) {
}
/**
diff --git a/apps/encryption/lib/Util.php b/apps/encryption/lib/Util.php
index 6ca4d2c1e1e..61656ddcda3 100644
--- a/apps/encryption/lib/Util.php
+++ b/apps/encryption/lib/Util.php
@@ -25,11 +25,7 @@ class Util {
private IConfig $config,
private IUserManager $userManager,
) {
- $this->files = $files;
- $this->crypt = $crypt;
$this->user = $userSession->isLoggedIn() ? $userSession->getUser() : false;
- $this->config = $config;
- $this->userManager = $userManager;
}
/**
@@ -140,4 +136,5 @@ class Util {
public function getStorage($path) {
return $this->files->getMount($path)->getStorage();
}
+
}
diff --git a/apps/encryption/tests/HookManagerTest.php b/apps/encryption/tests/HookManagerTest.php
deleted file mode 100644
index f831690580f..00000000000
--- a/apps/encryption/tests/HookManagerTest.php
+++ /dev/null
@@ -1,52 +0,0 @@
-<?php
-
-/**
- * SPDX-FileCopyrightText: 2016-2024 Nextcloud GmbH and Nextcloud contributors
- * SPDX-FileCopyrightText: 2016 ownCloud, Inc.
- * SPDX-License-Identifier: AGPL-3.0-only
- */
-namespace OCA\Encryption\Tests;
-
-use OCA\Encryption\HookManager;
-use OCA\Encryption\Hooks\Contracts\IHook;
-use OCP\IConfig;
-use Test\TestCase;
-
-class HookManagerTest extends TestCase {
-
- /**
- * @var HookManager
- */
- private static $instance;
-
-
- public function testRegisterHookWithArray(): void {
- self::$instance->registerHook([
- $this->getMockBuilder(IHook::class)->disableOriginalConstructor()->getMock(),
- $this->getMockBuilder(IHook::class)->disableOriginalConstructor()->getMock(),
- $this->createMock(IConfig::class)
- ]);
-
- $hookInstances = self::invokePrivate(self::$instance, 'hookInstances');
- // Make sure our type checking works
- $this->assertCount(2, $hookInstances);
- }
-
-
-
- public static function setUpBeforeClass(): void {
- parent::setUpBeforeClass();
- // have to make instance static to preserve data between tests
- self::$instance = new HookManager();
- }
-
-
- public function testRegisterHooksWithInstance(): void {
- $mock = $this->getMockBuilder(IHook::class)->disableOriginalConstructor()->getMock();
- /** @var IHook $mock */
- self::$instance->registerHook($mock);
-
- $hookInstances = self::invokePrivate(self::$instance, 'hookInstances');
- $this->assertCount(3, $hookInstances);
- }
-}
diff --git a/apps/encryption/tests/Hooks/UserHooksTest.php b/apps/encryption/tests/Hooks/UserHooksTest.php
deleted file mode 100644
index 072a20de846..00000000000
--- a/apps/encryption/tests/Hooks/UserHooksTest.php
+++ /dev/null
@@ -1,370 +0,0 @@
-<?php
-
-/**
- * SPDX-FileCopyrightText: 2016-2024 Nextcloud GmbH and Nextcloud contributors
- * SPDX-FileCopyrightText: 2016 ownCloud, Inc.
- * SPDX-License-Identifier: AGPL-3.0-only
- */
-namespace OCA\Encryption\Tests\Hooks;
-
-use OCA\Encryption\Crypto\Crypt;
-use OCA\Encryption\Hooks\UserHooks;
-use OCA\Encryption\KeyManager;
-use OCA\Encryption\Recovery;
-use OCA\Encryption\Session;
-use OCA\Encryption\Users\Setup;
-use OCA\Encryption\Util;
-use OCP\IUser;
-use OCP\IUserManager;
-use OCP\IUserSession;
-use PHPUnit\Framework\MockObject\MockObject;
-use Psr\Log\LoggerInterface;
-use Test\TestCase;
-
-/**
- * Class UserHooksTest
- *
- * @group DB
- * @package OCA\Encryption\Tests\Hooks
- */
-class UserHooksTest extends TestCase {
- /**
- * @var \PHPUnit\Framework\MockObject\MockObject
- */
- private $utilMock;
- /**
- * @var \PHPUnit\Framework\MockObject\MockObject
- */
- private $recoveryMock;
- /**
- * @var \PHPUnit\Framework\MockObject\MockObject
- */
- private $sessionMock;
- /**
- * @var \PHPUnit\Framework\MockObject\MockObject
- */
- private $keyManagerMock;
- /**
- * @var \PHPUnit\Framework\MockObject\MockObject
- */
- private $userManagerMock;
-
- /**
- * @var \PHPUnit\Framework\MockObject\MockObject
- */
- private $userSetupMock;
- /**
- * @var \PHPUnit\Framework\MockObject\MockObject
- */
- private $userSessionMock;
- /**
- * @var MockObject|IUser
- */
- private $user;
- /**
- * @var \PHPUnit\Framework\MockObject\MockObject
- */
- private $cryptMock;
- /**
- * @var \PHPUnit\Framework\MockObject\MockObject
- */
- private $loggerMock;
- /**
- * @var UserHooks
- */
- private $instance;
-
- private $params = ['uid' => 'testUser', 'password' => 'password'];
-
- public function testLogin(): void {
- $this->userSetupMock->expects($this->once())
- ->method('setupUser')
- ->willReturnOnConsecutiveCalls(true, false);
-
- $this->keyManagerMock->expects($this->once())
- ->method('init')
- ->with('testUser', 'password');
-
- $this->assertNull($this->instance->login($this->params));
- }
-
- public function testLogout(): void {
- $this->sessionMock->expects($this->once())
- ->method('clear');
- $this->instance->logout();
- $this->addToAssertionCount(1);
- }
-
- public function testPostCreateUser(): void {
- $this->userSetupMock->expects($this->once())
- ->method('setupUser');
-
- $this->instance->postCreateUser($this->params);
- $this->addToAssertionCount(1);
- }
-
- public function testPostDeleteUser(): void {
- $this->keyManagerMock->expects($this->once())
- ->method('deletePublicKey')
- ->with('testUser');
-
- $this->instance->postDeleteUser($this->params);
- $this->addToAssertionCount(1);
- }
-
- public function testPrePasswordReset(): void {
- $params = ['uid' => 'user1'];
- $expected = ['user1' => true];
- $this->instance->prePasswordReset($params);
- $passwordResetUsers = $this->invokePrivate($this->instance, 'passwordResetUsers');
-
- $this->assertSame($expected, $passwordResetUsers);
- }
-
- public function testPostPasswordReset(): void {
- $params = ['uid' => 'user1', 'password' => 'password'];
- $this->invokePrivate($this->instance, 'passwordResetUsers', [['user1' => true]]);
- $this->keyManagerMock->expects($this->once())->method('backupUserKeys')
- ->with('passwordReset', 'user1');
- $this->keyManagerMock->expects($this->once())->method('deleteUserKeys')
- ->with('user1');
- $this->userSetupMock->expects($this->once())->method('setupUser')
- ->with('user1', 'password');
-
- $this->instance->postPasswordReset($params);
- $passwordResetUsers = $this->invokePrivate($this->instance, 'passwordResetUsers');
- $this->assertEmpty($passwordResetUsers);
- }
-
- /**
- * @dataProvider dataTestPreSetPassphrase
- */
- public function testPreSetPassphrase($canChange): void {
- /** @var UserHooks | \PHPUnit\Framework\MockObject\MockObject $instance */
- $instance = $this->getMockBuilder(UserHooks::class)
- ->setConstructorArgs(
- [
- $this->keyManagerMock,
- $this->userManagerMock,
- $this->loggerMock,
- $this->userSetupMock,
- $this->userSessionMock,
- $this->utilMock,
- $this->sessionMock,
- $this->cryptMock,
- $this->recoveryMock
- ]
- )
- ->setMethods(['setPassphrase'])
- ->getMock();
-
- $userMock = $this->createMock(IUser::class);
-
- $this->userManagerMock->expects($this->once())
- ->method('get')
- ->with($this->params['uid'])
- ->willReturn($userMock);
- $userMock->expects($this->once())
- ->method('canChangePassword')
- ->willReturn($canChange);
-
- if ($canChange) {
- // in this case the password will be changed in the post hook
- $instance->expects($this->never())->method('setPassphrase');
- } else {
- // if user can't change the password we update the encryption
- // key password already in the pre hook
- $instance->expects($this->once())
- ->method('setPassphrase')
- ->with($this->params);
- }
-
- $instance->preSetPassphrase($this->params);
- }
-
- public function dataTestPreSetPassphrase() {
- return [
- [true],
- [false]
- ];
- }
-
- public function XtestSetPassphrase() {
- $this->sessionMock->expects($this->once())
- ->method('getPrivateKey')
- ->willReturn(true);
-
- $this->cryptMock->expects($this->exactly(4))
- ->method('encryptPrivateKey')
- ->willReturn(true);
-
- $this->cryptMock->expects($this->any())
- ->method('generateHeader')
- ->willReturn(Crypt::HEADER_START . ':Cipher:test:' . Crypt::HEADER_END);
-
- $this->keyManagerMock->expects($this->exactly(4))
- ->method('setPrivateKey')
- ->willReturnCallback(function ($user, $key): void {
- $header = substr($key, 0, strlen(Crypt::HEADER_START));
- $this->assertSame(
- Crypt::HEADER_START,
- $header, 'every encrypted file should start with a header');
- });
-
- $this->assertNull($this->instance->setPassphrase($this->params));
- $this->params['recoveryPassword'] = 'password';
-
- $this->recoveryMock->expects($this->exactly(3))
- ->method('isRecoveryEnabledForUser')
- ->with('testUser1')
- ->willReturnOnConsecutiveCalls(true, false);
-
-
- $this->instance = $this->getMockBuilder(UserHooks::class)
- ->setConstructorArgs(
- [
- $this->keyManagerMock,
- $this->userManagerMock,
- $this->loggerMock,
- $this->userSetupMock,
- $this->userSessionMock,
- $this->utilMock,
- $this->sessionMock,
- $this->cryptMock,
- $this->recoveryMock
- ]
- )->setMethods(['initMountPoints'])->getMock();
-
- $this->instance->expects($this->exactly(3))->method('initMountPoints');
-
- $this->params['uid'] = 'testUser1';
-
- // Test first if statement
- $this->assertNull($this->instance->setPassphrase($this->params));
-
- // Test Second if conditional
- $this->keyManagerMock->expects($this->exactly(2))
- ->method('userHasKeys')
- ->with('testUser1')
- ->willReturn(true);
-
- $this->assertNull($this->instance->setPassphrase($this->params));
-
- // Test third and final if condition
- $this->utilMock->expects($this->once())
- ->method('userHasFiles')
- ->with('testUser1')
- ->willReturn(false);
-
- $this->cryptMock->expects($this->once())
- ->method('createKeyPair');
-
- $this->keyManagerMock->expects($this->once())
- ->method('setPrivateKey');
-
- $this->recoveryMock->expects($this->once())
- ->method('recoverUsersFiles')
- ->with('password', 'testUser1');
-
- $this->assertNull($this->instance->setPassphrase($this->params));
- }
-
- public function testSetPassphraseResetUserMode(): void {
- $params = ['uid' => 'user1', 'password' => 'password'];
- $this->invokePrivate($this->instance, 'passwordResetUsers', [[$params['uid'] => true]]);
- $this->sessionMock->expects($this->never())->method('getPrivateKey');
- $this->keyManagerMock->expects($this->never())->method('setPrivateKey');
- $this->assertTrue($this->instance->setPassphrase($params));
- $this->invokePrivate($this->instance, 'passwordResetUsers', [[]]);
- }
-
- public function XtestSetPasswordNoUser() {
- $userSessionMock = $this->getMockBuilder(IUserSession::class)
- ->disableOriginalConstructor()
- ->getMock();
-
- $userSessionMock->expects($this->any())->method('getUser')->willReturn(null);
-
- $this->recoveryMock->expects($this->once())
- ->method('isRecoveryEnabledForUser')
- ->with('testUser')
- ->willReturn(false);
-
- $userHooks = $this->getMockBuilder(UserHooks::class)
- ->setConstructorArgs(
- [
- $this->keyManagerMock,
- $this->userManagerMock,
- $this->loggerMock,
- $this->userSetupMock,
- $userSessionMock,
- $this->utilMock,
- $this->sessionMock,
- $this->cryptMock,
- $this->recoveryMock
- ]
- )->setMethods(['initMountPoints'])->getMock();
-
- /** @var UserHooks $userHooks */
- $this->assertNull($userHooks->setPassphrase($this->params));
- }
-
- protected function setUp(): void {
- parent::setUp();
- $this->loggerMock = $this->createMock(LoggerInterface::class);
- $this->keyManagerMock = $this->getMockBuilder(KeyManager::class)
- ->disableOriginalConstructor()
- ->getMock();
- $this->userManagerMock = $this->getMockBuilder(IUserManager::class)
- ->disableOriginalConstructor()
- ->getMock();
- $this->userSetupMock = $this->getMockBuilder(Setup::class)
- ->disableOriginalConstructor()
- ->getMock();
-
- $this->user = $this->createMock(IUser::class);
- $this->user->expects($this->any())
- ->method('getUID')
- ->willReturn('testUser');
-
- $this->userSessionMock = $this->createMock(IUserSession::class);
- $this->userSessionMock->expects($this->any())
- ->method('getUser')
- ->willReturn($this->user);
-
- $utilMock = $this->getMockBuilder(Util::class)
- ->disableOriginalConstructor()
- ->getMock();
-
- $sessionMock = $this->getMockBuilder(Session::class)
- ->disableOriginalConstructor()
- ->getMock();
-
- $this->cryptMock = $this->getMockBuilder(Crypt::class)
- ->disableOriginalConstructor()
- ->getMock();
- $recoveryMock = $this->getMockBuilder(Recovery::class)
- ->disableOriginalConstructor()
- ->getMock();
-
- $this->sessionMock = $sessionMock;
- $this->recoveryMock = $recoveryMock;
- $this->utilMock = $utilMock;
- $this->utilMock->expects($this->any())->method('isMasterKeyEnabled')->willReturn(false);
-
- $this->instance = $this->getMockBuilder(UserHooks::class)
- ->setConstructorArgs(
- [
- $this->keyManagerMock,
- $this->userManagerMock,
- $this->loggerMock,
- $this->userSetupMock,
- $this->userSessionMock,
- $this->utilMock,
- $this->sessionMock,
- $this->cryptMock,
- $this->recoveryMock
- ]
- )->setMethods(['setupFS'])->getMock();
- }
-}
diff --git a/apps/encryption/tests/Listeners/UserEventsListenersTest.php b/apps/encryption/tests/Listeners/UserEventsListenersTest.php
new file mode 100644
index 00000000000..cb31523f105
--- /dev/null
+++ b/apps/encryption/tests/Listeners/UserEventsListenersTest.php
@@ -0,0 +1,258 @@
+<?php
+
+declare(strict_types=1);
+
+/**
+ * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
+ * SPDX-License-Identifier: AGPL-3.0-or-later
+ */
+namespace OCA\Encryption\Tests\Listeners;
+
+use OC\Core\Events\BeforePasswordResetEvent;
+use OC\Core\Events\PasswordResetEvent;
+use OC\Files\SetupManager;
+use OCA\Encryption\KeyManager;
+use OCA\Encryption\Listeners\UserEventsListener;
+use OCA\Encryption\Services\PassphraseService;
+use OCA\Encryption\Session;
+use OCA\Encryption\Users\Setup;
+use OCA\Encryption\Util;
+use OCP\IUser;
+use OCP\IUserManager;
+use OCP\IUserSession;
+use OCP\User\Events\BeforePasswordUpdatedEvent;
+use OCP\User\Events\PasswordUpdatedEvent;
+use OCP\User\Events\UserCreatedEvent;
+use OCP\User\Events\UserDeletedEvent;
+use OCP\User\Events\UserLoggedInEvent;
+use OCP\User\Events\UserLoggedOutEvent;
+use PHPUnit\Framework\MockObject\MockObject;
+use Test\TestCase;
+
+/**
+ * @group DB
+ */
+class UserEventsListenersTest extends TestCase {
+
+ protected Util&MockObject $util;
+ protected Setup&MockObject $userSetup;
+ protected Session&MockObject $session;
+ protected KeyManager&MockObject $keyManager;
+ protected IUserManager&MockObject $userManager;
+ protected IUserSession&MockObject $userSession;
+ protected SetupManager&MockObject $setupManager;
+ protected PassphraseService&MockObject $passphraseService;
+
+ protected UserEventsListener $instance;
+
+ public function setUp(): void {
+ parent::setUp();
+
+ $this->util = $this->createMock(Util::class);
+ $this->userSetup = $this->createMock(Setup::class);
+ $this->session = $this->createMock(Session::class);
+ $this->keyManager = $this->createMock(KeyManager::class);
+ $this->userManager = $this->createMock(IUserManager::class);
+ $this->userSession = $this->createMock(IUserSession::class);
+ $this->setupManager = $this->createMock(SetupManager::class);
+ $this->passphraseService = $this->createMock(PassphraseService::class);
+
+ $this->instance = new UserEventsListener(
+ $this->util,
+ $this->userSetup,
+ $this->session,
+ $this->keyManager,
+ $this->userManager,
+ $this->userSession,
+ $this->setupManager,
+ $this->passphraseService,
+ );
+ }
+
+ public function testLogin(): void {
+ $this->userSetup->expects(self::once())
+ ->method('setupUser')
+ ->willReturn(true);
+
+ $this->keyManager->expects(self::once())
+ ->method('init')
+ ->with('testUser', 'password');
+
+ $this->util->method('isMasterKeyEnabled')->willReturn(false);
+
+ $user = $this->createMock(IUser::class);
+ $user->expects(self::any())
+ ->method('getUID')
+ ->willReturn('testUser');
+ $event = $this->createMock(UserLoggedInEvent::class);
+ $event->expects(self::atLeastOnce())
+ ->method('getUser')
+ ->willReturn($user);
+ $event->expects(self::atLeastOnce())
+ ->method('getPassword')
+ ->willReturn('password');
+
+ $this->instance->handle($event);
+ }
+
+ public function testLoginMasterKey(): void {
+ $this->util->method('isMasterKeyEnabled')->willReturn(true);
+
+ $this->userSetup->expects(self::never())
+ ->method('setupUser');
+
+ $this->keyManager->expects(self::once())
+ ->method('init')
+ ->with('testUser', 'password');
+
+ $user = $this->createMock(IUser::class);
+ $user->expects(self::any())
+ ->method('getUID')
+ ->willReturn('testUser');
+
+ $event = $this->createMock(UserLoggedInEvent::class);
+ $event->expects(self::atLeastOnce())
+ ->method('getUser')
+ ->willReturn($user);
+ $event->expects(self::atLeastOnce())
+ ->method('getPassword')
+ ->willReturn('password');
+
+ $this->instance->handle($event);
+ }
+
+ public function testLogout(): void {
+ $this->session->expects(self::once())
+ ->method('clear');
+
+ $event = $this->createMock(UserLoggedOutEvent::class);
+ $this->instance->handle($event);
+ }
+
+ public function testUserCreated(): void {
+ $this->userSetup->expects(self::once())
+ ->method('setupUser')
+ ->with('testUser', 'password');
+
+ $event = $this->createMock(UserCreatedEvent::class);
+ $event->expects(self::atLeastOnce())
+ ->method('getUid')
+ ->willReturn('testUser');
+ $event->expects(self::atLeastOnce())
+ ->method('getPassword')
+ ->willReturn('password');
+
+ $this->instance->handle($event);
+ }
+
+ public function testUserDeleted(): void {
+ $this->keyManager->expects(self::once())
+ ->method('deletePublicKey')
+ ->with('testUser');
+
+ $event = $this->createMock(UserDeletedEvent::class);
+ $event->expects(self::atLeastOnce())
+ ->method('getUid')
+ ->willReturn('testUser');
+ $this->instance->handle($event);
+ }
+
+ public function testBeforePasswordUpdated(): void {
+ $this->passphraseService->expects(self::never())
+ ->method('setPassphraseForUser');
+
+ $user = $this->createMock(IUser::class);
+ $user->expects(self::atLeastOnce())
+ ->method('canChangePassword')
+ ->willReturn(true);
+
+ $event = $this->createMock(BeforePasswordUpdatedEvent::class);
+ $event->expects(self::atLeastOnce())
+ ->method('getUser')
+ ->willReturn($user);
+ $event->expects(self::atLeastOnce())
+ ->method('getPassword')
+ ->willReturn('password');
+ $this->instance->handle($event);
+ }
+
+ public function testBeforePasswordUpdated_CannotChangePassword(): void {
+ $this->passphraseService->expects(self::once())
+ ->method('setPassphraseForUser')
+ ->with('testUser', 'password');
+
+ $user = $this->createMock(IUser::class);
+ $user->expects(self::atLeastOnce())
+ ->method('getUID')
+ ->willReturn('testUser');
+ $user->expects(self::atLeastOnce())
+ ->method('canChangePassword')
+ ->willReturn(false);
+
+ $event = $this->createMock(BeforePasswordUpdatedEvent::class);
+ $event->expects(self::atLeastOnce())
+ ->method('getUser')
+ ->willReturn($user);
+ $event->expects(self::atLeastOnce())
+ ->method('getPassword')
+ ->willReturn('password');
+ $this->instance->handle($event);
+ }
+
+ public function testPasswordUpdated(): void {
+ $this->passphraseService->expects(self::once())
+ ->method('setPassphraseForUser')
+ ->with('testUser', 'password');
+
+ $event = $this->createMock(PasswordUpdatedEvent::class);
+ $event->expects(self::atLeastOnce())
+ ->method('getUid')
+ ->willReturn('testUser');
+ $event->expects(self::atLeastOnce())
+ ->method('getPassword')
+ ->willReturn('password');
+
+ $this->instance->handle($event);
+ }
+
+ public function testBeforePasswordReset(): void {
+ $this->passphraseService->expects(self::once())
+ ->method('setProcessingReset')
+ ->with('testUser');
+
+ $event = $this->createMock(BeforePasswordResetEvent::class);
+ $event->expects(self::atLeastOnce())
+ ->method('getUid')
+ ->willReturn('testUser');
+ $this->instance->handle($event);
+ }
+
+ public function testPasswordReset(): void {
+ // backup required
+ $this->keyManager->expects(self::once())
+ ->method('backupUserKeys')
+ ->with('passwordReset', 'testUser');
+ // delete old keys
+ $this->keyManager->expects(self::once())
+ ->method('deleteUserKeys')
+ ->with('testUser');
+ // create new keys
+ $this->userSetup->expects(self::once())
+ ->method('setupUser')
+ ->with('testUser', 'password');
+ // reset ends
+ $this->passphraseService->expects(self::once())
+ ->method('setProcessingReset')
+ ->with('testUser', false);
+
+ $event = $this->createMock(PasswordResetEvent::class);
+ $event->expects(self::atLeastOnce())
+ ->method('getUid')
+ ->willReturn('testUser');
+ $event->expects(self::atLeastOnce())
+ ->method('getPassword')
+ ->willReturn('password');
+ $this->instance->handle($event);
+ }
+
+}
diff --git a/apps/encryption/tests/PassphraseServiceTest.php b/apps/encryption/tests/PassphraseServiceTest.php
new file mode 100644
index 00000000000..c2dc9d8173c
--- /dev/null
+++ b/apps/encryption/tests/PassphraseServiceTest.php
@@ -0,0 +1,196 @@
+<?php
+
+declare(strict_types=1);
+
+/**
+ * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
+ * SPDX-License-Identifier: AGPL-3.0-or-later
+ */
+namespace OCA\Encryption\Tests;
+
+use OCA\Encryption\Crypto\Crypt;
+use OCA\Encryption\KeyManager;
+use OCA\Encryption\Recovery;
+use OCA\Encryption\Services\PassphraseService;
+use OCA\Encryption\Session;
+use OCA\Encryption\Util;
+use OCP\IUser;
+use OCP\IUserManager;
+use OCP\IUserSession;
+use PHPUnit\Framework\MockObject\MockObject;
+use Psr\Log\LoggerInterface;
+use Test\TestCase;
+
+/**
+ * @group DB
+ */
+class PassphraseServiceTest extends TestCase {
+
+ protected Util&MockObject $util;
+ protected Crypt&MockObject $crypt;
+ protected Session&MockObject $session;
+ protected Recovery&MockObject $recovery;
+ protected KeyManager&MockObject $keyManager;
+ protected IUserManager&MockObject $userManager;
+ protected IUserSession&MockObject $userSession;
+
+ protected PassphraseService $instance;
+
+ public function setUp(): void {
+ parent::setUp();
+
+ $this->util = $this->createMock(Util::class);
+ $this->crypt = $this->createMock(Crypt::class);
+ $this->session = $this->createMock(Session::class);
+ $this->recovery = $this->createMock(Recovery::class);
+ $this->keyManager = $this->createMock(KeyManager::class);
+ $this->userManager = $this->createMock(IUserManager::class);
+ $this->userSession = $this->createMock(IUserSession::class);
+
+ $this->instance = new PassphraseService(
+ $this->util,
+ $this->crypt,
+ $this->session,
+ $this->recovery,
+ $this->keyManager,
+ $this->createMock(LoggerInterface::class),
+ $this->userManager,
+ $this->userSession,
+ );
+ }
+
+ public function testSetProcessingReset(): void {
+ $this->instance->setProcessingReset('userId');
+ $this->assertEquals(['userId' => true], $this->invokePrivate($this->instance, 'passwordResetUsers'));
+ }
+
+ public function testUnsetProcessingReset(): void {
+ $this->instance->setProcessingReset('userId');
+ $this->assertEquals(['userId' => true], $this->invokePrivate($this->instance, 'passwordResetUsers'));
+ $this->instance->setProcessingReset('userId', false);
+ $this->assertEquals([], $this->invokePrivate($this->instance, 'passwordResetUsers'));
+ }
+
+ /**
+ * Check that the passphrase setting skips if a reset is processed
+ */
+ public function testSetPassphraseResetUserMode(): void {
+ $this->session->expects(self::never())
+ ->method('getPrivateKey');
+ $this->keyManager->expects(self::never())
+ ->method('setPrivateKey');
+
+ $this->instance->setProcessingReset('userId');
+ $this->assertTrue($this->instance->setPassphraseForUser('userId', 'password'));
+ }
+
+ public function testSetPassphrase_currentUser() {
+ $instance = $this->getMockBuilder(PassphraseService::class)
+ ->onlyMethods(['initMountPoints'])
+ ->setConstructorArgs([
+ $this->util,
+ $this->crypt,
+ $this->session,
+ $this->recovery,
+ $this->keyManager,
+ $this->createMock(LoggerInterface::class),
+ $this->userManager,
+ $this->userSession,
+ ])
+ ->getMock();
+
+ $user = $this->createMock(IUser::class);
+ $user->method('getUID')->willReturn('testUser');
+ $this->userSession->expects(self::atLeastOnce())
+ ->method('getUser')
+ ->willReturn($user);
+ $this->userManager->expects(self::atLeastOnce())
+ ->method('get')
+ ->with('testUser')
+ ->willReturn($user);
+ $this->session->expects(self::any())
+ ->method('getPrivateKey')
+ ->willReturn('private-key');
+ $this->crypt->expects(self::any())
+ ->method('encryptPrivateKey')
+ ->with('private-key')
+ ->willReturn('encrypted-key');
+ $this->crypt->expects(self::any())
+ ->method('generateHeader')
+ ->willReturn('crypt-header: ');
+
+ $this->keyManager->expects(self::atLeastOnce())
+ ->method('setPrivateKey')
+ ->with('testUser', 'crypt-header: encrypted-key');
+
+ $this->assertTrue($instance->setPassphraseForUser('testUser', 'password'));
+ }
+
+ public function testSetPassphrase_currentUserFails() {
+ $instance = $this->getMockBuilder(PassphraseService::class)
+ ->onlyMethods(['initMountPoints'])
+ ->setConstructorArgs([
+ $this->util,
+ $this->crypt,
+ $this->session,
+ $this->recovery,
+ $this->keyManager,
+ $this->createMock(LoggerInterface::class),
+ $this->userManager,
+ $this->userSession,
+ ])
+ ->getMock();
+
+ $user = $this->createMock(IUser::class);
+ $user->method('getUID')->willReturn('testUser');
+ $this->userManager->expects(self::atLeastOnce())
+ ->method('get')
+ ->with('testUser')
+ ->willReturn($user);
+ $this->userSession->expects(self::atLeastOnce())
+ ->method('getUser')
+ ->willReturn($user);
+ $this->session->expects(self::any())
+ ->method('getPrivateKey')
+ ->willReturn('private-key');
+ $this->crypt->expects(self::any())
+ ->method('encryptPrivateKey')
+ ->with('private-key')
+ ->willReturn(false);
+
+ $this->keyManager->expects(self::never())
+ ->method('setPrivateKey');
+
+ $this->assertFalse($instance->setPassphraseForUser('testUser', 'password'));
+ }
+
+ public function testSetPassphrase_currentUserNotExists() {
+ $instance = $this->getMockBuilder(PassphraseService::class)
+ ->onlyMethods(['initMountPoints'])
+ ->setConstructorArgs([
+ $this->util,
+ $this->crypt,
+ $this->session,
+ $this->recovery,
+ $this->keyManager,
+ $this->createMock(LoggerInterface::class),
+ $this->userManager,
+ $this->userSession,
+ ])
+ ->getMock();
+
+ $user = $this->createMock(IUser::class);
+ $user->method('getUID')->willReturn('testUser');
+ $this->userManager->expects(self::atLeastOnce())
+ ->method('get')
+ ->with('testUser')
+ ->willReturn(null);
+ $this->userSession->expects(self::never())
+ ->method('getUser');
+ $this->keyManager->expects(self::never())
+ ->method('setPrivateKey');
+
+ $this->assertFalse($instance->setPassphraseForUser('testUser', 'password'));
+ }
+
+}
diff --git a/apps/files/l10n/ar.js b/apps/files/l10n/ar.js
index f2367e66c32..0757599ae32 100644
--- a/apps/files/l10n/ar.js
+++ b/apps/files/l10n/ar.js
@@ -49,7 +49,7 @@ OC.L10N.register(
"Upload (max. %s)" : "الرفع ( حد اقصى. %s ) ",
"Files compatibility" : "توافقية الملفات",
"Allow to restrict filenames to ensure files can be synced with all clients. By default all filenames valid on POSIX (e.g. Linux or macOS) are allowed." : "السماح بتقييد أسماء الملفات لضمان إمكانية مزامنة الملفات مع جميع العملاء. بشكل افتراضي، يُسمح بجميع أسماء الملفات الصالحة على POSIX (مثل Linux أو macOS).",
- "Enforce Windows compatibility" : "فرض التوافق مع وندوز",
+ "Enforce Windows compatibility" : "فرض التوافق مع ويندوز",
"This will block filenames not valid on Windows systems, like using reserved names or special characters. But this will not enforce compatibility of case sensitivity." : "سيؤدي هذا إلى حظر أسماء الملفات غير الصالحة على أنظمة Windows؛ مثل استخدام الأسماء المحجوزة أو الأحرف الخاصة. لكن هذا لن يفرض توافق حساسية الحالة.",
"Accept" : "قبول",
"Reject" : "رفض",
@@ -71,7 +71,7 @@ OC.L10N.register(
"Target folder does not exist any more" : "المجلد المراد غير موجود بعد الان",
"Reload current directory" : "إعادة تحميل المجلد الحالي",
"Go to the \"{dir}\" directory" : "إنتقل إلى المجلد \"{dir}\"",
- "Drag and drop files here to upload" : "إسحَب و أفلِت الملفات هنا لرفعها",
+ "Drag and drop files here to upload" : "قم بسحب وإسقاط الملفات هنا لرفعها",
"Your have used your space quota and cannot upload files anymore" : "لقد استنفذت حصتك التخزينية و لم يعد بإمكانك رفع أي ملفات بعدُ",
"You don’t have permission to upload or create files here" : "لا تملك الصلاحية لرفع او انشاء ملف هنا ",
"Favorite" : "المفضلة",
@@ -195,7 +195,7 @@ OC.L10N.register(
"Leave these shares" : "مغادرة هاتين المشاركتين",
"Disconnect storage" : "قطع اتصال التخزين",
"Disconnect storages" : "فصل وحدات تخزين",
- "Delete file" : "إحذِف الملف",
+ "Delete file" : "حذف الملف",
"Delete files" : "حذف الملفات",
"Delete folder" : "حذف مجلد",
"Delete folders" : "حذف مجلدين",
diff --git a/apps/files/l10n/ar.json b/apps/files/l10n/ar.json
index 0b957d89b1f..98c90da6fec 100644
--- a/apps/files/l10n/ar.json
+++ b/apps/files/l10n/ar.json
@@ -47,7 +47,7 @@
"Upload (max. %s)" : "الرفع ( حد اقصى. %s ) ",
"Files compatibility" : "توافقية الملفات",
"Allow to restrict filenames to ensure files can be synced with all clients. By default all filenames valid on POSIX (e.g. Linux or macOS) are allowed." : "السماح بتقييد أسماء الملفات لضمان إمكانية مزامنة الملفات مع جميع العملاء. بشكل افتراضي، يُسمح بجميع أسماء الملفات الصالحة على POSIX (مثل Linux أو macOS).",
- "Enforce Windows compatibility" : "فرض التوافق مع وندوز",
+ "Enforce Windows compatibility" : "فرض التوافق مع ويندوز",
"This will block filenames not valid on Windows systems, like using reserved names or special characters. But this will not enforce compatibility of case sensitivity." : "سيؤدي هذا إلى حظر أسماء الملفات غير الصالحة على أنظمة Windows؛ مثل استخدام الأسماء المحجوزة أو الأحرف الخاصة. لكن هذا لن يفرض توافق حساسية الحالة.",
"Accept" : "قبول",
"Reject" : "رفض",
@@ -69,7 +69,7 @@
"Target folder does not exist any more" : "المجلد المراد غير موجود بعد الان",
"Reload current directory" : "إعادة تحميل المجلد الحالي",
"Go to the \"{dir}\" directory" : "إنتقل إلى المجلد \"{dir}\"",
- "Drag and drop files here to upload" : "إسحَب و أفلِت الملفات هنا لرفعها",
+ "Drag and drop files here to upload" : "قم بسحب وإسقاط الملفات هنا لرفعها",
"Your have used your space quota and cannot upload files anymore" : "لقد استنفذت حصتك التخزينية و لم يعد بإمكانك رفع أي ملفات بعدُ",
"You don’t have permission to upload or create files here" : "لا تملك الصلاحية لرفع او انشاء ملف هنا ",
"Favorite" : "المفضلة",
@@ -193,7 +193,7 @@
"Leave these shares" : "مغادرة هاتين المشاركتين",
"Disconnect storage" : "قطع اتصال التخزين",
"Disconnect storages" : "فصل وحدات تخزين",
- "Delete file" : "إحذِف الملف",
+ "Delete file" : "حذف الملف",
"Delete files" : "حذف الملفات",
"Delete folder" : "حذف مجلد",
"Delete folders" : "حذف مجلدين",
diff --git a/apps/files_sharing/l10n/ar.js b/apps/files_sharing/l10n/ar.js
index 5d33b18fa81..090310deca7 100644
--- a/apps/files_sharing/l10n/ar.js
+++ b/apps/files_sharing/l10n/ar.js
@@ -138,11 +138,13 @@ OC.L10N.register(
"Copy to clipboard" : "نسخ الرابط إلى الحافظة",
"Send link via email" : "إرسال رابط عبر الإيميل",
"Enter an email address or paste a list" : "أدخِل عنوان إيميل أو إلصِق قائمةً",
- "Remove email" : "إحذِف الإيميل",
+ "Remove email" : "حذف البريد الإلكتروني",
"Automatically copying failed, please copy the share link manually" : "تعذّر النسخ التلقائي. قم رجاءً بنسخ رابط المشاركة يدويّاً",
"Link copied to clipboard" : "تمّ نسخ الرابط إلى الحافظة",
"Email already added" : "الإيميل سبقت إضافته سلفاً",
"Invalid email address" : "عنوان الإيميل غير صحيح",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["{count} عنوان بريد إلكتروني مضاف مسبقاً","{count} عنوان بريد إلكتروني مضاف مسبقاً","{count} عناوين بريد إلكتروني مضافة مسبقاً","{count} عناوين بريد إلكتروني مضافة مسبقاً","{count} عناوين بريد إلكتروني مضافة مسبقاً","{count} عناوين بريد إلكتروني مضافة مسبقاً"],
+ "_{count} email address added_::_{count} email addresses added_" : ["تمت إضافة {count} عنوان بريد إلكتروني ","تم إضافة {count} عنوان بريد إلكتروني ","تم إضافة {count} عناوين بريد إلكتروني ","تمت إضافة {count} عناوين بريد إلكتروني ","تمت إضافة {count} عناوين بريد إلكتروني ","تمت إضافة {count} عناوين بريد إلكتروني "],
"What are you requesting?" : "ماذا تطلب؟",
"Request subject" : "موضوع الطلب",
"Birthday party photos, History assignment…" : "صور حفلات الميلاد، أو تاريخ المهام، ....",
@@ -199,7 +201,7 @@ OC.L10N.register(
"Copy internal link to clipboard" : "إنسَخ رابطاً داخليّاً إلى الحافظة",
"Only works for people with access to this folder" : "يعمل فقط عند الأشخاص الذين يمكنهم الوصول إلى هذا المجلد",
"Only works for people with access to this file" : "يعمل فقط عند الأشخاص الذين يمكنهم الوصول إلى هذا الملف",
- "Link copied" : "تمّ نَسْخ الرابط",
+ "Link copied" : "تمّ نسخ الرابط",
"Please enter the following required information before creating the share" : "الرجاء إدخال المعلومات المطلوبة التالية قبل إنشاء المشاركة",
"Password protection (enforced)" : "الحماية بكلمة مرور (إلزامي)",
"Password protection" : "الحماية بكلمة مرور",
diff --git a/apps/files_sharing/l10n/ar.json b/apps/files_sharing/l10n/ar.json
index f4e421916d6..d296e2916aa 100644
--- a/apps/files_sharing/l10n/ar.json
+++ b/apps/files_sharing/l10n/ar.json
@@ -136,11 +136,13 @@
"Copy to clipboard" : "نسخ الرابط إلى الحافظة",
"Send link via email" : "إرسال رابط عبر الإيميل",
"Enter an email address or paste a list" : "أدخِل عنوان إيميل أو إلصِق قائمةً",
- "Remove email" : "إحذِف الإيميل",
+ "Remove email" : "حذف البريد الإلكتروني",
"Automatically copying failed, please copy the share link manually" : "تعذّر النسخ التلقائي. قم رجاءً بنسخ رابط المشاركة يدويّاً",
"Link copied to clipboard" : "تمّ نسخ الرابط إلى الحافظة",
"Email already added" : "الإيميل سبقت إضافته سلفاً",
"Invalid email address" : "عنوان الإيميل غير صحيح",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["{count} عنوان بريد إلكتروني مضاف مسبقاً","{count} عنوان بريد إلكتروني مضاف مسبقاً","{count} عناوين بريد إلكتروني مضافة مسبقاً","{count} عناوين بريد إلكتروني مضافة مسبقاً","{count} عناوين بريد إلكتروني مضافة مسبقاً","{count} عناوين بريد إلكتروني مضافة مسبقاً"],
+ "_{count} email address added_::_{count} email addresses added_" : ["تمت إضافة {count} عنوان بريد إلكتروني ","تم إضافة {count} عنوان بريد إلكتروني ","تم إضافة {count} عناوين بريد إلكتروني ","تمت إضافة {count} عناوين بريد إلكتروني ","تمت إضافة {count} عناوين بريد إلكتروني ","تمت إضافة {count} عناوين بريد إلكتروني "],
"What are you requesting?" : "ماذا تطلب؟",
"Request subject" : "موضوع الطلب",
"Birthday party photos, History assignment…" : "صور حفلات الميلاد، أو تاريخ المهام، ....",
@@ -197,7 +199,7 @@
"Copy internal link to clipboard" : "إنسَخ رابطاً داخليّاً إلى الحافظة",
"Only works for people with access to this folder" : "يعمل فقط عند الأشخاص الذين يمكنهم الوصول إلى هذا المجلد",
"Only works for people with access to this file" : "يعمل فقط عند الأشخاص الذين يمكنهم الوصول إلى هذا الملف",
- "Link copied" : "تمّ نَسْخ الرابط",
+ "Link copied" : "تمّ نسخ الرابط",
"Please enter the following required information before creating the share" : "الرجاء إدخال المعلومات المطلوبة التالية قبل إنشاء المشاركة",
"Password protection (enforced)" : "الحماية بكلمة مرور (إلزامي)",
"Password protection" : "الحماية بكلمة مرور",
diff --git a/apps/files_sharing/l10n/cs.js b/apps/files_sharing/l10n/cs.js
index 2f0f146f871..6a13664237b 100644
--- a/apps/files_sharing/l10n/cs.js
+++ b/apps/files_sharing/l10n/cs.js
@@ -143,6 +143,8 @@ OC.L10N.register(
"Link copied to clipboard" : "Odkaz zkopírován do schánky",
"Email already added" : "E-mail již byl přidán",
"Invalid email address" : "Neplatná e-mailová adresa",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["{count} e-mailová adresa už přidána","{count} e-mailové adresy už přidány","{count} e-mailových adres už přidáno","{count} e-mailové adresy už přidány"],
+ "_{count} email address added_::_{count} email addresses added_" : ["{count} e-mailová adresa přidána","{count} e-mailové adresy přidány","{count} e-mailových adres přidáno","{count} e-mailové adresy přidány"],
"What are you requesting?" : "Co požadujete?",
"Request subject" : "Předmět požadavku",
"Birthday party photos, History assignment…" : "Fotky z narozeninové oslavy, přiřazení historie…",
diff --git a/apps/files_sharing/l10n/cs.json b/apps/files_sharing/l10n/cs.json
index d8706f28d66..059bfc9457e 100644
--- a/apps/files_sharing/l10n/cs.json
+++ b/apps/files_sharing/l10n/cs.json
@@ -141,6 +141,8 @@
"Link copied to clipboard" : "Odkaz zkopírován do schánky",
"Email already added" : "E-mail již byl přidán",
"Invalid email address" : "Neplatná e-mailová adresa",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["{count} e-mailová adresa už přidána","{count} e-mailové adresy už přidány","{count} e-mailových adres už přidáno","{count} e-mailové adresy už přidány"],
+ "_{count} email address added_::_{count} email addresses added_" : ["{count} e-mailová adresa přidána","{count} e-mailové adresy přidány","{count} e-mailových adres přidáno","{count} e-mailové adresy přidány"],
"What are you requesting?" : "Co požadujete?",
"Request subject" : "Předmět požadavku",
"Birthday party photos, History assignment…" : "Fotky z narozeninové oslavy, přiřazení historie…",
diff --git a/apps/files_sharing/l10n/da.js b/apps/files_sharing/l10n/da.js
index bfa41f4d6c0..7112b5f668a 100644
--- a/apps/files_sharing/l10n/da.js
+++ b/apps/files_sharing/l10n/da.js
@@ -118,6 +118,7 @@ OC.L10N.register(
"Link copied" : "Link kopieret",
"Password protection" : "Adgangskodebeskyttelse",
"Enter a password" : "Indtast en adgangskode",
+ "Create share" : "Opret share",
"Add another link" : "Tilføj et link mere",
"Create a new share link" : "Opret et nyt delingslink",
"View only" : "Kun visning",
diff --git a/apps/files_sharing/l10n/da.json b/apps/files_sharing/l10n/da.json
index f8af618d99d..ab7e6e702a6 100644
--- a/apps/files_sharing/l10n/da.json
+++ b/apps/files_sharing/l10n/da.json
@@ -116,6 +116,7 @@
"Link copied" : "Link kopieret",
"Password protection" : "Adgangskodebeskyttelse",
"Enter a password" : "Indtast en adgangskode",
+ "Create share" : "Opret share",
"Add another link" : "Tilføj et link mere",
"Create a new share link" : "Opret et nyt delingslink",
"View only" : "Kun visning",
diff --git a/apps/files_sharing/l10n/de_DE.js b/apps/files_sharing/l10n/de_DE.js
index 14892dc6616..af8ece2fb68 100644
--- a/apps/files_sharing/l10n/de_DE.js
+++ b/apps/files_sharing/l10n/de_DE.js
@@ -143,6 +143,8 @@ OC.L10N.register(
"Link copied to clipboard" : "Link wurde in die Zwischenablage kopiert",
"Email already added" : "E-Mail-Adresse wurde bereits hinzugefügt",
"Invalid email address" : "Ungültige E-Mail-Adresse",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["{count} E-Mail-Adresse bereits hinzugefügt","{count} E-Mail-Adressen bereits hinzugefügt"],
+ "_{count} email address added_::_{count} email addresses added_" : ["{count} E-Mail-Adresse hinzugefügt","{count} E-Mail-Adressen hinzugefügt"],
"What are you requesting?" : "Was fragen Sie an?",
"Request subject" : "Betreff der Anfrage",
"Birthday party photos, History assignment…" : "Fotos von Geburtstagsfeiern, Geschichtsaufgaben…",
diff --git a/apps/files_sharing/l10n/de_DE.json b/apps/files_sharing/l10n/de_DE.json
index d08d1f8c733..6c8bfbe69de 100644
--- a/apps/files_sharing/l10n/de_DE.json
+++ b/apps/files_sharing/l10n/de_DE.json
@@ -141,6 +141,8 @@
"Link copied to clipboard" : "Link wurde in die Zwischenablage kopiert",
"Email already added" : "E-Mail-Adresse wurde bereits hinzugefügt",
"Invalid email address" : "Ungültige E-Mail-Adresse",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["{count} E-Mail-Adresse bereits hinzugefügt","{count} E-Mail-Adressen bereits hinzugefügt"],
+ "_{count} email address added_::_{count} email addresses added_" : ["{count} E-Mail-Adresse hinzugefügt","{count} E-Mail-Adressen hinzugefügt"],
"What are you requesting?" : "Was fragen Sie an?",
"Request subject" : "Betreff der Anfrage",
"Birthday party photos, History assignment…" : "Fotos von Geburtstagsfeiern, Geschichtsaufgaben…",
diff --git a/apps/files_sharing/l10n/en_GB.js b/apps/files_sharing/l10n/en_GB.js
index 0e6cdbc3b70..0e5555b5ced 100644
--- a/apps/files_sharing/l10n/en_GB.js
+++ b/apps/files_sharing/l10n/en_GB.js
@@ -143,6 +143,8 @@ OC.L10N.register(
"Link copied to clipboard" : "Link copied to clipboard",
"Email already added" : "Email already added",
"Invalid email address" : "Invalid email address",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["{count} email address already added","{count} email addresses already added"],
+ "_{count} email address added_::_{count} email addresses added_" : ["{count} email address added","{count} email addresses added"],
"What are you requesting?" : "What are you requesting?",
"Request subject" : "Request subject",
"Birthday party photos, History assignment…" : "Birthday party photos, History assignment…",
@@ -309,9 +311,11 @@ OC.L10N.register(
"Create file request" : "Create file request",
"Upload files to {foldername}" : "Upload files to {foldername}",
"Public file share" : "Public file share",
+ "Publicly shared file." : "Publicly shared file.",
"No file" : "No file",
"The file shared with you will show up here" : "The file shared with you will show up here",
"Public share" : "Public share",
+ "Publicly shared files." : "Publicly shared files.",
"No files" : "No files",
"Files and folders shared with you will show up here" : "Files and folders shared with you will show up here",
"Overview of shared files." : "Overview of shared files.",
diff --git a/apps/files_sharing/l10n/en_GB.json b/apps/files_sharing/l10n/en_GB.json
index 20ab935c3b4..b2d61c35869 100644
--- a/apps/files_sharing/l10n/en_GB.json
+++ b/apps/files_sharing/l10n/en_GB.json
@@ -141,6 +141,8 @@
"Link copied to clipboard" : "Link copied to clipboard",
"Email already added" : "Email already added",
"Invalid email address" : "Invalid email address",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["{count} email address already added","{count} email addresses already added"],
+ "_{count} email address added_::_{count} email addresses added_" : ["{count} email address added","{count} email addresses added"],
"What are you requesting?" : "What are you requesting?",
"Request subject" : "Request subject",
"Birthday party photos, History assignment…" : "Birthday party photos, History assignment…",
@@ -307,9 +309,11 @@
"Create file request" : "Create file request",
"Upload files to {foldername}" : "Upload files to {foldername}",
"Public file share" : "Public file share",
+ "Publicly shared file." : "Publicly shared file.",
"No file" : "No file",
"The file shared with you will show up here" : "The file shared with you will show up here",
"Public share" : "Public share",
+ "Publicly shared files." : "Publicly shared files.",
"No files" : "No files",
"Files and folders shared with you will show up here" : "Files and folders shared with you will show up here",
"Overview of shared files." : "Overview of shared files.",
diff --git a/apps/files_sharing/l10n/gl.js b/apps/files_sharing/l10n/gl.js
index 12778011c94..927b6b043cb 100644
--- a/apps/files_sharing/l10n/gl.js
+++ b/apps/files_sharing/l10n/gl.js
@@ -143,6 +143,8 @@ OC.L10N.register(
"Link copied to clipboard" : "A ligazón foi copiada no portapapeis",
"Email already added" : "O correo xa foi engadido",
"Invalid email address" : "Enderezo de correo incorrecto",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["Xa foi engadido {count} enderezo de correo","Xa foron engadidos {count} enderezos de correo"],
+ "_{count} email address added_::_{count} email addresses added_" : ["Foi engadido {count} enderezo de correo","Foron engadidos {count} enderezos de correo"],
"What are you requesting?" : "Que é o que está a solicitar?",
"Request subject" : "Asunto da solicitude",
"Birthday party photos, History assignment…" : "Fotos da festa de aniversario, tarefa de historia…",
diff --git a/apps/files_sharing/l10n/gl.json b/apps/files_sharing/l10n/gl.json
index 02fe744d0dc..c1d517e1fcf 100644
--- a/apps/files_sharing/l10n/gl.json
+++ b/apps/files_sharing/l10n/gl.json
@@ -141,6 +141,8 @@
"Link copied to clipboard" : "A ligazón foi copiada no portapapeis",
"Email already added" : "O correo xa foi engadido",
"Invalid email address" : "Enderezo de correo incorrecto",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["Xa foi engadido {count} enderezo de correo","Xa foron engadidos {count} enderezos de correo"],
+ "_{count} email address added_::_{count} email addresses added_" : ["Foi engadido {count} enderezo de correo","Foron engadidos {count} enderezos de correo"],
"What are you requesting?" : "Que é o que está a solicitar?",
"Request subject" : "Asunto da solicitude",
"Birthday party photos, History assignment…" : "Fotos da festa de aniversario, tarefa de historia…",
diff --git a/apps/files_sharing/l10n/lt_LT.js b/apps/files_sharing/l10n/lt_LT.js
index f3ccc0227d1..b29138419a9 100644
--- a/apps/files_sharing/l10n/lt_LT.js
+++ b/apps/files_sharing/l10n/lt_LT.js
@@ -66,13 +66,16 @@ OC.L10N.register(
"Could not create share" : "Nepavyko sukurti viešinio",
"Invalid permissions" : "Neteisingi leidimai",
"Invalid date, date format must be YYYY-MM-DD" : "Netinkamas datos formavimas. Privaloma: YYYY-MM-DD",
+ "Please specify a valid account to share with" : "Nurodykite tinkamą paskyrą, su kuria bendrinti",
"Group sharing is disabled by the administrator" : "Administratorius išjungė bendrinimą su grupėmis",
"Please specify a valid group" : "Nurodykite teisingą grupę",
"Public link sharing is disabled by the administrator" : "Administratorius išjungė bendrinimą per viešąsias nuorodas",
"Public upload disabled by the administrator" : "Įkėlimas viešam naudojimui yra išjungtas",
"Public upload is only possible for publicly shared folders" : "Duomenų įkėlimas viešam naudojimui galimas tik į viešai dalinamus aplankus",
+ "Please specify a valid email address" : "Nurodykite tinkamą el. pašto adresą",
"Sharing %s sending the password by Nextcloud Talk failed because Nextcloud Talk is not enabled" : "Nepavyko išsiųsti slaptažodžio bendrinimui %s panaudojant Nextcloud Talk, kadangi Nextcloud Talk neįjungtas ",
"Sharing %1$s failed because the back end does not allow shares from type %2$s" : "Bendrinimas %1$s nepavyko, kadangi sistema nepalaiko bendrinimų šio tipo %2$s",
+ "Please specify a valid team" : "Nurodykite tinkamą komandą",
"Sharing %s failed because the back end does not support room shares" : "Bendrinimas %s nepavyko, kadangi sistema nepalaiko kambario bendrinimo",
"Unknown share type" : "Nežinomas viešinio tipas",
"Not a directory" : "Nėra katalogas",
@@ -92,14 +95,22 @@ OC.L10N.register(
"You received {share} to group {group} as a share by {user}" : "Jūs gavote bendrinimą {share} prie grupės {group}, kuriuo pasidalino {user}",
"Accept" : "Priimti",
"Decline" : "Atmesti",
+ "Open \"%s\"" : "Atverti „%s“",
"People" : "Žmonės",
"Expiration date" : "Pabaigos data",
"Set a password" : "Nustatyti slaptažodį",
"Password" : "Slaptažodis",
"Share link" : "Bendrinimo nuoroda",
"Copy to clipboard" : "Kopijuoti į iškarpinę",
+ "Send link via email" : "Siųsti nuorodą el. paštu",
+ "Enter an email address or paste a list" : "Įveskite el. pašto adresą arba įdėkite sąrašą",
+ "Remove email" : "Šalinti el. paštą",
+ "Automatically copying failed, please copy the share link manually" : "Nepavyko automatiškai nukopijuoti, nukopijuokite nuorodą rankiniu būdu",
"Link copied to clipboard" : "Nuoroda nukopijuota į iškarpinę",
+ "Email already added" : "El. paštas jau pridėtas",
+ "Invalid email address" : "Neteisingas el. pašto adresas",
"Select" : "Žymėti",
+ "Previous step" : "Ankstesnis žingsnis",
"Cancel" : "Atsisakyti",
"Close" : "Užverti",
"Continue" : "Tęsti",
@@ -132,6 +143,7 @@ OC.L10N.register(
"Enter a password" : "Enter a password",
"Expiration date (enforced)" : "Galiojimo pabaigos data (nustatyta)",
"Create share" : "Sukurti viešinį",
+ "Generate QR code" : "Generuoti QR kodą",
"Add another link" : "Pridėti kitą nuorodą",
"Create a new share link" : "Sukurti naują bendrinimo nuorodą",
"Shared via link by {initiator}" : "{initiator} bendrina per nuorodą",
@@ -149,8 +161,11 @@ OC.L10N.register(
"Guest" : "Svečias",
"Group" : "Grupė",
"Email" : "El. paštas",
+ "Team" : "Komanda",
"on {server}" : "serveryje {server}",
"Terms of service" : "Naudojimosi sąlygos",
+ "Nickname" : "Slapyvardis",
+ "Enter your nickname" : "Įveskite savo slapyvardį",
"Allow upload and editing" : "Leisti įkelti ir redaguoti",
"Allow editing" : "Leisti redaguoti",
"Advanced settings" : "Išplėstiniai nustatymai",
diff --git a/apps/files_sharing/l10n/lt_LT.json b/apps/files_sharing/l10n/lt_LT.json
index a765471f982..db8b3d16758 100644
--- a/apps/files_sharing/l10n/lt_LT.json
+++ b/apps/files_sharing/l10n/lt_LT.json
@@ -64,13 +64,16 @@
"Could not create share" : "Nepavyko sukurti viešinio",
"Invalid permissions" : "Neteisingi leidimai",
"Invalid date, date format must be YYYY-MM-DD" : "Netinkamas datos formavimas. Privaloma: YYYY-MM-DD",
+ "Please specify a valid account to share with" : "Nurodykite tinkamą paskyrą, su kuria bendrinti",
"Group sharing is disabled by the administrator" : "Administratorius išjungė bendrinimą su grupėmis",
"Please specify a valid group" : "Nurodykite teisingą grupę",
"Public link sharing is disabled by the administrator" : "Administratorius išjungė bendrinimą per viešąsias nuorodas",
"Public upload disabled by the administrator" : "Įkėlimas viešam naudojimui yra išjungtas",
"Public upload is only possible for publicly shared folders" : "Duomenų įkėlimas viešam naudojimui galimas tik į viešai dalinamus aplankus",
+ "Please specify a valid email address" : "Nurodykite tinkamą el. pašto adresą",
"Sharing %s sending the password by Nextcloud Talk failed because Nextcloud Talk is not enabled" : "Nepavyko išsiųsti slaptažodžio bendrinimui %s panaudojant Nextcloud Talk, kadangi Nextcloud Talk neįjungtas ",
"Sharing %1$s failed because the back end does not allow shares from type %2$s" : "Bendrinimas %1$s nepavyko, kadangi sistema nepalaiko bendrinimų šio tipo %2$s",
+ "Please specify a valid team" : "Nurodykite tinkamą komandą",
"Sharing %s failed because the back end does not support room shares" : "Bendrinimas %s nepavyko, kadangi sistema nepalaiko kambario bendrinimo",
"Unknown share type" : "Nežinomas viešinio tipas",
"Not a directory" : "Nėra katalogas",
@@ -90,14 +93,22 @@
"You received {share} to group {group} as a share by {user}" : "Jūs gavote bendrinimą {share} prie grupės {group}, kuriuo pasidalino {user}",
"Accept" : "Priimti",
"Decline" : "Atmesti",
+ "Open \"%s\"" : "Atverti „%s“",
"People" : "Žmonės",
"Expiration date" : "Pabaigos data",
"Set a password" : "Nustatyti slaptažodį",
"Password" : "Slaptažodis",
"Share link" : "Bendrinimo nuoroda",
"Copy to clipboard" : "Kopijuoti į iškarpinę",
+ "Send link via email" : "Siųsti nuorodą el. paštu",
+ "Enter an email address or paste a list" : "Įveskite el. pašto adresą arba įdėkite sąrašą",
+ "Remove email" : "Šalinti el. paštą",
+ "Automatically copying failed, please copy the share link manually" : "Nepavyko automatiškai nukopijuoti, nukopijuokite nuorodą rankiniu būdu",
"Link copied to clipboard" : "Nuoroda nukopijuota į iškarpinę",
+ "Email already added" : "El. paštas jau pridėtas",
+ "Invalid email address" : "Neteisingas el. pašto adresas",
"Select" : "Žymėti",
+ "Previous step" : "Ankstesnis žingsnis",
"Cancel" : "Atsisakyti",
"Close" : "Užverti",
"Continue" : "Tęsti",
@@ -130,6 +141,7 @@
"Enter a password" : "Enter a password",
"Expiration date (enforced)" : "Galiojimo pabaigos data (nustatyta)",
"Create share" : "Sukurti viešinį",
+ "Generate QR code" : "Generuoti QR kodą",
"Add another link" : "Pridėti kitą nuorodą",
"Create a new share link" : "Sukurti naują bendrinimo nuorodą",
"Shared via link by {initiator}" : "{initiator} bendrina per nuorodą",
@@ -147,8 +159,11 @@
"Guest" : "Svečias",
"Group" : "Grupė",
"Email" : "El. paštas",
+ "Team" : "Komanda",
"on {server}" : "serveryje {server}",
"Terms of service" : "Naudojimosi sąlygos",
+ "Nickname" : "Slapyvardis",
+ "Enter your nickname" : "Įveskite savo slapyvardį",
"Allow upload and editing" : "Leisti įkelti ir redaguoti",
"Allow editing" : "Leisti redaguoti",
"Advanced settings" : "Išplėstiniai nustatymai",
diff --git a/apps/files_sharing/l10n/sr.js b/apps/files_sharing/l10n/sr.js
index 0e9e2be90ed..213f1f8ec85 100644
--- a/apps/files_sharing/l10n/sr.js
+++ b/apps/files_sharing/l10n/sr.js
@@ -143,6 +143,8 @@ OC.L10N.register(
"Link copied to clipboard" : "Веза копирана у оставу",
"Email already added" : "И-мејл је већ додат",
"Invalid email address" : "Неисправна и-мејл адреса",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["{count} и-мејл адреса је већ додата","{count} и-мејл адресе су већ додате","{count} и-мејл адреса је већ додато"],
+ "_{count} email address added_::_{count} email addresses added_" : ["Додата је {count} и-мејл адреса","Додате су {count} и-мејл адресе","Додато је {count} и-мејл адреса"],
"What are you requesting?" : "Шта захтевате?",
"Request subject" : "Тема захтева",
"Birthday party photos, History assignment…" : "Слике за рођенданске забаве, рад из историје…",
diff --git a/apps/files_sharing/l10n/sr.json b/apps/files_sharing/l10n/sr.json
index c12b4f74df6..47ee9aaec69 100644
--- a/apps/files_sharing/l10n/sr.json
+++ b/apps/files_sharing/l10n/sr.json
@@ -141,6 +141,8 @@
"Link copied to clipboard" : "Веза копирана у оставу",
"Email already added" : "И-мејл је већ додат",
"Invalid email address" : "Неисправна и-мејл адреса",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["{count} и-мејл адреса је већ додата","{count} и-мејл адресе су већ додате","{count} и-мејл адреса је већ додато"],
+ "_{count} email address added_::_{count} email addresses added_" : ["Додата је {count} и-мејл адреса","Додате су {count} и-мејл адресе","Додато је {count} и-мејл адреса"],
"What are you requesting?" : "Шта захтевате?",
"Request subject" : "Тема захтева",
"Birthday party photos, History assignment…" : "Слике за рођенданске забаве, рад из историје…",
diff --git a/apps/files_sharing/l10n/sv.js b/apps/files_sharing/l10n/sv.js
index 8d1e6ed9002..ae50dfb5cfe 100644
--- a/apps/files_sharing/l10n/sv.js
+++ b/apps/files_sharing/l10n/sv.js
@@ -143,6 +143,8 @@ OC.L10N.register(
"Link copied to clipboard" : "Länken kopierad till urklipp",
"Email already added" : "E-post har redan lagts till",
"Invalid email address" : "Ogiltig e-postadress",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["{count} e-postadress som redan har lagts till","{count} e-postadresser som redan har lagts till"],
+ "_{count} email address added_::_{count} email addresses added_" : ["{count} e-postadress har lagts till","{count} e-postadresser har lagts till"],
"What are you requesting?" : "Vad begär du?",
"Request subject" : "Begär ämne",
"Birthday party photos, History assignment…" : "Foton på födelsedagsfester, historiauppgift...",
diff --git a/apps/files_sharing/l10n/sv.json b/apps/files_sharing/l10n/sv.json
index 96b1e18d410..e79a9e5aa1d 100644
--- a/apps/files_sharing/l10n/sv.json
+++ b/apps/files_sharing/l10n/sv.json
@@ -141,6 +141,8 @@
"Link copied to clipboard" : "Länken kopierad till urklipp",
"Email already added" : "E-post har redan lagts till",
"Invalid email address" : "Ogiltig e-postadress",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["{count} e-postadress som redan har lagts till","{count} e-postadresser som redan har lagts till"],
+ "_{count} email address added_::_{count} email addresses added_" : ["{count} e-postadress har lagts till","{count} e-postadresser har lagts till"],
"What are you requesting?" : "Vad begär du?",
"Request subject" : "Begär ämne",
"Birthday party photos, History assignment…" : "Foton på födelsedagsfester, historiauppgift...",
diff --git a/apps/files_sharing/l10n/zh_HK.js b/apps/files_sharing/l10n/zh_HK.js
index 6f9511ff702..9587be32c93 100644
--- a/apps/files_sharing/l10n/zh_HK.js
+++ b/apps/files_sharing/l10n/zh_HK.js
@@ -143,6 +143,8 @@ OC.L10N.register(
"Link copied to clipboard" : "已複製連結至剪貼板",
"Email already added" : "已加入電郵地址",
"Invalid email address" : "電郵地址無效",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["已添加 {count} 個電郵地址"],
+ "_{count} email address added_::_{count} email addresses added_" : ["添加了{count}個電郵地址"],
"What are you requesting?" : "您正在請求什麼?",
"Request subject" : "請求主題",
"Birthday party photos, History assignment…" : "生日派對照片、歷史作業 …",
@@ -309,9 +311,11 @@ OC.L10N.register(
"Create file request" : "創建檔案請求",
"Upload files to {foldername}" : "上傳檔案至 {foldername}",
"Public file share" : "公共檔案分享",
+ "Publicly shared file." : "公開分享的檔案。",
"No file" : "無檔案",
"The file shared with you will show up here" : "其它人與您分享的檔案將會顯示在此處",
"Public share" : "公共分享",
+ "Publicly shared files." : "公開分享的檔案。",
"No files" : "沒有檔案",
"Files and folders shared with you will show up here" : "其它人與您分享的檔案與資料夾將會顯示在此處",
"Overview of shared files." : "已分享檔案的概覽。",
diff --git a/apps/files_sharing/l10n/zh_HK.json b/apps/files_sharing/l10n/zh_HK.json
index 90e74843baa..9919d983117 100644
--- a/apps/files_sharing/l10n/zh_HK.json
+++ b/apps/files_sharing/l10n/zh_HK.json
@@ -141,6 +141,8 @@
"Link copied to clipboard" : "已複製連結至剪貼板",
"Email already added" : "已加入電郵地址",
"Invalid email address" : "電郵地址無效",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["已添加 {count} 個電郵地址"],
+ "_{count} email address added_::_{count} email addresses added_" : ["添加了{count}個電郵地址"],
"What are you requesting?" : "您正在請求什麼?",
"Request subject" : "請求主題",
"Birthday party photos, History assignment…" : "生日派對照片、歷史作業 …",
@@ -307,9 +309,11 @@
"Create file request" : "創建檔案請求",
"Upload files to {foldername}" : "上傳檔案至 {foldername}",
"Public file share" : "公共檔案分享",
+ "Publicly shared file." : "公開分享的檔案。",
"No file" : "無檔案",
"The file shared with you will show up here" : "其它人與您分享的檔案將會顯示在此處",
"Public share" : "公共分享",
+ "Publicly shared files." : "公開分享的檔案。",
"No files" : "沒有檔案",
"Files and folders shared with you will show up here" : "其它人與您分享的檔案與資料夾將會顯示在此處",
"Overview of shared files." : "已分享檔案的概覽。",
diff --git a/apps/files_sharing/l10n/zh_TW.js b/apps/files_sharing/l10n/zh_TW.js
index bdc0a3b4e28..0b573da3b43 100644
--- a/apps/files_sharing/l10n/zh_TW.js
+++ b/apps/files_sharing/l10n/zh_TW.js
@@ -143,6 +143,8 @@ OC.L10N.register(
"Link copied to clipboard" : "已複製連結至剪貼簿",
"Email already added" : "已新增電子郵件",
"Invalid email address" : "無效的電子郵件地址",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["已新增 {count} 個電子郵件地址"],
+ "_{count} email address added_::_{count} email addresses added_" : ["新增了 {count} 個電子郵件地址"],
"What are you requesting?" : "您正在請求什麼?",
"Request subject" : "請求主旨",
"Birthday party photos, History assignment…" : "生日派對照片、歷史作業……",
diff --git a/apps/files_sharing/l10n/zh_TW.json b/apps/files_sharing/l10n/zh_TW.json
index 22b6be9becd..62978271640 100644
--- a/apps/files_sharing/l10n/zh_TW.json
+++ b/apps/files_sharing/l10n/zh_TW.json
@@ -141,6 +141,8 @@
"Link copied to clipboard" : "已複製連結至剪貼簿",
"Email already added" : "已新增電子郵件",
"Invalid email address" : "無效的電子郵件地址",
+ "_{count} email address already added_::_{count} email addresses already added_" : ["已新增 {count} 個電子郵件地址"],
+ "_{count} email address added_::_{count} email addresses added_" : ["新增了 {count} 個電子郵件地址"],
"What are you requesting?" : "您正在請求什麼?",
"Request subject" : "請求主旨",
"Birthday party photos, History assignment…" : "生日派對照片、歷史作業……",
diff --git a/apps/files_sharing/src/mixins/SharesMixin.js b/apps/files_sharing/src/mixins/SharesMixin.js
index 994a0fe5cb3..8b467a4e0bc 100644
--- a/apps/files_sharing/src/mixins/SharesMixin.js
+++ b/apps/files_sharing/src/mixins/SharesMixin.js
@@ -3,11 +3,11 @@
* SPDX-License-Identifier: AGPL-3.0-or-later
*/
+import { getCurrentUser } from '@nextcloud/auth'
+import { showError, showSuccess } from '@nextcloud/dialogs'
import { emit } from '@nextcloud/event-bus'
import { fetchNode } from '../services/WebdavClient.ts'
-import { showError, showSuccess } from '@nextcloud/dialogs'
-import { getCurrentUser } from '@nextcloud/auth'
-// eslint-disable-next-line import/no-unresolved, n/no-missing-import
+
import PQueue from 'p-queue'
import debounce from 'debounce'
@@ -92,10 +92,10 @@ export default {
// Datepicker language
lang() {
const weekdaysShort = window.dayNamesShort
- ? window.dayNamesShort // provided by nextcloud
+ ? window.dayNamesShort // provided by Nextcloud
: ['Sun.', 'Mon.', 'Tue.', 'Wed.', 'Thu.', 'Fri.', 'Sat.']
const monthsShort = window.monthNamesShort
- ? window.monthNamesShort // provided by nextcloud
+ ? window.monthNamesShort // provided by Nextcloud
: ['Jan.', 'Feb.', 'Mar.', 'Apr.', 'May.', 'Jun.', 'Jul.', 'Aug.', 'Sep.', 'Oct.', 'Nov.', 'Dec.']
const firstDayOfWeek = window.firstDay ? window.firstDay : 0
@@ -156,7 +156,7 @@ export default {
methods: {
/**
- * Fetch webdav node
+ * Fetch WebDAV node
*
* @return {Node}
*/
@@ -169,6 +169,7 @@ export default {
logger.error('Error:', error)
}
},
+
/**
* Check if a share is valid before
* firing the request
@@ -192,19 +193,7 @@ export default {
},
/**
- * @param {string} date a date with YYYY-MM-DD format
- * @return {Date} date
- */
- parseDateString(date) {
- if (!date) {
- return
- }
- const regex = /([0-9]{4}-[0-9]{2}-[0-9]{2})/i
- return new Date(date.match(regex)?.pop())
- },
-
- /**
- * @param {Date} date
+ * @param {Date} date the date to format
* @return {string} date a date with YYYY-MM-DD format
*/
formatDateToString(date) {
@@ -316,11 +305,17 @@ export default {
// clear any previous errors
this.$delete(this.errors, propertyNames[0])
- showSuccess(t('files_sharing', 'Share {propertyName} saved', { propertyName: propertyNames[0] }))
- } catch ({ message }) {
+ showSuccess(this.updateSuccessMessage(propertyNames))
+ } catch (error) {
+ logger.error('Could not update share', { error, share: this.share, propertyNames })
+
+ const { message } = error
if (message && message !== '') {
this.onSyncError(propertyNames[0], message)
- showError(t('files_sharing', message))
+ showError(message)
+ } else {
+ // We do not have information what happened, but we should still inform the user
+ showError(t('files_sharing', 'Could not update share'))
}
} finally {
this.saving = false
@@ -334,6 +329,32 @@ export default {
},
/**
+ * @param {string[]} names Properties changed
+ */
+ updateSuccessMessage(names) {
+ if (names.length !== 1) {
+ return t('files_sharing', 'Share saved')
+ }
+
+ switch (names[0]) {
+ case 'expireDate':
+ return t('files_sharing', 'Share expire date saved')
+ case 'hideDownload':
+ return t('files_sharing', 'Share hide-download state saved')
+ case 'label':
+ return t('files_sharing', 'Share label saved')
+ case 'note':
+ return t('files_sharing', 'Share note for recipient saved')
+ case 'password':
+ return t('files_sharing', 'Share password saved')
+ case 'permissions':
+ return t('files_sharing', 'Share permissions saved')
+ default:
+ return t('files_sharing', 'Share saved')
+ }
+ },
+
+ /**
* Manage sync errors
*
* @param {string} property the errored property, e.g. 'password'
diff --git a/apps/files_versions/l10n/ar.js b/apps/files_versions/l10n/ar.js
index 0c490be919d..b57e1c4641f 100644
--- a/apps/files_versions/l10n/ar.js
+++ b/apps/files_versions/l10n/ar.js
@@ -14,7 +14,7 @@ OC.L10N.register(
"Current version" : "النسخة الحالية",
"Initial version" : "النسخة الأساسية",
"Version name" : "اسم النسخة",
- "Named versions are persisted, and excluded from automatic cleanups when your storage quota is full." : "النُّسخ المُسمّاة محفوظة و مُستثناة من التنظيف التلقائي عند تضاؤل المٌتاح من الحصة التخزينية.",
+ "Named versions are persisted, and excluded from automatic cleanups when your storage quota is full." : "النسخ المسماة محفوظة ومستثناة من التنظيف التلقائي عند امتلاء الحصة التخزينية الخاصة بك.",
"Cancel" : "إلغاء",
"Remove version name" : "إلغ اسم النسخة",
"Save version name" : "إحفظ اسم النسخة",
diff --git a/apps/files_versions/l10n/ar.json b/apps/files_versions/l10n/ar.json
index 8e872f4bb62..f2ffb1a8930 100644
--- a/apps/files_versions/l10n/ar.json
+++ b/apps/files_versions/l10n/ar.json
@@ -12,7 +12,7 @@
"Current version" : "النسخة الحالية",
"Initial version" : "النسخة الأساسية",
"Version name" : "اسم النسخة",
- "Named versions are persisted, and excluded from automatic cleanups when your storage quota is full." : "النُّسخ المُسمّاة محفوظة و مُستثناة من التنظيف التلقائي عند تضاؤل المٌتاح من الحصة التخزينية.",
+ "Named versions are persisted, and excluded from automatic cleanups when your storage quota is full." : "النسخ المسماة محفوظة ومستثناة من التنظيف التلقائي عند امتلاء الحصة التخزينية الخاصة بك.",
"Cancel" : "إلغاء",
"Remove version name" : "إلغ اسم النسخة",
"Save version name" : "إحفظ اسم النسخة",
diff --git a/apps/settings/l10n/ar.js b/apps/settings/l10n/ar.js
index aefe3ce3b04..4ff4b2da34b 100644
--- a/apps/settings/l10n/ar.js
+++ b/apps/settings/l10n/ar.js
@@ -124,7 +124,7 @@ OC.L10N.register(
"Security & setup warnings" : "تحذيرات الإعداد و الأمان",
"Background jobs" : "المهام في الخلفية Background jobs",
"Unlimited" : "غير محدود",
- "Verifying" : "التحقُّق",
+ "Verifying" : "التحقق",
"Allowed admin IP ranges" : "نطاقات عناوين IP المسموحة للمشرفين",
"Admin IP filtering isn’t applied." : "غربلة عناوين IP الخاصة بالمشرفين غير مُطبَّقة.",
"Configuration key \"%1$s\" expects an array (%2$s found). Admin IP range validation will not be applied." : "مفتاح التهيئة \"%1$s\" يتوقع مصفوفة (%2$s موجودة). غربلة عناوين IP الخاصة بالمشرفين غير مُطبَّقة.",
@@ -264,7 +264,7 @@ OC.L10N.register(
"Push service" : "خدمة الدفش push",
"Valid enterprise license" : "رخصة مؤسسية سارية",
"Free push service" : "خدمة الدفش push المجانية",
- "This is the unsupported community build of Nextcloud. Given the size of this instance, performance, reliability and scalability cannot be guaranteed. Push notifications are limited to avoid overloading our free service. Learn more about the benefits of Nextcloud Enterprise at {link}." : "هذه تجهيزة مجتمعية community build لنكست كلاود غير مدعومة. نظراً لحجم هذا الخادوم، لا يمكن ضمان الأداء و الموثوقية و قابلية التوسع. إشعارات الدفش Push notifications تمّ تقييد عددها لتجنب الضغط الزائد على خدمتنا المجانية. تعرّف على المزيد حول مزايا نسخة نسكت كلاود المؤسسية على الرابط: {link}.",
+ "This is the unsupported community build of Nextcloud. Given the size of this instance, performance, reliability and scalability cannot be guaranteed. Push notifications are limited to avoid overloading our free service. Learn more about the benefits of Nextcloud Enterprise at {link}." : "هذه تجهيزة مجتمعية community build لنكست كلاود غير مدعومة. نظراً لحجم هذا الخادم، لا يمكن ضمان الأداء و الموثوقية و قابلية التوسع. إشعارات الدفش Push notifications تمّ تقييد عددها لتجنب الضغط الزائد على خدمتنا المجانية. تعرّف على المزيد حول مزايا نسخة نسكت كلاود المؤسسية على الرابط: {link}.",
"Random generator" : "مُولِّد عشوائي",
"No suitable source for randomness found by PHP which is highly discouraged for security reasons." : "لم يتمكن PHP من العثور على أي مصدر مناسب للعشوائية randomness؛ الأمر الذي لا يُنصح به لأسباب أمنية.",
"Secure" : "آمِن",
@@ -304,7 +304,7 @@ OC.L10N.register(
"- %.1f GiB available in %s (Nextcloud temporary directory)" : "- %.1f غيغا بايت متاحة فثي %s (المجلد المؤقت لنكست كلاود)",
"Temporary directory is correctly configured:\n%s" : "المجلد المؤقت تمّ تكوينه بالشكل الصحيح:\n%s",
"This instance uses an S3 based object store as primary storage, and has enough space in the temporary directory.\n%s" : "يستخدم هذا الخادم مخزن العناصر object store المستند إلى S3 كمخزن أساسي، و لا زالت لديه سعة تخزين كافية في المجلد المؤقت.\n%s",
- "This instance uses an S3 based object store as primary storage. The uploaded files are stored temporarily on the server and thus it is recommended to have 50 GiB of free space available in the temp directory of PHP. To improve this please change the temporary directory in the php.ini or make more space available in that path. \nChecking the available space in the temporary path resulted in %.1f GiB instead of the recommended 50 GiB. Path: %s" : "يستخدم هذا الخادوم مخزن العناصر object store المستند إلى S3 كمخزن أساسي. الملفات التي تمّ رفعها يتم تخزينها مؤقتًا على الخادوم، و لهذا يُوصى بتوفير مساحة خالية تبلغ 50 غيغا بايت في الدليل المؤقت لـ PHP. لمزيد التحسين، يُرجى تغيير الدليل المؤقت في php.ini أو توفير مساحة أكبر في هذا المسار. \nنتيجة فحص المساحة المتوفرة في المسار المؤقت: %.1f جيجا بايت؛ بدلاً عن: 50 جيجا بايت الموصى بها. المسار: %s",
+ "This instance uses an S3 based object store as primary storage. The uploaded files are stored temporarily on the server and thus it is recommended to have 50 GiB of free space available in the temp directory of PHP. To improve this please change the temporary directory in the php.ini or make more space available in that path. \nChecking the available space in the temporary path resulted in %.1f GiB instead of the recommended 50 GiB. Path: %s" : "يستخدم هذا الخادم مخزن العناصر object store المستند إلى S3 كمخزن أساسي. الملفات التي تمّ رفعها يتم تخزينها مؤقتًا على الخادم، و لهذا يُوصى بتوفير مساحة خالية تبلغ 50 غيغا بايت في الدليل المؤقت لـ PHP. لمزيد التحسين، يُرجى تغيير الدليل المؤقت في php.ini أو توفير مساحة أكبر في هذا المسار. \nنتيجة فحص المساحة المتوفرة في المسار المؤقت: %.1f جيجا بايت؛ بدلاً عن: 50 جيجا بايت الموصى بها. المسار: %s",
"Database transaction isolation level" : "مستوى عزل معاملات قاعدة البيانات",
"Your database does not run with \"READ COMMITTED\" transaction isolation level. This can cause problems when multiple actions are executed in parallel." : "لا تعمل قاعدة البيانات الخاصة بك بمستوى عزل المعاملة \"READ COMMITTED\". هذا يمكن أن يسبب مشاكل عند تنفيذ عدة إجراءات بالتوازي.",
"Was not able to get transaction isolation level: %s" : "تعذّر العثور على مستوى عزل المعاملات transaction isolation level: %s",
@@ -390,7 +390,7 @@ OC.L10N.register(
"Update to {update}" : "التحديث إلى {update}",
"Remove" : "حذف",
"Disable" : "إيقاف",
- "Featured" : "مُميّزةٌ",
+ "Featured" : "مميزة",
"This app is supported via your current Nextcloud subscription." : "هذا التطبيق مدعوم من خلال اشتراكك الحالي في نكست كلود.",
"Featured apps are developed by and within the community. They offer central functionality and are ready for production use." : "تم تطوير التطبيقات المميزة من قبل المجتمع وداخله. و هي توفر وظائف مركزية وجاهزة للعمل في بيئة العمل الفعلية.",
"Community rating: {score}/5" : "تقييم المجتمع: {score}/5",
@@ -717,7 +717,7 @@ OC.L10N.register(
"Date of birth" : "تاريخ الميلاد",
"Full name" : "الاسم الكامل",
"Additional email" : "إيميل إضافي",
- "Fediverse (e.g. Mastodon)" : "الحساب على شبكات التواصل الاجتماعي \"فيدي فيرس\" Fedivers (مثل Mastodon)",
+ "Fediverse (e.g. Mastodon)" : "الحساب على شبكات التواصل الاجتماعي Fediverse (مثل Mastodon)",
"Headline" : "العنوان",
"Organisation" : "المؤسسة",
"Phone number" : "رقم الهاتف",
@@ -736,11 +736,11 @@ OC.L10N.register(
"Active apps" : "التطبيقات النشطة",
"Disabled apps" : "التطبيقات المعطلة",
"Updates" : "التحديثات",
- "App bundles" : "حِزَم التطبيقات App bundles",
- "Featured apps" : "تطبيقات مُميّزة",
+ "App bundles" : "حزم التطبيقات App bundles",
+ "Featured apps" : "تطبيقات مميزة",
"Supported apps" : "التطبيقات المدعومة",
"Show to everyone" : "عرض للكل",
- "Show to logged in accounts only" : "أظهِر للحسابات الداخلة فقط",
+ "Show to logged in accounts only" : "عرض للحسابات المسجل دخولها فقط",
"Hide" : "إخفاء",
"Download and enable" : "تنزيل وتمكين",
"Allow untested app" : "إسمَح للتطبيقات غير المُجازة untested app",
@@ -762,10 +762,10 @@ OC.L10N.register(
"Could not set group sorting" : "تعذّر تعيين كيفية ترتيب المجموعات",
"There were too many requests from your network. Retry later or contact your administrator if this is an error." : "الطلبات الموجودة أكثر من الحد المسموح به في الشبكة. أعد المحاولة أو اتصل بمشرف النظام.",
"Error" : "خطأ",
- "Account documentation" : "وثائق الحساب",
+ "Account documentation" : "وثائق المستخدم",
"Administration documentation" : "وثائق الإشراف",
- "Forum" : "المُنتَدَى",
- "Nextcloud help & privacy resources" : "مصادر للمساعدة و الخصوصية حول نكست كلاود",
+ "Forum" : "المنتدى",
+ "Nextcloud help & privacy resources" : "مصادر للمساعدة والخصوصية حول نكست كلاود",
"General documentation" : "وثائق عامة",
"Legal notice" : "ملاحظة قانونية",
"Privacy policy" : "سياسة الخصوصية",
@@ -863,7 +863,7 @@ OC.L10N.register(
"Add user to group" : "إضافة مستخدِمٍ إلى مجموعةٍ",
"Set user as admin for" : "تعيين المستخدم كمسؤول عن",
"Select user quota" : "تعيين حصة المستخدم",
- "Delete user" : "إحذِف المستخدم",
+ "Delete user" : "حذف المستخدم",
"Wipe all devices" : "مَحْو جميع الأجهزة",
"Disable user" : "إيقاف المستخدم",
"Enable user" : "تمكين المستخدم",
diff --git a/apps/settings/l10n/ar.json b/apps/settings/l10n/ar.json
index e3cc18d03ac..46a77e3c13f 100644
--- a/apps/settings/l10n/ar.json
+++ b/apps/settings/l10n/ar.json
@@ -122,7 +122,7 @@
"Security & setup warnings" : "تحذيرات الإعداد و الأمان",
"Background jobs" : "المهام في الخلفية Background jobs",
"Unlimited" : "غير محدود",
- "Verifying" : "التحقُّق",
+ "Verifying" : "التحقق",
"Allowed admin IP ranges" : "نطاقات عناوين IP المسموحة للمشرفين",
"Admin IP filtering isn’t applied." : "غربلة عناوين IP الخاصة بالمشرفين غير مُطبَّقة.",
"Configuration key \"%1$s\" expects an array (%2$s found). Admin IP range validation will not be applied." : "مفتاح التهيئة \"%1$s\" يتوقع مصفوفة (%2$s موجودة). غربلة عناوين IP الخاصة بالمشرفين غير مُطبَّقة.",
@@ -262,7 +262,7 @@
"Push service" : "خدمة الدفش push",
"Valid enterprise license" : "رخصة مؤسسية سارية",
"Free push service" : "خدمة الدفش push المجانية",
- "This is the unsupported community build of Nextcloud. Given the size of this instance, performance, reliability and scalability cannot be guaranteed. Push notifications are limited to avoid overloading our free service. Learn more about the benefits of Nextcloud Enterprise at {link}." : "هذه تجهيزة مجتمعية community build لنكست كلاود غير مدعومة. نظراً لحجم هذا الخادوم، لا يمكن ضمان الأداء و الموثوقية و قابلية التوسع. إشعارات الدفش Push notifications تمّ تقييد عددها لتجنب الضغط الزائد على خدمتنا المجانية. تعرّف على المزيد حول مزايا نسخة نسكت كلاود المؤسسية على الرابط: {link}.",
+ "This is the unsupported community build of Nextcloud. Given the size of this instance, performance, reliability and scalability cannot be guaranteed. Push notifications are limited to avoid overloading our free service. Learn more about the benefits of Nextcloud Enterprise at {link}." : "هذه تجهيزة مجتمعية community build لنكست كلاود غير مدعومة. نظراً لحجم هذا الخادم، لا يمكن ضمان الأداء و الموثوقية و قابلية التوسع. إشعارات الدفش Push notifications تمّ تقييد عددها لتجنب الضغط الزائد على خدمتنا المجانية. تعرّف على المزيد حول مزايا نسخة نسكت كلاود المؤسسية على الرابط: {link}.",
"Random generator" : "مُولِّد عشوائي",
"No suitable source for randomness found by PHP which is highly discouraged for security reasons." : "لم يتمكن PHP من العثور على أي مصدر مناسب للعشوائية randomness؛ الأمر الذي لا يُنصح به لأسباب أمنية.",
"Secure" : "آمِن",
@@ -302,7 +302,7 @@
"- %.1f GiB available in %s (Nextcloud temporary directory)" : "- %.1f غيغا بايت متاحة فثي %s (المجلد المؤقت لنكست كلاود)",
"Temporary directory is correctly configured:\n%s" : "المجلد المؤقت تمّ تكوينه بالشكل الصحيح:\n%s",
"This instance uses an S3 based object store as primary storage, and has enough space in the temporary directory.\n%s" : "يستخدم هذا الخادم مخزن العناصر object store المستند إلى S3 كمخزن أساسي، و لا زالت لديه سعة تخزين كافية في المجلد المؤقت.\n%s",
- "This instance uses an S3 based object store as primary storage. The uploaded files are stored temporarily on the server and thus it is recommended to have 50 GiB of free space available in the temp directory of PHP. To improve this please change the temporary directory in the php.ini or make more space available in that path. \nChecking the available space in the temporary path resulted in %.1f GiB instead of the recommended 50 GiB. Path: %s" : "يستخدم هذا الخادوم مخزن العناصر object store المستند إلى S3 كمخزن أساسي. الملفات التي تمّ رفعها يتم تخزينها مؤقتًا على الخادوم، و لهذا يُوصى بتوفير مساحة خالية تبلغ 50 غيغا بايت في الدليل المؤقت لـ PHP. لمزيد التحسين، يُرجى تغيير الدليل المؤقت في php.ini أو توفير مساحة أكبر في هذا المسار. \nنتيجة فحص المساحة المتوفرة في المسار المؤقت: %.1f جيجا بايت؛ بدلاً عن: 50 جيجا بايت الموصى بها. المسار: %s",
+ "This instance uses an S3 based object store as primary storage. The uploaded files are stored temporarily on the server and thus it is recommended to have 50 GiB of free space available in the temp directory of PHP. To improve this please change the temporary directory in the php.ini or make more space available in that path. \nChecking the available space in the temporary path resulted in %.1f GiB instead of the recommended 50 GiB. Path: %s" : "يستخدم هذا الخادم مخزن العناصر object store المستند إلى S3 كمخزن أساسي. الملفات التي تمّ رفعها يتم تخزينها مؤقتًا على الخادم، و لهذا يُوصى بتوفير مساحة خالية تبلغ 50 غيغا بايت في الدليل المؤقت لـ PHP. لمزيد التحسين، يُرجى تغيير الدليل المؤقت في php.ini أو توفير مساحة أكبر في هذا المسار. \nنتيجة فحص المساحة المتوفرة في المسار المؤقت: %.1f جيجا بايت؛ بدلاً عن: 50 جيجا بايت الموصى بها. المسار: %s",
"Database transaction isolation level" : "مستوى عزل معاملات قاعدة البيانات",
"Your database does not run with \"READ COMMITTED\" transaction isolation level. This can cause problems when multiple actions are executed in parallel." : "لا تعمل قاعدة البيانات الخاصة بك بمستوى عزل المعاملة \"READ COMMITTED\". هذا يمكن أن يسبب مشاكل عند تنفيذ عدة إجراءات بالتوازي.",
"Was not able to get transaction isolation level: %s" : "تعذّر العثور على مستوى عزل المعاملات transaction isolation level: %s",
@@ -388,7 +388,7 @@
"Update to {update}" : "التحديث إلى {update}",
"Remove" : "حذف",
"Disable" : "إيقاف",
- "Featured" : "مُميّزةٌ",
+ "Featured" : "مميزة",
"This app is supported via your current Nextcloud subscription." : "هذا التطبيق مدعوم من خلال اشتراكك الحالي في نكست كلود.",
"Featured apps are developed by and within the community. They offer central functionality and are ready for production use." : "تم تطوير التطبيقات المميزة من قبل المجتمع وداخله. و هي توفر وظائف مركزية وجاهزة للعمل في بيئة العمل الفعلية.",
"Community rating: {score}/5" : "تقييم المجتمع: {score}/5",
@@ -715,7 +715,7 @@
"Date of birth" : "تاريخ الميلاد",
"Full name" : "الاسم الكامل",
"Additional email" : "إيميل إضافي",
- "Fediverse (e.g. Mastodon)" : "الحساب على شبكات التواصل الاجتماعي \"فيدي فيرس\" Fedivers (مثل Mastodon)",
+ "Fediverse (e.g. Mastodon)" : "الحساب على شبكات التواصل الاجتماعي Fediverse (مثل Mastodon)",
"Headline" : "العنوان",
"Organisation" : "المؤسسة",
"Phone number" : "رقم الهاتف",
@@ -734,11 +734,11 @@
"Active apps" : "التطبيقات النشطة",
"Disabled apps" : "التطبيقات المعطلة",
"Updates" : "التحديثات",
- "App bundles" : "حِزَم التطبيقات App bundles",
- "Featured apps" : "تطبيقات مُميّزة",
+ "App bundles" : "حزم التطبيقات App bundles",
+ "Featured apps" : "تطبيقات مميزة",
"Supported apps" : "التطبيقات المدعومة",
"Show to everyone" : "عرض للكل",
- "Show to logged in accounts only" : "أظهِر للحسابات الداخلة فقط",
+ "Show to logged in accounts only" : "عرض للحسابات المسجل دخولها فقط",
"Hide" : "إخفاء",
"Download and enable" : "تنزيل وتمكين",
"Allow untested app" : "إسمَح للتطبيقات غير المُجازة untested app",
@@ -760,10 +760,10 @@
"Could not set group sorting" : "تعذّر تعيين كيفية ترتيب المجموعات",
"There were too many requests from your network. Retry later or contact your administrator if this is an error." : "الطلبات الموجودة أكثر من الحد المسموح به في الشبكة. أعد المحاولة أو اتصل بمشرف النظام.",
"Error" : "خطأ",
- "Account documentation" : "وثائق الحساب",
+ "Account documentation" : "وثائق المستخدم",
"Administration documentation" : "وثائق الإشراف",
- "Forum" : "المُنتَدَى",
- "Nextcloud help & privacy resources" : "مصادر للمساعدة و الخصوصية حول نكست كلاود",
+ "Forum" : "المنتدى",
+ "Nextcloud help & privacy resources" : "مصادر للمساعدة والخصوصية حول نكست كلاود",
"General documentation" : "وثائق عامة",
"Legal notice" : "ملاحظة قانونية",
"Privacy policy" : "سياسة الخصوصية",
@@ -861,7 +861,7 @@
"Add user to group" : "إضافة مستخدِمٍ إلى مجموعةٍ",
"Set user as admin for" : "تعيين المستخدم كمسؤول عن",
"Select user quota" : "تعيين حصة المستخدم",
- "Delete user" : "إحذِف المستخدم",
+ "Delete user" : "حذف المستخدم",
"Wipe all devices" : "مَحْو جميع الأجهزة",
"Disable user" : "إيقاف المستخدم",
"Enable user" : "تمكين المستخدم",
diff --git a/apps/settings/l10n/cs.js b/apps/settings/l10n/cs.js
index d010d2efa31..c8b68aa385d 100644
--- a/apps/settings/l10n/cs.js
+++ b/apps/settings/l10n/cs.js
@@ -185,13 +185,17 @@ OC.L10N.register(
"HTTPS access and URLs" : "HTTPS přístup a URL adresy",
"Accessing site insecurely via HTTP. You are strongly advised to set up your server to require HTTPS instead. Without it some important web functionality like \"copy to clipboard\" or \"service workers\" will not work!" : "Přistupujete přes přes nezabezpečený HTTP protokol. Důrazně doporučujeme nastavit svůj server tak, aby namísto toho vyžadoval HTTPS. Bez šifrovaného spojení nebudou k dispozici některé důležité funkce, jako např. „zkopírovat do schránky“ nebo tzv. „service workers“!",
"Accessing site insecurely via HTTP." : "Nezabezpečený přístup k webu přes HTTP.",
+ "You are accessing your instance over a secure connection, however your instance is generating insecure URLs. This likely means that your instance is behind a reverse proxy and the Nextcloud `overwrite*` config values are not set correctly." : "K vámi využívané instanci sice přistupujete přes zabezpečené připojení, nicméně tato vytváří nezabezpečené URL adresy. To s nejvyšší pravděpodobností znamená, že se nachází za reverzní proxy a hodnoty nastavení Nextcloud `overwrite*` nejsou nastavené správně.",
+ "Your instance is generating insecure URLs. If you access your instance over HTTPS, this likely means that your instance is behind a reverse proxy and the Nextcloud `overwrite*` config values are not set correctly." : "Vaše instance vytváří nezabezpečené URL adresy. Pokud k ní přistupujete přes HTTPs, pak to nejspíš znamená, že se nachází za reverzní proxy a hodnoty nastavení Nextcloud `overwrite*` nejsou nastavené správně.",
"You are accessing your instance over a secure connection, and your instance is generating secure URLs." : "K vámi využívané instanci přistupujete přes zabezpečené spojení a ta vytváří bezpečné URL adresy.",
"Internet connectivity" : "Připojení k internetu",
"Internet connectivity is disabled in configuration file." : "Připojení k Internetu je vypnuté v souboru s nastaveními.",
"This server has no working internet connection: Multiple endpoints could not be reached. This means that some of the features like mounting external storage, notifications about updates or installation of third-party apps will not work. Accessing files remotely and sending of notification emails might not work, either. Establish a connection from this server to the internet to enjoy all features." : "Tento server nemá funkční připojení k Internetu: Nedaří se připojit k vícero koncovým bodům. Některé moduly jako např. externí úložiště, upozorňování na dostupnost aktualizací, nebo instalace aplikací třetích stran kvůli tomu nebudou fungovat. Přístup k souborům z jiných míst a odesílání e-mailů s upozorněními také nemusí fungovat. Pokud chcete využívat všechny možnosti tohoto serveru, doporučujeme zprovoznit připojení k Internetu.",
"JavaScript modules support" : "Podpora JavaScript modulů",
+ "Unable to run check for JavaScript support. Please remedy or confirm manually if your webserver serves `.mjs` files using the JavaScript MIME type." : "Nepodařilo se spustit kontrolu ohledně podpory pro JavaScript. Prosím napravte to ručně nebo ručně potvrďte, zda váš webový server poskytuje soubory `.mjs` pomocí MIME typu JavaScript.",
"Your webserver does not serve `.mjs` files using the JavaScript MIME type. This will break some apps by preventing browsers from executing the JavaScript files. You should configure your webserver to serve `.mjs` files with either the `text/javascript` or `application/javascript` MIME type." : "Váš webový server neposkytuje soubory `.mjs` pomocí MIME typu JavaScript. Toto rozbíjí některé aplikace tím, že prohlížečům brání ve spouštění JavaScript souborů. Měli byste svůj webový server nastavit tak, aby soubory `.mjs` poskytoval s MIME typem buď `text/javascript` nebo `application/javascript` .",
"JavaScript source map support" : "Podpora JavaScript zdrojových map",
+ "Your webserver is not set up to serve `.js.map` files. Without these files, JavaScript Source Maps won't function properly, making it more challenging to troubleshoot and debug any issues that may arise." : "Váš webový server není nastavený tak, aby poskytoval soubory `.js.map`. Bez nich nebudou správně fungovat zdrojové mapy JavaScriptu, což ztíží řešení problémů a ladění jakéhokoli problému, který může nastat. ",
"Old server-side-encryption" : "Původní šifrování na straně serveru",
"Disabled" : "Vypnuto",
"The old server-side-encryption format is enabled. We recommend disabling this." : "Je zapnutý starý formát šifrování na straně serveru. Je doporučeno toto vypnout.",
@@ -199,9 +203,12 @@ OC.L10N.register(
"Server has no maintenance window start time configured. This means resource intensive daily background jobs will also be executed during your main usage time. We recommend to set it to a time of low usage, so users are less impacted by the load caused from these heavy tasks." : "Na serveru není nastavený žádný začátek doby údržby. To znamená, že každodenní na systémové prostředky náročné úlohy na pozadí budou spouštěné také v průběhu běžného uživatelského provozu. Doporučujeme nastavit na dobu nízkého využívání, čímž se vytěžování těmito náročnými úlohami bude méně dotýkat uživatelů.",
"Maintenance window to execute heavy background jobs is between {start}:00 UTC and {end}:00 UTC" : "Okno pro údržbu, ve kterém vykonávat náročné úlohy na pozadí je mezi {start}:00 UTC a {end}:00 UTC",
"Memcache" : "Memcache",
+ "Memcached is configured as distributed cache, but the wrong PHP module (\"memcache\") is installed. Please install the PHP module \"memcached\"." : "Coby distribuovaná mezipaměť je nastaveno Memcached, ale je nainstalován nesprávný PHP modul („memcache“). Nainstalujte správný PHP modul „memcached“.",
+ "Memcached is configured as distributed cache, but the PHP module \"memcached\" is not installed. Please install the PHP module \"memcached\"." : "Coby distribuovaná mezipaměť je nastaveno Memcached, ale potřebný PHP modul – „memcached“ – není nainstalován. Prosíme napravte to.",
"No memory cache has been configured. To enhance performance, please configure a memcache, if available." : "Nebylo nastaveno žádné ukládání do mezipaměti. Pokud chcete vylepšit výkon, nastavte memcache (pokud je k dispozici).",
"Configured" : "Nastaveno",
"Mimetype migrations available" : "Migraci mimetypů k dispozici",
+ "One or more mimetype migrations are available. Occasionally new mimetypes are added to better handle certain file types. Migrating the mimetypes take a long time on larger instances so this is not done automatically during upgrades. Use the command `occ maintenance:repair --include-expensive` to perform the migrations." : "Je k dispozici jedna nebo více migrací mimetypu. Občas jsou přidány nové mimetypy pro lepší obsloužení určitých typů souborů. Na velkých instancích zabere migrace mimetypů dlouhou dobu, proto se neprovádí automaticky při aktualizacích. Pro provedení migrací použijte příkaz `occ maintenance:repair --include-expensive`.",
"MySQL Unicode support" : "Podpora MySQL Unicode",
"You are not using MySQL" : "Nepoužíváte MySQL",
"MySQL is used as database and does support 4-byte characters" : "Jako databáze je používáno MySQL a nepodporuje 4bajtové znaky",
@@ -241,7 +248,11 @@ OC.L10N.register(
"OPcache is disabled. For better performance, it is recommended to apply \"opcache.enable=1\" to your PHP configuration." : "OPcache je vypnuté. Pro lepší výkon je doporučeno mít v nastavení PHP „opcache.enable=1“.",
"The shared memory based OPcache is disabled. For better performance, it is recommended to apply \"opcache.file_cache_only=0\" to your PHP configuration and use the file cache as second level cache only." : "OPcache založené na sdílené paměti je vypnuté. Pro lepší výkon je doporučeno mít v nastavení PHP „opcache.file_cache_only=0“ a používat souborovou mezipaměť až jako druhý stupeň mezipaměti.",
"OPcache is not working as it should, opcache_get_status() returns false, please check configuration." : "OPcache nefunguje, jak by mělo, opcache_get_status() vrací negaci (false) – zkontrolujte nastavení.",
+ "The maximum number of OPcache keys is nearly exceeded. To assure that all scripts can be kept in the cache, it is recommended to apply \"opcache.max_accelerated_files\" to your PHP configuration with a value higher than \"%s\"." : "Je téměř vyčerpán nejvyšší umožněný počet klíčů v OPcache. Aby bylo zajištěno, že se do mezipaměti vejdou veškeré skripty, je doporučeno přidat do nastavení PHP volbu „opcache.max_accelerated_files“ s hodnotou vyšší než „%s“.",
+ "The OPcache buffer is nearly full. To assure that all scripts can be hold in cache, it is recommended to apply \"opcache.memory_consumption\" to your PHP configuration with a value higher than \"%s\"." : "Vyrovnávací paměť OPcache je téměř zaplněná. Aby bylo zajištěno, že se do mezipaměti vejdou veškeré skripty, je doporučeno přidat do nastavení PHP volbu „opcache.memory_consumption“ s hodnotou vyšší než „%s“.",
+ "The OPcache interned strings buffer is nearly full. To assure that repeating strings can be effectively cached, it is recommended to apply \"opcache.interned_strings_buffer\" to your PHP configuration with a value higher than \"%s\"." : "Vnitřní vyrovnávací paměť OPcache pro řetězce je téměř zaplněná. Aby bylo zajištěno, že ukládání řetězců do mezipaměti bude skutečně prováděno, je doporučeno přidat do nastavení PHP volbu „opcache.interned_strings_buffer“ s hodnotou vyšší než „%s“.",
"OPcache is configured to remove code comments. With OPcache enabled, \"opcache.save_comments=1\" must be set for Nextcloud to function." : "OPcache je nastaveno tak, aby odebíralo komentáře kódu. Pokud je OPcache zapnuté, pak aby Nextcloud fungoval, je třeba, aby bylo nastaveno také „opcache.save_comments=1“.",
+ "Nextcloud is not allowed to use the OPcache API. With OPcache enabled, it is highly recommended to include all Nextcloud directories with \"opcache.restrict_api\" or unset this setting to disable OPcache API restrictions, to prevent errors during Nextcloud core or app upgrades." : "Pro Nextcloud nemáte zpřístupněno používání aplikačního program. rozhraní (API) OPcache. Pokud je OPcache zapnuté, je velmi doporučováno zahrnout veškeré složky s Nextcloud do „opcache.restrict_api“ nebo toto nastavení zrušit a vypnout tak omezení k OPcache API. Zabráníte tak chybám při aktualizacím jádra Nextcloud či aplikací.",
"Checking from CLI, OPcache checks have been skipped." : "Kontroluje se z CLI – kontroly OPcache byly přeskočeny.",
"The PHP OPcache module is not properly configured. %s." : "PHP modul OPcache není správně nastavený: %s.",
"Correctly configured" : "Správně nastaveno",
@@ -253,6 +264,7 @@ OC.L10N.register(
"Push service" : "Služba push",
"Valid enterprise license" : "Platná licence pro podniky",
"Free push service" : "Push služba zdarma",
+ "This is the unsupported community build of Nextcloud. Given the size of this instance, performance, reliability and scalability cannot be guaranteed. Push notifications are limited to avoid overloading our free service. Learn more about the benefits of Nextcloud Enterprise at {link}." : "Toto je nepodporované komunitní sestavení Nextcloud. Vzhledem k velikosti této instance, není garantován výkon, spolehlivost, ani škálovatelnost. Aby se zabránilo přetěžování našich služeb, které jsou zdarma, okamžitá oznámení jsou omezena. Zjistěte více o výhodách Nextcloud pro podniky na {link}.",
"Random generator" : "Generátor náhodnosti",
"No suitable source for randomness found by PHP which is highly discouraged for security reasons." : "PHP nenalezlo žádný použitelný zdroj náhodnosti a to je velmi nedoporučováno z bezpečnostních důvodů.",
"Secure" : "Zabezpečené",
@@ -260,21 +272,34 @@ OC.L10N.register(
"The read-only config has been enabled. This prevents setting some configurations via the web-interface. Furthermore, the file needs to be made writable manually for every update." : "Nastavení bylo přepnuto do režimu pouze pro čtení. To brání změnám prostřednictvím webového rozhraní. Dále při každé aktualizaci, je třeba soubor ručně zpřístupnit pro zápis.",
"Nextcloud configuration file is writable" : "Do souboru s nastaveními pro Nextcloud lze zapisovat",
"Scheduling objects table size" : "Velikost tabulky objektů plánování",
+ "You have more than %s rows in the scheduling objects table. Please run the expensive repair jobs via occ maintenance:repair --include-expensive." : "Máte více než %s řádků v tabulce objektů plánování. Prosím spusťte na výkon náročné opravné úlohy pomocí occ maintenance:repair --include-expensive.",
"Scheduling objects table size is within acceptable range." : "Velikost tabulky objektů plánování je v přijatelném rozsahu.",
"HTTP headers" : "HTTP hlavičky",
+ "- The `%1$s` HTTP header is not set to `%2$s`. Some features might not work correctly, as it is recommended to adjust this setting accordingly." : "- HTTP hlavička `%1$s` není nastavena na `%2$s`. To značí možné ohrožení bezpečnosti a soukromí a je doporučeno toto nastavení upravit.",
+ "- The `%1$s` HTTP header is not set to `%2$s`. This is a potential security or privacy risk, as it is recommended to adjust this setting accordingly." : "- HTTP hlavička `%1$s` není nastavena na `%2$s`. To značí možné ohrožení bezpečnosti a soukromí a je doporučeno toto nastavení upravit.",
+ "- The `%1$s` HTTP header does not contain `%2$s`. This is a potential security or privacy risk, as it is recommended to adjust this setting accordingly." : "- HTTP hlavička `%1$s` neobsahuje `%2$s`. To značí možné ohrožení bezpečnosti a soukromí a je doporučeno toto nastavení upravit.",
+ "- The `%1$s` HTTP header is not set to `%2$s`, `%3$s`, `%4$s`, `%5$s` or `%6$s`. This can leak referer information. See the {w3c-recommendation}." : "- HTTP hlavička `%1$s` není nastavena na `%2$s`, `%3$s`, `%4$s`, `%5$s` nebo `%6$s`. Kvůli tomu může uniknout referer údaj. Podrobnosti viz {w3c-recommendation}.",
+ "- The `Strict-Transport-Security` HTTP header is not set to at least `%d` seconds (current value: `%d`). For enhanced security, it is recommended to use a long HSTS policy." : "- HTTP hlavička `Strict-Transport-Security` není nastavena na hodnotu alespoň `%d` sekund (stávající hodnota: `%d`). Pro zlepšení zabezpečení je doporučeno použít dlouhou zásadu HSTS.",
"- The `Strict-Transport-Security` HTTP header is malformed: `%s`. For enhanced security, it is recommended to enable HSTS." : "- HTTP hlavička `Strict-Transport-Security` HTTP nemá správnou podobu: `%s`. Pro zlepšení zabezpečení je doporučeno zapnout HSTS.",
+ "- The `Strict-Transport-Security` HTTP header is not set (should be at least `%d` seconds). For enhanced security, it is recommended to enable HSTS." : "- HTTP hlavička `Strict-Transport-Security` není nastavena (mělo by být alespoň `%d` sekund). Pro zlepšení zabezpečení je doporučeno HSTS zapnout.",
"Some headers are not set correctly on your instance" : "Některé hlavičky nejsou ve vaší instanci správně nastaveny",
"Could not check that your web server serves security headers correctly. Please check manually." : "Nepodařilo se zkontrolovat že vámi využívaný webový server je správně nastaven tak, aby správně poskytoval hlavičky zabezpečení. Zkontrolujte to ručně.",
"Could not check that your web server serves security headers correctly, unable to query `%s`" : "Nepodařilo se zkontrolovat zda vámi využívaný webový server je správně nastaven tak, aby správně poskytoval hlavičky zabezpečení – nepodařil ose dotázat `%s`",
"Your server is correctly configured to send security headers." : "Server je správně nastaven pro odesílání hlaviček zabezpečení",
"Database version" : "Verze databáze",
+ "MariaDB version 10.3 detected, this version is end-of-life and only supported as part of Ubuntu 20.04. MariaDB >=%1$s and <=%2$s is suggested for best performance, stability and functionality with this version of Nextcloud." : "Zjištěna verze MariaDB 10.3 – tuto verzi už její vývojáři nepodporují (existuje ještě podpora coby součásti Ubuntu 20.04) Pro nejlepší výkon, stabilitu a funkčnost s touto verzí Nextcloud je doporučeno MariaDB >=%1$s a <=%2$s.",
+ "MariaDB version \"%1$s\" detected. MariaDB >=%2$s and <=%3$s is suggested for best performance, stability and functionality with this version of Nextcloud." : "Zjištěna verze MariaDB „%1$s“. Pro nejlepší výkon, stabilitu a funkčnost s touto verzí Nextcloud je doporučeno MariaDB >=%2$s a <=%3$s.",
+ "MySQL version \"%1$s\" detected. MySQL >=%2$s and <=%3$s is suggested for best performance, stability and functionality with this version of Nextcloud." : "Zjištěna verze MySQL „%1$s“. Pro nejlepší výkon, stabilitu a funkčnost s touto verzí Nextcloud je doporučeno MySQL >=%2$s a <=%3$s.",
+ "PostgreSQL version \"%s\" detected. PostgreSQL >=12 and <=16 is suggested for best performance, stability and functionality with this version of Nextcloud." : "Zjištěna verze PostgreSQL „%s“. Pro nejlepší výkon, stabilitu a funkčnost s touto verzí Nextcloud je doporučeno PostgreSQL >=12 a <=16.",
"SQLite is currently being used as the backend database. For larger installations we recommend that you switch to a different database backend. This is particularly recommended when using the desktop client for file synchronisation. To migrate to another database use the command line tool: \"occ db:convert-type\"." : "Coby databáze je nyní využíváno SQLite. V případě vícero uživatelů a/nebo množství dat doporučujeme přejít na robustnější databázi. Zejména je toto doporučeno v případě synchronizace souborů prostřednictvím klienta pro počítač. Na jinou databázi je možné přejít pomocí nástroje pro příkazový řádek: „occ db:convert-type“.",
"Unknown database platform" : "Neznámá databázová platforma",
"Architecture" : "Architektura",
"64-bit" : "64bit",
"It seems like you are running a 32-bit PHP version. Nextcloud needs 64-bit to run well. Please upgrade your OS and PHP to 64-bit!" : "Zdá se, že provozujete 32bitovou verzi PHP. Aby správně fungoval, potřebuje Nextcloud 64bit. Přejděte na 64bit instalaci operačního systému a PHP!",
"Temporary space available" : "Dočasný prostor k dispozici",
+ "Error while checking the temporary PHP path - it was not properly set to a directory. Returned value: %s" : "Chyba při kontrole popisu umístění dočasných souborů PHP – nebylo správně nastaveno na složku. Vrácená hodnota: %s",
"The PHP function \"disk_free_space\" is disabled, which prevents the check for enough space in the temporary directories." : "PHP funkce „disk_free_space“ je vypnutá, což brání v kontrolách zda je k dispozici dostatek místa ve složkách pro dočasná data.",
+ "Error while checking the available disk space of temporary PHP path or no free disk space returned. Temporary path: %s" : "Chyba při zjišťování dostupného prostoru na úložišti v popisu umístění dočasných souborů pro PHP, případně vráceno že není žádné volné místo. Popis umístění pro dočasné soubory: %s",
"- %.1f GiB available in %s (PHP temporary directory)" : "- %.1f GiB k dispozici v %s (složka pro dočasná data PHP)",
"- %.1f GiB available in %s (Nextcloud temporary directory)" : "- %.1f GiB k dispozici v %s (složka pro dočasná data Nextcloud)",
"Temporary directory is correctly configured:\n%s" : "Dočasná složka je správně nastavena:\n%s",
@@ -289,6 +314,8 @@ OC.L10N.register(
"Your web server is not properly set up to resolve `.well-known` URLs, failed on:\n`%s`" : "Tento webový server není správně nastaven pro překlad `.well-known` URL adres. Chybující URL je `%s`.",
"Your server is correctly configured to serve `.well-known` URLs." : "Server je správně nastaven poskytování `.well-known` URL adres.",
"Font file loading" : "Načítání souboru s písmem",
+ "Could not check for {extension} loading support. Please check manually if your webserver serves `.{extension}` files." : "Nepodařilo se zkontrolovat podporu načítání {extension}. Zkontrolujte ručně, zda vámi využívaný webový server poskytuje `.{extension}` soubory.",
+ "Your web server is not properly set up to deliver .{extension} files. This is typically an issue with the Nginx configuration. For Nextcloud 15 it needs an adjustment to also deliver .{extension} files. Compare your Nginx configuration to the recommended configuration in our documentation." : "Váš webový server není správně nastaven k doručování .{extension} souborů. To je obvykle chyba v nastavení Nginx. Nextcloud 15 také potřebuje úpravu k doručování .{extension} souborů. Porovnejte své nastavení Nginx s doporučeným nastavením v naší dokumentaci.",
"Profile information" : "Informace o profilu",
"Profile picture, full name, email, phone number, address, website, Twitter, organisation, role, headline, biography, and whether your profile is enabled" : "Profilový obrázek, celé jméno, e-mailová adresa, telefonní číslo, adresa, webové stránky, Twitter, organizace, role, úvod, životopis a to, zda je profil zapnutý",
"Nextcloud settings" : "Nastavení Nextcloud",
@@ -860,6 +887,7 @@ OC.L10N.register(
"SMTP Username" : "SMTP uživatelské jméno ",
"Follow us on Twitter" : "Následujte nás na Twitteru",
"To allow this check to run you have to make sure that your webserver can connect to itself. Therefor it must be able to resolve and connect to at least one its `trusted_domains` or the `overwrite.cli.url`." : "Aby bylo možné možné povolit spuštění této kontroly je třeba zajistit, aby se webový server mohl připojit sám na sebe. Proto je třeba, aby bylo možné přeložit na IP adresu a připojit se k alespoň jedné z `trusted_domains` nebo `overwrite.cli.url`.",
+ "Could not check for JavaScript support via any of your `trusted_domains` nor `overwrite.cli.url`. This may be the result of a server-side DNS mismatch or outbound firewall rule. Please check manually if your webserver serves `.mjs` files using the JavaScript MIME type." : "Nepodařilo se zkontrolovat ohledně podpory pro JavaScript prostřednictvím libovolné z vašich `trusted_domains` ani `overwrite.cli.url`. Může se jednat o důsledek neshody DNS na straně serveru nebo odchozího pravidla na bráně firewall. Ručně zkontrolujte zda váš webový server poskytuje `.mjs` soubory pomocí MIME typu JavaScript.",
"Active accounts" : "Aktivní účty",
"To allow this check to run you have to make sure that your Web server can connect to itself. Therefore it must be able to resolve and connect to at least one of its `trusted_domains` or the `overwrite.cli.url`. This failure may be the result of a server-side DNS mismatch or outbound firewall rule." : "Aby bylo možné možné povolit spuštění této kontroly je třeba zajistit, aby se webový server mohl připojit sám na sebe. Proto je třeba, aby bylo možné přeložit na IP adresu a připojit se k alespoň jedné z `trusted_domains` nebo `overwrite.cli.url`. Tento nezdar může být výsledkem neshody DNS na serveru nebo odchozího pravidla na bráně firewall."
},
diff --git a/apps/settings/l10n/cs.json b/apps/settings/l10n/cs.json
index 5fbb562865e..58039ffbeff 100644
--- a/apps/settings/l10n/cs.json
+++ b/apps/settings/l10n/cs.json
@@ -183,13 +183,17 @@
"HTTPS access and URLs" : "HTTPS přístup a URL adresy",
"Accessing site insecurely via HTTP. You are strongly advised to set up your server to require HTTPS instead. Without it some important web functionality like \"copy to clipboard\" or \"service workers\" will not work!" : "Přistupujete přes přes nezabezpečený HTTP protokol. Důrazně doporučujeme nastavit svůj server tak, aby namísto toho vyžadoval HTTPS. Bez šifrovaného spojení nebudou k dispozici některé důležité funkce, jako např. „zkopírovat do schránky“ nebo tzv. „service workers“!",
"Accessing site insecurely via HTTP." : "Nezabezpečený přístup k webu přes HTTP.",
+ "You are accessing your instance over a secure connection, however your instance is generating insecure URLs. This likely means that your instance is behind a reverse proxy and the Nextcloud `overwrite*` config values are not set correctly." : "K vámi využívané instanci sice přistupujete přes zabezpečené připojení, nicméně tato vytváří nezabezpečené URL adresy. To s nejvyšší pravděpodobností znamená, že se nachází za reverzní proxy a hodnoty nastavení Nextcloud `overwrite*` nejsou nastavené správně.",
+ "Your instance is generating insecure URLs. If you access your instance over HTTPS, this likely means that your instance is behind a reverse proxy and the Nextcloud `overwrite*` config values are not set correctly." : "Vaše instance vytváří nezabezpečené URL adresy. Pokud k ní přistupujete přes HTTPs, pak to nejspíš znamená, že se nachází za reverzní proxy a hodnoty nastavení Nextcloud `overwrite*` nejsou nastavené správně.",
"You are accessing your instance over a secure connection, and your instance is generating secure URLs." : "K vámi využívané instanci přistupujete přes zabezpečené spojení a ta vytváří bezpečné URL adresy.",
"Internet connectivity" : "Připojení k internetu",
"Internet connectivity is disabled in configuration file." : "Připojení k Internetu je vypnuté v souboru s nastaveními.",
"This server has no working internet connection: Multiple endpoints could not be reached. This means that some of the features like mounting external storage, notifications about updates or installation of third-party apps will not work. Accessing files remotely and sending of notification emails might not work, either. Establish a connection from this server to the internet to enjoy all features." : "Tento server nemá funkční připojení k Internetu: Nedaří se připojit k vícero koncovým bodům. Některé moduly jako např. externí úložiště, upozorňování na dostupnost aktualizací, nebo instalace aplikací třetích stran kvůli tomu nebudou fungovat. Přístup k souborům z jiných míst a odesílání e-mailů s upozorněními také nemusí fungovat. Pokud chcete využívat všechny možnosti tohoto serveru, doporučujeme zprovoznit připojení k Internetu.",
"JavaScript modules support" : "Podpora JavaScript modulů",
+ "Unable to run check for JavaScript support. Please remedy or confirm manually if your webserver serves `.mjs` files using the JavaScript MIME type." : "Nepodařilo se spustit kontrolu ohledně podpory pro JavaScript. Prosím napravte to ručně nebo ručně potvrďte, zda váš webový server poskytuje soubory `.mjs` pomocí MIME typu JavaScript.",
"Your webserver does not serve `.mjs` files using the JavaScript MIME type. This will break some apps by preventing browsers from executing the JavaScript files. You should configure your webserver to serve `.mjs` files with either the `text/javascript` or `application/javascript` MIME type." : "Váš webový server neposkytuje soubory `.mjs` pomocí MIME typu JavaScript. Toto rozbíjí některé aplikace tím, že prohlížečům brání ve spouštění JavaScript souborů. Měli byste svůj webový server nastavit tak, aby soubory `.mjs` poskytoval s MIME typem buď `text/javascript` nebo `application/javascript` .",
"JavaScript source map support" : "Podpora JavaScript zdrojových map",
+ "Your webserver is not set up to serve `.js.map` files. Without these files, JavaScript Source Maps won't function properly, making it more challenging to troubleshoot and debug any issues that may arise." : "Váš webový server není nastavený tak, aby poskytoval soubory `.js.map`. Bez nich nebudou správně fungovat zdrojové mapy JavaScriptu, což ztíží řešení problémů a ladění jakéhokoli problému, který může nastat. ",
"Old server-side-encryption" : "Původní šifrování na straně serveru",
"Disabled" : "Vypnuto",
"The old server-side-encryption format is enabled. We recommend disabling this." : "Je zapnutý starý formát šifrování na straně serveru. Je doporučeno toto vypnout.",
@@ -197,9 +201,12 @@
"Server has no maintenance window start time configured. This means resource intensive daily background jobs will also be executed during your main usage time. We recommend to set it to a time of low usage, so users are less impacted by the load caused from these heavy tasks." : "Na serveru není nastavený žádný začátek doby údržby. To znamená, že každodenní na systémové prostředky náročné úlohy na pozadí budou spouštěné také v průběhu běžného uživatelského provozu. Doporučujeme nastavit na dobu nízkého využívání, čímž se vytěžování těmito náročnými úlohami bude méně dotýkat uživatelů.",
"Maintenance window to execute heavy background jobs is between {start}:00 UTC and {end}:00 UTC" : "Okno pro údržbu, ve kterém vykonávat náročné úlohy na pozadí je mezi {start}:00 UTC a {end}:00 UTC",
"Memcache" : "Memcache",
+ "Memcached is configured as distributed cache, but the wrong PHP module (\"memcache\") is installed. Please install the PHP module \"memcached\"." : "Coby distribuovaná mezipaměť je nastaveno Memcached, ale je nainstalován nesprávný PHP modul („memcache“). Nainstalujte správný PHP modul „memcached“.",
+ "Memcached is configured as distributed cache, but the PHP module \"memcached\" is not installed. Please install the PHP module \"memcached\"." : "Coby distribuovaná mezipaměť je nastaveno Memcached, ale potřebný PHP modul – „memcached“ – není nainstalován. Prosíme napravte to.",
"No memory cache has been configured. To enhance performance, please configure a memcache, if available." : "Nebylo nastaveno žádné ukládání do mezipaměti. Pokud chcete vylepšit výkon, nastavte memcache (pokud je k dispozici).",
"Configured" : "Nastaveno",
"Mimetype migrations available" : "Migraci mimetypů k dispozici",
+ "One or more mimetype migrations are available. Occasionally new mimetypes are added to better handle certain file types. Migrating the mimetypes take a long time on larger instances so this is not done automatically during upgrades. Use the command `occ maintenance:repair --include-expensive` to perform the migrations." : "Je k dispozici jedna nebo více migrací mimetypu. Občas jsou přidány nové mimetypy pro lepší obsloužení určitých typů souborů. Na velkých instancích zabere migrace mimetypů dlouhou dobu, proto se neprovádí automaticky při aktualizacích. Pro provedení migrací použijte příkaz `occ maintenance:repair --include-expensive`.",
"MySQL Unicode support" : "Podpora MySQL Unicode",
"You are not using MySQL" : "Nepoužíváte MySQL",
"MySQL is used as database and does support 4-byte characters" : "Jako databáze je používáno MySQL a nepodporuje 4bajtové znaky",
@@ -239,7 +246,11 @@
"OPcache is disabled. For better performance, it is recommended to apply \"opcache.enable=1\" to your PHP configuration." : "OPcache je vypnuté. Pro lepší výkon je doporučeno mít v nastavení PHP „opcache.enable=1“.",
"The shared memory based OPcache is disabled. For better performance, it is recommended to apply \"opcache.file_cache_only=0\" to your PHP configuration and use the file cache as second level cache only." : "OPcache založené na sdílené paměti je vypnuté. Pro lepší výkon je doporučeno mít v nastavení PHP „opcache.file_cache_only=0“ a používat souborovou mezipaměť až jako druhý stupeň mezipaměti.",
"OPcache is not working as it should, opcache_get_status() returns false, please check configuration." : "OPcache nefunguje, jak by mělo, opcache_get_status() vrací negaci (false) – zkontrolujte nastavení.",
+ "The maximum number of OPcache keys is nearly exceeded. To assure that all scripts can be kept in the cache, it is recommended to apply \"opcache.max_accelerated_files\" to your PHP configuration with a value higher than \"%s\"." : "Je téměř vyčerpán nejvyšší umožněný počet klíčů v OPcache. Aby bylo zajištěno, že se do mezipaměti vejdou veškeré skripty, je doporučeno přidat do nastavení PHP volbu „opcache.max_accelerated_files“ s hodnotou vyšší než „%s“.",
+ "The OPcache buffer is nearly full. To assure that all scripts can be hold in cache, it is recommended to apply \"opcache.memory_consumption\" to your PHP configuration with a value higher than \"%s\"." : "Vyrovnávací paměť OPcache je téměř zaplněná. Aby bylo zajištěno, že se do mezipaměti vejdou veškeré skripty, je doporučeno přidat do nastavení PHP volbu „opcache.memory_consumption“ s hodnotou vyšší než „%s“.",
+ "The OPcache interned strings buffer is nearly full. To assure that repeating strings can be effectively cached, it is recommended to apply \"opcache.interned_strings_buffer\" to your PHP configuration with a value higher than \"%s\"." : "Vnitřní vyrovnávací paměť OPcache pro řetězce je téměř zaplněná. Aby bylo zajištěno, že ukládání řetězců do mezipaměti bude skutečně prováděno, je doporučeno přidat do nastavení PHP volbu „opcache.interned_strings_buffer“ s hodnotou vyšší než „%s“.",
"OPcache is configured to remove code comments. With OPcache enabled, \"opcache.save_comments=1\" must be set for Nextcloud to function." : "OPcache je nastaveno tak, aby odebíralo komentáře kódu. Pokud je OPcache zapnuté, pak aby Nextcloud fungoval, je třeba, aby bylo nastaveno také „opcache.save_comments=1“.",
+ "Nextcloud is not allowed to use the OPcache API. With OPcache enabled, it is highly recommended to include all Nextcloud directories with \"opcache.restrict_api\" or unset this setting to disable OPcache API restrictions, to prevent errors during Nextcloud core or app upgrades." : "Pro Nextcloud nemáte zpřístupněno používání aplikačního program. rozhraní (API) OPcache. Pokud je OPcache zapnuté, je velmi doporučováno zahrnout veškeré složky s Nextcloud do „opcache.restrict_api“ nebo toto nastavení zrušit a vypnout tak omezení k OPcache API. Zabráníte tak chybám při aktualizacím jádra Nextcloud či aplikací.",
"Checking from CLI, OPcache checks have been skipped." : "Kontroluje se z CLI – kontroly OPcache byly přeskočeny.",
"The PHP OPcache module is not properly configured. %s." : "PHP modul OPcache není správně nastavený: %s.",
"Correctly configured" : "Správně nastaveno",
@@ -251,6 +262,7 @@
"Push service" : "Služba push",
"Valid enterprise license" : "Platná licence pro podniky",
"Free push service" : "Push služba zdarma",
+ "This is the unsupported community build of Nextcloud. Given the size of this instance, performance, reliability and scalability cannot be guaranteed. Push notifications are limited to avoid overloading our free service. Learn more about the benefits of Nextcloud Enterprise at {link}." : "Toto je nepodporované komunitní sestavení Nextcloud. Vzhledem k velikosti této instance, není garantován výkon, spolehlivost, ani škálovatelnost. Aby se zabránilo přetěžování našich služeb, které jsou zdarma, okamžitá oznámení jsou omezena. Zjistěte více o výhodách Nextcloud pro podniky na {link}.",
"Random generator" : "Generátor náhodnosti",
"No suitable source for randomness found by PHP which is highly discouraged for security reasons." : "PHP nenalezlo žádný použitelný zdroj náhodnosti a to je velmi nedoporučováno z bezpečnostních důvodů.",
"Secure" : "Zabezpečené",
@@ -258,21 +270,34 @@
"The read-only config has been enabled. This prevents setting some configurations via the web-interface. Furthermore, the file needs to be made writable manually for every update." : "Nastavení bylo přepnuto do režimu pouze pro čtení. To brání změnám prostřednictvím webového rozhraní. Dále při každé aktualizaci, je třeba soubor ručně zpřístupnit pro zápis.",
"Nextcloud configuration file is writable" : "Do souboru s nastaveními pro Nextcloud lze zapisovat",
"Scheduling objects table size" : "Velikost tabulky objektů plánování",
+ "You have more than %s rows in the scheduling objects table. Please run the expensive repair jobs via occ maintenance:repair --include-expensive." : "Máte více než %s řádků v tabulce objektů plánování. Prosím spusťte na výkon náročné opravné úlohy pomocí occ maintenance:repair --include-expensive.",
"Scheduling objects table size is within acceptable range." : "Velikost tabulky objektů plánování je v přijatelném rozsahu.",
"HTTP headers" : "HTTP hlavičky",
+ "- The `%1$s` HTTP header is not set to `%2$s`. Some features might not work correctly, as it is recommended to adjust this setting accordingly." : "- HTTP hlavička `%1$s` není nastavena na `%2$s`. To značí možné ohrožení bezpečnosti a soukromí a je doporučeno toto nastavení upravit.",
+ "- The `%1$s` HTTP header is not set to `%2$s`. This is a potential security or privacy risk, as it is recommended to adjust this setting accordingly." : "- HTTP hlavička `%1$s` není nastavena na `%2$s`. To značí možné ohrožení bezpečnosti a soukromí a je doporučeno toto nastavení upravit.",
+ "- The `%1$s` HTTP header does not contain `%2$s`. This is a potential security or privacy risk, as it is recommended to adjust this setting accordingly." : "- HTTP hlavička `%1$s` neobsahuje `%2$s`. To značí možné ohrožení bezpečnosti a soukromí a je doporučeno toto nastavení upravit.",
+ "- The `%1$s` HTTP header is not set to `%2$s`, `%3$s`, `%4$s`, `%5$s` or `%6$s`. This can leak referer information. See the {w3c-recommendation}." : "- HTTP hlavička `%1$s` není nastavena na `%2$s`, `%3$s`, `%4$s`, `%5$s` nebo `%6$s`. Kvůli tomu může uniknout referer údaj. Podrobnosti viz {w3c-recommendation}.",
+ "- The `Strict-Transport-Security` HTTP header is not set to at least `%d` seconds (current value: `%d`). For enhanced security, it is recommended to use a long HSTS policy." : "- HTTP hlavička `Strict-Transport-Security` není nastavena na hodnotu alespoň `%d` sekund (stávající hodnota: `%d`). Pro zlepšení zabezpečení je doporučeno použít dlouhou zásadu HSTS.",
"- The `Strict-Transport-Security` HTTP header is malformed: `%s`. For enhanced security, it is recommended to enable HSTS." : "- HTTP hlavička `Strict-Transport-Security` HTTP nemá správnou podobu: `%s`. Pro zlepšení zabezpečení je doporučeno zapnout HSTS.",
+ "- The `Strict-Transport-Security` HTTP header is not set (should be at least `%d` seconds). For enhanced security, it is recommended to enable HSTS." : "- HTTP hlavička `Strict-Transport-Security` není nastavena (mělo by být alespoň `%d` sekund). Pro zlepšení zabezpečení je doporučeno HSTS zapnout.",
"Some headers are not set correctly on your instance" : "Některé hlavičky nejsou ve vaší instanci správně nastaveny",
"Could not check that your web server serves security headers correctly. Please check manually." : "Nepodařilo se zkontrolovat že vámi využívaný webový server je správně nastaven tak, aby správně poskytoval hlavičky zabezpečení. Zkontrolujte to ručně.",
"Could not check that your web server serves security headers correctly, unable to query `%s`" : "Nepodařilo se zkontrolovat zda vámi využívaný webový server je správně nastaven tak, aby správně poskytoval hlavičky zabezpečení – nepodařil ose dotázat `%s`",
"Your server is correctly configured to send security headers." : "Server je správně nastaven pro odesílání hlaviček zabezpečení",
"Database version" : "Verze databáze",
+ "MariaDB version 10.3 detected, this version is end-of-life and only supported as part of Ubuntu 20.04. MariaDB >=%1$s and <=%2$s is suggested for best performance, stability and functionality with this version of Nextcloud." : "Zjištěna verze MariaDB 10.3 – tuto verzi už její vývojáři nepodporují (existuje ještě podpora coby součásti Ubuntu 20.04) Pro nejlepší výkon, stabilitu a funkčnost s touto verzí Nextcloud je doporučeno MariaDB >=%1$s a <=%2$s.",
+ "MariaDB version \"%1$s\" detected. MariaDB >=%2$s and <=%3$s is suggested for best performance, stability and functionality with this version of Nextcloud." : "Zjištěna verze MariaDB „%1$s“. Pro nejlepší výkon, stabilitu a funkčnost s touto verzí Nextcloud je doporučeno MariaDB >=%2$s a <=%3$s.",
+ "MySQL version \"%1$s\" detected. MySQL >=%2$s and <=%3$s is suggested for best performance, stability and functionality with this version of Nextcloud." : "Zjištěna verze MySQL „%1$s“. Pro nejlepší výkon, stabilitu a funkčnost s touto verzí Nextcloud je doporučeno MySQL >=%2$s a <=%3$s.",
+ "PostgreSQL version \"%s\" detected. PostgreSQL >=12 and <=16 is suggested for best performance, stability and functionality with this version of Nextcloud." : "Zjištěna verze PostgreSQL „%s“. Pro nejlepší výkon, stabilitu a funkčnost s touto verzí Nextcloud je doporučeno PostgreSQL >=12 a <=16.",
"SQLite is currently being used as the backend database. For larger installations we recommend that you switch to a different database backend. This is particularly recommended when using the desktop client for file synchronisation. To migrate to another database use the command line tool: \"occ db:convert-type\"." : "Coby databáze je nyní využíváno SQLite. V případě vícero uživatelů a/nebo množství dat doporučujeme přejít na robustnější databázi. Zejména je toto doporučeno v případě synchronizace souborů prostřednictvím klienta pro počítač. Na jinou databázi je možné přejít pomocí nástroje pro příkazový řádek: „occ db:convert-type“.",
"Unknown database platform" : "Neznámá databázová platforma",
"Architecture" : "Architektura",
"64-bit" : "64bit",
"It seems like you are running a 32-bit PHP version. Nextcloud needs 64-bit to run well. Please upgrade your OS and PHP to 64-bit!" : "Zdá se, že provozujete 32bitovou verzi PHP. Aby správně fungoval, potřebuje Nextcloud 64bit. Přejděte na 64bit instalaci operačního systému a PHP!",
"Temporary space available" : "Dočasný prostor k dispozici",
+ "Error while checking the temporary PHP path - it was not properly set to a directory. Returned value: %s" : "Chyba při kontrole popisu umístění dočasných souborů PHP – nebylo správně nastaveno na složku. Vrácená hodnota: %s",
"The PHP function \"disk_free_space\" is disabled, which prevents the check for enough space in the temporary directories." : "PHP funkce „disk_free_space“ je vypnutá, což brání v kontrolách zda je k dispozici dostatek místa ve složkách pro dočasná data.",
+ "Error while checking the available disk space of temporary PHP path or no free disk space returned. Temporary path: %s" : "Chyba při zjišťování dostupného prostoru na úložišti v popisu umístění dočasných souborů pro PHP, případně vráceno že není žádné volné místo. Popis umístění pro dočasné soubory: %s",
"- %.1f GiB available in %s (PHP temporary directory)" : "- %.1f GiB k dispozici v %s (složka pro dočasná data PHP)",
"- %.1f GiB available in %s (Nextcloud temporary directory)" : "- %.1f GiB k dispozici v %s (složka pro dočasná data Nextcloud)",
"Temporary directory is correctly configured:\n%s" : "Dočasná složka je správně nastavena:\n%s",
@@ -287,6 +312,8 @@
"Your web server is not properly set up to resolve `.well-known` URLs, failed on:\n`%s`" : "Tento webový server není správně nastaven pro překlad `.well-known` URL adres. Chybující URL je `%s`.",
"Your server is correctly configured to serve `.well-known` URLs." : "Server je správně nastaven poskytování `.well-known` URL adres.",
"Font file loading" : "Načítání souboru s písmem",
+ "Could not check for {extension} loading support. Please check manually if your webserver serves `.{extension}` files." : "Nepodařilo se zkontrolovat podporu načítání {extension}. Zkontrolujte ručně, zda vámi využívaný webový server poskytuje `.{extension}` soubory.",
+ "Your web server is not properly set up to deliver .{extension} files. This is typically an issue with the Nginx configuration. For Nextcloud 15 it needs an adjustment to also deliver .{extension} files. Compare your Nginx configuration to the recommended configuration in our documentation." : "Váš webový server není správně nastaven k doručování .{extension} souborů. To je obvykle chyba v nastavení Nginx. Nextcloud 15 také potřebuje úpravu k doručování .{extension} souborů. Porovnejte své nastavení Nginx s doporučeným nastavením v naší dokumentaci.",
"Profile information" : "Informace o profilu",
"Profile picture, full name, email, phone number, address, website, Twitter, organisation, role, headline, biography, and whether your profile is enabled" : "Profilový obrázek, celé jméno, e-mailová adresa, telefonní číslo, adresa, webové stránky, Twitter, organizace, role, úvod, životopis a to, zda je profil zapnutý",
"Nextcloud settings" : "Nastavení Nextcloud",
@@ -858,6 +885,7 @@
"SMTP Username" : "SMTP uživatelské jméno ",
"Follow us on Twitter" : "Následujte nás na Twitteru",
"To allow this check to run you have to make sure that your webserver can connect to itself. Therefor it must be able to resolve and connect to at least one its `trusted_domains` or the `overwrite.cli.url`." : "Aby bylo možné možné povolit spuštění této kontroly je třeba zajistit, aby se webový server mohl připojit sám na sebe. Proto je třeba, aby bylo možné přeložit na IP adresu a připojit se k alespoň jedné z `trusted_domains` nebo `overwrite.cli.url`.",
+ "Could not check for JavaScript support via any of your `trusted_domains` nor `overwrite.cli.url`. This may be the result of a server-side DNS mismatch or outbound firewall rule. Please check manually if your webserver serves `.mjs` files using the JavaScript MIME type." : "Nepodařilo se zkontrolovat ohledně podpory pro JavaScript prostřednictvím libovolné z vašich `trusted_domains` ani `overwrite.cli.url`. Může se jednat o důsledek neshody DNS na straně serveru nebo odchozího pravidla na bráně firewall. Ručně zkontrolujte zda váš webový server poskytuje `.mjs` soubory pomocí MIME typu JavaScript.",
"Active accounts" : "Aktivní účty",
"To allow this check to run you have to make sure that your Web server can connect to itself. Therefore it must be able to resolve and connect to at least one of its `trusted_domains` or the `overwrite.cli.url`. This failure may be the result of a server-side DNS mismatch or outbound firewall rule." : "Aby bylo možné možné povolit spuštění této kontroly je třeba zajistit, aby se webový server mohl připojit sám na sebe. Proto je třeba, aby bylo možné přeložit na IP adresu a připojit se k alespoň jedné z `trusted_domains` nebo `overwrite.cli.url`. Tento nezdar může být výsledkem neshody DNS na serveru nebo odchozího pravidla na bráně firewall."
},"pluralForm" :"nplurals=4; plural=(n == 1 && n % 1 == 0) ? 0 : (n >= 2 && n <= 4 && n % 1 == 0) ? 1: (n % 1 != 0 ) ? 2 : 3;"
diff --git a/apps/settings/l10n/lt_LT.js b/apps/settings/l10n/lt_LT.js
index c7a99df19b4..7e058467ce2 100644
--- a/apps/settings/l10n/lt_LT.js
+++ b/apps/settings/l10n/lt_LT.js
@@ -234,6 +234,7 @@ OC.L10N.register(
"Change password" : "Pakeisti slaptažodį",
"Your profile picture" : "Jūsų profilio paveikslėlis",
"Remove profile picture" : "Šalinti profilio paveikslėlį",
+ "The file must be a PNG or JPG" : "Failas privalo būti PNG ar JPG tipo",
"Picture provided by original account" : "Paveikslėlis gautas iš pagrindinės paskyros",
"Set as profile picture" : "Nustatyti kaip profilio paveikslėlį",
"Please note that it can take up to 24 hours for your profile picture to be updated everywhere." : "Turėkite omenyje, kad gali praeiti iki 24 valandų, kol jūsų profilis bus visur atnaujintas.",
diff --git a/apps/settings/l10n/lt_LT.json b/apps/settings/l10n/lt_LT.json
index d391e272309..b8267dbb2ad 100644
--- a/apps/settings/l10n/lt_LT.json
+++ b/apps/settings/l10n/lt_LT.json
@@ -232,6 +232,7 @@
"Change password" : "Pakeisti slaptažodį",
"Your profile picture" : "Jūsų profilio paveikslėlis",
"Remove profile picture" : "Šalinti profilio paveikslėlį",
+ "The file must be a PNG or JPG" : "Failas privalo būti PNG ar JPG tipo",
"Picture provided by original account" : "Paveikslėlis gautas iš pagrindinės paskyros",
"Set as profile picture" : "Nustatyti kaip profilio paveikslėlį",
"Please note that it can take up to 24 hours for your profile picture to be updated everywhere." : "Turėkite omenyje, kad gali praeiti iki 24 valandų, kol jūsų profilis bus visur atnaujintas.",
diff --git a/apps/systemtags/l10n/ar.js b/apps/systemtags/l10n/ar.js
index a8ad779a517..af865aca33f 100644
--- a/apps/systemtags/l10n/ar.js
+++ b/apps/systemtags/l10n/ar.js
@@ -56,9 +56,9 @@ OC.L10N.register(
"Delete" : "حذف ",
"Reset" : "إعادة تعيين",
"Loading …" : "التحميل جارٍ …",
- "Public" : "للعامة",
- "Restricted" : "مقيدة",
- "Invisible" : "غير مرئية",
+ "Public" : "عام",
+ "Restricted" : "مقيد",
+ "Invisible" : "غير مرئي",
"Created tag" : "تم إنشاء الوسم",
"Failed to create tag" : "فشل إنشاء الوسم",
"Updated tag" : "تم تحديث الوسم",
diff --git a/apps/systemtags/l10n/ar.json b/apps/systemtags/l10n/ar.json
index c3540824d74..1a64707f0b0 100644
--- a/apps/systemtags/l10n/ar.json
+++ b/apps/systemtags/l10n/ar.json
@@ -54,9 +54,9 @@
"Delete" : "حذف ",
"Reset" : "إعادة تعيين",
"Loading …" : "التحميل جارٍ …",
- "Public" : "للعامة",
- "Restricted" : "مقيدة",
- "Invisible" : "غير مرئية",
+ "Public" : "عام",
+ "Restricted" : "مقيد",
+ "Invisible" : "غير مرئي",
"Created tag" : "تم إنشاء الوسم",
"Failed to create tag" : "فشل إنشاء الوسم",
"Updated tag" : "تم تحديث الوسم",
diff --git a/apps/user_ldap/l10n/ar.js b/apps/user_ldap/l10n/ar.js
index 1c9110fe253..b38cfaec553 100644
--- a/apps/user_ldap/l10n/ar.js
+++ b/apps/user_ldap/l10n/ar.js
@@ -10,7 +10,7 @@ OC.L10N.register(
"No action specified" : "لم يتم تحديد أيّ إجراءٍ",
"No configuration specified" : "لم يتم تحديد أيّ إعداداتٍ",
"No data specified" : "لم يتم تحديد أيّ بياناتٍ",
- "Invalid data specified" : "البيانات المُحدّدة غير صالحة",
+ "Invalid data specified" : "البيانات المحددة غير صالحة",
" Could not set configuration %s" : "تعذّر تعيين الإعداد %s",
"Action does not exist" : "الإجراء غير موجود",
"Renewing …" : "التجديد جارٍ …",
@@ -86,7 +86,7 @@ OC.L10N.register(
"Only from these groups:" : "فقط من هذه المجموعات:",
"Search groups" : "مجموعات البحث",
"Available groups" : "المجموعات المتاحة",
- "Selected groups" : "المجموعات المُحدّدة",
+ "Selected groups" : "المجموعات المحددة",
"Edit LDAP Query" : "تحرير استعلام من خادم LDAP",
"LDAP Filter:" : "تصفية LDAP:",
"The filter specifies which LDAP groups shall have access to the %s instance." : "يحدد عامل التصفية أي مجموعات من LDAP سوف يكون لها حق الوصول إلى التطبيق %s.",
@@ -202,8 +202,8 @@ OC.L10N.register(
"User profile Address will be set from the specified attribute" : "خانة العنوان في الملف الشخصي للمستخدم سيتم تعيينها من الخاصّية المُحدّدة",
"Twitter Field" : "خانة حساب تويتر",
"User profile Twitter will be set from the specified attribute" : "خانة حساب تويتر في الملف الشخصي للمستخدم سوف يتم تعيينها من الخاصّية المُحدّدة",
- "Fediverse Field" : "خانة حساب الـ\"فيدي فيرس\" Fediverse",
- "User profile Fediverse will be set from the specified attribute" : "خانة حساب الـ\"فيدي فيرس\" Fediverse في الملف الشخصي للمستخدم سوف يتم تعيينها من الخاصّية المُحدّدة",
+ "Fediverse Field" : "خانة حساب Fediverse",
+ "User profile Fediverse will be set from the specified attribute" : "خانة حساب Fediverse في الملف الشخصي للمستخدم سوف يتم تعيينها من الخاصّية المُحدّدة",
"Organisation Field" : "خانة المؤسسة organization",
"User profile Organisation will be set from the specified attribute" : "خانة المنظمة organization في الملف الشخصي للمستخدم سوف يتم تعيينها من الخاصّية المُحدّدة",
"Role Field" : "خانة الوظيفة role",
diff --git a/apps/user_ldap/l10n/ar.json b/apps/user_ldap/l10n/ar.json
index 559eb77dadb..dcd769a68fb 100644
--- a/apps/user_ldap/l10n/ar.json
+++ b/apps/user_ldap/l10n/ar.json
@@ -8,7 +8,7 @@
"No action specified" : "لم يتم تحديد أيّ إجراءٍ",
"No configuration specified" : "لم يتم تحديد أيّ إعداداتٍ",
"No data specified" : "لم يتم تحديد أيّ بياناتٍ",
- "Invalid data specified" : "البيانات المُحدّدة غير صالحة",
+ "Invalid data specified" : "البيانات المحددة غير صالحة",
" Could not set configuration %s" : "تعذّر تعيين الإعداد %s",
"Action does not exist" : "الإجراء غير موجود",
"Renewing …" : "التجديد جارٍ …",
@@ -84,7 +84,7 @@
"Only from these groups:" : "فقط من هذه المجموعات:",
"Search groups" : "مجموعات البحث",
"Available groups" : "المجموعات المتاحة",
- "Selected groups" : "المجموعات المُحدّدة",
+ "Selected groups" : "المجموعات المحددة",
"Edit LDAP Query" : "تحرير استعلام من خادم LDAP",
"LDAP Filter:" : "تصفية LDAP:",
"The filter specifies which LDAP groups shall have access to the %s instance." : "يحدد عامل التصفية أي مجموعات من LDAP سوف يكون لها حق الوصول إلى التطبيق %s.",
@@ -200,8 +200,8 @@
"User profile Address will be set from the specified attribute" : "خانة العنوان في الملف الشخصي للمستخدم سيتم تعيينها من الخاصّية المُحدّدة",
"Twitter Field" : "خانة حساب تويتر",
"User profile Twitter will be set from the specified attribute" : "خانة حساب تويتر في الملف الشخصي للمستخدم سوف يتم تعيينها من الخاصّية المُحدّدة",
- "Fediverse Field" : "خانة حساب الـ\"فيدي فيرس\" Fediverse",
- "User profile Fediverse will be set from the specified attribute" : "خانة حساب الـ\"فيدي فيرس\" Fediverse في الملف الشخصي للمستخدم سوف يتم تعيينها من الخاصّية المُحدّدة",
+ "Fediverse Field" : "خانة حساب Fediverse",
+ "User profile Fediverse will be set from the specified attribute" : "خانة حساب Fediverse في الملف الشخصي للمستخدم سوف يتم تعيينها من الخاصّية المُحدّدة",
"Organisation Field" : "خانة المؤسسة organization",
"User profile Organisation will be set from the specified attribute" : "خانة المنظمة organization في الملف الشخصي للمستخدم سوف يتم تعيينها من الخاصّية المُحدّدة",
"Role Field" : "خانة الوظيفة role",
diff --git a/apps/user_ldap/l10n/gl.js b/apps/user_ldap/l10n/gl.js
index 1762815e36b..82812905004 100644
--- a/apps/user_ldap/l10n/gl.js
+++ b/apps/user_ldap/l10n/gl.js
@@ -51,7 +51,7 @@ OC.L10N.register(
"An unspecified error occurred. Please check log and settings." : "Produciuse un erro non especificado. Comprobe o rexistro e os axustes.",
"The search filter is invalid, probably due to syntax issues like uneven number of opened and closed brackets. Please revise." : "O filtro de busca é incorrecto, probabelmente por mor de erros de sintaxe como un número impar de chaves de apertura/peche. Revíseo.",
"A connection error to LDAP/AD occurred. Please check host, port and credentials." : "Produciuse un erro de conexión a LDAP/AD. Verifique a máquina, o porto e as credenciais.",
- "The \"%uid\" placeholder is missing. It will be replaced with the login name when querying LDAP/AD." : "Falta o marcador de posición «%u id». Substitúese polo nome de acceso ao consultar LDAP/AD.",
+ "The \"%uid\" placeholder is missing. It will be replaced with the login name when querying LDAP/AD." : "Falta o marcador de substitución «%uid». Substitúese polo nome de acceso ao consultar LDAP/AD.",
"Please provide a login name to test against" : "Forneza o nome de acceso para facer a proba",
"The group box was disabled, because the LDAP/AD server does not support memberOf." : "Desactivouse a caixa de grupo porque o servidor LDAP/AD non admite memberOf.",
"Password change rejected. Hint: " : "Contrasinal rexeitado. Consello:",
@@ -191,7 +191,7 @@ OC.L10N.register(
"Set the user's email from their LDAP attribute. Leave it empty for default behaviour." : "Definir o correo do usuario dende un atributo LDAP. Déixeo baleiro para un comportamento predeterminado.",
"User Home Folder Naming Rule" : "Regra de nomeado do cartafol do usuario",
"Leave empty for username (default). Otherwise, specify an LDAP/AD attribute." : "Deixar baleiro para o nome de usuario (predeterminado). Noutro caso, especifique un atributo LDAP/AD.",
- "\"$home\" Placeholder Field" : "Campo de marcador de posición «$home»",
+ "\"$home\" Placeholder Field" : "Campo de marcador de substitución «$home»",
"$home in an external storage configuration will be replaced with the value of the specified attribute" : "Nunha configuración de almacenamento externo substituirase $home polo valor do atributo especificado",
"User Profile Attributes" : "Atributos do perfil de usuario",
"Phone Field" : "Campo de teléfono",
diff --git a/apps/user_ldap/l10n/gl.json b/apps/user_ldap/l10n/gl.json
index e29dd2926d3..a46b652b144 100644
--- a/apps/user_ldap/l10n/gl.json
+++ b/apps/user_ldap/l10n/gl.json
@@ -49,7 +49,7 @@
"An unspecified error occurred. Please check log and settings." : "Produciuse un erro non especificado. Comprobe o rexistro e os axustes.",
"The search filter is invalid, probably due to syntax issues like uneven number of opened and closed brackets. Please revise." : "O filtro de busca é incorrecto, probabelmente por mor de erros de sintaxe como un número impar de chaves de apertura/peche. Revíseo.",
"A connection error to LDAP/AD occurred. Please check host, port and credentials." : "Produciuse un erro de conexión a LDAP/AD. Verifique a máquina, o porto e as credenciais.",
- "The \"%uid\" placeholder is missing. It will be replaced with the login name when querying LDAP/AD." : "Falta o marcador de posición «%u id». Substitúese polo nome de acceso ao consultar LDAP/AD.",
+ "The \"%uid\" placeholder is missing. It will be replaced with the login name when querying LDAP/AD." : "Falta o marcador de substitución «%uid». Substitúese polo nome de acceso ao consultar LDAP/AD.",
"Please provide a login name to test against" : "Forneza o nome de acceso para facer a proba",
"The group box was disabled, because the LDAP/AD server does not support memberOf." : "Desactivouse a caixa de grupo porque o servidor LDAP/AD non admite memberOf.",
"Password change rejected. Hint: " : "Contrasinal rexeitado. Consello:",
@@ -189,7 +189,7 @@
"Set the user's email from their LDAP attribute. Leave it empty for default behaviour." : "Definir o correo do usuario dende un atributo LDAP. Déixeo baleiro para un comportamento predeterminado.",
"User Home Folder Naming Rule" : "Regra de nomeado do cartafol do usuario",
"Leave empty for username (default). Otherwise, specify an LDAP/AD attribute." : "Deixar baleiro para o nome de usuario (predeterminado). Noutro caso, especifique un atributo LDAP/AD.",
- "\"$home\" Placeholder Field" : "Campo de marcador de posición «$home»",
+ "\"$home\" Placeholder Field" : "Campo de marcador de substitución «$home»",
"$home in an external storage configuration will be replaced with the value of the specified attribute" : "Nunha configuración de almacenamento externo substituirase $home polo valor do atributo especificado",
"User Profile Attributes" : "Atributos do perfil de usuario",
"Phone Field" : "Campo de teléfono",
diff --git a/apps/weather_status/l10n/nl.js b/apps/weather_status/l10n/nl.js
index 63f3b953e33..136af4bb8aa 100644
--- a/apps/weather_status/l10n/nl.js
+++ b/apps/weather_status/l10n/nl.js
@@ -14,6 +14,10 @@ OC.L10N.register(
"{temperature} {unit} clear sky" : "{temperature}{unit} onbewolkt",
"{temperature} {unit} cloudy later today" : "{temperature} {unit} bewolkt later vandaag",
"{temperature} {unit} cloudy" : "{temperature} {unit} bewolkt",
+ "{temperature} {unit} snow and thunder later today" : "{temperature}{unit} sneeuw en onweer later vandaag",
+ "{temperature} {unit} snow and thunder" : "{temperature}{unit} sneeuw en onweer",
+ "{temperature} {unit} snow later today" : "{temperature}{unit} sneeuw later vandaag",
+ "{temperature} {unit} snow" : "{temperature}{unit}sneeuw",
"{temperature} {unit} fair weather later today" : "{temperature} {unit} lekker weer later vandaag",
"{temperature} {unit} fair weather" : "{temperature} {unit} lekker weer",
"{temperature} {unit} partly cloudy later today" : "{temperature} {unit} half bewolkt later vandaag",
diff --git a/apps/weather_status/l10n/nl.json b/apps/weather_status/l10n/nl.json
index 2b9302cbb87..410344d4802 100644
--- a/apps/weather_status/l10n/nl.json
+++ b/apps/weather_status/l10n/nl.json
@@ -12,6 +12,10 @@
"{temperature} {unit} clear sky" : "{temperature}{unit} onbewolkt",
"{temperature} {unit} cloudy later today" : "{temperature} {unit} bewolkt later vandaag",
"{temperature} {unit} cloudy" : "{temperature} {unit} bewolkt",
+ "{temperature} {unit} snow and thunder later today" : "{temperature}{unit} sneeuw en onweer later vandaag",
+ "{temperature} {unit} snow and thunder" : "{temperature}{unit} sneeuw en onweer",
+ "{temperature} {unit} snow later today" : "{temperature}{unit} sneeuw later vandaag",
+ "{temperature} {unit} snow" : "{temperature}{unit}sneeuw",
"{temperature} {unit} fair weather later today" : "{temperature} {unit} lekker weer later vandaag",
"{temperature} {unit} fair weather" : "{temperature} {unit} lekker weer",
"{temperature} {unit} partly cloudy later today" : "{temperature} {unit} half bewolkt later vandaag",